使用cdev_add注册字符设备

本文介绍Linux 2.6内核中字符设备驱动的基础概念及实现方法,包括设备号的分配与管理、cdev结构的使用及其操作函数,以及字符设备的注册与注销流程。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

在前面已经提到设备号有主设备号和次设备号,其中主设备号表示设备类 型,对应于确定的驱动程序,具备相同主设备号的设备之间共用同一个驱动程 序,而用次设备号来标识具体物理设备。因此在创建字符设备之前,必须先获 得设备的编号(可能需要分配多个设备号)。

在 Linux 2.6 的版本中,用 dev_t 类型来描述设备号(dev_t 是 32 位数值类型,其 中高 12 位表示主设备号,低 20 位表示次设备号)。用两个宏 MAJOR 和 MINOR 分别 获得 dev_t 设备号的主设备号和次设备号,而且用 MKDEV 宏来实现逆过程,即组合 主设备号和次设备号而获得 dev_t 类型设备号。

分配设备号有静态和动态的两种方法。静态分配(register_chrdev_region()函数)是 指在事先知道设备主设备号的情况下,通过参数函数指定第一个设备号(它的次设备 号通常为 0)而向系统申请分配一定数目的设备号。动态分配(alloc_chrdev_region()) 是指通过参数仅设置第一个次设备号(通常为 0,事先不会知道主设备号)和要分配 的设备数目而系统动态分配所需的设备号。通过 unregister_chrdev_region()函数释放已分配的(无论是静态的还是动态的)设 备号。

在Linux 2.6内核中的字符设备用cdev结构来描述,其定义如下:

struct cdev   
{  
    struct kobject kobj;  
    struct module *owner; //所属模块  
    const struct file_operations *ops; //文件操作结构  
    struct list_head list;  
    dev_t dev; //设备号,int 类型,高12位为主设备号,低20位为次设备号  
    unsigned int count;  
};  

