Linux中字符设备的创建过程

描述

Linux中很多设备都是字符设备,使用ls -l查看/dev下的设备,前面带c的都是字符设备。

存储

字符设备的创建比较容易,而且有一套固定的模式,掌握了模式完全可以“套模板”。更多的精力应该花在业务逻辑的部分。

Linux的每个设备都有一个主设备号和次设备号,创建字符设备第一步就是给它分配设备号。如果是自己用,可以任意给定一个没有用过的设备号,也可以让系统自动分配。但是如果要发布给不同的人不同的机器使用,就不能随便。推荐使用系统自动分配的方式。

下面代码中根据有没有给定设备号决定是指定还是自动分配设备号,dev变量里包含了主设备号和次设备号。MKDEV就是将两个数字合成的。register_chrdev_region()函数注册字符设备,alloc_chrdev_region()函数自动分配设备号。主设备号是关键,次设备号可以看成是该类设备的计数。通过MAJOR()可以得到主设备号。

存储

字符设备一般会创建一个数据结构,里边包含了cdev结构、设备数据和其他数据,可以根据自己实际需求添加。然后创建该结构的指针。

存储

分配完设备号,接下来给字符设备数据结构分配存储空间。

存储

创建设备类

存储

初始化字符设备、添加字符设备到内核和创建设备。

存储

这里涉及到一个结构file_operations,这个结构里包含了文件操作函数列表,所有操作这个字符设备的动作函数地址都保存在这个结构里。

存储

结构的定义在头文件fs.h中,这个结构有很多内容,不过常用的就那么几个。

存储

如果没遇到问题,到这里字符设备就创建成功了。如果失败,就会goto到错误处理的地方,这里用goto是因为失败后要把前面可能申请的资源释放掉,而且有顺序,这里一定要注意。

存储

在exit函数中要有对应的注销或释放资源的操作,顺序跟创建时相反,后创建的先释放,先创建的后释放。

存储

下面要完成的工作就是实现file_operations结构中的函数。对于简单的字符设备,最常见的操作是open、read、write、ioctl、release等。

open函数这里只是简单的获取了一下字符数据结构体的指针。container_of函数很有意思,可以根据结构成员的地址找到结构体数据的地址。

存储

read函数要实现将内核空间的数据传到用户空间的功能,copy_to_user()函数就是干这个的,第一个和第三个参数就是read函数的形参,中间是内核的数据。这里的count是字节个数。

存储

write函数实现的功能与read相反,使用copy_from_user()和memdup_user()函数可以实现。

ioctl函数实现一些设备独有操作的函数,i2c设备和spi设备就大量使用了ioctl。

release函数就是在设备文件关闭时需要做的操作。这里的关闭是指打开该文件的文件描述符个数为0了,如果不为0只会将计数减1而不会真正调用release函数。

具体的实例可以参看Linux源码中i2c-dev.c中i2c设备驱动的实现。

存储

必要的头文件:

存储

编译模块并插入到内核后,在/dev目录下会创建对应的设备文件。

存储

在/sys/class下会创建对应的设备类目录。

存储

ok,以上就是简单的创建字符设备的过程,通过字符设备可以实现内核与用户空间的数据交换。学习字符设备的创建也是学习linux驱动开发的第一步。

打开APP阅读更多精彩内容
声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉

全部0条评论

快来发表一下你的评论吧 !

×
20
完善资料,
赚取积分