内核input子系统分析

打开/driver/input/input.c 这就是input代码的核心

找到

static int __init input_init(void)

{

err = class_register(&input_class);注册一个类,注册后会在sys/class出现input目录.

......

err = register_chrdev(INPUT_MAJOR, "input", &input_fops);注册一个主设备号为13的input设备,

}

打开input_fops

static const struct file_operations input_fops = {
    .owner = THIS_MODULE,
    .open = input_open_file,
};

只有一个往内核中注册一个open函数,那么read,write,等函数也应该注册,则应该在input_open_file中注册,

分析static int input_open_file(struct inode *inode, struct file *file)

{

struct input_handler *handler = input_table[iminor(inode) >> 5];把input_table数组中的某一个成员赋值给handler,这个下面会具体分析

if (!handler || !(new_fops = fops_get(handler->fops)))获取handler中的file_operations结构体
        return -ENODEV;

old_fops = file->f_op;
    file->f_op = new_fops;

err = new_fops->open(inode, file);打开这个handler->fops->open函数

}

问1:input_table[iminor(inode) >> 5]这个数组是有谁来赋值的

int input_register_handler(struct input_handler *handler)

{

INIT_LIST_HEAD(&handler->h_list);

input_table[handler->minor >> 5] = handler;把handler赋值给input_table数组,位置以此设备号除以32

list_add_tail(&handler->node, &input_handler_list);把这个handler的节点加入input_handler_list链表中

list_for_each_entry(dev, &input_dev_list, node)一次遍历input_dev_list链表中的input_dev设备,然后对传进的handler进行匹配,寻找input_dev_list中有没有支持这个handler的设备
        input_attach_handler(dev, handler);具体的匹配方法接下来会详细分析

}

问2:那谁来调用input_register_handler,也就是handler从哪里来

搜索得到

Evbug.c (drivers\input):106:    return input_register_handler(&evbug_handler);
Evdev.c (drivers\input):745:    return input_register_handler(&evdev_handler);
Input.c (drivers\input):1193:int input_register_handler(struct input_handler *handler)
Input.c (drivers\input):1219:EXPORT_SYMBOL(input_register_handler);
Input.h (include\linux):1130:int input_register_handler(struct input_handler *);
Joydev.c (drivers\input):647:    return input_register_handler(&joydev_handler);
Keyboard.c (drivers\char):1367:    error = input_register_handler(&kbd_handler);
---- = input_register_handler(&kbd_handler); Matches (1 in 1 files) ----
Keyboard.c (drivers\char):1367:    error = input_register_handler(&kbd_handler);
那就打开keyboard.c以它为例详细分析这个是怎么注册的
static struct input_handler kbd_handler = {
    .event        = kbd_event,
    .connect    = kbd_connect,
    .disconnect    = kbd_disconnect,
    .start        = kbd_start,
    .name        = "kbd",
    .id_table    = kbd_ids,
};
int __init kbd_init(void)

{

error = input_register_handler(&kbd_handler);

}

其实就是吧kbd_handler这个结构体传递给input_table;

再总结一下:

int __init kbd_init(void)->error = input_register_handler(&kbd_handler)

{  input_table[handler->minor >> 5] = handler;

list_add_tail(&handler->node, &input_handler_list);

list_for_each_entry(dev, &input_dev_list, node)
        input_attach_handler(dev, handler);

}

分析如何匹配的 input_attach_handler(dev, kbd_hander)

{

id = input_match_device(handler->id_table, dev);如果匹配成功返回一个id

error = handler->connect(handler, dev, id);然后会根据返回的id找到设备和kbd_handler进行连接,这个连接connect函数就是kbd_handler->kbd_connect

}

接下来详细分析input_match_device

{

for (; id->flags || id->driver_info; id++) {

if (id->flags & INPUT_DEVICE_ID_MATCH_BUS)
            if (id->bustype != dev->id.bustype)
                continue;

.......
             //针对handler->id->flags,比较不同的类型,如果比较成功
                    //进入下面的宏比较,否则进入下一个id

MATCH_BIT(evbit,  EV_MAX);
          .............
           //安位比较所支持事件的类型,只有所有未匹配成功才能成功返回,否则进入下一id
        return id;
    }

return NULL;

}

