为了对输入子系统有一个清晰的认识,本节将分析输入系统的初始化过程。在 Linux 中,输入子系统作为一个模块存在,向上,为用户层提供接口函数,向下,为驱动层程序提供统一的接口函数。这样,就能够使输入设备的事件通过输入子系统发送给用户层应用程序,用户层应用程序也可以通过输入子系统通知驱动程序完成某项功能。
子系统初始化函数 input_init()
输入子系统作为一个模块存在,必然有一个初始化函数。在/drivers/input/input.c 文件中定义了输入子系统的初始化函数 input_init(),该函数的代码如下:
C/C++代码
- static int __init input_init(void)
- {
- int err;
- input_init_abs_bypass();
- err = class_register(&input_class);
-
-
-
- if (err) {
- printk(KERN_ERR "input: unable to register input_dev class\n");
- return err;
- }
-
- err = input_proc_init();
- if (err)
- goto fail1;
-
- err = register_chrdev(INPUT_MAJOR, "input", &input_fops);
-
-
-
-
- if (err) {
- printk(KERN_ERR "input: unable to register char major %d",
- INPUT_MAJOR);
- goto fail2;
- }
-
- return 0;
-
- fail2: input_proc_exit();
- fail1: class_unregister(&input_class);
- return err;
- }
文件打开函数 input_open_file()
文件操作指针中定义了 input_open_file()函数,该函数将控制转到 input_handler 中定义的fops 文件指针的 open()函数。该函数在 input_handler 中实现,这样就使不同的 handler 处理器对应了不同的文件打开方法,为完成不同功能提供了方便。input_open_file()函数的代码如下:
C/C++代码
- static int input_open_file(struct inode *inode, struct file *file)
- {
- struct input_handler *handler;
- const struct file_operations *old_fops, *new_fops = NULL;
- int err;
-
- err = mutex_lock_interruptible(&input_mutex);
- if (err)
- return err;
-
-
- handler = input_table[iminor(inode) >> 5];
-
-
-
- if (handler)
- new_fops = fops_get(handler->fops);
-
- mutex_unlock(&input_mutex);
-
-
-
-
-
-
- if (!new_fops || !new_fops->open) {
- fops_put(new_fops);
- err = -ENODEV;
- goto out;
- }
-
- old_fops = file->f_op;
- file->f_op = new_fops;
-
- err = new_fops->open(inode, file);
- if (err) {
- fops_put(file->f_op);
- file->f_op = fops_get(old_fops);
- }
- fops_put(old_fops);
- out:
- return err;
- }
阅读(657) | 评论(0) | 转发(0) |