linux 2.6 输入子系统 键盘驱动的实现

好久没来了!写点东西.关于linux 2.6下面的键盘驱动的实现.
    2.6内核采用"input sub system" 的概念.将输入驱动分成三块: driver,input core和Event handler. "一个输入事件,如鼠标移动,键盘按键按下,joystick的移动等等通过 Driver -> InputCore -> Eventhandler -> userspace 的顺序到达用户空间传给应用程序。"
    关于输入子系统详细的文章有一个老兄写了一篇:
    http://blog.csdn.net/colorant/archive/2007/04/12/1561837.aspx
    在这里我肤浅得讲一下如何实现一个驱动.todo

正文

2.6输入子系统使得用户空间可以通过字符设备接口毫无遗漏的获得原始的输入消息。比如说有些智能鼠标,除了坐标滚球,三键,滚轮之外还有其他的输入装置,,比如控制系统音量的按钮.

如何注册设备

      

/*
       *Input interface, not finished yet
       */

       int key,code;
       button_dev = input_allocate_device(); /*static struct input_dev *button_dev defined outside*/
       init_input_dev(button_dev);
      
       set_bit(EV_KEY,button_dev->evbit);
       //@set_bit(EV_REP,button_dev->evbit);       //repeat
       for(key=0;key<16;key++)
              set_bit(keypad_keycode[key],button_dev->keybit);
       button_dev->name = PCA9554_DRV_NAME;
       button_dev->id.bustype = BUS_HOST;
      
       input_register_device(button_dev);
                     

如上述代码所示,首先分配一个输入设备,之后对其进行初始化,然后配置参数. “set_bit(EV_KEY,button_dev->evbit);”将该输入设备定义为键盘输入,这样input core在分发消息的时候会将其发送到keyboard event handler.

接着就是初始化键盘定义,这里keybit是由若干个long型变量组成的数组,我所用的kernel直至最大按键数目为0x1ff.
       for(key=0;key<16;key++)
              set_bit(keypad_keycode[key],button_dev->keybit);

我的开发板上有16个按键,因为项目是播放器,所以我对其作了定义:
int keypad_keycode[] = {
              KEY_UP,  KEY_DOWN,     KEY_A,   KEY_B,              // 1 - 4
              KEY_LEFT, KEY_RIGHT,    KEY_C,   KEY_D,             // 5 - 8
              KEY_PLAYPAUSE,KEY_PLAY,KEY_NEXTSONG,KEY_VOLUMEUP,   // 9 - 12
              KEY_CANCEL,KEY_OK,KEY_PREVIOUSSONG,KEY_VOLUMEDOWN,  // 13 - 16
              KEY_UNKNOWN,
       };


 

键盘定义示意图

 

input.cinput_event函数中会对keybit进行判断,确定某个按键对应的位是否被打开,否则不予上传键值.

              case EV_KEY:
                     printk("haigang;:%s: code:%d,value=%d/n",__FUNCTION__,code,value);
                     if (code > KEY_MAX || !test_bit(code, dev->keybit) || !!test_bit(code, dev->key) == value){
                            printk("haigang:%s: code too big/n",__FUNCTION__,code);
                            return;
                     }
                     if (value == 2)
                            break;
                     change_bit(code, dev->key);
                     if (test_bit(EV_REP, dev->evbit) && dev->rep[REP_PERIOD] && dev->rep[REP_DELAY] && dev->timer.data && value) {
                            dev->repeat_key = code;
                            mod_timer(&dev->timer, jiffies + msecs_to_jiffies(dev->rep[REP_DELAY]));
                     }
                     break;

 

input_event部分代码

 


如何传送输入键值比较简单,只需要使用   函数void input_event即可,该函数定义

void input_event(struct input_dev *dev, unsigned int type, unsigned int code, int value)

include/linux/input.h中对该函数作了封装
static inline void input_report_key(struct input_dev *dev, unsigned int code, int value)
{
       input_event(dev, EV_KEY, code, !!value);
}

这里code是键值,value应该是键盘状态,按下或者释放(仅是猜测,有待考证)




我的驱动中的建值传输代码,当键值是0x16时候,表示松开键盘事件
       unsigned char key;
       static unsigned char oldKey=0x16;
       key = pca9554_readKey();

       //@printk("KEY:%d/n",key);
      
       if(key!=0x16){
              input_report_key(button_dev,keypad_keycode[key],1);
              input_sync(button_dev);
       }else{
              input_report_key(button_dev,keypad_keycode[oldKey],0);
              input_sync(button_dev);
       }

       set_irq_type(IRQ_KEYPAD,IRQT_FALLING);
       s3c_gpio_cfgpin(S3C_GPF4,S3C_GPF4_EINT4);/*reconfigure the GPF4 as EINT4*/

       oldKey=key;

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值