字符设备驱动之cdev_init()系列函数
1.内核中每个字符设备都对应一个 cdev 结构的变量,下面是它的定义:
linux-2.6.22/include/linux/cdev.h
struct cdev {
13 struct kobject kobj;
14 struct module *owner;
15 const struct file_operations *ops;
16 struct list_head list;
17 dev_t dev;
18 unsigned int count;
19};
1>kobj是一个嵌入在该结构中的内核对象。它用于该数据结构的一般管理。
2>owner指向提供驱动程序的模块
3>ops是一组文件操作,实现了与硬件通信的具体操作。
4>dev指定了设备号
5>count表示与该设备关联的从设备的数目
6>list用来实现一个链表,其中包含所有表示该设备的设备特殊文件的inode.
2.一个 cdev 一般它有两种定义初始化方式:静态的和动态的
1>静态内存定义初始化:
struct cdev my_cdev;
cdev_init(&my_cdev, &fops);
my_cdev.owner = THIS_MODULE;
2>动态内存定义初始化:
struct cdev *my_cdev = cdev_alloc();
my_cdev->ops = &fops;
my_cdev->owner = THIS_MODULE;
两种使用方式的功能是一样的,只是使用的内存区不一样,一般视实际的数据结构需求而定。
3. 下面是具体实现
1>struct cdve * cdev_alloc(void)
struct cdev *cdev_alloc(void)
{
struct cdev *p = kzalloc(sizeof(struct cdev), GFP_KERNEL);
if (p) {
INIT_LIST_HEAD(&p->list);
kobject_init(&p->kobj, &ktype_cdev_dynamic);
}
return p;
}
2>void cdev_init(struct cdev *cdev, const struct file_operations *fops)函数
void cdev_init(struct cdev *cdev, const struct file_operations *fops)
{
memset(cdev, 0, sizeof *cdev);
INIT_LIST_HEAD(&cdev->list);
kobject_init(&cdev->kobj, &ktype_cdev_default);
cdev->ops = fops;
}
两个函数完成都功能基本一致,只是 cdev_init() 还多赋了一个 cdev->ops 的值。
cdev_init的参数fops包含了一些函数指针,指向处理与设备实际通信的函数
4.初始化 cdev 后,需要把它添加到系统中去。为此可以调用 cdev_add() 函数。传入 cdev 结构的指针,起始设备编号,以及设备编号范围。
int cdev_add(struct cdev *p, dev_t dev, unsigned count)
{
p->dev = dev;
p->count = count;
return kobj_map(cdev_map, dev, count, NULL, exact_match, exact_lock, p);
}
1>cdev_add的count参数表示该设备提供的从设备号的数量。在cdev_add成功返回后,设备进入活动状态。
2>kobj_map() 内核中所有都字符设备都会记录在一个 kobj_map 结构的 cdev_map 变量中。这个结构的变量中包含一个散列表用来快速存取所有的对象。kobj_map() 函数就是用来把字符设备编号和 cdev 结构变量一起保存到 cdev_map 这个散列表里。当后续要打开一个字符设备文件时,通过调用 kobj_lookup() 函数,根据设备编号就可以找到 cdev 结构变量,从而取出其中的 ops 字段。
5.当一个字符设备驱动不再需要的时候(比如模块卸载),就可以用 cdev_del() 函数来释放 cdev 占用的内存
void cdev_del(struct cdev *p)
{
cdev_unmap(p->dev, p->count);
kobject_put(&p->kobj);
}
其中 cdev_unmap() 调用 kobj_unmap() 来释放 cdev_map 散列表中的对象。kobject_put() 释放 cdev 结构本身。
分享到:
相关推荐
Linux 2.6标准字符设备驱动 实现了open() ,read(),write(),close(),ioctl()接口,并附带测试应用程序
Linux字符设备驱动模板, (1)分配一个file_operation结构体; (2)设置file_operation结构体; (3)注册 cdev (4)入口(模块加载); (5) 出口(卸载函数);
cdev_alloc函数[归类].pdf
本文档详细介绍了嵌入式linux中字符设备的cdev结构体的知识,通俗易懂,适合学习。
在Linux内核里面,设备(device)主要分为字符设备,块设备,网络设备,字符设备驱动是Linux驱动基础,在看《Linux 设备驱动开发详解》这本书的过程中,把字符设备相知识记录整理如下。 字符设备驱动的组成 字符设备...
static int __init i2c_dev_init(void) { printk("i2c init ...\n"); int res = register_chrdev(I2C_MAJOR,"i2c",&i2cdev_fops);//注册设备号 if(res) goto out; i2c_dev_class =class_create(THIS_MODULE,"i...
linux字符cdev和inode的联系,对字符设备有更深理解
第8章 编写i2c设备驱动程序模块的方法 61 第9章 用户进程访问i2c设备的步骤 64 讨论和总结 65 i2c操作中的同步问题 65 总结各个模块初始化函数的作用 65 对i2c框架代码的修改 66 有关i2c设备私有数据结构的讨论 68 ...
讨口饭吃,这个我的文章里面有的,囊中羞涩的老铁直接去文章cv
讨口饭吃,这个我的文章里面有的,囊中羞涩的老铁直接去文章cv
6.1.4 Linux字符设备驱动的组成 6.2一个字符设备驱动例子——virtualchar 6.2.1头文件、宏及设备结构体 6.2.2加载与卸载设备驱动 6.2.3驱动函数实现 6.2.4驱动设备私有数据 6.3对virtualchar设备的访问 第7章Linux...
基于linux2.4及以前的老方法字符设备注册,和基于linux2.6及之后的新方法字符设备注册,以及新方法注册过程中两种不同的初始化方法的区别
linux内核4.7版本设备驱动介绍,包含如下模块:class、pci、platform、pinctrl、kset、kobject、bus、device、device_driver、i2c、tty、regmap、misc、spin_lock、inode、mutex、cdev、gpio_keys、usb
.............\4.13.5字符设备驱动的file_operations 结构体中成员.c .............\4.14.1头文件、宏及设备结构体.c .............\4.14.2加载与卸载设备驱动.c .............\4.14.3读写函数.c ................
mini241o 按键的3种驱动 字符设备 混杂设备 platform驱动
此API(已通过Linux v4.4进行了稳定化)将旧sysfs接口与GPIO弃用,该接口计划在2020年之后(即将到来)从上游内核中删除。 如果您不需要定位较早的内核,则鼓励使用此API,而不要使用此板条箱的前身使用的sysfs ...
此API已通过Linux v4.4 gpio-cdev API文档进行了稳定化处理rust-gpio-cdev是一个Rust库/板条箱,可提供对GPIO字符设备ABI的访问。 此API(已通过Linux v4.4进行了稳定化)将旧sysfs接口与GPIO弃用,该接口计划在2020...
//向系统注册一个字符设备 cdev_add(&bio_dev.cdev, bio_dev.devno, 1); //MIO_PIN_50申请GPIO口 ret = gpio_request(MIO_PIN_50, "key"); //将原子变量置0,相当于初始化 atomic64_set(&bio_dev.state, 0);
Linux是文件型系统,... 例如,键盘这种设备提供的就是一个数据流,当你敲入“cnblogs”这个字 符串时, 键盘驱动程序会按照和输入完全相同的顺序返回这个由七个字符组成的数据流。它们是顺序的,先返回c,最后是s。