LDD3学习笔记(4):字符驱动2-阿里云开发者社区

开发者社区> 宋金时> 正文

LDD3学习笔记(4):字符驱动2

简介: 1、重要的数据结构 注册设备编号仅仅是驱动代码需要完成的任务之一,还有很多基础性的驱动操作需要驱动代码来完成,这里有3个重要的内核数据结构需要了解一下分别是:file_operations、file、inode。
+关注继续查看

1、重要的数据结构

注册设备编号仅仅是驱动代码需要完成的任务之一,还有很多基础性的驱动操作需要驱动代码来完成,这里有3个重要的内核数据结构需要了解一下分别是:file_operationsfileinode

1.1、文件操作

File_operation结构的功能是建立一个字符驱动与设备编号的连接。通常结构中的每个成员必须指向驱动中的函数,这些函数实现一个特别的操作,对于不支持的操作置为NULL

__user这种注解是一种文档形式,表明一个指针是一个不能被直接解除引用的用户空间地址,对于正常的编译没有效果。

Struct module* owner

指向拥有这个结构的模块的指针。

Loff_t(*llseek) (struct file* , loff_t , int);

Llseek用来改变文件中当前读/写位置,并且新位置作为返回值,错误返回负值。

Ssize_t*read) (struct file* , char __user* , size_t , loff_t*);

用来从设备中获取数据,如果这个方法是NULL,将导致系统调用-EINVAL失败,返回非负值代笔成功读取的字节数(返回值是一个signed size 类型,通常是目标平台本地的整数类型)。

Ssize_t *aio_read) (struct kiocb * , char __user* , size_t , loff_t);

初始化一个异步读,肯恩在函数返回前不结束的读操作,如果这个方法是NULL,所有的操作将会由read代替进行(同步读)。

Ssize_t*write)(struct file* , const char __user * , size_t , loff_t *;

发送数据给设备,如果这个方法NULL,将返回-EINVAL给调用这个方法的程序,如果非负,返回值代表成功写的字节数。

Ssize_t*aio_write(struct kiocb* , const char __user* , size_t , loff_f *);

初始化设备上的一个异步写。

Int *readdir(struct file * , void * , filldir_t);

对于设备文件这个成员应当为NULL,它用来读取目录,并且仅对文件系统有用。

Unsigned int *poll(struct file* , struct poll_table_struct *);

Poll方法为3个系统调用的后端:poll epoll select,它们都用作查询一个或多个文件描述符的读和写是否会拥塞。Poll应当方法返回一个位掩码指示读或写是否拥塞,并提供给内核信息,使调用进程睡眠直到I/O变为可能,如果一个驱动的poll方法为NULL,则设备被假定为不阻塞可读可写。

Int *ioctl)(struct inode* struct file * unsigned int unsigned long;

Ioctl系统调用提供了发出设备特定命令的方法,如果设备不提供ioctl,对于任何事先未定义的请求,系统调用都会返回一个错误。

Int *mmap(struct file * , struct vm_area_struct *);

Mmap用来请求将设备内存映射到进程的地址空间。如果这个方法是NULLmmap系统调用返回-ENODEY

Int *open)(struct inode* struct file*);

这常常是对设备文件进行的第一个操作,不要求驱动声明一个对应的方法,如果这个方法是NULL,设备打开一只成功,但是你的驱动不会得到通知。

Int *flush(struct file *);

Flush操作在进程关闭它的设备文件描述符的拷贝时调用,它应当执行设备的任何未完成的操作。

Int *release)(struct inode * , struct file*);

在文件结构被释放时引用这个操作,如同openrelease可以为NULL

Int (*fsync)(struct file* , struct dentry * , int );

这个方法是fsync系统调用的后端,用户调用来刷新任何挂着的数据,如果这个指针是NULL,系统返回-EINVAL

Int *aio_fsync) (struct kiocb* , int);

Fsync方法的异步版本。

Int *fasync)(int , struct file* , int );

这个操作用来通知设备它的FASYNC标志发生改变,如果驱动不支持异步通知,这个成员可以使NULL

Int*lock)(struct file * , int , struct file_lock*;

Lock发放用来实现文件加锁,加锁对常规文件时必不可少的特性,但是设备驱动几乎从不实现它。

Ssize_t(*readv)(struct file* , const struct iovec* , unsigned long , loff_f*);

Ssize_t(*writev)(struct file* , const struct iovec* , unsigned long , loff_t*);

这个方法实现发散/汇聚读和写操作,应用程序偶尔需要做一个包含多个内存区的单个读或写操作,这个系统调用允许他们这样做而不必对数据进行额外拷贝,如果这些函数指针为NULLreadwrite方法被调用(可能多于一次)。

Ssize_t*sendfile)(struct file* , loff_t* , size_tread_actor_t,void *);

这个方法实现sendfile系统调用的读,使用最少的拷贝从一个文件描述符搬移数据到另一个,设备驱动常常使sendfileNULL

