在前文 Linux/Android——input子系统核心 (三)中概括了总体的结构,以及介绍了input核心的职责,其中有说道注册input设备时会去匹配已有的事件处理器handler,

而这个handler也是存放在一个链表里面的,这里介绍下input子系统中的事件处理input_handler机制.

撰写不易,转载需注明出处:http://blog.csdn.net/jscese/article/details/42238377#t6

evdev:

/kernel/drivers/input下众多事件处理器handler其中的一个,可以看下源码/kernel/drivers/input/evdev.c中的模块init:

[objc] view plaincopyprint?
  1. static int __init evdev_init(void)
  2. {
  3. return input_register_handler(&evdev_handler);
  4. }

这个初始化就是往input核心中注册一个input_handler类型的evdev_handler,调用的是input.c提供的接口,input_handler结构前文有介绍,看下evdev_handler的赋值:

[objc] view plaincopyprint?
  1. static struct input_handler evdev_handler = {
  2. .event      = evdev_event,
  3. .connect    = evdev_connect,
  4. .disconnect = evdev_disconnect,
  5. .fops       = &evdev_fops,
  6. .minor      = EVDEV_MINOR_BASE,
  7. .name       = "evdev",
  8. .id_table   = evdev_ids,
  9. };

赋值各个函数指针!

input_register_handler:

可以看到上面的evdev handler 就是调用这个接口注册到input核心中的,同样evdev.c同目录下也还有其它的handler,有兴趣可以看看它们的init函数,都是会调用到这个接口去注册的.

[objc] view plaincopyprint?
  1. /**
  2. * input_register_handler - register a new input handler
  3. * @handler: handler to be registered
  4. *
  5. * This function registers a new input handler (interface) for input
  6. * devices in the system and attaches it to all input devices that
  7. * are compatible with the handler.
  8. */
  9. int input_register_handler(struct input_handler *handler)
  10. {
  11. struct input_dev *dev;
  12. int retval;
  13. retval = mutex_lock_interruptible(&input_mutex);
  14. if (retval)
  15. return retval;
  16. INIT_LIST_HEAD(&handler->h_list);
  17. if (handler->fops != NULL) {
  18. if (input_table[handler->minor >> 5]) {
  19. retval = -EBUSY;
  20. goto out;
  21. }
  22. input_table[handler->minor >> 5] = handler; //给input.c定义的全局handler 数组赋值,evdev handler的次设备号为64,这里除以32,赋值在input_table[2]
  23. }
  24. list_add_tail(&handler->node, &input_handler_list);  //添加进handler 链表
  25. list_for_each_entry(dev, &input_dev_list, node)   //同样遍历input_dev这个链表,依次调用下面的input_attach_handler去匹配input_dev,这个跟input_dev注册的时候的情形类似
  26. input_attach_handler(dev, handler);
  27. input_wakeup_procfs_readers();
  28. out:
  29. mutex_unlock(&input_mutex);
  30. return retval;
  31. }

input核心中保存的handler数组:

[objc] view plaincopyprint?
  1. static struct input_handler *input_table[8];

这是保存注册到input核心中的handler数组,因为在之前input注册的时候注册的字符设备主设备号为13.字符设备的次设备号为0~255,可以有256个设备,

这里后面会看到一个handler可以connect处理32个input设备,所以input体系中,最多拥有8个handler

这个匹配过程和上一篇中的过程是一样的,最后匹配上的话会调用匹配上的handler 中connect指针指向的函数.

另外可以注意的是evdev是匹配所有设备的,因为:

[objc] view plaincopyprint?
  1. static const struct input_device_id evdev_ids[] = {
  2. { .driver_info = 1 },   /* Matches all devices */
  3. { },            /* Terminating zero entry */
  4. };

如果没有特定的handler添加进handler链表,那么在匹配的时候,只要有这个evdev的handler,最后都会匹配到evdev,这个具体可以去看看上篇的匹配过程.

我这边调试的是usb触摸屏,所以用的是evdev的handler,下面看下evdev的connect.

evdev_connect:

注册的evdev_handler中connect指向的函数为evdev_connect:

