在Linux下的输入设备键盘、触摸屏、鼠标等都能够用输入子系统来实现驱动。输入子系统分为三层,核心层和设备驱动层。事件层。核心层和事件层由Linux输入子系统本身实现,设备驱动层由我们实现。我们在设备驱动层将输入事件上报给核心层input.c,核心层找到匹配的事件层,将事件交给事件层处理,事件层处理完后传递到用户空间。

我们终于要搞清楚的是在用户空间调用open和read终于在内核中是如何处理的,向内核上报的事件又是谁处理的,处理完后是如何传递到用户空间的?

上面两个图是输入子系统的框架。

以下以按键驱动为例分析输入子系统的工作流程。

设备驱动层:

在设备驱动层的init入口函数中调用input_allocate_device(),分配返回一个input_dev结构体,填充该结构体,调用input_register_device(button_dev),注冊设备。跟踪input_register_device例如以下:

list_add_tail(&dev->node, &input_dev_list);将input_dev结构体放进input_dev_list链表中

list_for_each_entry(handler, &input_handler_list, node)遍历input_handler_list链表中的每个handler

input_attach_handler(dev, handler);将input_dev和handler比較

input_match_device(handler->id_table, dev);<*input.c*>

handler->connect(handler, dev, id);<*input.c*>比較的方式是通过对照handler的id_table和input_dev。若找
 到匹配的handler。则调用handler的connect方法。

在evdev_connect(struct input_handler *handler, struct input_dev *dev,const struct input_device_id *id)中调用了
input_register_handle,调用handle之前填充handle中成员dev和handler结构体。

evdev->handle.dev = input_get_device(dev);

evdev->handle.handler = handler;

在input_register_handle中list_add_tail_rcu(&handle->d_node, &dev->h_list)list_add_tail(&handle->h_node, &h andler->h_list)两个函数将handle增加进handler和dev的h_list链表中。

通过dev->h_list能够找到handle。通过
handle找到handle.handler,同理。通过handler->h_list能够找到handle,再找到handle.dev。

关键的问题来了?handler究竟是什么?由谁注冊的?handler是事件层调用核心层的函数注冊的。

事件层:(evdev.c,keyboard.c,ts.c)

static struct input_handler evdev_handler = {//handler结构体

.event
= evdev_event,

.connect
= evdev_connect,

.disconnect
= evdev_disconnect,

.fops = &evdev_fops,

.minor
= EVDEV_MINOR_BASE,

.name = "evdev",

.id_table
= evdev_ids,

};

在evdev_init(void)入口函数中调用input_register_handler(&evdev_handler)来注冊handler,注冊的handler会放入input_table[ ]数组中。

核心层:(input.c)

核心层的入口函数input_init(void)注冊设备input。

当在应用层调用open时会在内核中调用input_open_file(struct inode *inode, struct file *file)。依据handler =
input_table[iminor(inode) >> 5];依据打开的文件的次设备号从数组input_table中得到已注冊的相应的handler。

new_fops = fops_get(handler->fops)从handler中得到新的fop。

file->f_op = new_fops;

err = new_fops->open(inode, file);open终于调用的是handler->fops->open。

原来的设备驱动的open等方法是自
己写的。如今输入子系统中的事件层帮我们写好了open等设备方法。同理在应用层调用read会调用事件层中的
read。即handler->fops->read,即evdev_fops.read,在read中会堵塞。直到在设备驱动层中过input_report_key
上报事件:

input_event(dev, EV_KEY, code, !!value);

input_handle_event(dev, type, code, value);

input_pass_event(dev, type, code, value);

handle->handler->event(handle,type, code, value);调用handler中的event事件方法。处理完上报的事
件后,唤醒休眠的read,再read出事件的处理结果。

总结:

在用户空间调用open将终于调用事件层中handler的fops的open设备方法。详细是匹配到evdev.c还是keyboard.c的handler要依据设备驱动init入口函数中填充的input_dev结构体的id_table。

用户空间调用read将调用handler的read。

在设备驱动中通过input_event上报事件到核心层。终于调用相应的handler的event方法来处理事件,处理完后通过read传递到用户空间。

这样就搞清楚了open是谁调用的?read是谁调用的?