Ssize_t*sendpage)(struct file* , struct page* , int ,size_t ,loff_t* , int);

Sendpagesendfile的另一半,它由内核调用来发送数据,一次一页,到对应的文件,设备驱动通常不实现sendpage

Unsigned long*get_unmapped_area)(struct file * , unsigned long ,unsigned long ,unsigned long);

这个方法的目的是在进程的地址空间找一个合适的位置来映射在底层设备上的内存段中,大部分驱动可以置这个方法为NULL

Int *check_flags)(int);

这个方法允许模块检查传递给fnctlF_SETEL...)调用的标志。

Int*dir_notify)(struct file* , unsigned long);

这个方法在应用程序使用fnctl来请求目录改变通知时调用,只对文件系统有用,驱动不需要实现dir_notify

Scull设备驱动只实现最重要的设备方法,它的file_operations结构是如下初始化的:

Struct file_operations scull_fops={.owner = THIS_MODULE,

.llseek = scull_llseek,

.read = scull_read,

.write = scull_write,

.ioctl = scull_ioctl,

.open = scull_open,

,release = scull_release,

};

1.2、文件结构

Struct file是设备驱动中第二个重要的数据结构,注意file与用户空间程序的FILE指针没有任何关系,一个FILE定义在C库中,从不出现在内核代码中,一个struct file是内核结构,从不出现在用户程序中。

Struct file代表一个打开的文件,在内核空间,系统中每个打开的文件都有一个关联的struct file ,在内核open时创建,并传递给操作文件的函数,直到最后关闭,然后内核释放这个数据结构。

在内核源码中,struct file的指针常常称为file或者filpfile pointer,为避免混淆,以后file指结构,filp指结构指针。下面是file结构的一些重要成员:

Mode_t f_mode:文件模式

确定文件时可读或是可写,通过位FMODE_READFMODE_WRITE确定。

Loff_t f_pos:当前读写位置

Loff_t在所有平台都是64位的,驱动如果需要知道文件的当前位置,可以读这个值。

Unsigned int f_flags:文件标志

O_RDONLY O_NONBLOCK O_SYNC,驱动应该检查O_NONBLOCK标志来查看是否是请求非阻塞操作,其他的很少使用。

Struct file_operations *f_op:和文件关联的操作

Void *private_data:open系统调用设置这个指针为NULL,在为驱动调用open方法之前,你可自由使用这个成员或者忽略它。

Struct dentry* f_dentry:关联到文件的目录入口结构。

1.3inode结构

Inode结构由内核在内部用来表示文件。

有用的成员:

Dev_t i_rdev:对于代表设备文件的节点,这个成员包含实际的设备编号。

Struct cdev* i_cdev:

Struct cdev是内核的内部结构,代表字符设备。

从一个inode中获取主次编号的宏:

Unsigned int iminor(struct inode * inode);

Unsigned int imajor(struct inode* inode);

版权声明:本文内容由阿里云实名注册用户自发贡献,版权归原作者所有,阿里云开发者社区不拥有其著作权,亦不承担相应法律责任。具体规则请查看《阿里云开发者社区用户服务协议》和《阿里云开发者社区知识产权保护指引》。如果您发现本社区中有涉嫌抄袭的内容,填写侵权投诉表单进行举报,一经查实,本社区将立刻删除涉嫌侵权内容。

相关文章
LDD3学习笔记(2):建立和运行模块
1、建立开发环境 在开始编写代码以及研究代码之前,你需要有一个开发环境,也就是一个linux系统。通常我们的做法是在windows下安装一个虚拟机软件,然后在虚拟机软件中安装一个linux系统的发行版本,在众多的发行版本中我推荐ubuntu,不过具体还要看个人爱好。
939 0
LDD3学习笔记(3):字符驱动1
1、scull( Simple Character Utility for Loading  Localities)的设计 Scull是一个字符驱动,它操作一块内存区域,就好像它是一个设备,因此在以下的介绍中我们可以互换的使用设备和scull操作的内存区。
708 0
领域驱动设计(DDD)-简单落地
一、序言     领域驱动设计是一种解决业务复杂性的设计思想,不是一种标准规则的解决方法。在本文中的实战示例可能会与常见的DDD规则方法不太一样,是简单、入门级别,新手可以快速实践版的DDD。如果不熟悉DDD设计思想可看下基础思想篇 二、设计阶段     领域建模设计阶段常见的方法有 四色建模法、EventSourcing等 推荐一篇博文正确理解领域建
4435 0
Leanote云笔记本的搭建
Leanote云笔记本的搭建
201 0
+关注
宋金时
小米Android Framework及系统稳定性专家,专注Android系统改进及增强,研究跟进最新Android系统及特性,提升系统稳定性和用户体验
261
文章
0
问答
文章排行榜
最热
最新
相关电子书
更多
《2021云上架构与运维峰会演讲合集》
立即下载
《零基础CSS入门教程》
立即下载
《零基础HTML入门教程》
立即下载