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

【Linux驱动】Linux 按键驱动开发指南

Linux 按键驱动开发指南

1、按键驱动开发基础

1.1. 按键驱动类型

Linux下的按键驱动主要有两种实现方式:

  • 输入子系统驱动:最常用,通过input子系统上报按键事件

  • 字符设备驱动:较少用,需要自己实现文件操作接口

1.2. 输入子系统框架

推荐使用input子系统实现按键驱动,主要组件:

  • input_register_device() - 注册输入设备

  • input_report_key() - 上报按键事件

  • input_sync() - 同步事件

2、设备树配置

2.1. 基本按键节点配置

/ {gpio-keys {compatible = "gpio-keys";#address-cells = <1>;#size-cells = <0>;button@1 {label = "Power Button";linux,code = <KEY_POWER>;  // 按键编码,定义在include/uapi/linux/input-event-codes.hgpios = <&gpio0 5 GPIO_ACTIVE_LOW>; // 使用的GPIO,低电平有效debounce-interval = <20>;  // 消抖时间(ms)};};
};

【linux驱动】【设备树】按键设备树讲解

2.2. 关键属性说明

  • compatible: 必须包含"gpio-keys"
  • linux,code: 按键键值,如KEY_POWER、KEY_VOLUMEUP等
  • gpios: 指定GPIO控制器、引脚和有效电平
  • debounce-interval: 硬件消抖时间

3、驱动代码实现

3.1. 基本驱动框架

#include <linux/module.h>
#include <linux/gpio.h>
#include <linux/interrupt.h>
#include <linux/input.h>
#include <linux/platform_device.h>
#include <linux/of_gpio.h>struct gpio_key_data {struct gpio_desc *gpiod;int code;int irq;
};struct gpio_key_drvdata {struct input_dev *input;struct gpio_key_data *data;int n_buttons;
};static irqreturn_t gpio_key_isr(int irq, void *dev_id)
{struct gpio_key_data *key_data = dev_id;struct gpio_key_drvdata *ddata = container_of(key_data, struct gpio_key_drvdata, data);int state = gpiod_get_value(key_data->gpiod);input_report_key(ddata->input, key_data->code, !state);input_sync(ddata->input);return IRQ_HANDLED;
}static int gpio_key_probe(struct platform_device *pdev)
{// 驱动初始化代码// 1. 分配input设备// 2. 解析设备树获取按键信息// 3. 申请GPIO和中断// 4. 注册input设备// ...
}static const struct of_device_id gpio_key_of_match[] = {{ .compatible = "gpio-keys", },{ },
};
MODULE_DEVICE_TABLE(of, gpio_key_of_match);static struct platform_driver gpio_key_driver = {.probe = gpio_key_probe,.driver = {.name = "gpio-keys",.of_match_table = gpio_key_of_match,},
};
module_platform_driver(gpio_key_driver);

4、应用层使用按键驱动

4.1. 通过input子系统访问

按键驱动注册后会在/dev/input/下生成设备节点,如event0

4.2. 读取按键事件示例代码

#include <stdio.h>
#include <linux/input.h>
#include <fcntl.h>
#include <unistd.h>int main()
{int fd = open("/dev/input/event0", O_RDONLY);struct input_event ev;while(1) {read(fd, &ev, sizeof(ev));if(ev.type == EV_KEY && ev.value >= 0 && ev.value <= 2) {printf("Key %d %s\n", ev.code, ev.value ? "pressed" : "released");}}close(fd);return 0;
}

4.3. 使用工具测试

  • evtest: 通用输入设备测试工具
  • cat /proc/bus/input/devices: 查看已注册的输入设备

5、调试与问题排查

5.1 检查设备树是否正确加载:

ls /proc/device-tree/gpio-keys/

5.2 查看GPIO状态:

cat /sys/kernel/debug/gpio

5.3 检查中断是否注册:

cat /proc/interrupts

5.4 查看input设备信息:

cat /proc/bus/input/devices

相关文章:

  • 【回溯 剪支 状态压缩】# P10419 [蓝桥杯 2023 国 A] 01 游戏|普及+
  • 【第三篇】 SpringBoot项目中的属性配置
  • 动态规划(4)可视化理解:图形化思考
  • SparkSQL基本操作
  • ​在 ASP.NET 中,HTTP 处理程序(HttpHandler)是处理 HTTP 请求的核心组件​
  • 嵌入式通信协议(二)——IIC总线
  • Flink Table SQL
  • Flink并行度与分区机制深度解析
  • 【c库主要功能】
  • 深入理解Java中的Minor GC、Major GC和Full GC
  • OpenHarmony 5.0状态栏息屏状态下充电然后亮屏会出现电量跳变情况
  • AI工程 新技术追踪 探讨
  • 23种设计模式考试趋势分析之——适配器(Adapter)设计模式——求三连
  • Android 自定义悬浮拖动吸附按钮
  • 【赵渝强老师】Memcached的路由算法
  • Serverless技术深度整合:从冷启动优化到边缘场景落地
  • 锂电池SOC估计EKF仿真模型
  • 人工智能赋能产业升级:AI在智能制造、智慧城市等领域的应用实践
  • 原型链的详细解释及使用场景
  • C++23 新特性:使某些视图的多参数构造函数显式化(P2711R1)
  • 聘期三年已至:37岁香港青年叶家麟卸任三亚市旅游发展局局长
  • 交响4K修复版《神女》昨晚上演,观众听到了阮玲玉的声音
  • 上海将建设万兆小区、园区及工厂,为模型训练数据的传输提供硬件支持
  • 被围观的“英之园”,谁建了潮汕天价违建?
  • 美国失去最后的AAA主权评级,继标普、惠誉后再遭穆迪降级
  • 秦洪看盘|风格有所转变,热钱回流高弹性品种