鼠标驱动之-sys节点-input子系统

首先需要了解sys节点和linux驱动编程的知识,在linux内核<linux/>下有着对应的实现。本例实现创建sys节点,外围程序通过input子系统控制鼠标位置。

第一步编写驱动代码,创建sys节点:

    1. #include <linux/module.h>
    2. #include <linux/platform_device.h>
    3. #include <linux/slab.h>
    4. #include <linux/input.h>
    5. #include <linux/kthread.h>
    6. #include <linux/semaphore.h>
    7. struct v_dev{
    8. struct platform_device *p_dev;
    9. struct input_dev *input;
    10. int x;
    11. int y;
    12. struct task_struct *run_thread;
    13. struct semaphore sem;
    14. };
    15. struct v_dev *vmouse_dev = NULL;
    16. static ssize_t write_pos(struct device *dev, struct device_attribute *attr,
    17. const char *buf, size_t count)
    18. {
    19. int x,y;
    20. sscanf(buf,"%d%d",&x,&y);
    21. vmouse_dev->x = x;
    22. vmouse_dev->y =y;
    23. //post 信号量
    24. up(&vmouse_dev->sem);
    25. return count;
    26. }
    27. static ssize_t show_pos(struct device *dev, struct device_attribute *attr,
    28. char *buf){
    29. return sprintf(buf,"(%d,%d)\n",vmouse_dev->x,vmouse_dev->y);
    30. }
    31. DEVICE_ATTR(pos,0644,show_pos,write_pos);
    32. static int vmouse_thread(void *data)
    33. {
    34. int x,y;
    35. struct v_dev *vmouse_dev = (struct v_dev*)data;
    36. struct semaphore *sema = &(vmouse_dev->sem);
    37. printk(KERN_INFO "vmouse thread running\n");
    38. while(1){
    39. //等待信号量
    40. while((down_interruptible(sema)) == -EINTR){} ;
    41. x = vmouse_dev->x;
    42. y = vmouse_dev->y;
    43. input_report_abs(vmouse_dev->input,ABS_X,x);
    44. input_report_abs(vmouse_dev->input,ABS_Y,y);
    45. input_sync(vmouse_dev->input);
    46. printk("vmouse thread report\n");
    47. }
    48. return 0;
    49. }
    50. static int vmouse_probe(struct platform_device *pdev)
    51. {
    52. int ret = -1;
    53. printk("%s debug \n",__func__);
    54. if(vmouse_dev->p_dev == pdev){
    55. printk("platform device is same\n");
    56. }
    57. vmouse_dev->input = input_allocate_device();
    58. if(!(vmouse_dev->input)){
    59. printk("%s request input deivce error\n",__func__);
    60. goto alloc_input;
    61. }
    62. vmouse_dev->input->name = "vmouse";
    63. set_bit(EV_ABS,vmouse_dev->input->evbit);
    64. input_set_abs_params(vmouse_dev->input, ABS_X, -1024, 1024, 0, 0);
    65. input_set_abs_params(vmouse_dev->input, ABS_Y, -1024, 1024, 0, 0);
    66. ret = input_register_device(vmouse_dev->input);
    67. if(ret < 0){
    68. printk("%s register input device error\n",__func__);
    69. goto input_register;
    70. }
    71. device_create_file(&pdev->dev,&dev_attr_pos);
    72. //初始化信号量,在线程中要用到
    73. sema_init(&(vmouse_dev->sem),0);
    74. vmouse_dev->run_thread = kthread_run(vmouse_thread,vmouse_dev,"vmouse_thread");
    75. return 0;
    76. input_register:
    77. input_free_device(vmouse_dev->input);
    78. alloc_input:
    79. kfree(vmouse_dev);
    80. return ret;
    81. }
    82. static struct platform_driver vmouse_driver = {
    83. .probe = vmouse_probe,
    84. .driver = {
    85. .owner = THIS_MODULE,
    86. .name = "v_mouse",
    87. },
    88. };
    89. static int __init vmouse_init(void)
    90. {
    91. int ret =-1;
    92. printk("%s\n", __func__);
    93. vmouse_dev = kzalloc(sizeof(struct v_dev),GFP_KERNEL);
    94. if(vmouse_dev == NULL){
    95. printk("%s alloc memory  error\n",__func__);
    96. return -ENOMEM;
    97. }
    98. vmouse_dev->p_dev= platform_device_register_simple("v_mouse",-1,NULL,0);
    99. if(!(vmouse_dev->p_dev)){
    100. printk("%s register platform device error\n",__func__);
    101. return ret;
    102. }
    103. ret = platform_driver_register(&vmouse_driver);
    104. if(ret < 0){
    105. printk("%s register driver error\n",__func__);
    106. return ret;
    107. }
    108. return 0;
    109. }
    110. static void __exit vmouse_exit(void)
    111. {
    112. printk("%s\n", __func__);
    113. if(vmouse_dev->input != NULL){
    114. input_unregister_device(vmouse_dev->input);
    115. }
    116. printk("%s debug__1\n",__func__);
    117. if(vmouse_dev != NULL){
    118. platform_device_unregister(vmouse_dev->p_dev);
    119. }
    120. printk("%s debug__2\n",__func__);
    121. platform_driver_unregister(&vmouse_driver);
    122. printk("%s debug__3\n",__func__);
    123. kfree(vmouse_dev);
    124. printk("%s debug__4\n",__func__);
    125. }
    126. module_init(vmouse_init);
    127. module_exit(vmouse_exit);
    128. MODULE_LICENSE("GPL");
    129. MODULE_AUTHOR("[email protected]“);
    130. 编写makefile:
    131. 1 ifeq ($(KERNELRELEASE),)
        2 #KERNEL_DIR:=/home/archermind/zhaoxi/bsw_ww02_2016/kernel/cht
        3 KERNEL_DIR:=/usr/src/linux-headers-3.13.0-32-generic
        4 PWD:=$(shell pwd)
        5 modules:
        6     $(MAKE) -C $(KERNEL_DIR) M=$(PWD) modules
        7 modules_install:
        8     $(MAKE) -C $(KERNEL_DIR) M=$(PWD) modules_install
        9 clean:
       10     rm -rf  .*.cmd *.ko  *.o modules.order  Module.symvers *mod.c
       11     .PHONY: modules modules_install clean
       12 else
       13     modules-objs := sys.o
       14     obj-m := sys.o
       15 endif
    132. 在当前makefile路径下执行make
    133. sudo insmod **.ko
    134. 加载驱动---会在/sys/devices/platform下出现节点,并且/dev/input下会出先对应的可读写的节点。
    135. 驱动的卸载sudo rmmod modules
    136. 编写测试程序:
      1. #include <fcntl.h>
      2. #include <linux/input.h>
      3. #include <stdio.h>
      4. //要看自己的节点了  ---》特别注意去看自己多出的到底是哪个节点这里需要改动哦
      5. #define EVENT_DEV "/dev/input/event5"
      6. int main(void)
      7. {
      8. struct input_event ev;
      9. int count,x,y;
      10. int fd = open(EVENT_DEV, O_RDWR);
      11. if(fd < 0){
      12. printf("open %s failed\n",EVENT_DEV);
      13. return 0;
      14. }
      15. while(1){
      16. count = read(fd, &ev,sizeof(struct input_event));
      17. if(EV_ABS == ev.type){
      18. if(ev.code == ABS_X){
      19. x = ev.value;
      20. }else if(ev.code == ABS_Y){
      21. y = ev.value;
      22. }
      23. printf("position: x=%d, y=%d\n",x,y);
      24. }else if(EV_SYN == ev.type){
      25. puts("sync!");
      26. }
      27. }
      28. return 0;
      29. }
      30. 先运行应用程序:./test_sys  
        在另一个终端执行:echo "90 90" >/sys/devices/platform/v_mouse/pos

        你就会看到你input设备上报的坐标,打印信息如下:

        position: x=90, y=0
        position: x=90, y=90
        sync!

      31. 完毕!
时间: 2024-11-08 10:03:04

鼠标驱动之-sys节点-input子系统的相关文章

Linux设备驱动之input子系统

作者:武汉华嵌嵌入式培训中心  讲师  李家凯 对于输入类设备如键盘.鼠标.触摸屏之类的Linux驱动,内核提供input子系统,使得这类设备的处理变得非常便捷.总体上来讲,input子系统由三部分组成: 事件驱动<-->input核心<-->设备驱动. 其中事件驱动负责与用户程序打交道,诸如设备节点/dev之类的,都由他负责,我们在写驱动时就不用实现这个了:设备驱动负责与硬件设备打交道,这里的交互很简单,只需要读取相关硬件的数据,然后抛给input核心就可以了: 举个例子,以按键

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

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

input子系统——架构、驱动、应用程序

一.input子系统架构 input子系统由驱动层drivers,输入子系统核心层input core,事件处理层event handler组成. 一个输入事件,通过输入设备发给系统如鼠标移动,键盘按键按下等通过device driver->input core(handler->event函数)->event handler->user space的顺序到达用户空间传给应用程序. 一个输出事件,通过系统发给输入设备,通过user space->event handler-&

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这

Linux input 子系统应用之按键驱动

硬件平台:s5pv210 软件平台:Linux2.6.35.7 应用程序:inputk2_app.c #include <stdio.h> #include <fcntl.h> #include <linux/input.h> #include <sys/types.h> #include <unistd.h> int main(int argc, char** argv) { int fd; int count; int i = 0; int

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子系统

input子系统 相关链接: http://bbs.ednchina.com/BLOG_ARTICLE_255205.HTM //good http://blog.chinaunix.net/uid-14027506-id-200921.html //good

Linux/Android——input子系统核心

之前的博客有涉及到linux的input子系统,这里学习记录一下input模块. input子系统,作为管理输入设备与系统进行交互的中枢,任何的输入设备驱动都要通过input向内核注册其设备, 常用的输入设备也就是鼠标,键盘,触摸屏. 稍微细分一点整个输入体系,就是 硬件驱动层,input核心中转层,事件处理层.层次之间传递都以event事件的形式,这其中input连接上下层,分别提供接口. 之前有分析usbtouchscreen的驱动,也就是硬件驱动部分,这里简单记录一下input核心中转处理

Linux/Android——input子系统核心 (三)【转】

本文转载自:http://blog.csdn.net/jscese/article/details/42123673 之前的博客有涉及到linux的input子系统,这里学习记录一下input模块. input子系统,作为管理输入设备与系统进行交互的中枢,任何的输入设备驱动都要通过input向内核注册其设备, 常用的输入设备也就是鼠标,键盘,触摸屏. 稍微细分一点整个输入体系,就是 硬件驱动层,input核心中转层,事件处理层.层次之间传递都以event事件的形式,这其中input连接上下层,分