忘记很早以前从哪里的找到的一个很简单的C代码,完成了字符设备注册,自动获取设备号(主/从),并且自动建立设备节点。
由于最近好几个人问我类似的问题,因此在这里贴出了,供大家参考。
感谢代码的原作者:zengxiaolong
以下代码在2..6.35-22内核下编译通过。
create_chrdev.c:
// create_chrdev.c
// ---------------------------------------------
#include <linux/types.h> //dev_t
#include <linux/cdev.h> //struct cdev
#include <linux/fs.h> //alloc_chrdev_region()
#include <linux/device.h> //class_create()
dev_t devid;
static struct cdev *led_cdev;
static int led_Major = 0;
static int led_Minor = 0;
static struct class *led_class;
static struct file_operations led_fops = {
.owner = THIS_MODULE,
};
static int __init hello_init(void)
{
int err;
//初始化cdev
led_cdev = cdev_alloc();
cdev_init(led_cdev, &led_fops);
led_cdev->owner = THIS_MODULE;
//动态获取主设备号(dev_t devid中包含"主设备号"和"次设备号"信息)
alloc_chrdev_region(&devid, 66, 1, "led");//从66开始分配1个设备号
led_Major = MAJOR(devid);
led_Minor = MINOR(devid);
printk(KERN_INFO "I was assigned major number %d.\n", led_Major);
printk(KERN_INFO "I was assigned minor number %d.\n", led_Minor);
//注册字符设备 (1)
err = cdev_add(led_cdev, devid, 1);
if (err) {
printk(KERN_NOTICE "Error %d adding device\n", err);
return -1;
}
led_class = class_create(THIS_MODULE, "led_class1");//在/sys目录下建立了一个一个led_class1的目录
if (IS_ERR(led_class)) {
printk(KERN_INFO "create class error\n");
return -1;
}
device_create(led_class,NULL,devid,NULL,"led" "%d", MINOR(devid));
//在2.6早期的版本应该使用class_device_create,早于2.6.27? 2.6.29? 忘记了。这一步在/dev目录下建立了一个led66的设备文件
return 0;
}
static void __exit hello_exit(void)
{
unregister_chrdev_region(devid, 1);
cdev_del(led_cdev);
device_destroy(led_class, devid);//在2.6早期的版本应该使用class_device_destroy,早于2.6.27? 2.6.29? 忘记了。
class_destroy(led_class);
}
module_init(hello_init);
module_exit(hello_exit);
MODULE_LICENSE("GPL");
MODULE_AUTHOR("zengxiaolong ");
MODULE_DESCRIPTION("A sample driver");
MODULE_SUPPORTED_DEVICE("testdevice");
Makefile(for PC):
obj-m := create_chrdev.o
KERNELDIR :=/lib/modules/$(shell uname -r)/build
default:
make -C $(KERNELDIR) M=$(shell pwd) modules
install:
insmod create_chrdev.ko
uninstall:
rmmod create_chrdev
clean:
make -C $(KERNELDIR) M=$(shell pwd) clean
Makefile(for ARM):
obj-m := create_chrdev.o
KERNELDIR := /usr/local/src/ARM2440/kernel-2.6.29 #it is your arm kernel folder
default:
make -C $(KERNELDIR) M=$(shell pwd) modules
install:
insmod create_chrdev.ko
uninstall:
rmmod create_chrdev
clean:
make -C $(KERNELDIR) M=$(shell pwd) clean