前言:
在上一篇博客《基于Dragon Board 410c开发板的触摸屏驱动编写》中,我已经描述了触摸屏驱动编写过程,以及其架构,这篇博客我会跟大家讲述一下按键和轨迹球驱动。
一、按键和轨迹球驱动
MSM具有按键和轨迹球的功能,对应的驱动程序在文件arch/arm/mach-msm/board-mahimahi-keypad.c中,接下来开始介绍此文件的实现流程。
(1)文件board-mahimahi-keypad.c中的全局定义代码如下所示:
static struct gpio_event_info *mahimahi_input_info[] = { ?
? ? &mahimahi_keypad_matrix_info.info, ? ? ?// 键盘矩阵 ?
? ? &mahimahi_keypad_key_info.info, ? ? ? ? // 键盘信息 ?
? ? &jogball_x_axis.info.info, ? ? ? ? ? ? ?// 轨迹球X方向信息 ?
? ? &jogball_y_axis.info.info, ? ? ? ? ? ? ?// 轨迹球Y方向信息 ?
}; ?
static struct gpio_event_platform_data mahimahi_input_data = { ?
? ? .names = { ?
? ? ? ? "mahimahi-keypad", ? ? ? ? ? ? ? ? ?// 按键设备 ?
? ? ? ? "mahimahi-nav", ? ? ? ? ? ? ? ? ? ? // 轨迹球设备 ?
? ? ? ? NULL, ?
? ? }, ?
? ? .info = mahimahi_input_info, ?
? ? .info_count = ARRAY_SIZE(mahimahi_input_info), ?
? ? .power = jogball_power, ?
};
static struct platform_device mahimahi_input_device = { ?
? ? .name = GPIO_EVENT_DEV_NAME, ?
? ? .id = 0, ?
? ? .dev = { ?
? ? ? ? .platform_data = &mahimahi_input_data, ?
? ? }, ?
};?
按键和轨迹球是通过GPIO系统来实现的,因此定义了gpio_event_info类型的数组。
“mahimahi-keypad”和“mahimahi-nav”分别是两个设备的名称。
gpio_event_info 指针各式的数组mahimahi_input_info中包含了mahimahi_keypad_matrix_info.info,mahimahi_keypad_key_ info.info,jogball_x_axis.info.info和jogball_y_axis.info.info。
按键驱动是一个利用GPIO矩阵的驱动,由gpio_event_matrix_info矩阵定义,定义还需要包含按键的GPIO矩阵和input设备的信息,内容如下所示:
static unsigned int mahimahi_col_gpios[] = { 33, 32, 31 };
static unsigned int mahimahi_row_gpios[] = { 42, 41, 40 };
#define KEYMAP_INDEX(col, row) ?((col)*ARRAY_
SIZE(mahimahi_row_gpios) + (row)) ?
#define KEYMAP_SIZE ? ? (ARRAY_SIZE(mahimahi_col_gpios) * \ ?
? ? ? ? ? ? ? ? ?ARRAY_SIZE(mahimahi_row_gpios)) ?
static const unsigned short mahimahi_keymap
[KEYMAP_SIZE] = { ? ? ? ? // 按键映射关系 ?
? ? [KEYMAP_INDEX(0, 0)] = KEY_VOLUMEUP, /* 115 */ ?
? ? [KEYMAP_INDEX(0, 1)] = KEY_VOLUMEDOWN, /* 114 */ ?
? ? [KEYMAP_INDEX(1, 1)] = MATRIX_KEY(1, BTN_MOUSE), ?
}; ?
static struct gpio_event_matrix_info mahimahi
_keypad_matrix_info = { ?
? ? .info.func = gpio_event_matrix_func, ? ??
// 关键函数实现 ?
? ? .keymap = mahimahi_keymap, ?
? ? .output_gpios = mahimahi_col_gpios, ?
? ? .input_gpios = mahimahi_row_gpios, ?
? ? .noutputs = ARRAY_SIZE(mahimahi_col_gpios), ?
? ? .ninputs = ARRAY_SIZE(mahimahi_row_gpios), ?
? ? .settle_time.tv.nsec = 40 * NSEC_PER_USEC, ?
? ? .poll_time.tv.nsec = 20 * NSEC_PER_MSEC, ?
? ? .flags = (GPIOKPF_LEVEL_TRIGGERED_IRQ | ?
? ? ? ? ? GPIOKPF_REMOVE_PHANTOM_KEYS | ?
? ? ? ? ? GPIOKPF_PRINT_UNMAPPED_KEYS), ?
}; ?
static struct gpio_event_direct_entry mahimahi_
keypad_key_map[] = { // Power按键 ?
? ? { ?
? ? ? ? .gpio ? = MAHIMAHI_GPIO_POWER_KEY, ?
? ? ? ? .code ? = KEY_POWER, ?
? ? }, ?
}; ?
static struct gpio_event_input_info mahimahi_
keypad_key_info = { ?
? ? .info.func = gpio_event_input_func, ? ? ??
// 关键函数实现 ?
? ? .info.no_suspend = true, ?
? ? .flags = 0, ?
? ? .type = EV_KEY, ?
? ? .keymap = mahimahi_keypad_key_map, ?
? ? .keymap_size = ARRAY_SIZE(mahimahi_keypad_key_map) ?
};?
keypad_key_matrix _info和keypad _info是gpio_event_matrix_info类型的结构体,分别负责两个和一个按键的处理,
实际上,MSM平台基本上只有三个按键:Power,音量增加按键和音量减少按键。音量增加和音量减少的扫描码分别是KEY_VOLUMEUP(=115)和KEY_VOLUMEDOWN(=114)。
提示:音量控制的两个按键在全键盘的qwerty.kl有所定义,同时符合Linux的input设备和Android的按键标准。
轨迹球部分也是由GPIO实现的,由X方向和Y方向两部分组成,内容如下所示:
static uint32_t jogball_x_gpios[] = { ?
? ? MAHIMAHI_GPIO_BALL_LEFT, MAHIMAHI_GPIO_BALL_RIGHT, ?
}; ?
static uint32_t jogball_y_gpios[] = { ?
? ? MAHIMAHI_GPIO_BALL_UP, MAHIMAHI_GPIO_BALL_DOWN, ?
}; ?
static struct jog_axis_info jogball_x_axis = { ?
// X轴的内容 ?
? ? .info = { ?
? ? ? ? .info.func = gpio_event_axis_func, ?
// 关键函数实现 ?
? ? ? ? .count = ARRAY_SIZE(jogball_x_gpios), ?
? ? ? ? .dev = 1, ?
? ? ? ? .type = EV_REL, ?
? ? ? ? .code = REL_X, ?
? ? ? ? .decoded_size = 1U << ARRAY_SIZE(jogball_x_gpios), ?
? ? ? ? .map = jogball_axis_map, ?
? ? ? ? .gpio = jogball_x_gpios, ?
? ? ? ? .flags = GPIOEAF_PRINT_UNKNOWN_DIRECTION, ?
? ? } ?
}; ?
static struct jog_axis_info jogball_y_axis = {?
// Y轴的内容 ?
? ? .info = { ?
? ? ? ? .info.func = gpio_event_axis_func, ?
// 关键函数实现 ?
? ? ? ? .count = ARRAY_SIZE(jogball_y_gpios) ?
? ? ? ? .dev = 1, ?
? ? ? ? .type = EV_REL, ?
? ? ? ? .code = REL_Y, ?
? ? ? ? .decoded_size = 1U << ARRAY_SIZE(jogball_y_gpios), ?
? ? ? ? .map = jogball_axis_map, ?
? ? ? ? .gpio = jogball_y_gpios, ?
? ? ? ? .flags = GPIOEAF_PRINT_UNKNOWN_DIRECTION, ?
? ? } ?
};?
在上述代码中,这里的轨迹球是用jog_axis_info类型的结构体进行定义的,这种设备的类型(type)是相对设备EV_REL。
注意:除了默认的AVRCP.kl和qwerty.kl之外,在MSM8916平台中新增了文件h2w_headset.kl文件。
评论