“扩展板蜂鸣器驱动移植”的版本间的差异
(创建页面,内容为“==实验原理== ==实验平台== ==实验步骤==”) |
(→实验步骤) |
||
(未显示同一用户的2个中间版本) | |||
第1行: | 第1行: | ||
==实验原理== | ==实验原理== | ||
+ | 打开扩展板原理图对照扩展板可以看到扩展板有1个蜂鸣器SP1,如下图: | ||
+ | |||
+ | <center>[[Image:59-1-1.png|640px]]</center><br> | ||
+ | |||
+ | 由上图可见通过TIM4_CH1电平改变控制电路的通断从而驱动蜂鸣器。 | ||
+ | |||
+ | <center>[[Image:59-1-1.png|640px]]</center> | ||
+ | |||
+ | <center>扩展板与底板接口对照图</center> | ||
+ | 查看原理图可知TIM4_CH1对应PB6,查看芯片手册可知PB6可以作为PWM TIMER4的通道1使用,本文实现如何通过PWM驱动蜂鸣器: | ||
+ | {|align="center" border=1 | ||
+ | |- | ||
+ | ! 原理图网络编号 !! 对应管脚 !! 管脚功能 !! 管脚功能码 | ||
+ | |- | ||
+ | | TIM4_CH1 | ||
+ | | PB6 | ||
+ | | TIM4_CH1 | ||
+ | | AF2 | ||
+ | |} | ||
+ | |||
==实验平台== | ==实验平台== | ||
+ | 华清远见开发环境,FS-MP1A平台 | ||
+ | |||
==实验步骤== | ==实验步骤== | ||
+ | <ol> | ||
+ | <li>导入交叉编译工具链</li> | ||
+ | linux@ubuntu:$ source /opt/st/stm32mp1/3.1-openstlinux-5.4-dunfell-mp1-20-06-24/environment-setup-cortexa7t2hf-neon-vfpv4-ostl-linux-gnueabi | ||
+ | <li>内核配置</li> | ||
+ | |||
+ | 内核中为pwm驱动蜂鸣器提供了标准的驱动,只要在内核中配置对应选项即可,驱动路径为: | ||
+ | drivers/input/misc/pwm-beeper.c | ||
+ | 执行make menuconfig配置内核对应选项 | ||
+ | |||
+ | <pre> | ||
+ | linux@ubuntu:$ make menuconfig | ||
+ | Device Drivers ---> | ||
+ | Input device support ---> | ||
+ | [*] Miscellaneous devices ---> | ||
+ | <*> PWM beeper support | ||
+ | </pre> | ||
+ | |||
+ | <li>修改设备树</li> | ||
+ | 参考linux内核文档: | ||
+ | |||
+ | <pre> | ||
+ | Documentation/devicetree/bindings/pwm/pwm-stm32.txt | ||
+ | Documentation/devicetree/bindings/input/pwm-beeper.txt | ||
+ | </pre> | ||
+ | 修改设备树文件: | ||
+ | arch/arm/boot/dts/stm32mp157a-fsmp1a-extended.dts | ||
+ | |||
+ | 由于timers4在stm32mp151.dtsi中已完成定义,这里需要在原有基础添加与硬件对应的相关信息,在文件stm32mp157a-fsmp1a-extended.dts末尾集成并添加timers4相关内容: | ||
+ | <pre> | ||
+ | &timers4 { | ||
+ | /* spare dmas for other usage */ | ||
+ | /delete-property/dmas; | ||
+ | /delete-property/dma-names; | ||
+ | status = "okay"; | ||
+ | pwm4: pwm { | ||
+ | pinctrl-0 = <&pwm4_pins_c>; | ||
+ | pinctrl-1 = <&pwm4_sleep_pins_c>; | ||
+ | pinctrl-names = "default", "sleep"; | ||
+ | #pwm-cells = <2>; | ||
+ | status = "okay"; | ||
+ | }; | ||
+ | timer@3 { | ||
+ | status = "disabled"; | ||
+ | }; | ||
+ | }; | ||
+ | |||
+ | </pre> | ||
+ | |||
+ | 仿照设备树对于pwm管脚的配置添加timers4 pwm输出管脚配置,在stm32mp157a-fsmp1a-extended.dts末尾添加如下内容: | ||
+ | <pre> | ||
+ | &pinctrl { | ||
+ | pwm4_pins_c: pwm4-0 { | ||
+ | pins { | ||
+ | pinmux = <STM32_PINMUX('B', 6, AF2)>; /* TIM4_CH1 */ | ||
+ | bias-pull-down; | ||
+ | drive-push-pull; | ||
+ | slew-rate = <0>; | ||
+ | }; | ||
+ | }; | ||
+ | |||
+ | pwm4_sleep_pins_c: pwm4-sleep-0 { | ||
+ | pins { | ||
+ | pinmux = <STM32_PINMUX('B', 6, ANALOG)>; /* TIM4_CH1 */ | ||
+ | }; | ||
+ | }; | ||
+ | }; | ||
+ | </pre> | ||
+ | 在根节点中添加蜂鸣器对应节点,红色字体部分为添加内容: | ||
+ | |||
+ | <div style="background-color:#F8F8F8;border:1px solid #E5E5E5;font-family: monospace,Courier;"> | ||
+ | / { | ||
+ | : model = "HQYJ STM32MP157 FSMP1A Discovery Board"; | ||
+ | : compatible = "st,stm32mp157", "st,stm32mp157a-fsmp1a-extended"; | ||
+ | : …… | ||
+ | <font color="#F00"> | ||
+ | : beeper { | ||
+ | :: compatible = "pwm-beeper"; | ||
+ | :: pwms = <&pwm4 0 4000000>; | ||
+ | : }; | ||
+ | </font> | ||
+ | }; | ||
+ | </div> | ||
+ | |||
+ | <li>重新编译内核和设备树文件</li> | ||
+ | linux@ubuntu:$ make -j4 uImage dtbs LOADADDR=0xC2000040 | ||
+ | |||
+ | <li>更新系统内核和设备树</li> | ||
+ | |||
+ | <li>测试</li> | ||
+ | 系统启动后可以查看目录/dev/input | ||
+ | |||
+ | <pre> | ||
+ | root@fsmp1a:~# ls /dev/input/ | ||
+ | by-path event0 event1 | ||
+ | </pre> | ||
+ | 如果系统中有多个input设备,这里可能会有很对eventx,确定哪个event文件是我们的设备文件,可以通过查看/dev/input/by-path或查看dmesg系统启动信息确认:<br> | ||
+ | |||
+ | 查看by-path目录下文件 | ||
+ | |||
+ | <pre> | ||
+ | root@fsmp1a:# ls /dev/input/by-path/ -l | ||
+ | total 0 | ||
+ | lrwxrwxrwx 1 root root 9 May 30 08:43 platform-40013000.i2c-event -> ../event1 | ||
+ | lrwxrwxrwx 1 root root 9 May 30 08:43 platform-beeper-event -> ../event0 | ||
+ | </pre> | ||
+ | 由显示信息可以确认event0是蜂鸣器的设备文件<br> | ||
+ | 编写测试程序测试:<br> | ||
+ | 编写测试程序或参考:“光盘资料:华清远见-FS-MP1A开发资料\02-程序源码\06-资源扩展板测试程序\01-beeper_test”<br> | ||
+ | pwm-beeper.c<br> | ||
+ | |||
+ | <pre> | ||
+ | #include <stdint.h> | ||
+ | #include <string.h> | ||
+ | #include <fcntl.h> | ||
+ | #include <stdio.h> | ||
+ | #include <stdlib.h> | ||
+ | #include <linux/input.h> | ||
+ | #include <unistd.h> | ||
+ | |||
+ | #define ESC_KEY 0x1b | ||
+ | |||
+ | int main(int argc, char *argv[]) | ||
+ | { | ||
+ | int fd, version, ret; | ||
+ | struct input_event event; | ||
+ | struct timeval time; | ||
+ | |||
+ | if ((fd = open(argv[1], O_RDWR)) < 0) { | ||
+ | perror("beep test"); | ||
+ | return 1; | ||
+ | } | ||
+ | |||
+ | event.type = EV_SND; | ||
+ | event.code = SND_TONE; | ||
+ | event.value = 1000; | ||
+ | time.tv_sec = 1; | ||
+ | time.tv_usec = 0; | ||
+ | event.time = time; | ||
+ | |||
+ | while( 1 ) | ||
+ | { | ||
+ | int key; | ||
+ | |||
+ | ret = write(fd, &event, sizeof(struct input_event)); | ||
+ | |||
+ | printf( "Freq = %d\n", event.value); | ||
+ | |||
+ | key = getchar(); | ||
+ | |||
+ | switch(key) { | ||
+ | case '+': | ||
+ | if( event.value < 20000) | ||
+ | event.value += 10; | ||
+ | break; | ||
+ | |||
+ | case '-': | ||
+ | if( event.value > 11 ) | ||
+ | event.value -= 10 ; | ||
+ | break; | ||
+ | |||
+ | case ESC_KEY: | ||
+ | case EOF: | ||
+ | event.code = SND_BELL; | ||
+ | event.value = 0000; | ||
+ | ret = write(fd, &event, sizeof(struct input_event)); | ||
+ | close(fd); | ||
+ | return 1; | ||
+ | default: | ||
+ | break; | ||
+ | } | ||
+ | } | ||
+ | } | ||
+ | |||
+ | </pre> | ||
+ | 交叉编译测试程序并将编译好的测试程序下载到板子上,执行程序如下: | ||
+ | <pre> | ||
+ | root@fsmp1a:# ./beeper_test /dev/input/event0 | ||
+ | Freq = 1000 | ||
+ | </pre> | ||
+ | 通过输入‘+’或者‘-’调整频率,输入esc退出程序 | ||
+ | <pre> | ||
+ | root@fsmp1a: # ./beeper_test /dev/input/event0 | ||
+ | Freq = 1000 | ||
+ | + | ||
+ | Freq = 1010 | ||
+ | - | ||
+ | Freq = 1000 | ||
+ | ^[ | ||
+ | </pre> | ||
+ | 调整频率后听不同频率蜂鸣器发出声音的不同。 | ||
+ | </ol> |
2021年3月23日 (二) 13:38的最新版本
实验原理
打开扩展板原理图对照扩展板可以看到扩展板有1个蜂鸣器SP1,如下图:
由上图可见通过TIM4_CH1电平改变控制电路的通断从而驱动蜂鸣器。
查看原理图可知TIM4_CH1对应PB6,查看芯片手册可知PB6可以作为PWM TIMER4的通道1使用,本文实现如何通过PWM驱动蜂鸣器:
原理图网络编号 | 对应管脚 | 管脚功能 | 管脚功能码 |
---|---|---|---|
TIM4_CH1 | PB6 | TIM4_CH1 | AF2 |
实验平台
华清远见开发环境,FS-MP1A平台
实验步骤
- 导入交叉编译工具链
- 内核配置
- 修改设备树
- model = "HQYJ STM32MP157 FSMP1A Discovery Board";
- compatible = "st,stm32mp157", "st,stm32mp157a-fsmp1a-extended";
- ……
- beeper {
- compatible = "pwm-beeper";
- pwms = <&pwm4 0 4000000>;
- };
- 重新编译内核和设备树文件
- 更新系统内核和设备树
- 测试
linux@ubuntu:$ source /opt/st/stm32mp1/3.1-openstlinux-5.4-dunfell-mp1-20-06-24/environment-setup-cortexa7t2hf-neon-vfpv4-ostl-linux-gnueabi
内核中为pwm驱动蜂鸣器提供了标准的驱动,只要在内核中配置对应选项即可,驱动路径为:
drivers/input/misc/pwm-beeper.c
执行make menuconfig配置内核对应选项
linux@ubuntu:$ make menuconfig Device Drivers ---> Input device support ---> [*] Miscellaneous devices ---> <*> PWM beeper support
参考linux内核文档:
Documentation/devicetree/bindings/pwm/pwm-stm32.txt Documentation/devicetree/bindings/input/pwm-beeper.txt
修改设备树文件:
arch/arm/boot/dts/stm32mp157a-fsmp1a-extended.dts
由于timers4在stm32mp151.dtsi中已完成定义,这里需要在原有基础添加与硬件对应的相关信息,在文件stm32mp157a-fsmp1a-extended.dts末尾集成并添加timers4相关内容:
&timers4 { /* spare dmas for other usage */ /delete-property/dmas; /delete-property/dma-names; status = "okay"; pwm4: pwm { pinctrl-0 = <&pwm4_pins_c>; pinctrl-1 = <&pwm4_sleep_pins_c>; pinctrl-names = "default", "sleep"; #pwm-cells = <2>; status = "okay"; }; timer@3 { status = "disabled"; }; };
仿照设备树对于pwm管脚的配置添加timers4 pwm输出管脚配置,在stm32mp157a-fsmp1a-extended.dts末尾添加如下内容:
&pinctrl { pwm4_pins_c: pwm4-0 { pins { pinmux = <STM32_PINMUX('B', 6, AF2)>; /* TIM4_CH1 */ bias-pull-down; drive-push-pull; slew-rate = <0>; }; }; pwm4_sleep_pins_c: pwm4-sleep-0 { pins { pinmux = <STM32_PINMUX('B', 6, ANALOG)>; /* TIM4_CH1 */ }; }; };
在根节点中添加蜂鸣器对应节点,红色字体部分为添加内容:
/ {
};
linux@ubuntu:$ make -j4 uImage dtbs LOADADDR=0xC2000040
系统启动后可以查看目录/dev/input
root@fsmp1a:~# ls /dev/input/ by-path event0 event1
如果系统中有多个input设备,这里可能会有很对eventx,确定哪个event文件是我们的设备文件,可以通过查看/dev/input/by-path或查看dmesg系统启动信息确认:
查看by-path目录下文件
root@fsmp1a:# ls /dev/input/by-path/ -l total 0 lrwxrwxrwx 1 root root 9 May 30 08:43 platform-40013000.i2c-event -> ../event1 lrwxrwxrwx 1 root root 9 May 30 08:43 platform-beeper-event -> ../event0
由显示信息可以确认event0是蜂鸣器的设备文件
编写测试程序测试:
编写测试程序或参考:“光盘资料:华清远见-FS-MP1A开发资料\02-程序源码\06-资源扩展板测试程序\01-beeper_test”
pwm-beeper.c
#include <stdint.h> #include <string.h> #include <fcntl.h> #include <stdio.h> #include <stdlib.h> #include <linux/input.h> #include <unistd.h> #define ESC_KEY 0x1b int main(int argc, char *argv[]) { int fd, version, ret; struct input_event event; struct timeval time; if ((fd = open(argv[1], O_RDWR)) < 0) { perror("beep test"); return 1; } event.type = EV_SND; event.code = SND_TONE; event.value = 1000; time.tv_sec = 1; time.tv_usec = 0; event.time = time; while( 1 ) { int key; ret = write(fd, &event, sizeof(struct input_event)); printf( "Freq = %d\n", event.value); key = getchar(); switch(key) { case '+': if( event.value < 20000) event.value += 10; break; case '-': if( event.value > 11 ) event.value -= 10 ; break; case ESC_KEY: case EOF: event.code = SND_BELL; event.value = 0000; ret = write(fd, &event, sizeof(struct input_event)); close(fd); return 1; default: break; } } }
交叉编译测试程序并将编译好的测试程序下载到板子上,执行程序如下:
root@fsmp1a:# ./beeper_test /dev/input/event0 Freq = 1000
通过输入‘+’或者‘-’调整频率,输入esc退出程序
root@fsmp1a: # ./beeper_test /dev/input/event0 Freq = 1000 + Freq = 1010 - Freq = 1000 ^[
调整频率后听不同频率蜂鸣器发出声音的不同。