keypad driver

2022-08-10 13:57:31 浏览数 (1)

大家好,又见面了,我是你们的朋友全栈君。

键盘是6×6矩阵式,在网上下了对应的PATCH,下载地址是

https://patchwork.kernel.org/patch/71857/

这个补丁会创建两个文件

arch/arm/plat-mxc/include/mach/mxc_keypad.h //mxc_keypad_platform_data键盘平台设备的结构体

/drivers/input/keyboard/mxc_keypad.c //驱动实现文件

打好补丁后,会发现这个驱动是一个通用的驱动,在mx21,25,27,31等板上都可以用,所以要自己实现相应的keymap,自己实现注册键盘设备,还有注册键盘相应的时钟。

1.添加键盘设备:

2.6.32内核源码里没有对mx21键盘的支持,所以我们要自己添加键盘的结构体

所以在arch/arm/mach-mx2/device.c里,加上

static struct resource mxc_keypad_resources[] = { { .start = KPP_BASE_ADDR, .end = KPP_BASE_ADDR SZ_4K – 1, .flags = IORESOURCE_MEM, },{ .start = MXC_INT_KPP, .end = MXC_INT_KPP, .flags = IORESOURCE_IRQ, }, };

struct platform_device mxc_keypad_device = { .name = “mxc-keypad”, //这个名字必须和驱动里的名字相对应 .id = 0, .num_resources = ARRAY_SIZE(mxc_keypad_resources), .resource = mxc_keypad_resources, }; 接下来在arch/arm/mach-mx2/device.h里最后一行加上结构体的声明

extern struct platform_device mxc_spi_device1; extern struct platform_device mxc_spi_device2; extern struct platform_device mxc_keypad_device; 然后进入到arch/arm/mach-mx2/mx21ads.c里

在注册的时候,添加键盘对应的注册函数,找到mxc_register_device函数,添加上

static void __init mx21ads_board_init(void) { mxc_gpio_setup_multiple_pins(mx21ads_pins, ARRAY_SIZE(mx21ads_pins), “mx21ads”);

mxc_register_device(&mxc_uart_device0, &uart_pdata); mxc_register_device(&mxc_uart_device2, &uart_norts_pdata); mxc_register_device(&mxc_uart_device3, &uart_pdata); mxc_register_device(&mxc_fb_device, &mx21ads_fb_data); mxc_register_device(&mxc_sdhc_device0, &mx21ads_sdhc_pdata); mxc_register_device(&mxc_nand_device, &mx21ads_nand_board_info); mxc_register_device(&mxc_keypad_device, &keypad_data); //keypad_data是struct mxc_keypad_platform_data结构体,需要我们自己完成,在第二步里我会具体说明

platform_add_devices(platform_devices, ARRAY_SIZE(platform_devices)); }

2.添加平台设备的数据(即定义keypad_data)

首先,根据你的键盘的实际情况,设置好keymap数组,这里我建立一个文件”mxc_keymap.h”,它是键盘的keymap数组,表明了键盘的实际按键排列情况。

#ifndef __MXC_KEYMAP_H__

#define __MXC_KEYMAP_H__

static unsigned int keymap[48] = { //这里定义大小为48不是说有48个按键,由于在补丁的驱动实现文件mxc_keypad.c里,在建立扫描码对应的键值时,每行默认为8个按键,所以这里每行都有两个是保留键。当然具体实现你也可以修改,在mxc_keypad.c里做相应变动即可 KEY_F1, //EXTRA 5 KEY_MACRO, //# KEY_0, //0 KEY_KPASTERISK, //* KEY_F8, //RECORD KEY_POWER, //POWER KEY_RESERVED, KEY_RESERVED, KEY_F2, //EXTRA 4 KEY_9, //9 KEY_8, KEY_7, KEY_F9, //EXTRA 1 KEY_VOLUMEDOWN, KEY_RESERVED, KEY_RESERVED, KEY_F3, //EXTRA 3 KEY_6, KEY_5, KEY_4, KEY_F10, //APP 4 KEY_VOLUMEUP, KEY_RESERVED, KEY_RESERVED, KEY_F4, //EXTRA 2 KEY_3, KEY_2, KEY_1, KEY_F11, //APP 3 KEY_DOWN, //DOWN KEY_RESERVED, KEY_RESERVED, KEY_BACKSPACE, //BACK KEY_RIGHT, //RIGHT KEY_F6, //ACTION KEY_LEFT, //LEFT KEY_HOME, //HOME KEY_F16, //APP 2 KEY_RESERVED, KEY_RESERVED, KEY_END, //END KEY_F5, //KEY 2 KEY_UP, //UP KEY_F7, //KEY 1 KEY_F12, //SEND KEY_F17, //APP 1 KEY_RESERVED, KEY_RESERVED, };

#endif /*MXC_KEYMAP_H*/

接下来,进入arch/arm/mach-mx2/mx21ads.c,首先添加包含文件mxc_keypad.h,然后添加静态结构体变量并初始化。

在文件开头包含文件的地方添加上

#include <mach/mxc_keypad.h>

然后定义静态结构体keypad_data

static struct mxc_keypad_platform_data keypad_data = { .matrix_key_rows = 6, //实际用到的行,MX21键盘模块最大支持8×8,但我这里只用到了6×6 .matrix_key_cols = 6, //实际用到的列 .matrix_key_map = keymap, //按键映射数组 .matrix_key_map_size = 48, //映射的按键数 .debounce_ms = SAMPLE_PERIOD, //消除抖动的延时,这里我设置的是20ms,可以根据实际情况修改 };

3.时钟注册

查看arch/arm/mach-mx2/clock-mx21.c

可以发现其实键盘的时钟结构体已经注册过了,

_REGISTER_CLOCK(“imx-i2c.0”, NULL, i2c_clk) _REGISTER_CLOCK(“mxc-keypad”, NULL, kpp_clk)

但是在mxc_keypad.c里获得时钟的时候,调用clk_get的参数与注册的时钟名不匹配,

keypad->clk = clk_get(NULL, “kpp”); if (IS_ERR(keypad->clk)) { dev_err(&pdev->dev, “failed to get keypad clock/n”); error = PTR_ERR(keypad->clk); goto failed_free_io; } 这里有两种方法,第一修改_REGISTER_CLOCK(“mxc-keypad”,NULL,kpp_clk)

改成 _REGISTER_CLOCK(NULL,”kpp”,kpp_clk);

第二种是修改mxc_keypad.c里的clk_get的参数,改成 clk_get(“mxc-keypad”,NULL);

个人倾向于第一种,呵呵,改驱动的代码万一改错麻烦就大得多了。

———————————————-

这里我以为已经改好了,所以写了个测试的应用程序keytest来测试

#include <stdio.h> #include <string.h> #include <unistd.h> #include <fcntl.h> #include <signal.h> #include <linux/input.h>

int fd; int loop_flag = 1;

void exit_loop() { loop_flag–; }

void deal_int(int signum) { if(fd > 0) close(fd); exit_loop(); }

int main() { struct input_event evt;

fd = open(“/dev/input/event0”,O_RDWR);

if(fd < 0) printf(“Can not open/n”);

while(loop_flag > 0) { read(fd,&evt,sizeof(struct input_event)); switch(evt.type) { case EV_KEY: printf(“Key –>/nCode: %d/nValue: %d/n”,evt.code,evt.value); break; case EV_MSC: printf(“MSC –>/nCode: %d/nValue: %d/n”,evt.code,evt.value); break; case EV_SYN: printf(“<– /n”); default: printf(“…/n”); } }

return 0; }

启动板子,发现键盘设备被识别,而且/dev/input/下有event0,

cat sys/class/input/input0/name得到 mxc-keypad

说明驱动正常,设备也注册好了

所以使用keytest来测试,结果发现没有发生EV_KEY,事件,这说明input_report_key函数没有成功

查看内核源码,发现input_report_key()实际上调用的是input_event(),这个函数首先会检测合法位,也就是说会检测按键的code是否已经添加到了struct input_dev的keybit里(input_dev->keybit)

所以回到驱动实现文件mxc_keypad.c里

发现static void mxc_keypad_build_keycode(struct mxc_keypad *keypad) 这个函数在添加扫描码的时候,使用循环来处理

for (i = 0; i < pdata->matrix_key_map_size; i ) { unsigned int key = pdata->matrix_key_map[i]; unsigned int row = KEY_ROW(key); unsigned int col = KEY_COL(key); unsigned int scancode = MATRIX_SCAN_CODE(row, col, MATRIX_ROW_SHIFT);

keycode = KEY_VAL(key); keypad->keycodes[scancode] = keycode; __set_bit(keycode, input_dev->keybit); } 发现错在KEY_ROW和KEY_COL这两个宏,查看include/linux/matrix_keypad.h

KEY_ROW(k) ( ((k) >> 24) & 0xff )

KEY_COL(k) (((k) >> 16) & 0xff)

也就是说这两个宏决定行列的规则是行是键值的高8位,列是键值的次高8位

查看include/linux/input.h,发现相关的KEY_*(KEY_0,KEY_UP等)的值都没有超过255,这样确定行列时,就不能使用这两个宏了,所以我注释了原来的代码,自己修改成了

/* MOD— for (i = 0; i < pdata->matrix_key_map_size; i ) { unsigned int key = pdata->matrix_key_map[i]; unsigned int row = KEY_ROW(key); unsigned int col = KEY_COL(key); unsigned int scancode = MATRIX_SCAN_CODE(row, col, MATRIX_ROW_SHIFT);

keycode = KEY_VAL(key); keypad->keycodes[scancode] = keycode; __set_bit(keycode, input_dev->keybit); } */ unsigned int row,col; unsigned int key; unsigned int scancode;

row = col = 0; for(i = 0;i < pdata->matrix_key_map_size;i ) { key = pdata->matrix_key_map[i]; scancode = MATRIX_SCAN_CODE(row,col,MATRIX_ROW_SHIFT); col ; if(col == MAX_MATRIX_KEY_COLS) { col = 0; row ; }

key = KEY_VAL(key); keypad->keycodes[scancode] = key; __set_bit(key,input_dev->keybit); } 这里如果哪位有更好的方法欢迎留言,因为我觉得这个方法不是很好。

重新编译,启动板子,运行keytest程序

发布者:全栈程序员栈长,转载请注明出处:https://javaforall.cn/130064.html原文链接:https://javaforall.cn

0 人点赞