Linux输入子系统

  1. <strong>地址   http://blog.csdn.net/zhenwenxian/article/details/6064234</strong>  
  2.   
  3. Linux输入子系统  
  4.   
  5. 按键与触摸屏的设备驱动,在Linux系统中实现这类设备驱动的方法是利用input子系统。  
  6.   
  7. Linux系统提供了input子系统,按键、触摸屏、键盘、鼠标等输入都可以利用input接口函数来实现设备驱动,按键和触摸屏设备驱动都可以作为input设备驱动而实现。  
  8.   
  9. 在Linux内核中,input设备用input_dev结构体描述,使用input子系统实现输入设备驱动的时候,驱动的核心工作是向系统报告按键、触摸屏、键盘、鼠标等输入事件(event,通过input_event结构体描述),不再需要关心文件操作接口,因为input子系统已经完成了文件操作接口。驱动报告的事件经过InputCore和 Eventhandler最终到达用户空间。  
  10.   
  11. 输入子系统支持的事件类型在文件  
  12. kernel/include/linux/input.h 中定义:  
  13.   
  14.  event 还定义了标准按键的编码等  
  15.   
  16.   
  17.    
  18.   
  19.    
  20.   
  21. 通过input子系统,具体的输入设备驱动只需要完成如下工作。  
  22.   
  23. l 在模块加载函数中告知input子系统它可以报告的事件。  
  24.   
  25. 设备驱动通过set_bit()告诉input子系统它支持哪些事件,如下所示:  
  26.   
  27. set_bit(EV_KEY, button_dev.evbit);  
  28.   
  29. l 在模块加载函数中注册输入设备。  
  30.   
  31. 注册输入设备的函数为:  
  32.   
  33. int input_register_device(struct input_dev *dev);  
  34.   
  35. l 在键被按下/抬起、触摸屏被触摸/抬起/移动、鼠标被移动/单击/抬起时通过input_ report_xxx()报告发生的事件及对应的键值/坐标等状态。  
  36.   
  37. 主要的事件类型包括EV_KEY(按键事件)、EV_REL(相对值,如光标移动,报告的是相对最后一次位置的偏移)和EV_ABS(绝对值,如触摸屏和操纵杆,它们工作在绝对坐标系统)。  
  38.   
  39. 用于报告EV_KEY、EV_REL和EV_ABS事件的函数分别为:  
  40.   
  41. void input_report_key(struct input_dev *dev, unsigned int code, int value);  
  42.   
  43. void input_report_rel(struct input_dev *dev, unsigned int code, int value);  
  44.   
  45. void input_report_abs(struct input_dev *dev, unsigned int code, int value);  
  46.   
  47. input_sync()用于事件同步,它告知事件的接收者驱动已经发出了一个完整的报告。  
  48.   
  49. 例如,在触摸屏设备驱动中,一次坐标及按下状态的整个报告过程如下:  
  50.   
  51. input_report_abs(input_dev, ABS_X, x);  //X坐标  
  52.   
  53. input_report_abs(input_dev, ABS_Y, y);   //Y坐标  
  54.   
  55. input_report_abs(input_dev, ABS_PRESSURE, pres); //压力  
  56.   
  57. input_sync(input_dev);  //同步  
  58.   
  59. 在模块卸载函数中注销输入设备。注销输入设备的函数为:  
  60.   
  61. void input_unregister_device(struct input_dev *dev);  
  62.   
  63. 代码清单12.28给出了一个最简单的使用input接口实现按键设备驱动的范例,它在中断服务程序中向系统报告按键及同步事件。  
  64.   
  65. input设备驱动  
  66.   
  67. 1  /*在按键中断中报告事件*/  
  68.   
  69. 2  static void button_interrupt(int irq, void *dummy, struct pt_regs *fp)  
  70.   
  71. 3  {  
  72.   
  73. 4    input_report_key(&button_dev, BTN_1, inb(BUTTON_PORT) &1);  
  74.   
  75. 5    input_sync(&button_dev);  
  76.   
  77. 6  }  
  78.   
  79. 7   
  80.   
  81. 8  static int _ _init button_init(void)  
  82.   
  83. 9  {  
  84.   
  85. 10   /*申请中断*/  
  86.   
  87. 11   if (request_irq(BUTTON_IRQ, button_interrupt, 0, "button", NULL))  
  88.   
  89. 12   {  
  90.   
  91. 13     printk(KERN_ERR "button.c: Can't allocate irq %d/n", button_irq);  
  92.   
  93. 14     return  - EBUSY;  
  94.   
  95. 15   }  
  96.   
  97. 16  
  98.   
  99. 17   button_dev.evbit[0] = BIT(EV_KEY);    //支持EV_KEY事件  
  100.   
  101. 18   button_dev.keybit[LONG(BTN_0)] = BIT(BTN_0);  
  102.   
  103. 19  
  104.   
  105. 20   input_register_device(&button_dev);   //注册input设备  
  106.   
  107. 21 }  
  108.   
  109. 22  
  110.   
  111. 23 static void _ _exit button_exit(void)  
  112.   
  113. 24 {  
  114.   
  115. 25   input_unregister_device(&button_dev);   //注销input设备  
  116.   
  117. 26   free_irq(BUTTON_IRQ, button_interrupt); //释放中断  
  118.   
  119. 27 }  
  120.   
  121.    
  122.   
  123. 触摸屏的注册  
  124.   
  125.  ts->input_dev = input_allocate_device();  
  126.  ts->input_dev->name = "synaptics-rmi-touchscreen";  
  127.  set_bit(EV_SYN, ts->input_dev->evbit);  
  128.  set_bit(EV_KEY, ts->input_dev->evbit);  
  129.  set_bit(BTN_TOUCH, ts->input_dev->keybit);  
  130.  set_bit(BTN_2, ts->input_dev->keybit);  
  131.  set_bit(EV_ABS, ts->input_dev->evbit);  
  132.   
  133.  /* ts->input_dev->name = ts->keypad_info->name; */  
  134.  ret = input_register_device(ts->input_dev);  
  135.   
  136.    

你可能感兴趣的:(工作,linux,report,input,button,linux内核)