正点原子嵌入式linux驱动开发——Linux按键输入

在前几篇笔记之中都是使用的GPIO输出功能,还没有用过GPIO输入功能,本章就来学习一下如果在Linux下编写GPIO输入驱动程序。正点原子STM32MP1开发板上有三个按键,就使用这些按键来完成GPIO输入驱动程序,同时利用原子操作来对按键值进行保护

Linux下按键驱动原理

按键驱动和LED驱动原理上来讲基本都是一样的,都是操作GPIO,只不过一个是读取GPIO的高低电平,一个是从GPIO输出高低电平。本章实现按键输入,在驱动程序中使用一个整形变量来表示按键值 ,应用程序通过read函数来读取按键值,判断按键有没有按下。在这里,这个保存按键值的变量就是个共享资源,驱动程序要向其写入按键值,应用程序要读取

按键值。所以要对其进行保护,对于整形变量而言首选的就是原子操作,使用原子操作对变量进行赋值以及读取。Linux下的按键驱动原理很简单,接下来开始编写驱动。

注意,本章例程只是为了演示Linux下GPIO输入驱动的编写,实际中的按键驱动并不会采用本章中所讲解的方法,Linux下的input子系统专门用于输入设备!

硬件原理图分析

开发板上有三个按键:KEY0、KEY1和WK_UP,原理图如下图所示:

从上图可以看出,按键KEY0 、KEY1和WK_UP这三个按键分别连接到正点原子STM32MP1开发板的PG3、PH7和PA0这三个IO上。本节只用到KEY0这个按键,从上图可以看出,KEY0接了一个10K的上拉电阻,因此KEY0没有按下的时候PG3应该是
高电平,当KEY0按下以后PG3就是低电平

实验程序编写

修改设备树文件

在根节点"/"下创建KEY节点,命名为"key",节点内容如下:

c 复制代码
示例代码29.3.1.1 创建KEY节点 
1 key { 
2     compatible = "alientek,key"; 
3     status = "okay"; 
4     key-gpio = <&gpiog 3 GPIO_ACTIVE_LOW>; 
5 };

按键驱动程序编写

这里总体跟驱动LED的结构是类似的。

首先在设备结构体key_dev中,定义一个原子变量atomic_t的keyvalue。

在初始化函数keyio_init中,与之前的区别就是在最后调用gpio_direction_input设定gpio为输入模式。

key_open中,只要调用keyio_init初始化按键。

key_read中,通过gpio_get_value来读取IO口当前电平,如果读到0,也就是低电平,在维持高电平,也就是按键持续按下进入while循环等待释放,释放后设置atomic_set为KEY0VALUE(0XF0)。然后通过atomic_read读取原子变量。

在mykey_init中,调用(atomic_t)ATOMIC_INIT(0)初始化原子变量,然后通过atomic_set设置原子变量为INVAKEY(0X00)。

编写测试APP

这里就是在open这个字符设备之后,在while死循环中,read值,以此来判断按键是否按下。

运行测试

编译驱动程序和测试APP

编译驱动程序

把Makefile的obj-m改成key.o,然后"make"就可以了。

编译测试APP

可以通过如下命令编译keyApp.c:

|-------------------------------------------------|
| arm-none-linux-gnueabihf-gcc keyApp.c -o keyApp |

运行测试

将上一小节编译出来的key.ko和keyApp这两个文件拷贝到rootfs/lib/modules/5.4.31目录中,重启开发板,进入到目录lib/modules/5.4.31中,输入如下命令加载key.ko驱动模块:

|---------------------------------------------------|
| depmod //第一次加载驱动的时候需要运行此命令 modprobe key.ko //加载驱动 |

加载成功后通过如下命令来测试:

|-------------------|
| ./keyApp /dev/key |

按下开发板上的KEY0,keyApp就会过去并输出案件信息,如下图所示:

从上图可以看出,当按下KEY0以后就会打印出"KEY0 Press, value = 0XF0"表示按键按下。但是可能会发现,有时候按下一次KEY0但是会输出好几行"KEY0 Press, value = 0XF0",这是因为代码没有做按键消抖处理。

如果要卸载驱动可以使用如下命令:

|--------------|
| rmmod key.ko |

总结

按键输入和之前的LED驱动是很相似的,只是在驱动之中要换成gpio_direction_input表示这个GPIO是输入的。然后使用原子变量的时候是在key_read之中,通过原子变量控制读操作不会产生并发竞争。

相关推荐
芋头莎莎4 分钟前
STM32 51单片机设计半导体制冷片温控设计
stm32·嵌入式硬件·51单片机
VertexGeek6 分钟前
Rust学习(四):作用域、所有权和生命周期:
java·学习·rust
脱了格子衬衫14 分钟前
使用源码编译安装 Tomcat
linux·tomcat
陈yanyu18 分钟前
Linux - 弯路系列3:安装和编译libvirt-4.5.0及虚拟网卡virbr0(virbr0-nic)创建
linux·运维·服务器
小小码神Sundayx22 分钟前
三、模板与配置(下)
笔记·微信小程序
feng68_25 分钟前
Linux编辑/etc/fstab文件不当,不使用快照;进入救援模式
linux
spy47_30 分钟前
JavaEE 重要的API阅读
java·笔记·java-ee·api文档阅读
码农研究僧1 小时前
详细分析ip addr show 查看网络配置的命令
linux·tcp/ip·ip addr show
抱走江江1 小时前
SpringCloud框架学习(第二部分:Consul、LoadBalancer和openFeign)
学习·spring·spring cloud
码狂☆1 小时前
ubuntu连接orangepi-zero-2w桌面的几种方法
linux·ubuntu