如果匹配成功就会进入连接,分析kbd_connect

static int kbd_connect(struct input_handler *handler, struct input_dev *dev,const struct input_device_id *id)

{

struct input_handle *handle;定义一个input_hanlde指针

handle = kzalloc(sizeof(struct input_handle), GFP_KERNEL);分配input_handle结构体空间

handle->dev = dev;使分配的handle中的dev指向和handler匹配的input_dev
    handle->handler = handler;使handle中的handler指向和input_dev匹配的input_handler
    handle->name = "kbd";

error = input_register_handle(handle);注册这个新分配的handle;

error = input_open_device(handle);

}

继续分析

int input_register_handle(struct input_handle *handle)
{
    struct input_handler *handler = handle->handler;

list_add_tail(&handle->d_node, &handle->dev->h_list); 把handle->d_node挂在input_dev->h_list上
    list_add_tail(&handle->h_node, &handler->h_list);吧handle->h_node挂载handler->h_list上
    这样就可以通过input_handler中的h_list找到handle,然后通过handle->dev找到匹配的input_dev

同样也可以通过input_dev中的h_list找到handl,然后通过handle->handler找到匹配的input_handler

这样input_handler,input
    if (handler->start)
        handler->start(handle);

return 0;
}继续分析

int input_open_device(struct input_handle *handle)
{
    struct input_dev *dev = handle->dev;
    int err;
    handle->open++;//handle的打开计数加1,注意和evdev的open的区别

if (!dev->users++ && dev->open)
        err = dev->open(dev);//如果此input-dev没有进程引用,并且定义了open
        //方法,就调用open方法,

if (err) //retval=1说明,没有成功打开,
        handle->open--;
}

总结connect的作用就是根据匹配成功的input_dev和input_handler,创建一个新的结构体input_handle,然后把结构体中的dev和handler分配指向input_dev和input_handler

这样的话,就可以通过input_handle这个结构体把input_dev和input_hanlder连接起来;

static int __init input_init(void)->err = register_chrdev(INPUT_MAJOR, "input", &input_fops)->input_fops-> input_open_file

{struct input_handler *handler = input_table[iminor(inode) >> 5];

if (!new_fops->open) {
        fops_put(new_fops);
        return -ENODEV;
    }

err = new_fops->open(inode, file);这个open函数就是kbd_handler中的fops中的open;这个keyboard中没有初始化fops;

}

对于注册输入设备:
input_register_device
    // 放入链表
    list_add_tail(&dev->node, &input_dev_list);
    
    // 对于每一个input_handler,都调用input_attach_handler
    list_for_each_entry(handler, &input_handler_list, node)
        input_attach_handler(dev, handler); // 根据input_handler的id_table判断能否支持这个input_dev

input_attach_handler
    id = input_match_device(handler->id_table, dev);
    
    error = handler->connect(handler, dev, id);

和input_handler类似,大家自行分析

时间: 2024-08-08 17:40:44

内核input子系统分析的相关文章

linux kernel input 子系统分析

Linux 内核为了处理各种不同类型的的输入设备 , 比如说鼠标 , 键盘 , 操纵杆 , 触摸屏 , 设计并实现了一个对上层应用统一的试图的抽象层 , 即是Linux 输入子系统 . 输入子系统的层次结构体如下 从底层到上层 , input 子系统由 设备驱动层 , 核心层 , 以及事件处理层3个部分组成 当一个鼠标移动, 一个按键按下或弹起 , 它都需要从底层设备驱动-->核心层-->事件处理层 -->用户空间 , 层层上报 , 一直到运用程序. 应用这个input  子系统有如下优

全网络对Linux input子系统最清晰、详尽的分析

Linux input分析之二:解构input_handler.input_core.input_device 输入输出是用户和产品交互的手段,因此输入驱动开发在Linux驱动开发中很常见.同时,input子系统的分层架构思想在Linux驱动设计中极具代表性和先进性,因此对Linux input子系统进行深入分析很有意义. 本文继续在<Linuxinput子系统分析之一:软件分层>的基础上继续深入研究Linux输入子系统的分层架构思想以及其实现.软件分层探讨的是输入消息从底层硬件到内核.应用层

