1. 从物理按键到屏幕响应:一条看不见的“高速公路”
你有没有想过,当你按下电视遥控器上的“音量+”键,或者敲击一个连接到安卓盒子的USB键盘时,屏幕上的反应是怎么发生的?这背后其实是一条非常精密、高效的数据处理流水线,就像一条从物理世界通往数字世界的“高速公路”。我们今天要聊的,就是这条高速公路的完整路线图——从硬件扫描码开始,一直到应用层的事件分发。
这条路的起点,是HID设备。HID,全称是Human Interface Device,也就是人机接口设备。我们日常用的键盘、鼠标、游戏手柄,还有各种智能电视、机顶盒的遥控器,都属于这个范畴。它们通过USB、蓝牙或者红外等方式连接到你的安卓设备上。当你按下一个键,设备内部的芯片会立刻生成一个代表这个按键的原始电信号,这个信号在Linux内核里,会被转换成一个标准化的数字,我们称之为扫描码。你可以把它想象成这个按键的“身份证号”,比如字母‘A’的扫描码可能是0x001e。
但这个“身份证号”只在Linux内核里通用,到了安卓系统这一层,它需要被翻译成安卓自己能理解的“语言”,也就是Android键值。这个翻译过程,以及翻译后的键值如何被精准地送到正确的App手里,就是整个事件传递链路的核心。我做过不少安卓电视和定制化遥控器的项目,经常需要调整这个映射关系,让一个特殊的按键能触发我们想要的功能。这个过程听起来有点技术,但拆开来看,每一步都很有意思,而且一旦搞明白,很多遥控器适配的难题就迎刃而解了。
2. 事件传递链路的三大核心“枢纽站”
这条高速公路不是一条直路,中间有几个关键的“枢纽站”负责处理和转发信息。理解了它们,你就掌握了整个流程的脉络。
2.1 第一站:EventHub —— 原始事件的“收费站”
EventHub是安卓输入子系统最底层的组件,你可以把它看作一个24小时不间断的“收费站”。它的工作非常纯粹:轮询所有连接到系统的输入设备(比如/dev/input/event0, event1这些设备节点),读取来自Linux内核evdev驱动上报的原始事件数据。
这些原始数据包就像一辆辆载着货物(扫描码、按下/松开状态、时间戳等)的卡车。EventHub不关心卡车里装的是什么,它的任务就是确保每一辆“卡车”都能被及时、准确地接收下来。它会为每个输入设备维护状态,比如当前哪些按键被按下了(元键状态,像Shift、Ctrl)。当有按键事件发生时,EventHub会从驱动读取到类似这样的信息:“扫描码0x001e,动作:按下(KEYDOWN)”。
这里有个小细节,EventHub在读取关键事件(比如电源键唤醒)时,会持有唤醒锁,确保系统不会在处理过程中休眠,这保证了响应的即时性。你可以通过getevent这个命令行工具来亲眼看看这些原始数据流,感受一下EventHub在忙些什么。
adb shell getevent -l
执行这个命令后,当你按下遥控器或键盘的某个键,终端会打印出类似下面的信息:
/dev/input/event3: EV_KEY KEY_VOLUMEUP DOWN
/dev/input/event3: EV_SYN SYN_REPORT 00000000
/dev/input/event3: EV_KEY KEY_VOLUMEUP UP
/dev/input/event3: EV_SYN SYN_REPORT 00000000
这表示event3这个设备上报了一个KEY_VOLUMEUP(音量增大)键的按下和松开事件。这就是最原始的数据。
2.2 第二站:InputReader —— 翻译官与加工厂
InputReader是EventHub的消费者,它从EventHub那里拿到原始的“卡车”(事件),然后开始进行深加工。它的核心职责有两个:翻译和封装。
首先,它是个“翻译官”。它手里有一本或多本“词典”,这就是按键布局映射文件(.kl文件,比如Generic.kl, Vendor_XXXX_Product_XXXX.kl)。这本词典定义了如何将Linux扫描码(如KEY_VOLUMEUP)转换成Android键值(如KEYCOD


1103

被折叠的 条评论
为什么被折叠?