[objc] view plaincopyprint?
  1. /*
  2. * Create new evdev device. Note that input core serializes calls
  3. * to connect and disconnect so we don't need to lock evdev_table here.
  4. */
  5. static int evdev_connect(struct input_handler *handler, struct input_dev *dev,
  6. const struct input_device_id *id)
  7. {
  8. struct evdev *evdev;
  9. int minor;
  10. int error;
  11. for (minor = 0; minor < EVDEV_MINORS; minor++)
  12. if (!evdev_table[minor])
  13. break;
  14. if (minor == EVDEV_MINORS) {
  15. pr_err("no more free evdev devices\n");
  16. return -ENFILE;
  17. }
  18. // 可以看到这里evdev handler匹配连接好的设备都以evdev 类型存在这个evdev_table数组的,这个数组大小为32个,这就是我上面说到的,为什么只有8个handler
  19. //这里是判断evdev的32个位置中是否有空
  20. evdev = kzalloc(sizeof(struct evdev), GFP_KERNEL); //为上面定义的*evdev分配内存空间
  21. if (!evdev)
  22. return -ENOMEM;
  23. INIT_LIST_HEAD(&evdev->client_list); //以下都是对这个 evdev的初始化了
  24. spin_lock_init(&evdev->client_lock);
  25. mutex_init(&evdev->mutex);
  26. init_waitqueue_head(&evdev->wait);
  27. dev_set_name(&evdev->dev, "event%d", minor);  //给这个evdev命名
  28. evdev->exist = true;
  29. evdev->minor = minor;   // 以minor为索引赋值
  30. evdev->handle.dev = input_get_device(dev);  //evdev中的handle变量的初始化 ,后面分析这个handle ,这里面保存的就是已经匹配成功的input_dev 和 handler
  31. evdev->handle.name = dev_name(&evdev->dev);
  32. evdev->handle.handler = handler;
  33. evdev->handle.private = evdev;
  34. evdev->dev.devt = MKDEV(INPUT_MAJOR, EVDEV_MINOR_BASE + minor);
  35. evdev->dev.class = &input_class;
  36. evdev->dev.parent = &dev->dev;
  37. evdev->dev.release = evdev_free;
  38. device_initialize(&evdev->dev);
  39. error = input_register_handle(&evdev->handle); //把这个evdev中初始化好的handle 注册到input核心中去,代表一个匹配成功的组合
  40. if (error)
  41. goto err_free_evdev;
  42. error = evdev_install_chrdev(evdev);  //把这个初始化好的evdev添加到上面说到过的evdev_table数组,以minor索引序号
  43. if (error)
  44. goto err_unregister_handle;
  45. error = device_add(&evdev->dev); //把这个device 添加到/sys/class/input/下面,所以我们可以看到/dev/input下面看到:event0~31 字样字符设备文件,这就是在上面命名的
  46. if (error)
  47. goto err_cleanup_evdev;
  48. return 0;
  49. err_cleanup_evdev:
  50. evdev_cleanup(evdev);
  51. err_unregister_handle:
  52. input_unregister_handle(&evdev->handle);
  53. err_free_evdev:
  54. put_device(&evdev->dev);
  55. return error;
  56. }

evdev:

这里的evdev变量的结构如下:

[objc] view plaincopyprint?
  1. struct evdev
  2. {
  3. int open; //打开标志
  4. int minor; //次设备号
  5. struct input_handle handle; //包含的handle
  6. wait_queue_head_t wait; //等待队列
  7. struct evdev_client __rcu *grab; //强制绑定的evdev_client结构
  8. struct list_head client_list; //evdev_client 链表,这说明一个evdev设备可以处理多个evdev_client,可以有多个进程访问evdev设备
  9. spinlock_t client_lock; /* protects client_list */
  10. struct mutex mutex;
  11. struct device dev;
  12. bool exist;
  13. };

关于这个结构变量我的理解是抽象出来一个设备,代表一个 input_dev与其匹配好的 handler的组合( handle),可以看作提供给事件处理层的一个封装.

input_handle:

这个代表一个匹配成功的input dev和 handler组合,定义在input.h中,每个evdev中包含一个input_handle,并且注册到input核心中:

