Android 蓝牙键值适配(最通俗易通的一篇文章)
Posted peak wang
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Android 蓝牙键值适配(最通俗易通的一篇文章)相关的知识,希望对你有一定的参考价值。
目录
以下图中TV VOD两个按键为例,文章中所涉及到的文件只写文件名,因每个方案的路径各不相同,请自行全局搜索文件。
1.获取按键的扫描码
android设备串口或adb shell下执行 getevent -l(小写的L),然后按下按键(以VOD键为例),打印结果如下:
1. getevent显示结果中的三个有效信息:
1)000c006b 前四位000c为键值类型(详情请跳到2与阅读hid.h),后四位为蓝牙键值;
2)KEY_BLUE 如果在linux_key.h中有此按键的定义,此处就会显示,如果linux_key.h中没有对此键的定义就会显示KEY_UNKNOWN.
3)/dev/input/event4: 设备节点,可以通过dumpsys input命令查看此节点使用KeyLayoutFile(即映射键值的.kl的文件)
console:/ # getevent -l
add device 1: /dev/input/event4
name: "语音助手"
could not get driver version for /dev/input/mouse2, Not a typewriter
add device 2: /dev/input/event3
name: "Hi Keypad"
could not get driver version for /dev/input/mouse1, Not a typewriter
add device 3: /dev/input/event1
name: "Hi mouse"
could not get driver version for /dev/input/mice, Not a typewriter
add device 4: /dev/input/event0
name: "Hi keyboard"
could not get driver version for /dev/input/mouse0, Not a typewriter
add device 5: /dev/input/event2
name: "qwerty"
/dev/input/event4: EV_MSC MSC_SCAN 000c006b
/dev/input/event4: EV_KEY KEY_BLUE DOWN
/dev/input/event4: EV_SYN SYN_REPORT 00000000
/dev/input/event4: EV_MSC MSC_SCAN 000c006b
/dev/input/event4: EV_KEY KEY_BLUE UP
/dev/input/event4: EV_SYN SYN_REPORT 00000000
2.hid-input.c
从getevent -l时已知VOD按键为000c006b,前四位详细定义可以参考kernel下的hid.h,此处只列举几个:
#define HID_UP_KEYBOARD 0x00070000
#define HID_UP_LED 0x00080000
#define HID_UP_BUTTON 0x00090000
#define HID_UP_ORDINAL 0x000a0000
#define HID_UP_TELEPHONY 0x000b0000
#define HID_UP_CONSUMER 0x000c0000
在hid-input.c中有个函数hidinput_configure_usage,会根据不同的hid执行不同的代码,为0007(HID_UP_KEYBOARD)时会处理hid_keyboard[256]的数组,把扫描码转换为10进制后就是该数组的下标,数组下标的数组元素就是linux键值,我的键值前面是000c所以跳过前面说的,case走的是HID_UP_CONSUMER,此case下已有对0x6b的处理,意思就是此键在linux中代表的是KEY_BLUE。
static void hidinput_configure_usage(struct hid_input *hidinput, struct hid_field *field,
struct hid_usage *usage)
/**省略**/
switch (usage->hid & HID_USAGE_PAGE)
case HID_UP_UNDEFINED:
goto ignore;
/**省略**/
/**省略**/
case HID_UP_KEYBOARD:
set_bit(EV_REP, input->evbit);
if ((usage->hid & HID_USAGE) < 256)
if (!hid_keyboard[usage->hid & HID_USAGE]) goto ignore;
map_key_clear(hid_keyboard[usage->hid & HID_USAGE]);
else
map_key(KEY_UNKNOWN);
break;
case HID_UP_CONSUMER: /* USB HUT v1.12, pages 75-84 */
switch (usage->hid & HID_USAGE)
case 0x06a: map_key_clear(KEY_GREEN); break;
case 0x06b: map_key_clear(KEY_BLUE); break;
case 0x06c: map_key_clear(KEY_YELLOW); break;
case 0x06d: map_key_clear(KEY_ZOOM); break;
/**省略**/
/**省略**/
那KEY_BLUE的键值是多少呢?
"KEY_BLUE", 0x191,
我们来查阅linux_key.h,搜索KEY_BLUE,我的sdk中KEY_BLUE的值为0x191,转换为十进制值为401。
3.KeyLayoutFile --> .kl
getevent -l时已知设备节点为/dev/input/event4,那我们执行下dumpsys input,结果如下:
5: 语音助手
Classes: 0x8000012b
Path: /dev/input/event4
Enabled: true
Descriptor: e0cab5a5e7db45d100777208b12fb1c21978adca
Location:
ControllerNumber: 0
UniqueId: 18:11:18:24:33:18
Identifier: bus=0x0005, vendor=0x2b54, product=0x1600, version=0x0101
KeyLayoutFile: /vendor/usr/keylayout/Vendor_2b54_Product_1600.kl
KeyCharacterMapFile: /vendor/usr/keychars/Generic.kcm
ConfigurationFile:
HaveKeyboardLayoutOverlay: false
结果中可以看出我的蓝牙遥控器的vendor=2b54,product=1600,所以系统优先加载的我自己加到ROM中的kl(.kl命名规则与加载优先级请自行百度),有的可能找不到,会加载Generic.kl等文件,请根据KeyLayoutFile:后的路径的文件修改映射关系,如果为默认的Generic.kl时,建议不要修改此文件,可以根据vendor与product规则自己新建一个文件。
那我们打开指向的这个kl文件,内容如下:
#key 222 "KEY_RED"
# key 400 "KEY_YELLOW"
key 401 PROG_GREEN
key 402 CHANNEL_UP
key 403 CHANNEL_DOWN
kl文件格式简单解析(详解与其它使用方法自行搜索):
key 可以理解成固定格式
222/401等数字对应的就是linux的键值
PROG_GREEN 这里对应的就是Android层的键值(此字符串要与InputEventLabels.h中的保持一致,否则会报Expected key code label异常,然后就加载其它默认的kl文件了,5.0以前是KeycodeLabels.h或KeyCodeLabels.h)
既然都理清楚了,那处理按键岂不是很简单了?
已知前面linux_key.h中查询VOD的linux键值为401,如果你想让此按键生效,把前面的#号去掉重启设备,使其映射关系生效即可,如果你想改变键值的功能,可以直接在此kl中把401对应的android层的KEY_BLUE改为你想要的键就可以了。(当然也可以修改linux的键值,但不提倡这么做.)
总结:
1.通过getevent -l获取键值设备节点;
2.hid-input.c中添加或修改你的键值;
3.通过设备节点找到对应的kl文件;
4.添加或修改映射关系;
以上是关于Android 蓝牙键值适配(最通俗易通的一篇文章)的主要内容,如果未能解决你的问题,请参考以下文章
一起Talk Android吧(第三百九十回:关于Android版本12适配蓝牙权限的问题)