news 2026/5/30 14:41:42

外设与接口:基于内核 gpio-keys 子系统的按键处理

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
外设与接口:基于内核 gpio-keys 子系统的按键处理

1 基本原理

在 Linux 中,gpio-keys是一个平台驱动(Platform Driver),它充当了物理 GPIO 硬件与 Linux 标准输入子系统(Input Subsystem)之间的“翻译官”。

整个处理流程自下而上分为四层:

  1. 硬件层 (Hardware)
    • 物理按键连接到芯片的 GPIO 引脚。
    • 产生电平变化(中断信号)。
  2. 驱动层 (Driver: gpio-keys)
    • 职责:在设备树(DTS)匹配成功后,申请 GPIO 资源,注册中断服务函数(ISR),并配置软件消抖定时器。
    • 转换:当中断发生时,驱动读取 GPIO 电平,将物理信号(High/Low)转换为逻辑事件(Key Code, Value)。
    • 关联机制:调用input_allocate_device()input_register_device()向内核注册一个输入设备。
  3. 核心层 (Input Core)
    • 职责:Linux 内核通用的输入管理层。它不关心底层是 GPIO、USB 还是蓝牙。
    • 接口gpio-keys通过input_report_key()input_sync()函数将事件上报给核心层。
    • 分发:核心层将事件分发给对应的 Handler(通常是evdev)。
  4. 用户层 (Userspace)
    • 接口:系统在/dev/input/目录下生成字符设备节点(如event3)。
    • 数据:应用程序读取标准的struct input_event结构体。

2 操作步骤

以下是基于 RK3566 Armbian 环境的标准操作流程。为了简化软件配置,优先选择硬件默认电平与电路逻辑相符的引脚。

  • 按键接 GND:选择默认上拉的引脚(如 GPIO0_B5)。
  • 按键接 VCC:选择默认下拉的引脚。

GPIO 编号计算公式(以 RK3566 为例):

Pin Index=(Group Index×8)+Pin Number\text{Pin Index} = (\text{Group Index} \times 8) + \text{Pin Number}Pin Index=(Group Index×8)+Pin Number

  • Group A=0, B=8, C=16, D=24
  • 示例:GPIO0_B5 =8+5=138 + 5 = 138+5=13

第一步:编写设备树插件源码 (.dts)

创建文件rk3566-user-button.dts,该文件描述了按键的物理属性和逻辑键值。

/dts-v1/;/plugin/;/{compatible="rockchip,rk3566";fragment@0{target-path="/";__overlay__{gpio-keys-user{compatible="gpio-keys";autorepeat;/* 开启后,所有按键都支持长按连发 */ /* --- 第一个按键(GPIO0_B5)--- */ user_button_1{label="User Button 1";linux,code=<148>;/* KEY_PROG1 */ gpios=<&gpio0131>;/* GPIO0_B5, Active Low */ debounce-interval=<20>;gpio-key,wakeup;};/* 如果需要,可以继续加 Button2, Button3... */};};};};

第二步:编译为 .dtbo 文件

使用 设备树编译器(dtc)将源码编译为.dtbo文件。如果没有,执行以下命令安装:

sudoapt-getupdatesudoapt-getinstalldevice-tree-compiler

编译.dts文件:

dtc -I dts -O dtb -o rk3566-user-button.dtbo rk3566-user-button.dts

第三步:安装并启用 Overlay

将编译好的.dtbo文件放入系统指定的 Overlay 目录,并修改启动配置。

  1. 复制文件:将编译好的.dtbo放入/boot/overlay-user/目录(如果没有这个目录则手动创建)。
sudomkdir-p /boot/overlay-user/sudocprk3566-user-button.dtbo /boot/overlay-user/
  1. 编辑配置:修改/boot/armbianEnv.txt文件,添加或修改user_overlays参数(注意不需要 .dtbo 后缀)。
user_overlays=rk3566-user-button

第四步:重启并验证

  1. 重启系统:
sudoreboot
  1. 检查驱动加载: 重启后,查看内核日志,确认gpio-keys是否加载成功:
dmesg|grepgpio-keys
  1. 功能测试:安装并运行 evtest
sudoaptinstallevtestsudoevtest
  • 选择对应的 Event ID。
  • 按下按键应显示Event: type 1 (EV_KEY), code 148, value 1
  • 松开按键应显示value 0

3 应用开发

一旦设备树生效,系统启动后会在/dev/input/下生成一个新的节点(例如/dev/input/event3)。

应用层不需要包含任何 GPIO 相关的头文件,只需要标准的 Input 子系统头文件。

#include<stdio.h>#include<unistd.h>#include<fcntl.h>#include<linux/input.h>#defineKEY_EVENT_PATH"/dev/input/event3"intmain(){intfd=open(KEY_EVENT_PATH,O_RDONLY);structinput_eventie;while(read(fd,&ie,sizeof(ie))>0){// EV_KEY 代表按键事件if(ie.type==EV_KEY){// value: 1=按下, 0=松开, 2=长按连发printf("Key Code: %d, Action: %s\n",ie.code,ie.value==1?"Pressed":(ie.value==0?"Released":"Repeat"));}}return0;}
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/5/29 15:54:57

测试的“元认知”:智能体如何评估自身可靠性?

在软件测试领域&#xff0c;自动化与智能化正以前所未有的速度重塑工作流程。随着人工智能代理&#xff08;智能体&#xff09;广泛应用于测试用例生成、缺陷预测和持续集成&#xff0c;一个关键问题浮出水面&#xff1a;这些智能体如何像人类测试专家一样&#xff0c;对自身行…

作者头像 李华
网站建设 2026/5/28 5:23:26

本凡码农引领杭州小程序开发解决方案赋能企业创新与发展

本凡码农的杭州小程序开发解决方案为企业提供了一种高效的数字化转型工具。我们的目标是帮助品牌快速适应市场变化&#xff0c;提升用户体验。通过定制化的小程序&#xff0c;企业能够实现从线上到线下的无缝连接&#xff0c;简化业务流程&#xff0c;从而更好地满足用户需求。…

作者头像 李华
网站建设 2026/5/30 1:57:55

Windows11系统文件wer.dll丢失或损坏问题 下载修复

在使用电脑系统时经常会出现丢失找不到某些文件的情况&#xff0c;由于很多常用软件都是采用 Microsoft Visual Studio 编写的&#xff0c;所以这类软件的运行需要依赖微软Visual C运行库&#xff0c;比如像 QQ、迅雷、Adobe 软件等等&#xff0c;如果没有安装VC运行库或者安装…

作者头像 李华
网站建设 2026/5/30 14:05:56

vue基于Spring Boot的公务员考试练习系统的应用和研究_1p390k6z

目录 具体实现截图项目介绍论文大纲核心代码部分展示项目运行指导结论源码获取详细视频演示 &#xff1a;文章底部获取博主联系方式&#xff01;同行可合作 具体实现截图 本系统&#xff08;程序源码数据库调试部署讲解&#xff09;同时还支持java、ThinkPHP、Node.js、Spring…

作者头像 李华
网站建设 2026/5/29 14:09:33

测试机器人:下一代助手:AI如何重塑软件测试的未来

在软件测试领域&#xff0c;传统的手动测试和脚本化自动化测试正面临日益增长的复杂性和效率挑战。随着人工智能&#xff08;AI&#xff09;技术的迅猛发展&#xff0c;下一代测试助手——智能测试机器人——正逐步从概念走向现实&#xff0c;为测试从业者带来革命性变革。 AI…

作者头像 李华