該代碼在FL2440開發板上測試經過,爲方便教學,將驅動中的platform_device和platform_driver故意分爲兩個驅動模塊。linux
[guowenxue@centos6 input_kbd]$ ls event_button.c kbd_device.c kbd_driver.c kbd_driver.h Makefile
驅動相關頭文件 kbd_driver.h:ios
1 /******************************************************************************** 2 * Copyright: (C) 2016 Guo Wenxue<guowenxue@aliyun.com> 3 * All rights reserved. 4 * 5 * Filename: kbd_driver.h 6 * Description: This head file is for s3c keyboard driver 7 * 8 * Version: 1.0.0(07/26/2016) 9 * Author: Guo Wenxue <guowenxue@aliyun.com> 10 * ChangeLog: 1, Release initial version on "07/26/2016 06:54:47 PM" 11 * 12 ********************************************************************************/ 13 14 #ifndef _KBD_DRIVER_H_ 15 #define _KBD_DRIVER_H_ 16 17 /* keyboard hardware informtation structure definition */ 18 typedef struct s3c_kbd_info_s 19 { 20 int code; /* input device key code */ 21 int nIRQ; /* keyboard IRQ number*/ 22 unsigned int setting; /* keyboard IRQ Pin Setting*/ 23 unsigned int gpio; /* keyboard GPIO port */ 24 } s3c_kbd_info_t; 25 26 /* keyboard platform device private data structure */ 27 typedef struct s3c_kbd_platform_data_s 28 { 29 s3c_kbd_info_t *keys; 30 int nkeys; 31 } s3c_kbd_platform_data_t; 32 33 #endif /* ----- #ifndef _KBD_DRIVER_H_ ----- */
platform_device相關驅動文件 kbd_device.c:shell
1 /********************************************************************************* 2 * Copyright: (C) 2016 Guo Wenxue<guowenxue@aliyun.com> 3 * All rights reserved. 4 * 5 * Filename: kbd_device.c 6 * Description: This file 7 * 8 * Version: 1.0.0(07/26/2016) 9 * Author: Guo Wenxue <guowenxue@aliyun.com> 10 * ChangeLog: 1, Release initial version on "07/26/2016 05:01:25 PM" 11 * 12 ********************************************************************************/ 13 14 #include <linux/module.h> 15 #include <linux/init.h> 16 #include <linux/platform_device.h> 17 #include <linux/input.h> 18 #include <mach/hardware.h> 19 #include <asm/gpio.h> 20 #include <asm/irq.h> 21 #include <mach/regs-gpio.h> 22 #include "kbd_driver.h" 23 24 static s3c_kbd_info_t s3c_kbd_gpios[] = { 25 [0] = { 26 .code = KEY_1, 27 .nIRQ = IRQ_EINT0, 28 .gpio = S3C2410_GPF(0), 29 .setting = S3C2410_GPF0_EINT0, 30 }, 31 [1] = { 32 .code = KEY_2, 33 .nIRQ = IRQ_EINT2, 34 .gpio = S3C2410_GPF(2), 35 .setting = S3C2410_GPF2_EINT2, 36 }, 37 [2] = { 38 .code = KEY_3, 39 .nIRQ = IRQ_EINT3, 40 .gpio = S3C2410_GPF(3), 41 .setting = S3C2410_GPF3_EINT3, 42 }, 43 [3] = { 44 .code = KEY_4, 45 .nIRQ = IRQ_EINT4, 46 .gpio = S3C2410_GPF(4), 47 .setting = S3C2410_GPF4_EINT4, 48 }, 49 }; 50 51 /* keyboard platform device private data */ 52 static s3c_kbd_platform_data_t s3c_kbd_data = { 53 .keys = s3c_kbd_gpios, 54 .nkeys = ARRAY_SIZE(s3c_kbd_gpios), 55 }; 56 57 static void platform_kbd_release(struct device * dev) 58 { 59 return; 60 } 61 62 static struct platform_device s3c_keyboard_device = { 63 .name = "s3c_kbd", 64 .id = 1, 65 .dev = 66 { 67 .platform_data = &s3c_kbd_data, 68 .release = platform_kbd_release, 69 }, 70 }; 71 72 73 static int __init s3c_keyboard_dev_init(void) 74 { 75 int rv; 76 77 rv = platform_device_register(&s3c_keyboard_device); 78 if(rv) 79 { 80 printk("S3C keyboard platform device register failure\n"); 81 return rv; 82 } 83 84 printk("S3C keyboard platform device register ok\n"); 85 return 0; 86 } 87 88 static void __exit s3c_keyboard_dev_exit(void) 89 { 90 printk("S3C keyboard device exit\n"); 91 92 platform_device_unregister(&s3c_keyboard_device); 93 return ; 94 } 95 96 module_init(s3c_keyboard_dev_init); 97 module_exit(s3c_keyboard_dev_exit); 98 99 MODULE_DESCRIPTION("FL2440 board keyboard input driver platform_device"); 100 MODULE_AUTHOR("Guo Wenxue<guowenxue@gmail.com>"); 101 MODULE_LICENSE("GPL"); 102 MODULE_ALIAS("platform:FL2440 keyboard device");
platform_driver相關驅動文件 kbd_driver.c:centos
/********************************************************************************* * Copyright: (C) 2016 Guo Wenxue<guowenxue@aliyun.com> * All rights reserved. * * Filename: kbd_driver.c * Description: This file * * Version: 1.0.0(07/26/2016) * Author: Guo Wenxue <guowenxue@aliyun.com> * ChangeLog: 1, Release initial version on "07/26/2016 05:01:25 PM" * ********************************************************************************/ #include <linux/module.h> #include <linux/init.h> #include <linux/platform_device.h> #include <linux/input.h> #include <linux/irq.h> #include <linux/interrupt.h> #include <mach/hardware.h> #include <asm/gpio.h> #include <asm/irq.h> #include <linux/slab.h> #include <mach/regs-gpio.h> #include "kbd_driver.h" /* 1HZ=100*jiffies 1*jiffies=10ms => 1HZ=100*10ms = 1s */ #define CANCEL_DITHERING_DELAY (HZ/50) /* Remove button push down dithering timer delay 20ms */ typedef struct s3c_kbd_s { struct timer_list *timers; /* every key get a cancel dithering timer */ struct input_dev *input_dev; s3c_kbd_platform_data_t *pdata; } s3c_kbd_t; /*--- end of struct s3c_kbd_s ---*/ s3c_kbd_t *s3c_kbd = NULL; static irqreturn_t s3c_kbd_intterupt(int irq, void *dev_id) { int i; int found = 0; struct platform_device *pdev = dev_id; s3c_kbd_t *s3c_kbd = NULL; s3c_kbd = platform_get_drvdata(pdev); for(i=0; i<s3c_kbd->pdata->nkeys; i++) { if(irq == s3c_kbd->pdata->keys[i].nIRQ) { found = 1; break; } } if(!found) /* An ERROR interrupt */ return IRQ_NONE; mod_timer(&s3c_kbd->timers[i], jiffies+CANCEL_DITHERING_DELAY); return IRQ_HANDLED; } static void cancel_dithering_timer_handler(unsigned long data) { int which =(int)data; unsigned int pinval; pinval = s3c2410_gpio_getpin(s3c_kbd->pdata->keys[which].gpio); if( pinval ) { //printk("s3c_kbd key[%d] code[%d] released\n", which, s3c_kbd->pdata->keys[which].code); input_event(s3c_kbd->input_dev, EV_KEY, s3c_kbd->pdata->keys[which].code, 0); } else { //printk("s3c_kbd key[%d] code[%d] pressed\n", which, s3c_kbd->pdata->keys[which].code); input_event(s3c_kbd->input_dev, EV_KEY, s3c_kbd->pdata->keys[which].code, 1); } input_sync(s3c_kbd->input_dev); } static int s3c_kbd_probe(struct platform_device *pdev) { int i = 0; int rv = -ENOMEM; struct input_dev *input_dev = NULL; s3c_kbd_platform_data_t *pdata = pdev->dev.platform_data; /* malloc s3c_kbd struct */ s3c_kbd = kmalloc(sizeof(s3c_kbd_t), GFP_KERNEL); if( !s3c_kbd ) { printk("error: s3c_kbd_probe kmalloc() for s3c_kbd failure\n"); goto fail; } memset(s3c_kbd, 0, sizeof(s3c_kbd_t)); /* malloc cancel dithering timer for every key */ s3c_kbd->timers = (struct timer_list *) kmalloc(pdata->nkeys*sizeof(struct timer_list), GFP_KERNEL); if( !s3c_kbd->timers ) { printk("error: s3c_kbd_probe kmalloc() for s3c_kbd timers failure\n"); goto fail; } memset(s3c_kbd->timers, 0, pdata->nkeys*sizeof(struct timer_list)); /* malloc input_dev for keyboard */ input_dev=input_allocate_device(); if( !input_dev ) { printk("error: s3c_kbd_probe input_allocate_device() failure\n"); goto fail; } /* setup input_dev */ input_dev->name = pdev->name; input_dev->dev.parent = &pdev->dev; input_dev->id.bustype = BUS_HOST; input_dev->id.vendor = 0x0001; input_dev->id.product = 0x0001; input_dev->id.version = 0x0100; set_bit(EV_KEY,input_dev->evbit); set_bit(EV_REP,input_dev->evbit); /* Initialize all the keys and interrupt */ for(i=0; i<pdata->nkeys; i++) { set_bit(pdata->keys[i].code, input_dev->keybit); s3c2410_gpio_cfgpin(pdata->keys[i].gpio, pdata->keys[i].setting); irq_set_irq_type(pdata->keys[i].nIRQ, IRQ_TYPE_EDGE_BOTH); rv = request_irq(pdata->keys[i].nIRQ, s3c_kbd_intterupt, IRQF_DISABLED, pdev->name, pdev); if( rv ) { printk("error: request IRQ[%d] for key<%d> failure\n", pdata->keys[i].nIRQ, i); rv = -EBUSY; goto fail; } //printk("s3c_kbd request IRQ[%d] for key<%d> ok\n", pdata->keys[i].nIRQ, i); /* Initialize all the keys cancel dithering timer */ setup_timer(&s3c_kbd->timers[i], cancel_dithering_timer_handler, i); } /* register input device */ rv = input_register_device(input_dev); if( rv ) { printk("error: s3c_kbd_probe input_register_device error!\n"); goto fail; } /* set s3c_kbd as private data in pdev */ s3c_kbd->input_dev = input_dev; s3c_kbd->pdata = pdata; platform_set_drvdata(pdev, s3c_kbd); printk("s3c_kbd_probe ok\n"); return 0; fail: while(i--) { disable_irq(pdata->keys[i].nIRQ); free_irq(pdata->keys[i].nIRQ, pdev); del_timer( &s3c_kbd->timers[i] ); } if(input_dev) { input_free_device(input_dev); } if(s3c_kbd && s3c_kbd->timers) { kfree(s3c_kbd->timers); } if(s3c_kbd) { kfree(s3c_kbd); } printk("s3c_kbd_probe failed\n"); return -ENODEV; } static int s3c_kbd_remove(struct platform_device *pdev) { int i = 0; s3c_kbd_t *s3c_kbd = platform_get_drvdata(pdev); for(i=0; i<s3c_kbd->pdata->nkeys; i++) { del_timer( &s3c_kbd->timers[i] ); disable_irq(s3c_kbd->pdata->keys[i].nIRQ); free_irq(s3c_kbd->pdata->keys[i].nIRQ, pdev); } input_unregister_device(s3c_kbd->input_dev); kfree(s3c_kbd->timers); kfree(s3c_kbd); printk("s3c_kbd_remove ok\n"); return 0; } static struct platform_driver s3c_keyboard_driver = { .probe = s3c_kbd_probe, .remove = s3c_kbd_remove, .driver = { .name = "s3c_kbd", .owner = THIS_MODULE, }, }; static int __init s3c_keyboard_drv_init(void) { int rv; rv = platform_driver_register(&s3c_keyboard_driver); if(rv) { printk("s3c keyboard platform driver register failure\n"); return rv; } printk("s3c keyboard platform driver register ok\n"); return 0; } static void __exit s3c_keyboard_drv_exit(void) { printk("s3c keyboard driver exit\n"); platform_driver_unregister(&s3c_keyboard_driver); return ; } module_init(s3c_keyboard_drv_init); module_exit(s3c_keyboard_drv_exit); MODULE_DESCRIPTION("FL2440 board keyboard input driver platform_driver"); MODULE_AUTHOR("Guo Wenxue<guowenxue@gmail.com>"); MODULE_LICENSE("GPL"); MODULE_ALIAS("platform:FL2440 keyboard driver");
驅動測試文件event_button.c:bash
/********************************************************************************* * Copyright: (C) 2012 Guo Wenxue<Email:guowenxue@gmail.com QQ:281143292> * All rights reserved. * * Filename: event_button.c * Description: This file used to test GPIO button driver builtin Linux kernel on ARM board * * Version: 1.0.0(07/13/2012~) * Author: Guo Wenxue <guowenxue@gmail.com> * ChangeLog: 1, Release initial version on "07/13/2012 02:46:18 PM" * ********************************************************************************/ #include <stdio.h> #include <unistd.h> #include <errno.h> #include <string.h> #include <stdlib.h> #include <unistd.h> #include <fcntl.h> #include <libgen.h> #include <getopt.h> #include <sys/types.h> #include <sys/ioctl.h> #include <linux/input.h> #include <linux/kd.h> #include <linux/keyboard.h> #if 0 /* Just for comment here, Reference to linux-3.3/include/linux/input.h */ struct input_event { struct timeval time; __u16 type; /* 0x00:EV_SYN 0x01:EV_KEY 0x04:EV_MSC 0x11:EV_LED*/ __u16 code; /* key value, which key */ __s32 value; /* 1: Pressed 0:Not pressed 2:Always Pressed */ }; #endif #define TRUE 1 #define FALSE 0 #define EV_RELEASED 0 #define EV_PRESSED 1 #define EV_REPEAT 2 #define BUTTON_CNT 5 #define MODE_POLL 0x01 #define MODE_NORMAL 0x02 void usage(char *name); void display_button_event(struct input_event *ev, int cnt); int main(int argc, char **argv) { char *kbd_dev = NULL; char kbd_name[256] = "Unknown"; int kbd_fd = -1; int rv, opt; int mode = MODE_NORMAL; int size = sizeof (struct input_event); struct input_event ev[BUTTON_CNT]; struct option long_options[] = { {"device", required_argument, NULL, 'd'}, {"poll", no_argument, NULL, 'p'}, {"help", no_argument, NULL, 'h'}, {NULL, 0, NULL, 0} }; while ((opt = getopt_long(argc, argv, "d:ph", long_options, NULL)) != -1) { switch (opt) { case 'd': kbd_dev = optarg; break; case 'p': mode = MODE_POLL; break; case 'h': usage(argv[0]); return 0; default: break; } } if(NULL == kbd_dev) { usage(argv[0]); return -1; } if ((getuid ()) != 0) printf ("You are not root! This may not work...\n"); if ((kbd_fd = open(kbd_dev, O_RDONLY)) < 0) { printf("Open %s failure: %s", kbd_dev, strerror(errno)); return -1; } ioctl (kbd_fd, EVIOCGNAME (sizeof (kbd_name)), kbd_name); printf ("Monitor input device %s (%s) event with %s mode:\n", kbd_dev, kbd_name, MODE_POLL==mode?"poll":"infilit loop"); #if 0 /* Not implement in the Linux GPIO button driver */ unsigned char key_b[BUTTON_CNT/8 + 1]; memset(key_b, 0, sizeof(key_b)); if(ioctl(kbd_fd, EVIOCGKEY(sizeof(key_b)), key_b) < 0) { printf("EVIOCGKEY ioctl get error: %s\n", strerror(errno)); return -1; } #endif #if 0 /* Not implement in the Linux GPIO button driver */ /* rep[0]表示在按é」®é‡å¤å‡ºçŽ°ä¹‹å‰ delay的时间,rep[1]表示按é」®é‡å¤å‡ºçŽ°çš„时间间éš」。 */ int rep[2] ={2500, 1000} ; if(ioctl(kbd_fd, EVIOCSREP, rep) < 0) { printf("EVIOCSREP ioctl get error: %s\n", strerror(errno)); return -1; } if(ioctl(kbd_fd, EVIOCGREP, rep) < 0) { printf("EVIOCGKEY ioctl get error: %s\n", strerror(errno)); return -1; } else { printf("repeate speed: [0]= %d, [1] = %d/n", rep[0], rep[1]); } #endif while (1) { if(MODE_POLL==mode) { fd_set rds; FD_ZERO(&rds); FD_SET(kbd_fd, &rds); rv = select(kbd_fd + 1, &rds, NULL, NULL, NULL); if (rv < 0) { printf("Select() system call failure: %s\n", strerror(errno)); goto CleanUp; } else if (FD_ISSET(kbd_fd, &rds)) { if ((rv = read (kbd_fd, ev, size*BUTTON_CNT )) < size) { printf("Reading data from kbd_fd failure: %s\n", strerror(errno)); break; } else { display_button_event(ev, rv/size); } } } else { if ((rv = read (kbd_fd, ev, size*BUTTON_CNT )) < size) { printf("Reading data from kbd_fd failure: %s\n", strerror(errno)); break; } else { display_button_event(ev, rv/size); } } } CleanUp: close(kbd_fd); return 0; } void usage(char *name) { char *progname = NULL; char *ptr = NULL; ptr = strdup(name); progname = basename(ptr); printf("Usage: %s [-p] -d <device>\n", progname); printf(" -d[device ] button device name\n"); printf(" -p[poll ] Use poll mode, or default use infinit loop.\n"); printf(" -h[help ] Display this help information\n"); free(ptr); return; } void display_button_event(struct input_event *ev, int cnt) { int i; struct timeval pressed_time, duration_time; for(i=0; i<cnt; i++) { //printf("type:%d code:%d value:%d\n", ev[i].type, ev[i].code, ev[i].value); if(EV_KEY==ev[i].type && EV_PRESSED==ev[i].value) { if(BTN_1 == ev[i].code) { pressed_time = ev[i].time; printf("S1 button key[%d] pressed time: %ld.%ld\n", ev[i].code, pressed_time.tv_sec, pressed_time.tv_usec); } else if(BTN_2 == ev[i].code) { pressed_time = ev[i].time; printf("S2 button key[%d] pressed time: %ld.%ld\n", ev[i].code, pressed_time.tv_sec, pressed_time.tv_usec); } else if(BTN_3 == ev[i].code) { pressed_time = ev[i].time; printf("S3 button key[%d] pressed time: %ld.%ld\n", ev[i].code, pressed_time.tv_sec, pressed_time.tv_usec); } else if(BTN_4 == ev[i].code) { pressed_time = ev[i].time; printf("S4 button key[%d] pressed time: %ld.%ld\n", ev[i].code, pressed_time.tv_sec, pressed_time.tv_usec); } else { pressed_time = ev[i].time; printf("button key[%d] pressed time: %ld.%ld\n", ev[i].code, pressed_time.tv_sec, pressed_time.tv_usec); } } if(EV_KEY==ev[i].type && EV_RELEASED==ev[i].value) { if(BTN_1 == ev[i].code) { timersub(&ev[i].time, &pressed_time, &duration_time); printf("S1 button key[%d] released time: %ld.%ld\n", ev[i].code, ev[i].time.tv_sec, ev[i].time.tv_usec); printf("S1 button key[%d] duration time: %ld.%ld\n", ev[i].code, duration_time.tv_sec, duration_time.tv_usec); } else if(BTN_2 == ev[i].code) { timersub(&ev[i].time, &pressed_time, &duration_time); printf("S2 button key[%d] released time: %ld.%ld\n", ev[i].code, ev[i].time.tv_sec, ev[i].time.tv_usec); printf("S2 button key[%d] duration time: %ld.%ld\n", ev[i].code, duration_time.tv_sec, duration_time.tv_usec); } else if(BTN_3 == ev[i].code) { timersub(&ev[i].time, &pressed_time, &duration_time); printf("S3 button key[%d] released time: %ld.%ld\n", ev[i].code, ev[i].time.tv_sec, ev[i].time.tv_usec); printf("S3 button key[%d] duration time: %ld.%ld\n", ev[i].code, duration_time.tv_sec, duration_time.tv_usec); } else if(BTN_4 == ev[i].code) { timersub(&ev[i].time, &pressed_time, &duration_time); printf("S4 button key[%d] released time: %ld.%ld\n", ev[i].code, ev[i].time.tv_sec, ev[i].time.tv_usec); printf("S4 button key[%d] duration time: %ld.%ld\n", ev[i].code, duration_time.tv_sec, duration_time.tv_usec); } else { timersub(&ev[i].time, &pressed_time, &duration_time); printf("button key[%d] released time: %ld.%ld\n", ev[i].code, ev[i].time.tv_sec, ev[i].time.tv_usec); printf("button key[%d] duration time: %ld.%ld\n", ev[i].code, duration_time.tv_sec, duration_time.tv_usec); } } } /* for(i=0; i<cnt; i++) */ }
驅動和測試程序編譯Makefile文件app
1 TEST_APP=event_button 2 3 KERNEL_VER = linux-3.0 4 LINUX_SRC ?= ../../linux/kernel/$(KERNEL_VER) 5 6 CROSS_COMPILE=/opt/buildroot-2012.08/arm920t/usr/bin/arm-linux- 7 8 PWD := $(shell pwd) 9 10 obj-m += kbd_device.o 11 obj-m += kbd_driver.o 12 13 modules: 14 @make -C $(LINUX_SRC) M=$(PWD) modules 15 @make clear 16 @chmod a+x *.ko && cp *.ko /tftp 17 @make testapp 18 19 clear: 20 @rm -f *.o *.cmd *.mod.c 21 @rm -rf *~ core .depend .tmp_versions Module.symvers modules.order -f 22 @rm -f .*ko.cmd .*.o.cmd .*.o.d 23 24 clean: clear 25 @rm -f *.ko ${TEST_APP} 26 27 testapp: 28 ${CROSS_COMPILE}gcc ${TEST_APP}.c -o ${TEST_APP}
編譯:oop
[guowenxue@centos6 input_kbd]$ make make[1]: Entering directory `/usr/local/src/.guowenxue/code.taobao.org/fl2440/trunk/src/linux/kernel/linux-3.0' CC [M] /usr/local/src/.guowenxue/code.taobao.org/fl2440/trunk/src/driver/input_kbd/kbd_device.o CC [M] /usr/local/src/.guowenxue/code.taobao.org/fl2440/trunk/src/driver/input_kbd/kbd_driver.o Building modules, stage 2. MODPOST 2 modules CC /usr/local/src/.guowenxue/code.taobao.org/fl2440/trunk/src/driver/input_kbd/kbd_device.mod.o LD [M] /usr/local/src/.guowenxue/code.taobao.org/fl2440/trunk/src/driver/input_kbd/kbd_device.ko CC /usr/local/src/.guowenxue/code.taobao.org/fl2440/trunk/src/driver/input_kbd/kbd_driver.mod.o LD [M] /usr/local/src/.guowenxue/code.taobao.org/fl2440/trunk/src/driver/input_kbd/kbd_driver.ko make[1]: Leaving directory `/usr/local/src/.guowenxue/code.taobao.org/fl2440/trunk/src/linux/kernel/linux-3.0' make[1]: Entering directory `/usr/local/src/.guowenxue/code.taobao.org/fl2440/trunk/src/driver/input_kbd' make[1]: Leaving directory `/usr/local/src/.guowenxue/code.taobao.org/fl2440/trunk/src/driver/input_kbd' make[1]: Entering directory `/usr/local/src/.guowenxue/code.taobao.org/fl2440/trunk/src/driver/input_kbd' /opt/buildroot-2012.08/arm920t/usr/bin/arm-linux-gcc event_button.c -o event_button make[1]: Leaving directory `/usr/local/src/.guowenxue/code.taobao.org/fl2440/trunk/src/driver/input_kbd'
運行測試:測試
列出已經安裝的驅動模塊 ~ >: lsmod kbd_driver 2470 0 - Live 0xbf01c000 kbd_device 973 0 - Live 0xbf018000 ~ >: 刪除驅動的platform_driver模塊 ~ >: rmmod kbd_driver s3c keyboard driver exit s3c_kbd_remove ok ~ >: 刪除驅動的paltform_device模塊 ~ >: rmmod kbd_device S3C keyboard device exit ~ >: 安裝驅動的paltform_device模塊 ~ >: insmod kbd_device.ko S3C keyboard platform device register ok ~ >: 安裝驅動的platform_driver模塊 ~ >: insmod kbd_driver.ko input: s3c_kbd as /devices/platform/s3c_kbd.1/input/input6 s3c_kbd_probe ok s3c keyboard platform driver register ok ~ >: 使用測試程序event_button分別測試4個按鍵: ~ >: event_button -p -d /dev/input/event1 Monitor input device /dev/input/event1 (s3c_kbd) event with poll mode: button key[2] pressed time: 1469541794.755122 button key[2] released time: 1469541794.980104 button key[2] duration time: 1469418914.857224 button key[3] pressed time: 1469541798.625197 button key[3] released time: 1469541798.865105 button key[3] duration time: 1469418918.742225 button key[4] pressed time: 1469541801.260260 button key[4] released time: 1469541801.535216 button key[4] duration time: 1469418921.412336 button key[5] pressed time: 1469541803.805109 button key[5] released time: 1469541804.107 button key[5] duration time: 1469418923.877227