资料下载
coding无法使用浏览器打开,必须用git工具下载:
git clone https:
视频观看
百问网驱动大全
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 设备树讲解
属性:
2.2 100ASK_IMX6ULL
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
joystick {
compatible = "gpio-keys";
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
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,表示按键没有被按下。
4. QEMU上机实验
IMX6ULL、STM32MP157的出厂系统都已经配置的GPIO按键。 可以执行以下命令确认设备节点:
cat /proc/bus/input/devices
然后执行hexdump /dev/input/event? (?表示某个数值),并且操作按键来观察输出信息。
也可以在QEMU上做实验:原理图如下:
4.1 设置工具链
在Ubuntu中执行:
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
-> Device Drivers
-> Input device support
-> Generic input layer
-> Keyboards
<M> GPIO Buttons
4.3 编译驱动
book@100ask:~/100ask_imx6ull-qemu$ cd linux-4.9.88
book@100ask:~/100ask_imx6ull-qemu/linux-4.9.88$ make modules
成功的话,可以得到:
drivers/input/keyboard/gpio_keys.ko
复制到如下目录:
$ cp drivers/input/keyboard/gpio_keys.ko ~/nfs_rootfs/
4.4 启动QEMU
在Ubuntu中执行:
$ cd ubuntu-18.04_imx6ul_qemu_system
$ ./qemu-imx6ull-gui.sh
4.5 挂载NFS、实验
在QEMU中执行:
$ 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
#### 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界面操作:
|