Linux input子系统编程、分析与模板

输入设备都有共性:中断驱动+字符IO,基于分层的思想,Linux内核将这些设备的公有的部分提取出来,基于cdev提供接口,设计了输入子系统,所有使用输入子系统构建的设备都使用主设备号13,同时输入子系统也支持自动创建设备文件,这些文件采用阻塞的IO读写方式,被创建在"/dev/input/"下.如下图所示.内核中的输入子系统自底向上分为设备驱动层,输入核心层,事件处理层.由于每种输入的设备上报的事件都各有不同,所以为了应用层能够很好识别上报的事件,内核中也为应用层封装了标准的接口来描述

linux input输入子系统分析《四》:input子系统整体流程全面分析

1      input输入子系统整体流程 本节分析input子系统在内核中的实现,包括输入子系统(Input Core),事件处理层(Event Handler)和设备驱动层.由于上节代码讲解了设备驱动层的写法,因此在开头部分会从设备驱动层做为线索,分析输入子系统和事件处理层是如何配合的,最后从用户角度出发,从"/dev/input/*"接口如何使用输入子系统提供的服务. 既然需要详细分析,有一个这样的流程图能够帮助我们在被绕进代码的过程中,找到出口,你能够知道你现在位于代码框架的什

input 输入子系统分析

//input 输入子系统分析//刘术河2016.08.22 linux-2.6.39-at91-2016.08.11-lsh\drivers\input\Input.c该文件下有input_register_device 和 input_register_handler 函数接口0.先分析input.c的核心层架构 input_init(void) class_register(&input_class); //注册一个类 register_chrdev(INPUT_MAJOR, "i

Android驱动之 Linux Input子系统之TP——A/B(Slot)协议

utm_source=tuicool&utm_medium=referral">点击打开链接 将A/B协议这部分单独拿出来说一方面是由于这部分内容是比較easy忽视的.周围大多数用到input子系统的开发者也不甚理解.还有一方面是由于这部分知识一旦扩展到TP(触摸屏Touch Panel)的多点触摸就要与Middleware/Framework一起结合起来看才干全然掌握,复杂性所在. 这里的Middleware/Framework是针对android来说的,本人从事android这

Android驱动之 Linux Input子系统之TP——A/B(Slot)协议【转】

转自:http://www.thinksaas.cn/topics/0/646/646797.html 将A/B协议这部分单独拿出来说一方面是因为这部分内容是比较容易忽视的,周围大多数用到input子系统的开发人员也不甚理解:另一方面是由于这部分知识一旦扩展到TP(触摸屏Touch Panel)的多点触摸就要与Middleware/Framework一起结合起来看才能完全掌握,复杂性所在.这里的Middleware/Framework是针对android来说的,本人从事android这几个层次的

input子系统详解

一.初识linux输入子系统 linux输入子系统(linux input subsystem)从上到下由三层实现,分别为:输入子系统事件处理层(EventHandler).输入子系统核心层(InputCore)和输入子系统设备驱动层. 对于输入子系统设备驱动层而言,主要实现对硬件设备的读写访问,中断设置,并把硬件产生的事件转换为核心层定义的规范提交给事件处理层.即将底层的硬件输入转化为统一事件形式,想输入核心(Input Core)汇报. 对于核心层而言,为设备驱动层提供了规范和接口.设备驱动

input子系统学习之四:核心层

核心层:给事件层和设备层提供接口,规范编程接口. 一.  输入子系统核心分析. 1.输入子系统核心对应与/drivers/input/input.c文件,这个也是作为一个模块注册到内核的.所以首先分析模块初始化函数. 1 .cnblogs 2. 输入子系统的核心其他部分都是提供的接口,向上连接事件处理层,向下连接驱动层.    向下对驱动层的接口主要有:    input_allocate_device    这个函数主要是分配一个input_dev接口,并初始化一些基本的成员,这就是我们不能简