通过 STM32F1 芯片的一个 IO口控制板载有源/无源蜂鸣器,实现蜂鸣器控制。
0 {- N. p1 B' L; d4 o" a1 H
蜂鸣器介绍 蜂鸣器是一种一体化结构的电子讯响器,采用直流电压供电,广泛应用于计算机、打印机、复印机、报警器、电子玩具、汽车电子设备、电话机、定时器等电子产品中作发声器件。蜂鸣器主要分为压电式蜂鸣器和电磁式蜂鸣器两种类型。 压电式蜂鸣器主要由多谐振荡器、压电蜂鸣片、阻抗匹配器及共鸣箱、外壳等组成。多谐振荡器由晶体管或集成电路构成,当接通电源后(1.5~15V 直流工作电压),多谐振荡器起振,输出 1.5~5kHZ 的音频信号,阻抗匹配器推动压电蜂鸣片发声。 电磁式蜂鸣器由振荡器、电磁线圈、磁铁、振动膜片及外壳等组成。接通电源后,振荡器产生的音频信号电流通过电磁线圈,使电磁线圈产生磁场,振动膜片在电磁线圈和磁铁的相互作用下,周期性地振动发声。 其实一句话就可概括它们之间的区别,想要压电式蜂鸣器发声,需提供一定频率的脉冲信号;想要电磁式蜂鸣器发声,只需提供电源即可。无源蜂鸣器,属于压电式蜂鸣器类型;有源蜂鸣器,属于电磁式蜂鸣器类型。这里说的有源,并不是指电源的意思,而是指蜂鸣器内部是否含有振荡电路,有源蜂鸣器内部自带振荡电路,只需提供电源即可发声,而无源蜂鸣器则需提供一定频率的脉冲信号才能发声,频率大小通常在 1.5-5KHz 之间。有源蜂鸣器实物如下图。
: o* L* w& Q9 ]
% u$ z: W/ l% g- a4 ?" h 如果给有源蜂鸣器加一个 1.5-5KHz 的脉冲信号,同样也会发声,而且改变这个频率,就可以调节蜂鸣器音调,产生各种不同音色、音调的声音。如果改变输出电平的高低电平占空比,则可以改变蜂鸣器的声音大小。 & K) h* f( Z4 P6 g0 Y
硬件设计 在前面我们已经对 STM32 的 GPIO 做了简单介绍,并且还使用了其中 IO 口直接控制开发板上的 LED。对于本章要实现蜂鸣器的控制,我们能否直接使用 STM32 的 IO 口驱动呢?根据 STM32F103 芯片数据手册可知, 单个 IO 口的最大输出电流是 25mA,而蜂鸣器的驱动电流是 30mA 左右,两者非常接近,有的朋友就想直接用 IO 口来驱动,但是有没有考虑到整个芯片的输出电流,整个芯片的输出电流最大也就 150mA,如果在驱动蜂鸣器上就耗掉了 30mA,那么 STM32 其他的 IO 口及外设电流就非常拮据了。所以我们不会直接使用 IO 口驱动蜂鸣器,而是通过三极管把电流放大后再驱动蜂鸣器,这样 STM32 的 IO 口只需要提供不到1mA 的电流就可控制蜂鸣器。所以我们也经常说到 STM32 芯片是用来做控制的,而不是驱动。开发板上的无源蜂鸣器模块电路原理图如下所示。 # h, K% `6 n! U3 c) `& E: r D
6 C" o% {! Y) p 开发板上的有源蜂鸣器模块电路原理图如下所示。 + L5 ?# n Q) L# d1 t# v G$ \
! O8 H1 O3 f( B U T
从这两个电路图可以看到, 无源蜂鸣器的控制需要给其一定频率的脉冲才能发声,仅给一个高电平或者低电平是不能发声的,因此蜂鸣器模块电路直接将STM32引脚接在NPN三极管的基极上, 然后三极管的发射极连接蜂鸣器,无需考虑外界对 PB5 脚产生的高低电平会对蜂鸣器发声干扰。如果使用的是有源蜂鸣器,它只要有电源就会发声,因此就必须考虑外界对 PB5 引脚电平的干扰问题。通过电阻 R06 和 PNP 三极管 TP6 进行电流放大,从而驱动蜂鸣器。电阻 R66 是一个上拉电阻, 用来防止蜂鸣器误发声。 当 PB5 引脚输出低电平时, PNP三极管导通,蜂鸣器发声;当 PB5 引脚输出高电平时,PNP 三极管截止,蜂鸣器停止发声。
7 ]" S2 E7 g2 [! u. B
软件设计 蜂鸣器初始化函数 打开工程中 beep.c 文件,里面代码如下: - #include "beep.h"/ J& k4 T, H- B9 y# }8 V
( a- b/ B& U" d2 c2 V% f- /****************************************************************
; [+ e5 W* @: m% D3 C - 4 v! b1 o$ q% V( z/ c: ]! ~9 l
- ***************
1 q* b% y' c$ M* H- a/ Q- l, h
: g% A8 b: q- ~( R6 z- * 函 数 名 : BEEP_Init+ U8 p8 W9 B. |, T' n. h! y
6 x) k! o$ v. z0 k- * 函数功能 : 蜂鸣器初始化+ e% o+ P5 ^/ E% ~8 g4 U# N* M7 \
- 9 l4 s, T) U3 Y: j6 n
- * 输 入 : 无
7 p0 n: I* [& {, [( n. u - 8 q- |2 b; M% }% e. x# G
- * 输 出 : 无
; Y A8 u; n3 q! G
3 Z0 O- J0 A) v1 q8 H. V' Z- ****************************************************************/
0 c% D; m# x/ ` - ) W7 Q# Z5 [" P. M$ q: Q: _
- void BEEP_Init() //端口初始化! c& V6 P6 x6 @% A
; U2 Q }2 Z6 j* g2 G0 X- {
+ p* O* Z* [; u* \- f, F. T- J
! S! P+ d [* b" Y; U s- Z( `! S, D- GPIO_InitTypeDef GPIO_InitStructure; //声明一个结构体变量,用来初始化 GPIO7 u/ D2 n# V N6 q5 ^
- ) o/ c% A! J% t0 [- `6 L ^
- RCC_APB2PeriphClockCmd(BEEP_PORT_RCC,ENABLE); /* 开启 GPIO 时钟 */
S2 H% Z- u7 O- C8 y
5 i4 |+ Q! Y& D: W: \& A& E; B- /* 配置 GPIO 的模式和 IO 口 */" i! F' v! q7 G& S+ \2 o
- 6 ?9 s0 |' Y! e3 J
- GPIO_InitStructure.GPIO_Pin=BEEP_PIN; //选择你要设置的 IO口
C% q0 B5 I4 A7 k- s" \9 z" o - ) W3 X" X3 r! l1 Y5 K
- GPIO_InitStructure.GPIO_Mode=GPIO_Mode_Out_PP; //设置推挽输出模式
: A+ y3 r1 ~ O; R8 J- x- `$ K2 R7 R
; ]1 V: M. i1 {" n9 B- GPIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz; //设置传输速率
3 J! w. s4 N+ v: R& k
" Y, |8 J k: ?0 A- GPIO_Init(BEEP_PORT,&GPIO_InitStructure); /* 初始化 GPIO*/3 R) v/ i0 `. n, J" s- u
- 4 g1 Z' r* }5 _* L: G
- }
复制代码
- ~7 m# t2 ]' w( l- k BEEP_Init()函数用来初始化蜂鸣器的端口及时钟,在函数内我们看到有几个参数不是库函数内的,比如 BEEP_PIN、BEEP_PORT、BEEP_PORT_RCC,这种情况一般是我们自己定义的宏,通常放在对应的头文件内,我们打开 beep.h,可以看到如下代码: - #ifndef _beep_H
# z- f" N5 o; s' g - # y3 ]+ O+ U* W, q* ^* b# o1 G
- #define _beep_H
# W- w* m+ M4 I! [' q& f. i: r+ @$ O
/ N( M* N% G ]1 k6 v- #include "system.h"
& q. L$ _9 _* ] |- g
! K. x- B' p9 T- /* 蜂鸣器时钟端口、引脚定义 */
0 A2 f! ?5 p% S
9 ^& f0 G; i! z- [: _8 G" f! x- #define BEEP_PORT GPIOB6 p9 h5 c7 [* V# M1 n# v4 \
- ; S' B2 E3 v* H( j* r
- #define BEEP_PIN GPIO_Pin_5
; ` `$ g$ H' L) K0 F* Q
3 S& B8 i, J" m$ z# G1 l4 { u- #define BEEP_PORT_RCC RCC_APB2Periph_GPIOB
( C! j! }- L$ i - # o6 ^/ o6 E9 u/ `: g
- #define beep PBout(5)
5 B7 J: ^ h5 |
! ~8 w" G9 h1 i5 h5 C- d- void BEEP_Init(void);& |! E: v$ ^7 u4 D
( `; B3 i. l0 n- #endif
复制代码
) w0 X. r; t% ~6 ? 里面就将蜂鸣器的GPIO 端口及管脚进行了宏定义,这样做是方便大家移植程序,只需要对这个宏修改就能实现蜂鸣器的初始化修改。
! W( G. Y" |! u$ r+ m
主函数 我们打开工程中 main.c 文件,里面代码如下: - /****************************************************************
0 c; T1 @: B6 J - : K! D- W- ?; ~( r1 T# s
- * 函 数 名 : main
$ W( }: m' Y& w4 @$ ~. b. W
% m+ }4 X" R9 W$ \- * 函数功能 : 主函数
7 j& n- t( A: V: J, e. Y - - z* f7 i! _; E
- * 输 入 : 无& ]7 k6 ^1 T7 z1 M* U1 w
- 1 z. L# k! X' i% `* v
- * 输 出 : 无
* z9 [8 M$ r5 U( ^6 f% i4 y8 o
# Z: E; o, s$ p! t- ****************************************************************/
$ c0 ^; u- ^2 Z3 y4 g7 o* y6 ? - ' w# a9 [$ D" N' e q
- int main()
( a5 F4 K: G7 G! X, F1 S - 2 d z1 V# n+ o; a) U8 X
- {& }# n* Q% E3 {& V$ u
q. ?3 W) G; ^/ z- u16 i=0;
' l! F" t4 y5 r& S* m. U, z
/ T( I5 u! `0 m& V- SysTick_Init(72);
1 }6 L$ p3 S' t, [; K" g4 F
. {0 g5 w, g0 Q- LED_Init();
! @0 [, l7 I+ [ y
8 b% }; w& W& Z g9 A- BEEP_Init();. ^" t# d- V5 j- p
! p/ Z/ D2 ]6 c; Y, a# w5 ~+ \# |- while(1)
[7 r4 [1 [0 b* F W
" x3 F3 A& R/ ?+ R4 k+ F) Y- {2 B% E* a# Z9 e% y
& P9 k3 F; ]$ Z( u" w% h- i++;0 h. r: r" Y8 N" R
- ) j" P+ o7 T( N; F4 r
- if(i%10==0)4 }: _5 }% n; w4 c% [ f p4 ?
- * @. I) {; e7 Z' f* y+ o; k
- {
- {7 J# e9 w- P - 5 _9 V: M: ^+ Z
- beep=!beep;% a+ V/ U5 @0 \+ O
, N6 w9 y- P I( h/ M( F' M- }
* s; K5 N, L6 w/ M1 T& H
' V0 L9 W3 ]8 |1 ?8 i- if(i%20000==0)
9 R* n" e# W, Z# V; Z4 V - 9 o- W) O; [# {0 ~. a+ S
- {
) Y+ [' b; C( {& L+ h
1 [& j) q" K5 X2 Y8 C+ c6 M- led1=!led1;4 c6 l2 }8 ?, {! N( C5 v- x
/ K- R6 i$ u: U6 ^- }0 L% Q- ~& b% S' ?4 N) k4 ^
9 c/ h- H* R; f- delay_us(10); f2 v4 K/ l* o
4 i* C9 }. o: ]+ {2 W; i ^5 r- }
, ~# T9 i# ^9 v! o* m! S - " l. L1 A5 j/ D9 _; |
- }
复制代码 X2 e. @ r3 B {" I
主函数实现的功能比较简单,首先将使用到的外设硬件进行初始化,然后进入 while 循环,间隔 100us 对蜂鸣器管脚电平翻转,这样就产生了一个频率为5KHz 的脉冲,因此无源蜂鸣器就会发声,当然有源蜂鸣器的控制可以更简单,只需要给它一个低电平即可,为了实现 2 块板子程序的兼容,这里就统一采用此种方式控制蜂鸣器。间隔 200ms 对 led1 管脚电平翻转,因为使用到了 delay_us延时函数,所以在 main 函数开始处就需要调用 SysTick_Init(72)初始化,这个在我们后面所有程序都会使用,后面就不重复。 将工程程序编译下载到开发板内,可以看到 D1 指示灯间隔 200ms 闪烁,蜂鸣器发声。 # W) Z `2 ~7 K6 T$ l. w4 o% }
|