前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >DRV_05_GPIO按键驱动分析与使用

DRV_05_GPIO按键驱动分析与使用

作者头像
韦东山
发布2021-12-08 11:01:55
1.9K0
发布2021-12-08 11:01:55
举报
文章被收录于专栏:韦东山嵌入式

资料下载

coding无法使用浏览器打开,必须用git工具下载:

代码语言:javascript
复制
git clone https://e.coding.net/weidongshan/linux/doc_and_source_for_drivers.git

视频观看

百问网驱动大全

GPIO按键驱动分析与使用

参考资料:

  • Linux 5.x内核
    • Documentation\devicetree\bindings\input\gpio-keys.txt
    • drivers\input\keyboard\gpio_keys.c
  • Linux 4.x内核
    • Documentation\devicetree\bindings\input\gpio-keys.txt
    • drivers\input\keyboard\gpio_keys.c
  • 设备树
    • IMX6ULL:Linux-4.9.88/arch/arm/boot/dts/100ask_imx6ull-14x14.dts
    • STM32MP157:Linux-5.4/arch/arm/boot/dts/stm32mp15xx-100ask.dtsi
    • QEMU:linux-4.9.88/arch/arm/boot/dts/100ask_imx6ull_qemu.dts

1. 驱动程序框架

2. 设备树示例

2.1 设备树讲解

属性:

  • 必备:compatible = "gpio-keys";
  • 可选:
    • autorepeat: 表示自动重复,按下按键不松开,驱动会自动重复上报按键值
  • 对于每一个GPIO按键,都是一个子节点,有这些属性:
    • gpios:使用哪个GPIO
    • interrupts:对应的中断
    • linux,code:对应的按键值
    • 注意gpiosinterrupts至少要保留一个,不能都省略
    • debounce-interval: 消除抖动的间隔,单位:ms,默认是5ms
2.2 100ASK_IMX6ULL
代码语言:javascript
复制
gpio-keys {
	compatible = "gpio-keys";
	pinctrl-names = "default";

	user1 {
		label = "User1 Button";
		gpios = <&gpio5 1 GPIO_ACTIVE_LOW>;
		gpio-key,wakeup;
		linux,code = <KEY_1>;
	};

	user2 {
		label = "User2 Button";
		gpios = <&gpio4 14 GPIO_ACTIVE_LOW>;
		gpio-key,wakeup;
		linux,code = <KEY_2>;
	};
};
2.3 100ASK_STM32MP157
代码语言:javascript
复制
joystick {
		compatible = "gpio-keys";
		#size-cells = <0>;
		button-0 {
				 label = "usr_button0";
				 linux,code = <KEY_A>;
				interrupt-parent = <&gpiog>;
				interrupts = <3 IRQ_TYPE_EDGE_RISING>;
		};
	   button-1 {
				 label = "usr_button1";
				 linux,code = <KEY_ENTER>;
				interrupt-parent = <&gpiog>;
				interrupts = <2 IRQ_TYPE_EDGE_RISING>;
		};

};
2.4 QEMU
代码语言:javascript
复制
gpio-keys@0 {
				compatible = "gpio-keys";
				pinctrl-names = "default";
				pinctrl-0 = <&pinctrl_gpio_keys>;
				status = "okay";

				Key0{
						label = "Key 0";
						gpios = <&gpio5 1 GPIO_ACTIVE_HIGH>;
						linux,code = <KEY_1>;
				};
};

gpio-keys@1 {
				compatible = "gpio-keys";
				pinctrl-names = "default";
				pinctrl-0 = <&pinctrl_gpio_key1>;
				status = "okay";

				Key0{
						label = "Key 1";
						gpios = <&gpio1 18 GPIO_ACTIVE_HIGH>;
						linux,code = <KEY_2>;
				};
};

3. gpio_keys.c驱动程序分析

3.1 套路
  • 根据设备树获得硬件信息:哪个GPIO、对于什么按键
  • 分配/设置/注册input_dev结构体
  • request_irq: 在中断处理函数中确定按键值、上报按键值
    • 有两种IRQ函数
    • gpio_keys_gpio_isr:设备树中的用gpios来描述用到的引脚
    • gpio_keys_irq_isr:设备树中的用interrupts来描述用到的引脚
3.2 gpio_keys_gpio_isr分析

理想状况是:按下、松开按键,各产生一次中断,也只产生一次中断。 但是对于机械开关,它的金属弹片会反复震动。GPIO电平会反复变化,最后才稳定。一般是几十毫秒才会稳定。 如果不处理抖动的话,用户只操作一次按键,会发生多次中断,驱动程序可能会上报多个数据。