下面一组函数用来对cdev结构进行操作:
struct cdev *cdev_alloc(void);//分配一个cdev  
void cdev_init(struct cdev *, const struct file_operations *);//初始化cdev的file_operation  
void cdev_put(struct cdev *p);// //减少使用计数  
//注册设备,通常发生在驱动模块的加载函数中  
int cdev_add(struct cdev *, dev_t, unsigned);   
//注销设备,通常发生在驱动模块的卸载函数中  
void cdev_del(struct cdev *);  

    使用cdev_add注册字符设备前应该先调用register_chrdev_region或alloc_chrdev_region分配设备号。register_chrdev_region函数用于指定设备号的情况,alloc_chrdev_region函数用于动态申请设备号,系统自动返回没有占用的设备号。

    int register_chrdev_region(dev_t from, unsigned count, const char *name) ;  
    int alloc_chrdev_region(dev_t *dev,unsigned baseminor,unsigned count,const char *name); 
    

    函数传入值

    first:要分配的设备号的初始值

     count:要分配(释放)的设备号数目

     name:要申请设备号的设备名称(在/proc/devices 和 sysfs 中显示) 

    dev:动态分配的第一个设备号

    函数返回值

                    成功:0(只限于两种注册函数)

                    出错: 1(只限于两种注册函数)

    alloc_chrdev_region申请一个动态主设备号,并申请一系列次设备号。baseminor为起始次设备号,count为次设备号的数量。注销设备号(cdev_del)后使用unregister_chrdev_region:
    1. void unregister_chrdev_region(dev_t from,unsigned count) ; 

    这里讲解 2.6 内核中的字符设备的注册和注销过程。

    在 Linux 内核中使用 struct cdev 结构来描述字符设备,我们在驱动程序中必 须将已分配到的设备号以及设备操作接口(即为 struct file_operations 结构)赋予 struct cdev 结构变量。首先使用 cdev_alloc()函数向系统申请分配 struct cdev 结构, 再用 cdev_init()函数初始化已分配到的结构并与 file_operations 结构关联起来。最 后调用 cdev_add()函数将设备号与 struct  cdev 结构进行关联并向内核正式报告新 设备的注册,这样新设备可以被用起来了。

    如果要从系统中删除一个设备,则要调用 cdev_del()函数。

    例1.4  cdev_add注册字符设备实例

    代码见光盘\src\1drivermodel\1-4cdev。核心代码如下所示:

    1. struct file_operations simple_fops = {  
    2.     .owner =    THIS_MODULE,  
    3.     .read =     simple_read,  
    4.     .write =    simple_write,  
    5.     .open =     simple_open,  
    6.     .release =  simple_release,  
    7. };  
    8. /*******************************************************  
    9.                 MODULE ROUTINE  
    10. *******************************************************/  
    11. void simple_cleanup_module(void)  
    12. {  
    13.     dev_t devno = MKDEV(simple_MAJOR, simple_MINOR);  
    14.     if (simple_devices)   
    15.     {  
    16.         cdev_del(&simple_devices->cdev);  
    17.         kfree(simple_devices);  
    18.     }  
    19.     unregister_chrdev_region(devno,1);  
    20. }  
    21. //模块初始化  
    22. int simple_init_module(void)  
    23. {  
    24.     int result;  
    25.     dev_t dev = 0;  
    26.     dev = MKDEV(simple_MAJOR, simple_MINOR);  
    27.     result = register_chrdev_region(dev, 1, "DEMO");//申请设备号  
    28.     if (result < 0)   
    29.     {  
    30.         printk(KERN_WARNING "DEMO: can't get major %d\n", simple_MAJOR);  
    31.         return result;  
    32.     }  
    33.     simple_devices = kmalloc(sizeof(struct simple_dev), GFP_KERNEL);  
    34.     if (!simple_devices)  
    35.     {  
    36.         result = -ENOMEM;  
    37.         goto fail;  
    38.     }  
    39.     memset(simple_devices, 0, sizeof(struct simple_dev));  
    40.     //初始化设备结构  
    41.     cdev_init(&simple_devices->cdev, &simple_fops);  
    42.     simple_devices->cdev.owner = THIS_MODULE;  
    43.     simple_devices->cdev.ops = &simple_fops;  
    44.     result = cdev_add (&simple_devices->cdev, dev, 1);//添加字符设备  
    45.     if(result)  
    46.     {  
    47.         printk(KERN_NOTICE "Error %d adding DEMO\n", result);  
    48.         goto fail;  
    49.     }  
    50.     return 0;  
    51. fail:  
    52.     simple_cleanup_module();  
    53.     return result;  
    54. }  
    55. module_init(simple_init_module);  
    56. module_exit(simple_cleanup_module);  

    本例的应用层代码与运行结果同上例。
    <think>好的,我现在需要回答用户关于cdev_add函数的问题。首先,我得确认自己对这个函数的理解是否正确。cdev_add是Linux内核中的一个函数,用于字符设备驱动开发。不过具体细节可能需要进一步回忆或查阅资料。 用户可能是在学习如何编写字符设备驱动,或者遇到了相关的问题。他们想知道这个函数的作用以及如何使用它。因此,我需要详细解释函数的功能、参数以及使用时的注意事项。 首先,cdev_add的作用是什么?我记得它负责将字符设备结构(cdev注册到内核中,使得设备可以被用户空间访问。需要确认的是,它和cdev_init的关系,是否在初始化之后调用。另外,参数部分包括指向cdev结构的指针、设备号以及设备数量。主设备号和次设备号是通过dev_t类型传递的,可能涉及MAJOR和MINOR宏的使用。 接下来,用户可能需要知道使用这个函数的步骤。比如先调用cdev_init初始化cdev结构,设置file_operations,然后分配设备号(可能使用alloc_chrdev_region),最后调用cdev_add注册。同时,错误处理也很重要,比如注册失败需要释放设备号。 另外,需要注意在模块退出时要调用cdev_del来注销设备,避免资源泄漏。是否提到模块的退出函数?可能有必要,因为这是良好的编程实践。 可能用户会混淆cdev_add和register_chrdev的区别。后者是更老的接口,而cdev_add更灵活,支持动态分配主设备号。这一点需要说明清楚,帮助用户理解两者的区别和应用场景。 有没有常见的错误?比如忘记初始化cdev,或者设备号未正确分配,导致cdev_add失败。或者设备数量参数错误,导致次设备号超出范围。需要提醒用户检查这些点。 此外,是否应该给出一个简单的代码示例?虽然用户没有明确要求,但举例能帮助理解。例如,展示cdev_add在驱动模块初始化函数中的使用,包括错误处理的步骤。 总结一下,回答的结构应该是:函数的作用、参数说明、使用步骤、注意事项、可能的错误处理,以及与其他相关函数的对比。确保用中文清晰表达,符合用户的要求。</think>cdev_add函数是Linux内核中字符设备驱动的核心接口之一,主要作用是将初始化好的字符设备(cdev结构体)注册到内核中。以下是逐步说明: 1. **核心功能** - 将字符设备与设备号绑定,使设备对用户空间可见 - 触发内核创建`/dev`下的设备节点(需配合udev/mdev机制) - 建立设备文件操作(file_operations)与驱动程序的关联 2. **函数原型** ```c int cdev_add(struct cdev *p, dev_t dev, unsigned count); ``` - `p`:指向已初始化的cdev结构体指针 - `dev`:起始设备号(包含主/次设备号) - `count`:连续设备号的数量 3. **典型使用场景** ```c // 驱动模块初始化函数中: dev_t devno = MKDEV(major, minor); // 合成设备号 struct cdev *my_cdev = cdev_alloc(); // 分配cdev结构体 cdev_init(my_cdev, &fops); // 初始化操作函数集 int ret = cdev_add(my_cdev, devno, 1); // 注册单个设备 ``` 4. **注意事项** - 必须先调用`cdev_init()`初始化cdev结构体 - 设备号需提前通过`alloc_chrdev_region()`或`register_chrdev_region()`获取 - `count`参数决定创建多少个连续的次设备号 - 返回0表示成功,负数表示错误 5. **错误处理** ```c if (ret < 0) { printk(KERN_ERR "添加设备失败:%d\n", ret); unregister_chrdev_region(devno, 1); // 回滚设备号注册 return ret; } ``` 6. **对比旧接口** - 相比`register_chrdev()`,`cdev_add`+`alloc_chrdev_region`的组合: * 支持动态分配主设备号 * 允许单个驱动管理多个设备 * 更精细的资源控制 7. **配套操作** - 模块卸载时必须调用`cdev_del()`注销设备 - 需配合`class_create()`和`device_create()`实现自动节点创建 实际开发中约90%的字符设备驱动都会使用此接口。典型应用场景包括自定义硬件驱动、虚拟设备(如`/dev/null`)等需要实现文件操作的设备驱动。
    评论
    添加红包

    请填写红包祝福语或标题

    红包个数最小为10个

    红包金额最低5元

    当前余额3.43前往充值 >
    需支付:10.00
    成就一亿技术人!
    领取后你会自动成为博主和红包主的粉丝 规则
    hope_wisdom
    发出的红包
    实付
    使用余额支付
    点击重新获取
    扫码支付
    钱包余额 0

    抵扣说明:

    1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
    2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

    余额充值