Linux输入子系统框架分析(1)的更多相关文章

  1. 7.Linux 输入子系统分析

    为什么要引入输入子系统? 在前面我们写了一些简单的字符设备的驱动程序,我们是怎么样打开一个设备并操作的呢? 一般都是在执行应用程序时,open一个特定的设备文件,如:/dev/buttons .... ...

  2. Linux输入子系统详解

    input输入子系统框架  linux输入子系统(linux input subsystem)从上到下由三层实现,分别为:输入子系统事件处理层(EventHandler).输入子系统核心层(Input ...

  3. Linux USB驱动框架分析 【转】

    转自:http://blog.chinaunix.net/uid-11848011-id-96188.html 初次接触与OS相关的设备驱动编写,感觉还挺有意思的,为了不至于忘掉看过的东西,笔记跟总结 ...

  4. linux输入子系统之按键驱动

    上一节中,我们讲解了Linux  input子系统的框架,到内核源码里详细分析了输入子系统的分离分层的框架等. 上一节文章链接:http://blog.csdn.net/lwj103862095/ar ...

  5. linux驱动基础系列--linux spi驱动框架分析

    前言 主要是想对Linux 下spi驱动框架有一个整体的把控,因此会忽略某些细节,同时里面涉及到的一些驱动基础,比如平台驱动.设备模型等也不进行详细说明原理.如果有任何错误地方,请指出,谢谢! spi ...

  6. Linux USB驱动框架分析【转】

    转自:http://blog.csdn.net/jeffade/article/details/7701431 Linux USB驱动框架分析(一) 初次接触和OS相关的设备驱动编写,感觉还挺有意思的 ...

  7. Linux输入子系统(Input Subsystem)

    Linux输入子系统(Input Subsystem) http://blog.csdn.net/lbmygf/article/details/7360084 input子系统分析  http://b ...

  8. linux输入子系统

    linux输入子系统(linux input subsystem)从上到下由三层实现,分别为:输入子系统事件处理层(EventHandler).输入子系统核心层(InputCore)和输入子系统设备驱 ...

  9. linux输入子系统概念介绍

    在此文章之前,我们讲解的都是简单的字符驱动,涉及的内容有字符驱动的框架.自动创建设备节点.linux中断.poll机制.异步通知.同步互斥.非阻塞.定时器去抖动. 上一节文章链接:http://blo ...

随机推荐

  1. 基于Keil软件的MCU环境搭建

    我们在开发一款新的MCU的时候,偶尔会遇到Keil软件没有对应的Device设备选型,以下,我们以STM32F407VGT6作为实例来演示整个环境的搭建过程: 一.如下所示,我需要选择的是ST公司的S ...

  2. 错误 在类中找不到main方法请将main方法定义为 public static void main String args否则JavaFX应用程序类必须扩展javafx-ap

    最近在使用eclipse编写java程序时遇到这样一个问题: 错误在类中找不到main方法,请将main方法定义为 public static void main(String[] args)否则 J ...

  3. hdu1285 确定比赛名次【拓扑排序】

    题目链接 确定比赛名次                                         Time Limit: 2000/1000 MS (Java/Others)    Memory ...

  4. f5到底刷新了点什么,你知道吗

    引言 前面翻到了http缓存相关内容,关于强制缓存和协商缓存,他们之间的差别可能大家比较清楚. 并且常规情况下是否该使用缓存以及使用哪种缓存, 相关文章多且全,这里不再赘述. 不过用户的不同行为会打破 ...

  5. APP开发,微信第三方登录的介绍

    去年做了一阵APP相关的开发,经常遇到第三方登陆的需求,比如微信.微博.fb的第三方登陆等等,其实主要的流程都大同小异,这里就以微信为例来介绍,希望对大家有帮助. 微信开放平台(open.weixin ...

  6. 码云,git使用 教程

    码云,git使用 教程 code cloud, git use tutorials 作者:韩梦飞沙 Author:han_meng_fei_sha 邮箱:313134555@qq.com E-mail ...

  7. Centos6 安装RabbitMq3.7.7

    安装包准备官网地址:    Erlang安装包下载:https://www.erlang-solutions.com/resources/download.html    RabbitMq安装包下载: ...

  8. Java基础(十三) 文件高级技术

    文件高级技术 一.常见文件类型处理 一)属性文件 属性文件很简单,一行表示一个属性,属性就是键值对,键和值用(=)或者(:)分隔. #ready to work name = tang age = p ...

  9. GPIO知识点整理

    //GPIO的作业,抄两次,注意:本文件是知识点的整理不是可以直接运行的程序. //STM32必须包含的头文件 #include "stm32f10x.h" //GPIO相关头文件 ...

  10. IDEA中maven模块变成灰色

    可能该模块被忽略,解决办法: