extern struct device *device_create(struct class *cls, struct device *parent,
dev_t devt, void *drvdata,
const char *fmt, ...)
__attribute__((format(printf, 5, 6)));</span>
之前写的字符类设备驱动,没有自动创建设备节点,因为只使用了register_chrdev()函数,只是注册了这个设备。然后在系统启动后,就要自己创建设备节点mknod,这样虽然是可行的,但是比较麻烦。于是想在__init函数里面,自动创建设备节点。
函数功能:
函数device_create()用于动态的建立逻辑设备,并对新的逻辑设备类进行相应初始化,将其与函数的第一个参数所代表的逻辑类关联起来,然后将此逻辑设备加到linux内核系统的设备驱动程序模型中。函数能够自动在/sys/devices/virtual目录下创建新的逻辑设备目录,在/dev目录下创建于逻辑类对应的设备文件
参数说明:
struct class cls:与即将创建额逻辑设备相关的逻辑类。
dev_t dev:设备号
void *drvdata: void类型的指针,代表回调函数的输入参数
const char *fmt: 逻辑设备的设备名,即在目录 /sys/devices/virtual创建的逻辑设备目录的目录名。
#include <linux/module.h>
#include <linux/kernel.h>
#include <linux/fs.h>
#include <linux/init.h>
#include <linux/delay.h>
#include <linux/irq.h>
#include <asm/uaccess.h>
#include <asm/irq.h>
#include <asm/io.h>
#include <linux/poll.h>
#include <linux/device.h>
static struct class *Myleddrv_class; //自动注册驱动主设备
static struct device *Myleddrv_dev;
volatile unsigned long *gpbcon = NULL; //控制寄存器
volatile unsigned long *gpbdat = NULL; //数据寄存器
static int Myleddrv_open(void)
{
printk("Hello Linux World!\n");
return 0;
}
static int Myleddrv_write(void)
{
return 0;
}
static struct file_operations Myleddrv_fops = {
.owner = THIS_MODULE, /* 这是一个宏,推向编译模块时自动创建的__this_module变量 */
.open = Myleddrv_open,
.write = Myleddrv_write,
};
static int major; //全局变量
static int Myleddrv_init(void)
{
major = register_chrdev(0, "Myleddrv", &Myleddrv_fops); // 注册, 告诉内核
Myleddrv_class = class_create(THIS_MODULE, "Myleddrv");
Myleddrv_dev = device_create(Myleddrv_class, NULL, MKDEV(major, 0), NULL, "Myleddrv"); /* /dev/Myleddrv */
return 0;
}
static void Myleddrv_exit(void)
{
unregister_chrdev(major, "Myleddrv"); // 卸载
printk("Myleddrv has been unregistered!\n");
device_unregister(Myleddrv_dev);
class_destroy(Myleddrv_class,major);
}
module_init(Myleddrv_init);
module_exit(Myleddrv_exit);
MODULE_LICENSE("GPL");