概述
前面已经成功的注册了一个字符设备驱动了,不过呢,还有一个结构体是空的,就是file_operations,这个结构体是字符设备驱动的核心东西了,前面也说过,字符设备驱动其实就是实现这个结构体里的函数了.这个结构体比较庞大啊,好像比我电脑的屏幕大多了,不过,这里的结构体是按需实现,也就是说你不需要全部实现,按特定情况实现就是了,比如一个字符设备驱动在最简的时候只需要实现 ioctl这一个函数就可以了!下面还是来看一下这个结构体吧:
struct file_operations {
struct module *owner;
loff_t (*llseek) (struct file *, loff_t, int);
ssize_t (*read) (struct file *, char __user *, size_t, loff_t *);
ssize_t (*write) (struct file *, const char __user *, size_t, loff_t *);
ssize_t (*aio_read) (struct kiocb *, const struct iovec *, unsigned long, loff_t);
ssize_t (*aio_write) (struct kiocb *, const struct iovec *, unsigned long, loff_t);
int (*readdir) (struct file *, void *, filldir_t);
unsigned int (*poll) (struct file *, struct poll_table_struct *);
long (*unlocked_ioctl) (struct file *, unsigned int, unsigned long);
long (*compat_ioctl) (struct file *, unsigned int, unsigned long);
int (*mmap) (struct file *, struct vm_area_struct *);
int (*open) (struct inode *, struct file *);
int (*flush) (struct file *, fl_owner_t id);
int (*release) (struct inode *, struct file *);
int (*fsync) (struct file *, loff_t, loff_t, int datasync);
int (*aio_fsync) (struct kiocb *, int datasync);
int (*fasync) (int, struct file *, int);
int (*lock) (struct file *, int, struct file_lock *);
ssize_t (*sendpage) (struct file *, struct page *, int, size_t, loff_t *, int);
unsigned long (*get_unmapped_area)(struct file *, unsigned long, unsigned long, unsigned long, unsigned long);
int (*check_flags)(int);
int (*flock) (struct file *, int, struct file_lock *);
ssize_t (*splice_write)(struct pipe_inode_info *, struct file *, loff_t *, size_t, unsigned int);
ssize_t (*splice_read)(struct file *, loff_t *, struct pipe_inode_info *, size_t, unsigned int);
int (*setlease)(struct file *, long, struct file_lock **);
long (*fallocate)(struct file *file, int mode, loff_t offset,loff_t len);
};
下面简单说一下常用的几个吧,其余的我也不知道了,又忘了,选择性失忆:
struct module *owner; //一般直接指定为THIS_MODULE
1:文件指针操作函数
这个函数是文件指针操作函数,对应于系统调用的 off_t lseek(int fildes, off_t offset, int whence),这个系统调用做什么的我又忘了,自己查去.当我们调用这个系统调用操作字符设备的时候,那么实际上就会调用这个结构体里的这个函数.第一个参数是文件指针,第二个参数是移动的字节数,第三个参数是相对位置.原形如下:
loff_t (*llseek) (struct file *filp, loff_t offset, int whence);
2:文件打开函数
此函数将在我们对字符设备节点进行打开操作的时候,也就是使用系统调用的open函数的时候调用.第一个参数是inode形的结构体,这表示一个文件,就是我们的字符设备文件结点,是内核内部用来表示文件的,第二个参数表示打开的文件,一个文件可以被打开多次,所以会有多个filp存在.正常情况下这个函数运行成功应该返回0;我们可以在这个函数里做一些对设备的打开操作,等.
int (*open) (struct inode * ind, struct file * filp);
3:读数据操作函数
这个函数就是用户空间要从字符设备里读数据的时候调用read系统调用的时候调用的,第一个参数是打开的文件指针,第二个参数是用户空间的缓冲区地址,第三个参数size_t size是要读的字节数,第4个参数loff_t * offset是文件指针;注意这个文件指针,我们在用户空间调用read,write系统调用每读一次文件的时候是不是文件指针都会自动移动吗?就是这个参数,在驱动里需要我们自己维护这个参数,当然内核不会帮你做,我也不可能帮你做的,上帝更不会了,所以不要忘了!当然如果你不需要这个参数,也可以不处理它.函数正常处理后应该返回读取的字节数,错误返回小于0的值;
ssize_t (*read) (struct file *filp, char __user * buff, size_t size, loff_t * offset);
注意:这里的读取是从内核空间把数据往用户空间copy,当然我们不能直接使用memcpy这样的函数,内核提供了2个函数来做这些事情,一个是内核向用户空间copy,一个是用户空间向内核copy.函数远行如下:
unsigned long copy_to_user(void __user *to, const void *from, unsigned long n);
unsigned long copy_from_user(void *to, const void __user *from, unsigned long n);
我们应该检测这个2个函数的返回值,以判断是否执行成功,因为内核认为这2个函数的执行是很重要的,所以一定要对它们进行检测,看是否运行成功.记住这函数如果copy成功,将会返回0,如果没有copy成功,将会返回没有copy成功的字节数.
4:写数据操作函数
这个函数就是用户空间向内核写数据的时候所调用的函数,这个函数的参数和读操作函数一样,在操作成功后,应该返回实际写入的字节数,如操作失败,则应该返回小于0的值.同样应该使用内核提供的copy_from_user函数来操作数据。
ssize_t (*write) (struct file *, const char __user *, size_t, loff_t *);
5:文件关闭函数
这个函数是在调用close系统调用的时候调用的,基本上做和open函数向反的工作,就不多写了。原形如下啊:
int (*release) (struct inode *, struct file *);
总结:今天没写什么,就是抄了不少东西过来,呵呵,这样抄一次,选择性失忆的可能性就小多了!ps:open和release如果不实现,内核是会提供一个默认的函数的。
最后
以上就是成就黄蜂为你收集整理的linux中字符三设备文件,Linux字符设备驱动(三)-文件操作函数实现的全部内容,希望文章能够帮你解决linux中字符三设备文件,Linux字符设备驱动(三)-文件操作函数实现所遇到的程序开发问题。
如果觉得靠谱客网站的内容还不错,欢迎将靠谱客网站推荐给程序员好友。
发表评论 取消回复