linux键盘驱动 终端,Linux的input输入子系统:设备驱动之按键驱动

环境:kernel-2.6.30.4,arm-linux-gcc-4.3.3,目标板TQ2440

一、设备层驱动程序:

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include /*寄存器设置*/

#include /*hardware*/

struct pin_desc {   //引脚描述,将一个引脚和一个中断号、名字、按键值进行绑定

int irq;

char *name;

unsigned int pin;

unsigned int key_val;

};

struct pin_desc pins_desc[4] = {//绑定四个引脚

{IRQ_EINT1, "K1", S3C2410_GPF1, KEY_L},

{IRQ_EINT4, "K2", S3C2410_GPF4, KEY_S},

{IRQ_EINT2, "K3", S3C2410_GPF2, KEY_ENTER},

{IRQ_EINT0, "K4", S3C2410_GPF0, KEY_LEFTSHIFT},

};

static struct input_dev *buttons_dev;

static struct pin_desc *irq_pd;

static struct timer_list buttons_timer;

//中断执行函数,当设定的4个外部中断被触发后将执行该函数

static irqreturn_t buttons_irq(int irq, void *dev_id)

{

irq_pd = (struct pin_desc *)dev_id;//获得相应中断的相应参数

mod_timer(&buttons_timer, jiffies+HZ/100);//开启定时器,时间到将执行定时器处理函数,时间为10ms

return IRQ_RETVAL(IRQ_HANDLED);

}

//定时器处理函数

static void buttons_timer_function(unsigned long data)

{

struct pin_desc * pindesc = irq_pd;//发生中断的那个引脚的信息

unsigned int pinval;

if (!pindesc)

return;

pinval = s3c2410_gpio_getpin(pindesc->pin);//获得发生中断的那个引脚的值

if (pinval)

{

input_event(buttons_dev, EV_KEY, pindesc->key_val, 0);//按下

input_sync(buttons_dev);

}

else

{

input_event(buttons_dev, EV_KEY, pindesc->key_val, 1);//松开

input_sync(buttons_dev);

}

}

static int buttons_open(struct input_dev *dev) { return 0;}

static void buttons_close(struct input_dev *dev) {}

static int buttons_init(void)

{

int i;

// 1. 分配一个input_dev结构体

buttons_dev = input_allocate_device();

//2. 设置

//2.1 能产生哪类事件

set_bit(EV_KEY, buttons_dev->evbit);//能产生按键类事件

set_bit(EV_REP, buttons_dev->evbit);//内产生重复类事件

//2.2 能产生这类操作里的哪些事件

set_bit(KEY_L, buttons_dev->keybit);

set_bit(KEY_S, buttons_dev->keybit);

set_bit(KEY_ENTER, buttons_dev->keybit);

set_bit(KEY_LEFTSHIFT, buttons_dev->keybit);

buttons_dev->name = "buttons_input";

buttons_dev->dev.init_name = "buttons_input";

buttons_dev->name = "buttons_input";

buttons_dev->dev.init_name = "buttons_input";

buttons_dev->open = buttons_open;

buttons_dev->close = buttons_close;

//3. 注册input_dev:将该设备加入到input_dev_list列表中,找寻和它匹配的input_handler

input_register_device(buttons_dev);

init_timer(&buttons_timer);//初始化一个定时器

buttons_timer.function = buttons_timer_function;//设置定时器的处理函数

add_timer(&buttons_timer);//将该定时器器加入到内核

//注册中断

//pins_desc参数就是当相应的中断发生,该值将成为对应中断函数的参数

for (i = 0; i < 4; i++)

{

request_irq(pins_desc[i].irq, buttons_irq, IRQ_TYPE_EDGE_BOTH, pins_desc[i].name, &pins_desc[i]);

}

printk("按键驱动输入子系统初始化成功!\n");

return 0;

}

static void buttons_exit(void)

{

int i;

for (i = 0;i < 4;i++)//中断注销

free_irq(pins_desc[i].irq, &pins_desc[i]);

del_timer(&buttons_timer);//删除定时器

input_unregister_device(buttons_dev);//注销驱动

input_free_device(buttons_dev);//释放内存

printk("按键驱动输入子系统卸载成功!\n");

}

module_init(buttons_init);

module_exit(buttons_exit);

MODULE_LICENSE("GPL");

Makefile:

ERN_DIR = /lib/modules/2.6.30.4-EmbedSky/build

all:

make -C $(KERN_DIR) M=`pwd` modules

clean:

make -C $(KERN_DIR) M=`pwd` modules clean

rm -rf modules.order

obj-m   += buttons.o

二、测试程序

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

int main(void)

{

int fd;

int key_value,i=0,count;

struct input_event ev_key;

fd = open("/dev/event1", 666);

if (fd < 0)

{

printf("can't open device buttons!\n");

exit(1);

}

for (;;)

{

count = read(fd,&ev_key,sizeof(struct input_event));

for(i=0; i

if(EV_KEY==ev_key.type)

printf("type:%d,code:%d,value:%d\n", ev_key.type,ev_key.code,ev_key.value);

if(EV_SYN==ev_key.type)

printf("syn event\n\n");

}

close(fd);

return 0;

}

我们也可以在开发板上:

#cat  /dev/tty     //事件将上传到shell程序的控制终端

源码下载:http://download.csdn.net/detail/hbuxiaofei/7903951

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值