keypad driver

键盘是6x6矩阵式,在网上下了对应的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键盘模块最大支持8x8,但我这里只用到了6x6
        .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程序

你可能感兴趣的:(keypad driver)