在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. numpy 用于图像处理

    1. 转换为灰度图 灰度图的数据可以看成是二维数组,元素取值为0 ~ 255,其中,0为黑色,255为白色.从0到255逐渐由暗色变为亮色. 灰度图转换(ITU-R 601-2亮度变换): L = R ...

  2. 003.MongoDB主要概念

    一 对比关系 SQL术语/概念 MongoDB术语/概念 解释/说明 database database 数据库 table collection 数据库表/集合 row document 数据记录行 ...

  3. MongoDB+php7搭建

    0x00前言: 今天一位非计算机专业的朋友问我怎么打开.bson文件,我第一反应.bson文件是什么,网上查了下是mongodb的传输文件.也就是类似于mysql的.sql文件一样 之前看过mongo ...

  4. ubuntu安装虚拟环境

    首先 sudo pip install virtualenv sudo pip install virtualenvwrapper 然后进行配置 sudo gedit /.bashrc export ...

  5. Oracle no TOP, how to get top from order

    On ROWNUM and Limiting Results Our technologist explains how ROWNUM works and how to make it work fo ...

  6. 分布式存储MooseFS

    MooseFS 简介 http://moosefs.org/ MFS特性 安装和配置简单方便   可靠性高(数据的多个拷贝被存储在多个不同的服务器上)   通过添加新的服务器或硬盘就可以实现容量的动态 ...

  7. Codeforces.1028F.Make Symmetrical(结论 暴力)

    题目链接 \(Description\) \(q\)次操作,每次给定点的坐标\((x,y)\),表示加入一个点\((x,y)\),或删除一个点\((x,y)\),或询问:至少需要在平面中加入多少个点, ...

  8. Sql的行列(纵横表)转换

    1.行转列: 表结构和数据: DROP TABLE IF EXISTS `kj`; CREATE TABLE `kj` ( `姓名` ) DEFAULT NULL, `课程` ) DEFAULT NU ...

  9. Java并发编程(十一)-- Java中的锁详解

    上一章我们已经简要的介绍了Java中的一些锁,本章我们就详细的来说说这些锁. synchronized锁 synchronized锁是什么? synchronized是Java的一个关键字,它能够将代 ...

  10. php 创建返回结果配置文件 实例

    <?phpclass validateReturn{    //get return msg    function caseReturn($aRerurn)    {        $strM ...