怎么处理按键抖动?

  • 在按键中断程序中,可以循环判断几十亳秒,发现电平稳定之后再上报
  • 使用定时器

显然第1种方法太耗时,违背“中断要尽快处理”的原则,你的系统会很卡。

怎么使用定时器?看下图:

核心在于:在GPIO中断中并不立刻记录按键值,而是修改定时器超时时间,10ms后再处理。 如果10ms内又发生了GPIO中断,那就认为是抖动,这时再次修改超时时间为10ms。 只有10ms之内再无GPIO中断发生,那么定时器的函数才会被调用。 在定时器函数中上报按键值。

3.3 gpio_keys_irq_isr分析

有个变量key_pressed,用来表示当前按键状态:初始值是false,表示按键没有被按下。

  • 发生中断
    • 上报"按下的值":input_event(input, EV_KEY, button->code, 1); input_sync(input);
    • 如果不延迟(!bdata->release_delay)
    • 马上上报"松开的值":input_event(input, EV_KEY, button->code, 0); input_sync(input);
    • 如果延迟(bdata->release_delay)
    • 启动定时器,过若干毫秒再上报"松开的值"
  • 所以,使用gpio_keys_irq_isr时,一次中断就会导致上报2个事件:按下、松开
  • 缺点:无法准确判断一个按键确实已经被松开了

4. QEMU上机实验

IMX6ULL、STM32MP157的出厂系统都已经配置的GPIO按键。 可以执行以下命令确认设备节点:

代码语言:javascript
复制
cat /proc/bus/input/devices

然后执行hexdump /dev/input/event?(?表示某个数值),并且操作按键来观察输出信息。

也可以在QEMU上做实验:原理图如下:

4.1 设置工具链

在Ubuntu中执行:

代码语言:javascript
复制
export ARCH=arm
export CROSS_COMPILE=arm-linux-gnueabihf-
export PATH=$PATH:/home/book/100ask_imx6ull-qemu/ToolChain/gcc-linaro-6.2.1-2016.11-x86_64_arm-linux-gnueabihf/bin
4.2 配置内核

QEMU的内核里已经配置了GPIO按键的设备树,只需要编译出gpio_keys驱动程序即可。 配置内核:执行make menuconfig

代码语言:javascript
复制
-> Device Drivers
  -> Input device support
    -> Generic input layer   
      -> Keyboards
         <M>   GPIO Buttons      
4.3 编译驱动
代码语言:javascript
复制
book@100ask:~/100ask_imx6ull-qemu$ cd linux-4.9.88
book@100ask:~/100ask_imx6ull-qemu/linux-4.9.88$ make modules

成功的话,可以得到:

代码语言:javascript
复制
drivers/input/keyboard/gpio_keys.ko

复制到如下目录:

代码语言:javascript
复制
$ cp drivers/input/keyboard/gpio_keys.ko ~/nfs_rootfs/
4.4 启动QEMU

在Ubuntu中执行:

代码语言:javascript
复制
$ cd ubuntu-18.04_imx6ul_qemu_system
$ ./qemu-imx6ull-gui.sh
4.5 挂载NFS、实验

在QEMU中执行:

代码语言:javascript
复制
$ mount -t nfs -o nolock,vers=3 10.0.2.2:/home/book/nfs_rootfs /mnt
$ insmod /mnt/gpio_keys.ko
$ cat /proc/bus/input/devices   // 确认设备节点
$ hexdump /dev/input/event3

在QEMU的GUI界面操作:

x6ull-gui.sh

代码语言:javascript
复制
#### 4.5 挂载NFS、实验

在QEMU中执行:

```shell
$ mount -t nfs -o nolock,vers=3 10.0.2.2:/home/book/nfs_rootfs /mnt
$ insmod /mnt/gpio_keys.ko
$ cat /proc/bus/input/devices   // 确认设备节点
$ hexdump /dev/input/event3

在QEMU的GUI界面操作:

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2021/08/26 ,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 资料下载
  • 视频观看
  • GPIO按键驱动分析与使用
    • 1. 驱动程序框架
      • 2. 设备树示例
        • 2.1 设备树讲解
        • 2.2 100ASK_IMX6ULL
        • 2.3 100ASK_STM32MP157
        • 2.4 QEMU
      • 3. gpio_keys.c驱动程序分析
        • 3.1 套路
        • 3.2 gpio_keys_gpio_isr分析
        • 3.3 gpio_keys_irq_isr分析
      • 4. QEMU上机实验
        • 4.1 设置工具链
        • 4.2 配置内核
        • 4.3 编译驱动
        • 4.4 启动QEMU
        • 4.5 挂载NFS、实验
    领券
    问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档