1 介绍
内核提供了三个函数来注册一组字符设备编号,这三个函数分别是 register_chrdev_region()、alloc_chrdev_region() 和 register_chrdev()。
(1)register_chrdev 比较老的内核注册的形式 早期的驱动
(2)register_chrdev_region/alloc_chrdev_region + cdev 新的驱动形式
区别:register_chrdev()函数是老版本里面的设备号注册函数,可以实现静态和动态注册两种方法,主要是通过给定的主设备号是否为0来进行区别,为0的时候为动态注册。register_chrdev_region以及alloc_chrdev_region就是将上述函数的静态和动态注册设备号进行了拆分的强化。
2 函数分析
2.1 register_chrdev_region(dev_t first,unsigned int count,char *name)
(1)First :要分配的设备编号范围的初始值, 这组连续设备号的起始设备号, 相当于register_chrdev()中主设备号
(2)Count:连续编号范围. 是这组设备号的大小(也是次设备号的个数)
(3)Name:编号相关联的设备名称. (/proc/devices); 本组设备的驱动名称
2.2 int alloc_chrdev_region(dev_t *dev, unsigned baseminor, unsigned count, const char *name)
(1)dev:是输出型参数,获得一个分配到的设备号。可以用MAJOR宏和MINOR宏,将主设备号和次设备号,提取打印出来,看是自动分配的是多少,方便我们在mknod创建设备文件时用到主设备号和次设备号。 mknod /dev/xxx c 主设备号 次设备号
(2)baseminor:次设备号的基准,从第几个次设备号开始分配。
(3)count:次设备号的个数。
(4)name:驱动的名字。
(5)返回值:小于0,则错误,自动分配设备号错误。否则分配得到的设备号就被第一个参数带出来。
注意:
(1)register_chrdev_region是在事先知道要使用的主、次设备号时使用的;要先查看cat /proc/devices去查看没有使用的。
(2)更简便、更智能的方法是让内核给我们自动分配一个主设备号,使用alloc_chrdev_region就可以自动分配了。
(3)自动分配的设备号,我们必须去知道他的主次设备号,否则后面没法去mknod创建他对应的设备文件。
2.3 cdev介绍
cdev是一个结构体,里面的成员来共同帮助我们注册驱动到内核中,表达字符设备的,将这个struct cdev结构体进行填充,主要填充的内容就是
struct cdev {
struct kobject kobj;
struct module *owner;//填充时,值要为 THIS_MODULE,表示模块
const struct file_operations *ops;//这个file_operations结构体,注册驱动的关键,要填充成这个结构体变量
struct list_head list;
dev_t dev;//设备号,主设备号+次设备号
unsigned int count;//次设备号个数
};
file_operations这个结构体变量,让cdev中的ops成员的值为file_operations结构体变量的值。这个结构体会被cdev_add函数想内核注册。
cdev结构体,可以用很多函数来操作他。如:
cdev_alloc:让内核为这个结构体分配内存的
cdev_init:将struct cdev类型的结构体变量和file_operations结构体进行绑定的
cdev_add:向内核里面添加一个驱动,注册驱动
cdev_del:从内核中注销掉一个驱动。注销驱动
设备号
(1)dev_t类型(包括了主设备号和次设备号 ,不同的内核中定义不一样有的是16位次设备号和16位主设备号构成 有的是20位次设备号12位主设备号 )
(2)MKDEV、MAJOR、MINOR三个宏
MKDEV: 是用来将主设备号和次设备号,转换成一个主次设备号的。(设备号)
MAJOR: 从设备号里面提取出来主设备号的。
MINOR宏:从设备号中提取出来次设备号的。