之前写的字符类设备驱动,没有自动创建设备节点,因为只使用了register_chrdev()函数,只是注册了这个设备。然后在系统启动后,就要自己创建设备节点mknod,这样虽然是可行的,但是比较麻烦。于是想在__init函数里面,自动创建设备节点。
经过查阅资料,发现创建设备节点使用了两个函数 class_create()和class_device_create(),当然在__exit()函数里,要使用class_destory()和class_device_desotry()注销创建的设备节点!
问题来了,编译了之后,发现报错error: implicit declaration of 'class_device_create'等几个错误。经过分析,应该是Linux内核版本不同的原因!早期的版本,使用的是上面说的两个函数,但是在2.6.29以后(我用的是2.6.32的),使用的函数则变成了 class_create()和device_create(),并且要在声明中加入#i nclude <linux/device.h> ,因为定义这些函数是在Linux2.6.32/include/linux/device.h里面!
经过这些修改后,驱动编译成功,就能够自动创建设备节点了!
当然,在exit函数中要把创建的class移除:
class_destory(&xxx_dev->cdev);
class_device_desotry(my_class,MKDEV(major_num,0));
下面以一个简单字符设备驱动来展示如何使用这几个函数
#include <linux/module.h> #include <linux/kernel.h> #include <linux/init.h> #include <linux/fs.h> #include <linux/cdev.h> #include <linux/device.h> MODULE_LICENSE ("GPL"); int hello_major = 555; int hello_minor = 0; int number_of_devices = 1; struct cdev cdev; dev_t dev = 0; struct file_operations hello_fops = { .owner = THIS_MODULE }; static void char_reg_setup_cdev (void) { int error, devno = MKDEV (hello_major, hello_minor); cdev_init (&cdev, &hello_fops); cdev.owner = THIS_MODULE; cdev.ops = &hello_fops; error = cdev_add (&cdev, devno , 1); if (error) printk (KERN_NOTICE "Error %d adding char_reg_setup_cdev", error); } struct class *my_class; static int __init hello_2_init (void) { int result; dev = MKDEV (hello_major, hello_minor); result = register_chrdev_region (dev, number_of_devices, "hello"); if (result<0) { printk (KERN_WARNING "hello: can't get major number %d/n", hello_major); return result; } char_reg_setup_cdev (); /* create your own class under /sysfs */ my_class = class_create(THIS_MODULE, "my_class"); if(IS_ERR(my_class)) { printk("Err: failed in creating class./n"); return -1; } /* register your own device in sysfs, and this will cause udev to create corresponding device node */ device_create( my_class, NULL, MKDEV(hello_major, 0), "hello" "%d", 0 ); printk (KERN_INFO "Registered character driver/n"); return 0; } static void __exit hello_2_exit (void) { dev_t devno = MKDEV (hello_major, hello_minor); cdev_del (&cdev); device_destroy(my_class, MKDEV(adc_major, 0)); //delete device node under /dev class_destroy(my_class); //delete class created by us unregister_chrdev_region (devno, number_of_devices); printk (KERN_INFO "char driver cleaned up/n"); } module_init (hello_2_init); module_exit (hello_2_exit);