当前位置: 首页 > news >正文

外设与接口:基于内核 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;}
http://www.jsqmd.com/news/89131/

相关文章:

  • sglang 大模型推理框架支持的EAGLE 1,2,3
  • 延凡科技 EMS 智慧云平台:3 万起订阅,中小用能单位的 “云端全能源管家”
  • 拦截器注册InterceptorRegistry 实现讲解
  • 汇编语言全接触-27.工具提示控件
  • 汇编语言全接触-26.启动画面
  • 我不是这样
  • 验证IP地址(一)
  • 医院管理|基于springboot 医院管理系统(源码+数据库+文档)
  • 浅谈:算法中的斐波那契数(一)
  • 测试的“元认知”:智能体如何评估自身可靠性?
  • 10.8 总结
  • 【Hadoop+Spark+python毕设】物联网网络安全威胁数据分析系统、计算机毕业设计、包括数据爬取、数据分析、数据可视化、Hadoop、实战教学
  • 9.28总结
  • 零基础学JAVA--Day34(Map接口+HashTable+HashMap+TreeSet+TreeMap+开发中如何选择集合实现类?(重要)) - 指南
  • 电影院购票|基于springboot 电影院购票系统(源码+数据库+文档)
  • C#+VisionMaster联合开发(二)_操作流程
  • 本地部署DeepSeek
  • AI驱动的手动测试变革:赋能而非替代
  • 航空机票预定系统|基于springboot 航空机票预定系统(源码+数据库+文档)
  • [Windows] 剪映自动预合成v1.0
  • 低代码平台的测试挑战:测试从业者的新战场
  • Go项目发布到Go官方仓库完整指南
  • 智能测试用例生成技术探秘
  • 瞬行PRO_9.9.9_高级版
  • 初识结构体(新手友好)
  • 学习测评|基于springboot学习测评系统(源码+数据库+文档)
  • 免费AI写论文神器来袭!6款工具一键生成初稿,写论文效率提升300%! - 麟书学长
  • leetcode 困难题 745.Prefix and Suffix Search 前缀和后缀搜索
  • EtherCAT 逐帧报文解析:配置SM/FMMU
  • 列车售票|基于springboot 列车售票系统(源码+数据库+文档)