扩展板蜂鸣器驱动移植
实验原理
打开扩展板原理图对照扩展板可以看到扩展板有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 ^[
调整频率后听不同频率蜂鸣器发出声音的不同。