1.file_operations数据结构
内核内部通过file结构识别设备,通过file_operations数据结构提供文件系统的入口点
函数。file_operations定义在<linux/fs.h>中的函数指针表。这个结构的每一个成员的名字都对应着一个系统调用。从某种意义上说,写驱动程序的任务之一就是完成file_operations中的函数指针。如果在2.4版本内核下开发的驱动很可能在2.6版本中无法使用,需要进行移植。通常file_operations提供了包括open()、write()、read()、release()、poll()、ioctl()等文件系统的入口函数。
下面简单描述一下几个重要的入口函数。
1)open():
static int mydriver_open(struct inode *inode, struct file *filp)
当上层对mydriver执行open操作时调用该函数,其中参数inode为设备特殊文件的inode(索引节点)结构指针,参数file是指向这一设备的文件结构指针。open()的主要任务是确定硬件处在就绪状态、验证次设备号的合法性(次设备号可以用MINOR(inode→i - rdev)取得)、控制使用设备的进程数、根据执行情况返回状态码(0表示成功,负数表示存在错误)等。
2)write():
static ssize_t mydriver_write(struct file *filp, const char *buf, size_t size, loff_t *offp)
当设备特殊文件进行write系统调用时,将调用驱动程序的write()函数,向设备发送数据。如果没有这个函数,write 系统调用会向调用程序返回一个-EINVAL。如果返回值非负,则表示成功写入的字节数。Write函数通常就是把数据从用户空间复制到内核空间,所以在write函数里经常会看到copy_from_user()函数。
3)read():
static ssize_t mydriver_read(struct file *filp, char *buf, size_t size, loff_t *offp)
当对设备特殊文件进行read系统调用时,将调用驱动程序read()函数,用来从设备中读取数据。当该函数指针被赋为NULL 值时,将导致read 系统调用出错并返回-EINVAL(“Invalid argument,非法参数”)。函数返回非负值表示成功读取的字节数(返回值为“signed size”数据类型,通常就是目标平台上的固有整数类型)。Read()函数则通常是把数据从内核空间复制到用户空间,一般都会调用copy_to_user()函数。
4)release():
static int mydriver_release(struct inode *inode, struct file *filp)
当最后一个打开设备的用户进程执行close()系统调用时,内核将调用驱动程序的release()函数,release()函数的主要任务是清理未结束的输入/输出操作、释放资源、用户自定义其他标志的复位等。
5)ioctl():
static int mydriver_ioctl(struct inode *inode,struct file *filp,unsigned int cmd,unsigned long arg)