[objc] view plaincopyprint?
  1. /**
  2. * struct input_handle - links input device with an input handler
  3. * @private: handler-specific data
  4. * @open: counter showing whether the handle is 'open', i.e. should deliver
  5. *    events from its device
  6. * @name: name given to the handle by handler that created it
  7. * @dev: input device the handle is attached to
  8. * @handler: handler that works with the device through this handle
  9. * @d_node: used to put the handle on device's list of attached handles
  10. * @h_node: used to put the handle on handler's list of handles from which
  11. *    it gets events
  12. */
  13. struct input_handle {
  14. voidvoid *private;  //指向上面封装的evdev
  15. int open;
  16. const charchar *name;
  17. struct input_dev *dev;   //input 设备
  18. struct input_handler *handler;  // 一个input的handler
  19. struct list_head    d_node;  //链表结构
  20. struct list_head    h_node;
  21. };

input_register_handle:

看看这个handle的注册,不要和handler搞混淆了,这不是一个概念~

[objc] view plaincopyprint?
  1. /**
  2. * input_register_handle - register a new input handle
  3. * @handle: handle to register
  4. *
  5. * This function puts a new input handle onto device's
  6. * and handler's lists so that events can flow through
  7. * it once it is opened using input_open_device().
  8. *
  9. * This function is supposed to be called from handler's
  10. * connect() method.
  11. */
  12. int input_register_handle(struct input_handle *handle)
  13. {
  14. struct input_handler *handler = handle->handler;
  15. struct input_dev *dev = handle->dev;  //取出两个成员
  16. ...
  17. /*
  18. * Filters go to the head of the list, normal handlers
  19. * to the tail.
  20. */
  21. if (handler->filter)
  22. list_add_rcu(&handle->d_node, &dev->h_list);
  23. else
  24. list_add_tail_rcu(&handle->d_node, &dev->h_list);
  25. //把这个handle的d_node 加到对应input_dev的h_list链表里面
  26. ...
  27. list_add_tail_rcu(&handle->h_node, &handler->h_list);
  28. //把这个handle的h_node 加到对应input_handler的h_list链表里面
  29. ...
  30. }

这个注册是把handle 本身的链表加入到它自己的input_dev 以及 input_handler的h_list链表中,这样以后就可以通过h_list遍历到这个handle,

这样就实现了三者的绑定联系.

另外在evdev中还有个结构:

[objc] view plaincopyprint?
  1. struct evdev_client {
  2. unsigned int head;  //buffer数组的索引头
  3. unsigned int tail;   //buffer数组的索引尾
  4. unsigned int packet_head; /* [future] position of the first element of next packet */
  5. spinlock_t buffer_lock; /* protects access to buffer, head and tail */
  6. struct wake_lock wake_lock;
  7. bool use_wake_lock;
  8. char name[28];
  9. struct fasync_struct *fasync;    //异步通知函数
  10. struct evdev *evdev;  //包含一个evdev变量
  11. struct list_head node;  //链表
  12. unsigned int bufsize;
  13. struct input_event buffer[];   //input_event数据结构的数组,input_event代表一个事件,基本成员:类型(type),编码(code),值(value)
  14. };

这个结构会在evdev被打开的时候 创建,这里关于evdev的初始以及在input系统中承接作用暂时介绍到这里,

前文 Linux/Android——输入子系统input_event传递 (二)中有记录从设备驱动传递上来的event是怎么到input核心,然后接着往上传递的,接下来就是用到evdev传递了.下篇介绍.

