* 问: 当应用程序(APP, 这里指测试函数) 通过open. read. write 等系统调用函数访问某个设备文件时,Linux系统怎么知道去哪调用哪个驱动程序的file_operations
结构中的open. write. read 等成员那???????
* 答:等应用程序操作设备文件时,Linux 系统就会根据设备文件的类型(是字符设备还是块设备)、主设备号找到找到在内核中注册的file_operations 结构(如果是块设备就是block_device_operations结构),其中次设备号仅供驱动程序自身来分辨它是同类设备中的第几个。
字符设备驱动程序的编写过程:
1、编写驱动程序初始化函数:
进行必要的初始化,向内核注册驱动程序register_chrdev( major, "name", &s3c2440_leds_fops) ;其中参数分别为主设备号、名字、file_operations结构
这样,主设备号就和file_operations结构联系起来了,这样当应用程序操作设备文件时,Linux 系统就会根据设备文件的类型(是字符设备还是块设备)、 主设备号找到找到在内核中注册的file_operations 结构了,进而调用结构中的函数进行相应的操作。
2、构造填充要用用到的file_operations结构体中的函数
注意:
* 1. file_operations 结构是字符设备驱动程序的核心* 2. 当应用程序 执行open. read. write 等系统调用函数时, 最终会调用该结构中对应open. read. write 等的函数
* 3. 当执行#insmod first_drv.ko 时,调用的是s3c2440_leds_drv_init()函数:此时仅仅是加载了模块,告诉内核有这个功能,用不用看应用程序是否调用了,
因此,一般不在初始化函数中 设置硬件的一些初始化操作,例如设置摸引脚为输出,因为加载了未必使用,或者在其他模块中有其他的用途,一般在file_operations结构中对应的函数里设置,即使用时才去设置他。
major = register_chrdev(0, "myleds", &jz2440_leds_fops) ; // 0,表示自动分配主设备号
// 用来自动创建设备节点/dev/myleds
myleds_class = class_create(THIS_MODULE, "myleds");
myleds_class_dev = class_device_create(myleds_class, NULL, MKDEV(major, 0), NULL, "myleds"); // dev/mtleds