open、release、
read、
write还记得这些file_operations中的函数指针么?静静地实现他们吧。
1、open方法
原型: int (*open)(struct inode *inode, struct file *filp);
任务:检查设备特定错误;如果设备首次打开,对其进行初始化;必要时,更新f_op指针;分配并填写置于filp->private_data里的数据结构;
2、release方法
原型: int (*release) (struct inode *, struct file *);
任务:open方法的逆操作
(1)释放由open分配的、保存在filp->private_data中的所有内容;
(2)在最后一次关闭操作时关闭设备;
个人认为谨记release中要做一些free操作是很有必要的。不知道你有没有,反正我经常是malloc后忘记free,很臭的一个毛病。
在Page 64前四段简明地表达了系统调用中open、fork、dup、close与内核中open、release、flush的对应关系,多看几遍这个,我想你有一种”哦,原来是这样“的感觉。(在这页提到一个dup的系统调用,第一次见,百度了一下是复制文件描述符,如果用到感觉会很方便,等用到的时候百度吧)
3、read & write
ssize_t read(struct file *filp, char __user *buff, size_t count, loff_t *offp);
ssize_t write(struct file *filp, const char __user *buff, size_t count, loff_t *offp);
Page 70、71中read 、write 方法的实现对于初学者来说,理解是比较困难的。本质上read、write 方法实现了copy_to_user、copy_from_user。理解的时候,可以根据用户层的read write方法对比来看。每一个参数都是一一对应的。
unsigned long copy_to_user(void __user *to, const void *from, unsigned long count); //read实现
unsigned long copy_from_user(void *to, const void __user *from, unsigned long count); //write实现
以下是我的read、write简单实现,以及对应用户层的read 、write 调用;你可以添加到自己的代码中运行看看。其中返回值并没有做处理,只是为了简要说明其功能。在驱动中后续会常用这两方法,后续完善(其实是有点犯懒了)
81 size_t scull_read(struct file *filp, char __user *buf, size_t count, loff_t *f_ops)
82 {
83 const char *kernel_buf = "hello scull read";
84
85 copy_to_user(buf, kernel_buf, 30);
86
87 printk(KERN_INFO "scull_read\n");
88
89 return 0;
90 }
91 size_t scull_write(struct file *filp, char __user *buf, size_t count, loff_t *f_ops)
92 {
93 printk(KERN_INFO "scull_write\n");
94 char kernel_buf[BUFF_SIZE];
95
96 copy_from_user(kernel_buf, buf, 30);
97 printk(KERN_INFO "%s\n", kernel_buf);
98 return 0;
99 }
15 read(fd, buff, 20);
16 puts(buff);
17 strcpy(buff, "I'm so happy");
18 write(fd, buff, 10);