Linux/Android——input_handler之evdev (四)相关推荐

  1. linux 串口工具_会C++就能开发Linux/Android应用!这款Yoxios X3串口屏火了...

    大家好,我是小月月.今天给大家介绍一款搭载YOXIOS操作系统的YOXIOS X3串口屏,只要你会C/C++就能开发Linux/Android应用!而且小月月带来了福利--"免费试用名额&q ...

  2. linux/android驱动工程师面试相关内容总结

    理论的东西不常用时就会慢慢的被遗忘,但是找工作就是一个如何让别人相信自己的过程,理论知识就是一个非常重要的途径. 一次次机会在错失,每次想找工作时,刷一下简历就去面试了,一次次因为理论被鄙视,也该长长 ...

  3. 软件开发全套视频教程汇总(javaSE,javaEE,linux,android开发,C# ,web前端,大数据云计算,数据挖掘,web前端,php开发,UI设计,C++开发,3D视频)

    软件开发全套视频教程汇总(javaSE,javaEE,linux,android开发,C# ,web前端,大数据云计算,数据挖掘,web前端,php开发,UI设计,C++开发,3D视频) 这是我以前学 ...

  4. ubuntu下搭建android开发环境(四)核心篇安装AndroidStudio、sdk、jdk

    [置顶] ubuntu下搭建android开发环境(四)核心篇安装AndroidStudio.sdk.jdk(by 星空武哥) <div class="article_manage c ...

  5. Android项目实战(四):ViewPager切换动画(3.0版本以上有效果)

    原文:Android项目实战(四):ViewPager切换动画(3.0版本以上有效果) 学习内容来自"慕课网" 一般APP进去之后都会有几张图片来导航,这里就学习怎么在这张图片切换 ...

  6. Android Activity的launchMode四种启动模式备忘

    Android Activity的launchMode四种启动模式备忘 Android的Activity的启动模式有四种,在AndroidManifest.xml通过配置Activity的androi ...

  7. android 测试工具,Android开源项目第四篇:开发及测试工具篇

    本文为那些不错的Android开源项目第四篇--开发工具篇,**主要介绍Android开发工具和测试工具相关的开源项目**. Android开源项目系列汇总已完成,包括: 1.Buck faceboo ...

  8. Linux C++多线程同步的四种方式

    目录 一.互斥锁 二.条件变量 三.读写锁 原文链接:Linux C++多线程同步的四种方式(非常详细)_Y先森0.0-CSDN博客 背景问题:在特定的应用场景下,多线程不进行同步会造成什么问题? 通 ...

  9. 史上最详细的Android Studio系列教程四--Gradle基础

    史上最详细的Android Studio系列教程四--Gradle基础 转载于:https://www.cnblogs.com/zhujiabin/p/5125917.html

最新文章

  1. 【指标统计】删除失败遥控
  2. python处理数据的包_在Python中利用Into包整洁地进行数据迁移的教程
  3. Android之访问网络,使用HttpURLConnection还是HttpClient?
  4. kotlin自定义View出现 java.lang.ClassNotFoundException
  5. java asynccontext,超时后的JavaEE 6 AsyncContext行为
  6. 搜狗520甜蜜告白攻势:爱的心动 让她看见
  7. 明年的现在我也想去“双选会”应聘!
  8. 装完机,启grub+Linux,linux一路填坑...
  9. AndroMDA 4.x架构
  10. shell错位_shell脚本出错!来大神指出错误出处!!!!
  11. 快速用JavaScript实现划词取词,可复制百度文库文字(获取鼠标选中区域文字)
  12. 【建模算法】基于遗传算法求解TSP问题(Python实现)
  13. 新手如何让淘宝店铺快速走向正轨? 新型模式带你走向巅峰
  14. DDR扫盲——DDR中的名词解析
  15. How to Become a Straight-A student
  16. 微信请勿打扰昵称,个性好看,来看看!
  17. 基于haar+adaboost的人脸检测、深度学习的人脸识别技术应用综述
  18. PLL中的locked信号解析
  19. 2022年圣诞节 | matlab实现炫酷的圣诞树
  20. GitHub 上值得收藏的100个精选前端项目!你知道几个?

热门文章

  1. java实现hj协议_环保 HJ212协议解析
  2. 上三角矩阵的压缩存储
  3. 备战第十二届蓝桥杯电子类《EDA设计与开发》国赛
  4. 故障案例 | 慢SQL引发MySQL高可用切换排查全过程
  5. [再寄小读者之数学篇](2014-05-27 特征值估计)
  6. 过一下hibernate4-3
  7. SR研究(1)RCAN论文阅读上
  8. 卷积神经网络常用模型,卷积神经网络数学建模
  9. 一梦江湖——华山快还钱是怎么会事呢?
  10. CellID 基站定位