在一个月前,改了一个遥控器的按键。
那时候还不知道那个遥控器有什么功能,只是找到键值转换表,更新一下就OK了。
本来乱七八糟的键盘控制就变得正常起来了,在android中,按键事件的处理流程在网上资料很多。
就我那个遥控器来说,属于一个单独的输入设备。当有按键按下后,设备的驱动程序就检测到按键事件得到扫描码,然后根据linux的input输入系统,进行上报,然后就到达android的Event_Hub了。android的Event_Hub再根据kl文件对扫描码(scan code)进行解析,得到了按键码(key code),按键码在一定程度上就能对系统进行控制了,如KeyUP,KeyDown,Power等。但是对于字符来说,一个按键码,比如A,可能对应的字符很多,如a,A,所以就有了按键字符映射kcm。当按下的键需要映射时,系统就根据kcm文件的内容进行解析,最后才得到正真的字符。所以,要配置遥控器的按键,很简单,修改kl文件的映射关系,只有扫描码和按键码对应就OK了。
因为遥控只有0-9这10个数字,不能输入字母。于是,就需要对数字按键进行多字符解析。
这几天搞定系统后就自己就搞这个。
在一个月前google,百度很久之后,知道有NUMERIC这个东西之后,出现了能用数字键盘打出字符的现象,但是,android的虚拟键盘上的数字键也能打出字母,跟理想不一样。纠结很久之后放弃。
现在,就来解决这个问题,解决问题的最后方法当然是源码!
理清后大概流程是这样:
在系统启动过程,android对输入服务的初始时会根据设备进行不同处理。
首先找到输入设备,获取设备信息。
在frameworks/base/services/input/EventHub.cpp中,首先new一个设备结构,然后存放设备的信息。
1013 Device* device = new Device(fd, deviceId, String8(devicePath), identifier);
1015 ALOGD("add device %d: %s\n", deviceId, devicePath);
1016 ALOGD(" bus: %04x\n"
1017 " vendor %04x\n"
1018 " product %04x\n"
1019 " version %04x\n",
1020 identifier.bus, identifier.vendor, identifier.product, identifier.version);
1021 ALOGD(" name: \"%s\"\n", identifier.name.string());
1022 ALOGD(" location: \"%s\"\n", identifier.location.string());
1023 ALOGD(" unique id: \"%s\"\n", identifier.uniqueId.string());
1024 ALOGD(" descriptor: \"%s\"\n", identifier.descriptor.string());
1025 ALOGD(" driver: v%d.%d.%d\n",
1026 driverVersion >> 16, (driverVersion >> 8) & 0xff, driverVersion & 0xff);
这些信息可以用命令得到:cat /proc/bus/input/device得到。
接着:就调用keymap去寻找配置文件信息。包括kl,kcm,idc(这个暂时还不知道干嘛用)
1028 // Load the configuration file for the device.
1029 loadConfigurationLocked(device);
其中loadConfigurationLocked会调用getInputDeviceConfigurationFilePathByDeviceIdentifier,跑到:frameworks/base/libs/androidfw/InputD