微波EDA网,见证研发工程师的成长!
首页 > 硬件设计 > 嵌入式设计 > S5PV210(TQ210)学习笔记——输入子系统驱动

S5PV210(TQ210)学习笔记——输入子系统驱动

时间:11-28 来源:互联网 点击:
前面的文章简单的讲述了字符设备驱动程序的编写,用字符设备驱动的方式实现了按键驱动,但是,出了我们的自己编写的针对我们的这个驱动程序的应用程序之外,其他应用程序都无法接收我们这个驱动的键值输入,为了让所有应用程序都可以接收我们的按键驱动解析的键值,Linux内核定义了“输入子系统”的概念,也就是说,只要我们按照这个模型进行驱动开发,并为其提供必须的接口函数,那么,Linux内核就可以正常来获取我们的键盘值了。

输入子系统的原理分析强烈推荐观看韦东山老师的视频讲座,讲的非常清楚,我这里是按照输入子系统的方式实现的按键驱动,下面是源码,放在这里做个备份:

  1. #include
  2. #include
  3. #include
  4. #include
  5. #includeinterrupt.h>
  6. #include
  7. staticstructinput_dev*buttons_dev;
  8. staticstructtimer_listtimer;
  9. structbutton_desc*button_desc=NULL;
  10. structbutton_desc{
  11. char*name;
  12. unsignedintpin;
  13. unsignedintirq;
  14. unsignedintval;
  15. };
  16. staticstructbutton_descbuttons_desc[8]={
  17. [0]={
  18. .name="S1",
  19. .pin=S5PV210_GPH0(0),
  20. .irq=IRQ_EINT(0),
  21. .val=KEY_L,
  22. },
  23. [1]={
  24. .name="S2",
  25. .pin=S5PV210_GPH0(1),
  26. .irq=IRQ_EINT(1),
  27. .val=KEY_S,
  28. },
  29. [2]={
  30. .name="S3",
  31. .pin=S5PV210_GPH0(2),
  32. .irq=IRQ_EINT(2),
  33. .val=KEY_C,
  34. },
  35. [3]={
  36. .name="S4",
  37. .pin=S5PV210_GPH0(3),
  38. .irq=IRQ_EINT(3),
  39. .val=KEY_ENTER,
  40. },
  41. [4]={
  42. .name="S5",
  43. .pin=S5PV210_GPH0(4),
  44. .irq=IRQ_EINT(4),
  45. .val=KEY_LEFTCTRL,
  46. },
  47. [5]={
  48. .name="S6",
  49. .pin=S5PV210_GPH0(5),
  50. .irq=IRQ_EINT(5),
  51. .val=KEY_MINUS,
  52. },
  53. [6]={
  54. .name="S7",
  55. .pin=S5PV210_GPH2(6),
  56. .irq=IRQ_EINT(22),
  57. .val=KEY_CAPSLOCK,
  58. },
  59. [7]={
  60. .name="S8",
  61. .pin=S5PV210_GPH2(7),
  62. .irq=IRQ_EINT(23),
  63. .val=KEY_SPACE,
  64. },
  65. };
  66. staticvoidtimer_function(unsignedlongdata){
  67. if(button_desc==NULL)
  68. return;
  69. if(gpio_get_value(button_desc->pin)){
  70. input_event(buttons_dev,EV_KEY,button_desc->val,0);
  71. }
  72. else{
  73. input_event(buttons_dev,EV_KEY,button_desc->val,1);
  74. }
  75. input_sync(buttons_dev);
  76. }
  77. staticirqreturn_tirq_handler(intirq,void*devid){
  78. button_desc=(structbutton_desc*)devid;
  79. mod_timer(&timer,jiffies+HZ/100);
  80. returnIRQ_RETVAL(IRQ_HANDLED);
  81. }
  82. staticintbuttons_init(void){
  83. inti;
  84. buttons_dev=input_allocate_device();
  85. if(buttons_dev==NULL){
  86. printk(KERN_ERR"Error:allocateinputdevicefailed!");
  87. return-ENOMEM;
  88. }
  89. __set_bit(EV_KEY,buttons_dev->evbit);
  90. __set_bit(EV_REP,buttons_dev->evbit);
  91. __set_bit(KEY_L,buttons_dev->keybit);
  92. __set_bit(KEY_S,buttons_dev->keybit);
  93. __set_bit(KEY_C,buttons_dev->keybit);
  94. __set_bit(KEY_SPACE,buttons_dev->keybit);
  95. __set_bit(KEY_MINUS,buttons_dev->keybit);
  96. __set_bit(KEY_ENTER,buttons_dev->keybit);
  97. __set_bit(KEY_LEFTCTRL,buttons_dev->keybit);
  98. __set_bit(KEY_CAPSLOCK,buttons_dev->keybit);
  99. printk("1");
  100. if(input_register_device(buttons_dev)){
  101. gotoerror_1;
  102. }
  103. printk("2");
  104. init_timer(&timer);
  105. timer.function=timer_function;
  106. add_timer(&timer);
  107. printk("3");
  108. for(i=0;i!=8;++i){
  109. if(request_irq(buttons_desc[i].irq,irq_handler,
  110. IRQF_TRIGGER_RISING|IRQF_TRIGGER_FALLING,buttons_desc[i].name,&buttons_desc[i])){
  111. gotoerror_2;
  112. }
  113. }
  114. printk("4");
  115. return0;
  116. error_2:
  117. for(--i;i>=0;--i){
  118. free_irq(buttons_desc[i].irq,&buttons_desc[i]);
  119. }
  120. input_unregister_device(buttons_dev);
  121. error_1:
  122. input_free_device(buttons_dev);
  123. return-EBUSY;
  124. }
  125. staticvoidbuttons_exit(void){
  126. inti;
  127. for(i=0;i!=8;++i){
  128. free_irq(buttons_desc[i].irq,&buttons_desc[i]);
  129. }
  130. input_unregister_device(buttons_dev);
  131. input_free_device(buttons_dev);
  132. }
  133. module_init(buttons_init);
  134. module_exit(buttons_exit);
  135. MODULE_LICENSE("GPL");


很显然,基于输入子系统的按键驱动比起直接编写的字符驱动设备要简单的多,不过按键消抖还是要加的,上面的代码加上了按键消抖。如果在开发过程中遇到了什么问题,还是欢迎留言讨论。

Copyright © 2017-2020 微波EDA网 版权所有

网站地图

Top