- /* STM32 嵌入式学习入门(5)——PWM的实现
) \1 a* @2 x' ^
9 t: x( [/ V' g3 ^- 上一篇博文介绍了定时器和PWM的基本的原理,本篇博文从代码层面来介绍PWM的具体实现。同样,还是以博主所用的开发板——正点原子开发板STM32F103ZET6为例。
% c0 ]4 r! c" z) E
* _7 s. A: C" ^0 E) q- 一、基于STM32的PWM输出配置步骤(初始化操作):. h6 O4 B5 x5 |% ?; g+ B
- 1 ^, Y+ s4 `7 d5 h$ m
- 1. 操作步骤(基于STM32固件库、使用定时器3的PWM功能):
& D1 U6 Q& H: L - (1)使能相关时钟(定时器3和相关IO口时钟。):
7 R9 N0 f6 Q# W2 `$ n9 F0 Y - //要使用什么外设就要先使能相关外设所挂载的时钟,这些内容在最开始GPIO那块就有提到STM32的GPIO介绍
$ j Z) a% H; Y$ n# S - ①使能定时器3时钟:RCC_APB1PeriphClockCmd();2 v: }8 t3 j, h* i7 P7 M! V
- ②使能GPIOB时钟:RCC_APB2PeriphClockCmd();
7 |, h: B4 c/ e7 z
6 H& I9 G! ^' D/ Q- r; |4 k- (2)初始化IO口为复用功能输出。函数:GPIO_Init();
# d* @5 N0 ^8 @: | Q: F" Y* E, ? - //同样,IO口初始化的操作在STM32的GPIO介绍那篇文章里也说到了。; I' ~5 I+ |! @" \$ j
- GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP; 9 t( G* V* P' i; u7 @& Z* q
- //这里我们是要把引脚用作定时器的PWM输出引脚,因此要重映射配置。所以需要开启AFIO时钟。同时设置重映射。
& U6 g0 h: y9 b0 D6 y - RCC_APB2PeriphClockCmd(RCC_APB2Periph_AFIO,ENABLE);3 F' C0 d/ S( ^/ ^& A/ o
- GPIO_PinRemapConfig(GPIO_PartialRemap_TIM3, ENABLE); @7 N7 ?/ i4 H7 |0 t: d3 I
- 5 M+ N2 a) I& i1 M5 k; `
- (3)初始化定时器:ARR,PSC等:TIM_TimeBaseInit();ARR寄存器在上一篇文章讲原理时候已经说到了。PWM原理
, Y5 \ B2 m' v9 ]% B/ h
* ] q3 T: W$ ?" {) Q1 x, t- Y- (4)初始化输出比较参数:TIM_OC2Init();
5 q( f/ Q5 r2 Q$ \5 r8 T1 p: w5 M - (5)使能预装载寄存器: TIM_OC2PreloadConfig(TIM3, TIM_OCPreload_Enable); \1 u2 f. u( v, w+ s3 v( q- w
- (6)使能定时器。TIM_Cmd();! n+ [! [* a$ r
- (7)不断改变比较值CCRx,达到不同的占空比效果:TIM_SetCompare2();
% q' N @2 ~7 f4 d- {5 Y - - k6 ~( U: @0 S& q8 G
- 2.初始化源代码:9 F' t5 X! Z( Z a9 E- S
- (1)以STM32F103ZET6为芯片的开发板的PWM初始化,这里只是初始化一个通道用作PWM输出 */
5 E9 h' x; E6 X, o. x' F9 A, ` - //TIM3 PWM部分初始化
S% z8 S8 T6 ?% g - //PWM输出初始化6 U0 I0 f6 l5 h7 Y5 l& }
- //arr:自动重装值7 c- f$ }/ `$ y- @$ u2 U& |( S
- //psc:时钟预分频数
' h1 i: |' z) n- J; S) m - void TIM3_PWM_Init(u16 arr,u16 psc)//STM32F103ZET6
. ~9 ~3 h( k$ h* j - { ! h" i9 w" ^9 h% m) z0 P# A6 w
- GPIO_InitTypeDef GPIO_InitStructure;
8 o) ~' T6 _3 @; C- I7 j, B( I9 A - TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure;
, E; U" F) a' m Z; X - TIM_OCInitTypeDef TIM_OCInitStructure;2 G3 a& _/ U9 i4 d) `
-
+ m% k. A) r6 N& s( g) h) Z( \4 M - RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3, ENABLE); //使能定时器3时钟& R' c2 O, X3 p$ D4 K7 q% U9 s* b
- RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB|RCC_APB2Periph_AFIO, ENABLE); //使能GPIO外设和AFIO复用功能模块时钟
' F. S; p$ k" w8 M% F -
m2 ?9 P! X/ Y; F% S- s9 u - GPIO_PinRemapConfig(GPIO_PartialRemap_TIM3, ENABLE); //Timer3部分重映射 TIM3_CH2->PB5
# W5 I6 s" ?. X# | - : x, o/ z. p8 ~7 c" d
- //设置该引脚为复用输出功能,输出TIM3 CH2的PWM脉冲波形 GPIOB.50 H& K' \2 a, a2 I* V% Z
- GPIO_InitStructure.GPIO_Pin = GPIO_Pin_5; //TIM_CH2
! @+ K: L+ `( x, E/ `$ n - GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP; //复用推挽输出
# a4 E! S& w* t" {" U - GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;( m, \0 }1 Z; |$ ^/ {7 V
- GPIO_Init(GPIOB, &GPIO_InitStructure);//初始化GPIO8 N7 k; a2 D5 S1 h
-
/ r0 j/ ~! X" U* S0 p6 ^ h - //初始化TIM3' {- p# B) _# g: M; m. ^! N/ n! z
- TIM_TimeBaseStructure.TIM_Period = arr; //设置在下一个更新事件装入活动的自动重装载寄存器周期的值
( S) p* I* }$ s4 R! \# W - TIM_TimeBaseStructure.TIM_Prescaler =psc; //设置用来作为TIMx时钟频率除数的预分频值
$ u- [5 i) V0 n5 P9 A2 ? - TIM_TimeBaseStructure.TIM_ClockDivision = 0; //设置时钟分割:TDTS = Tck_tim
, f8 |. G" N" h4 }8 l' g( `) V* E - TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up; //TIM向上计数模式
. ~ S( X- U* O& c - TIM_TimeBaseInit(TIM3, &TIM_TimeBaseStructure); //根据TIM_TimeBaseInitStruct中指定的参数初始化TIMx的时间基数单位9 N4 R+ d+ i8 z; H8 c' ~, d
-
4 Q3 s( j" w- f, h f - //初始化TIM3 CH2 PWM模式
+ U' x& l; C) o2 t4 ?4 g7 X - TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM2; //选择定时器模式:TIM脉冲宽度调制模式2
; v' o2 L" o+ e8 | - TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable; //比较输出使能
3 b4 o2 C3 A: j# W7 O - TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_High; //输出极性:TIM输出比较极性高
, G# g9 A3 P2 f5 i7 }& v! Z" g - TIM_OC2Init(TIM3, &TIM_OCInitStructure); //根据T指定的参数初始化外设TIM3 OC2
' R7 O$ |3 B6 h2 |0 Z$ g - 3 ]+ v) M& H# p& M* _
- TIM_OC2PreloadConfig(TIM3, TIM_OCPreload_Enable); //使能TIM3在CCR2上的预装载寄存器: K$ x- I( W8 t7 d6 z: W
- 4 {' C6 F# Y2 c6 o5 s$ C( Z
- TIM_Cmd(TIM3, ENABLE); //使能TIM3# k- G9 n/ Q& v' [6 q9 a7 ]# `% A
- }+ Y9 ?$ p8 l# V7 A- ^2 Y
- /* 9 B1 K+ j: ~% l6 M! L2 }
- (2)下面的代码是博主做嵌入式循迹小车的项目中的PWM初始化,用了TIM1的两个通道(通道1和通道4)去分别控制两个驱动轮的转速,从而实现让小车转向的功能。和上面的初始化代码的不同主要在于小车项目中用了两个通道,同时项目使用的开发板的芯片是STM32F103RCT6。在这里贴出源代码,做一比较。 */4 p3 {7 @0 ?+ H) E
- //PWM输出初始化' D% `% w7 z8 `( v3 u/ {9 W
- //arr:自动重装值8 C; i/ U8 g2 q) p7 a H8 x
- //psc:时钟预分频数3 h9 H, S7 v$ N
- void TIM1_PWM_Init(u16 arr,u16 psc)//STM32F103RCT6
h' Z6 H: a' H" y; _2 z - {
! v/ o9 t; Y& T2 ]- l - GPIO_InitTypeDef GPIO_InitStructure;
8 ~6 s- Q8 h; R, V - TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure;. ?1 F: c9 J$ ^0 {& {9 I
- TIM_OCInitTypeDef TIM_OCInitStructure;
9 g) S* g0 q! Z9 e0 s3 ?/ X1 w6 U -
+ `- }/ Z0 N! J* i - RCC_APB2PeriphClockCmd(RCC_APB2Periph_TIM1,ENABLE);4 d1 F6 n7 E1 ?# Z& b0 l
-
O5 B( l- s4 q B - RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE); //使能GPIOA外设时钟使能
! H3 s/ {# {# C: x* K' Z! H -
+ @; `# K) ?% v7 M3 W" M - //设置该引脚为复用输出功能,输出TIM1 CH1和CH4的PWM脉冲波形+ y* k6 y# q* |: f5 z( Z- G
- GPIO_InitStructure.GPIO_Pin = GPIO_Pin_8|GPIO_Pin_11;//TIM_CH1
/ L2 O1 A8 f4 t& J - GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP; //复用推挽输出2 ~; u4 L# V. U0 e( ?. d
- GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
$ t( L$ C; l7 p+ j4 G6 j - GPIO_Init(GPIOA,&GPIO_InitStructure);
7 Q9 ~6 {& b5 e; ~ - ; S; n% U9 J: z" K, Q$ j
- TIM_TimeBaseStructure.TIM_Period = arr; //设置在下一个更新事件装入活动的自动重装载寄存器周期的值 80K
- _7 {$ [0 Z" H0 t/ ` - TIM_TimeBaseStructure.TIM_Prescaler = psc; //设置用来作为TIMx时钟频率除数的预分频值 不分频+ c4 A4 L$ a$ e0 \5 B/ a
- TIM_TimeBaseStructure.TIM_ClockDivision = 0; //设置时钟分割:TDTS = Tck_tim
- F _8 e! y8 \# a - TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up; //TIM向上计数模式 6 p7 C0 i: Z4 Z1 B" n% c
- TIM_TimeBaseInit(TIM1,&TIM_TimeBaseStructure); //根据TIM_TimeBaseInitStruct中指定的参数初始化TIMx的时间基数单位
3 V. h# F8 I0 Z - I0 \0 t6 x* ~; ~6 @$ {
- TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM2; //选择定时器模式:TIM脉冲宽度调制模式2
+ b) Y9 J2 M3 t2 Q+ |$ O) \. c$ [ - TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable; //比较输出使能
8 K9 I* L$ P& q" p: P9 \/ d$ K5 F - TIM_OCInitStructure.TIM_Pulse = 0; //设置待装入捕获比较寄存器的脉冲值9 ]2 K, e% j5 d& F6 _% ~
- TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_High; //输出极性:TIM输出比较极性高9 `8 U! _8 v, Y! r; s6 W- X3 K* [
- ! ]6 C: h2 G9 ^% J C
- TIM_OC1Init(TIM1, &TIM_OCInitStructure); //根据TIM_OC1InitStruct中指定的参数初始化外设TIM1的通道1
; Z& h/ b8 ~* X7 H( q! e$ ] - TIM_OC4Init(TIM1, &TIM_OCInitStructure); //根据TIM_OC2InitStruct中指定的参数初始化外设TIM1的通道4; ~' e0 b% H* A/ P9 Q) @5 a
-
# Y7 G0 g& \! m9 ]) f% K - TIM_CtrlPWMOutputs(TIM1,ENABLE); //MOE 主输出使能
- d* x- s! S! F7 P, c - + @9 x; x% j7 B" }- ^" i
- TIM_OC1PreloadConfig(TIM1,TIM_OCPreload_Enable); //CH1预装载使能 通道1! p! @4 ]5 C4 s, e- B+ I& d- ^
- TIM_OC4PreloadConfig(TIM1,TIM_OCPreload_Enable); //CH4预装载使能 通道4( d* x( @ N& z3 n' z" J$ I
-
+ q0 m. m% @0 X - TIM_ARRPreloadConfig(TIM1,ENABLE); //使能TIMx在ARR上的预装载寄存器(x==1)+ r1 m: T6 ^6 L
- 7 z* \$ o3 S% k( s$ B2 m
- TIM_Cmd(TIM1,ENABLE); //使能TIM1+ ]( P3 p0 z$ f. s9 ]% T( l
-
+ Q, S; A/ \( y. y* w- K - }$ h- @8 ~! l: h+ o
" V( O' r# @: R- q. T- /* 初始化函数在主函数的开头处调用,调用函数的两个参数根据自己需要去设定。) h1 T4 u& b9 v1 ]9 n- {& L9 N
- , c9 R5 I, L) N8 s, t1 b _! e$ x
" ]3 ]* M- h |$ `- f w- 二、PWM的操作:- |$ B# z+ n$ }7 l8 T+ {
- H6 q; I( _5 S) S4 D+ f1 `, O& Z
- 上面的初始化结束后就可以利用相关引脚输出PWM波了。这个输出过程简单说就是改变比较值CCRx的操作,就是往CCR寄存器里写值。这一点上一篇文章里有分析到。下面先介绍一下往CCRx寄存器里写值操作的函数(来自官方库函数):TIM_SetCompare1(); */! S; f0 ^1 d- X4 g
- /**# D( P: ]. U5 {' q. \- d9 H
- * @brief Sets the TIMx Capture Compare1 Register value4 n+ I9 y4 F2 e( F; I9 B+ ?/ R
- * @param TIMx: where x can be 1 to 17 except 6 and 7 to select the TIM peripheral.
% `6 l. ?' }, D( {& {) j% f. Y - * @param Compare1: specifies the Capture Compare1 register new value.8 w+ `: R& Z; G& }
- * @retval None# f) g' b% O6 g! T
- */# P7 M, }& ^- _2 K
- void TIM_SetCompare1(TIM_TypeDef* TIMx, uint16_t Compare1)
0 g8 @: P' N! h6 q( W; Z2 o - {
9 _# Q4 g1 Q% ]( b7 r9 K5 E - /* Check the parameters */
" w, X6 n& m$ W6 k - assert_param(IS_TIM_LIST8_PERIPH(TIMx));//有效性判断
. M9 A8 v3 Q5 ^ - /* Set the Capture Compare1 Register value */
4 d: P6 |% n3 l6 Q9 H3 G$ A2 j/ Y8 z - TIMx->CCR1 = Compare1;
7 u* ^- @3 G3 T, F; T8 H' l$ v - } n" _# ]! v2 u6 x: e/ a
- /* 说明:和这个函数类似的一共有四个:TIM_SetCompare1(); TIM_SetCompare2(); TIM_SetCompare3(); TIM_SetCompare4();其实现框架都是一样的,四个函数分别对应四个通道,使用哪个通道调用哪个函数,要对应起来。0 ]6 \6 A6 a. H* x1 e- X" _# w
- i: y6 E& Q5 v/ U* g. E% m- 7 b' S; J' K4 D' U# X2 j+ S
- 下面先介绍还是以自己嵌入式循迹小车的项目为例,比如小车直行时就是在执行下面这个函数。 *// ~$ ` M' v. P1 y, K
- u8 Case_F_SIGN(void)//直行2 G6 x9 [' c; `8 ?9 p- L
- {
" R. D$ e" K4 m9 F( W - while(1)
6 T$ y% g3 z4 Q4 P, |& T - {
/ [9 K' e% X0 N6 m, o& X; h! T" } - delay_ms(10);
- x$ y I% I. C; v - pwmvalR=413;* c+ F7 h5 J3 X* j! v4 v' ^
- pwmvalL=413;) x9 F$ h. T1 T0 |
- TIM_SetCompare1(TIM1,pwmvalR);
# m& D( K& d. _0 a$ \ - TIM_SetCompare4(TIM1,pwmvalL);
# B/ M4 d) D) f F. X" t - t=SensorScan();# Y6 p B: H) j
- if(t!=F_SIGN)! w8 g" C1 C: _: _
- return t;
" R% f6 j* n; ^% x, S/ ^ - }
: M! X0 C, H. u! \, I/ D% v. x - }; [$ O9 K# D" t' G$ }
- /* - s' U8 n7 a T8 r, J8 ?
- pwmvalR和pwmvalL是两个全局变量,它们的值是根据项目需要和小车硬件情况实际测出来的,然后调用两个函数,写入相关寄存器,,下面是调用SensorScan();看小车是不是还沿着轨迹在行驶,如果不是,那么跳出while(1)循环,否则,返回主调函数当前小车的状态。这段代码现在看写得不是特别好了,但是说明这个函数的用法还是没问题的。7 |" |. G; n* }' h# ^* n, w# g
- 4 J6 q4 @( Q( A' r( F6 [
4 t5 _% W9 ^6 M. z1 R) [- 高级控制定时器(TIM1 和TIM8)
$ v y% a0 c4 `/ w- X - $ M' ~! P* l, a4 b# L I( ?
- TIM1和TIM8定时器的功能包括: / l* v& D$ v& T! E' }/ q
- ● 16位向上、向下、向上/ 下自动装载计数器
& r; D+ L" ?& |- K# [ - ● 16位可编程( 可以实时修改)预分频器,计数器时钟频率的分频系数为1~65535 之间的任意数值
6 l1 v- I# q1 Z6 ]/ ~ - ● 多达4个独立通道: 6 l3 b& a# s* q" [. ?# U5 ~
- ─ 输入捕获 " V; F* E- ^* K1 r. Q9 F
- ─ 输出比较
# D0 r' @8 L {# l - ─ PWM生成(边缘或中间对齐模式)
) _. o9 G1 M; r; k: f8 H) v - ─ 单脉冲模式输出 , Y8 L8 t, s3 D/ q" @5 T
- ● 死区时间可编程的互补输出 ' J: s+ j% m8 [4 o p1 P
- ● 使用外部信号控制定时器和定时器互联的同步电路 ( V2 e1 e+ E. e* m: u7 M4 q9 `' g; W
- ● 允许在指定数目的计数器周期之后更新定时器寄存器的重复计数器
# l( K4 f( b0 T - ● 刹车输入信号可以将定时器输出信号置于复位状态或者一个已知状态 $ P5 S" I5 ` c
- ● 如下事件发生时产生中断/DMA :
% U% i' c- {& k; S9 @ - ─ 更新:计数器向上溢出/ 向下溢出,计数器初始化(通过软件或者内部/ 外部触发)
$ x. w+ c8 }0 l3 _2 X, K# x& S2 _ - ─ 触发事件(计数器启动、停止、初始化或者由内部/ 外部触发计数) 7 H q6 a' W8 h& j P2 L
- ─ 输入捕获 + O+ R1 M/ n8 K) N( M* g# c3 |0 K
- ─ 输出比较
/ Z# K1 |8 u6 W+ c/ a. a5 | - ─ 刹车信号输入
D. i" h! b; P; J, C! G8 `; E- Q - ● 支持针对定位的增量(正交)编码器和霍尔传感器电路 9 E) o$ a4 h! z( ^9 r0 W1 w
- ● 触发输入作为外部时钟或者按周期的电流管理/ q) u* w+ m( T4 I
- " Q4 [" b- F' m% O: C' z% U
- 通用定时器(TIMx), P6 ~, J0 i) U
- 1 J7 r5 x. n& {1 R" H! e
- 通用TIMx (TIM2、TIM3、TIM4和TIM5)定时器功能包括: 8 f, S5 P4 P' q! Z
- ● 16位向上、向下、向上/ 向下自动装载计数器 4 R5 K/ f J( u
- ● 16位可编程( 可以实时修改)预分频器,计数器时钟频率的分频系数为1~65536 之间的任意数值
7 |1 E& S; A0 x1 Q - ● 4个独立通道: 3 b% E ]" ]3 R" ^
- ─ 输入捕获 , m. i6 N- W6 B2 ^ E8 ~
- ─ 输出比较
6 M: ]" E3 A+ B) V, }9 }$ _ - ─ PWM生成(边缘或中间对齐模式) * y" w. S+ _. T& F4 P1 D. w
- ─ 单脉冲模式输出 3 Q: M% C9 I2 d7 N$ l
- ● 使用外部信号控制定时器和定时器互连的同步电路 # d( ^( o" R! C" }* x& l
- ● 如下事件发生时产生中断/DMA : / I2 O0 I3 H% K
- ─ 更新:计数器向上溢出/ 向下溢出,计数器初始化(通过软件或者内部/ 外部触发) 8 t$ ?: Y1 G _- N: Z4 d: {
- ─ 触发事件(计数器启动、停止、初始化或者由内部/ 外部触发计数) / b( ?" y, @; B* _9 L( p
- ─ 输入捕获
5 ], v; J" X) V1 V. ]; Z - ─ 输出比较 $ y7 U% e. ~+ N8 s
- ● 支持针对定位的增量(正交)编码器和霍尔传感器电路 ( t3 z2 C% H! R; V& D) Q7 z+ W
- ● 触发输入作为外部时钟或者按周期的电流管理( M5 I0 q% `) E6 A4 F! Y
- . ?( ?0 ]& }5 c( q% m/ C2 Q
- 基本定时器(TIM6 和TIM7)9 I( q8 o7 A9 d7 A0 N. a
- + i7 }" H1 y$ p$ p1 C* |. m
- TIM6和TIM7定时器的主要功能包括: $ q) I" H7 E4 s9 g9 H+ s! E1 }! z) [
- ● 16位自动重装载累加计数器 6 ^. W+ }+ c/ y7 L/ b. f
- ● 16位可编程( 可实时修改)预分频器,用于对输入的时钟按系数为1~65536 之间的任意数值
: ?6 B: m# l2 R5 I - 分频 ) C/ c* t9 C" P7 K" D
- ● 触发DAC的同步电路
$ x' ]/ F& i. F2 P9 l! G0 G; _0 \ - ● 在更新事件(计数器溢出)时产生中断/DMA 请求 }5 Y v- e$ z6 b3 u8 o/ q: d
1 T) M; r& j5 Y+ h) K3 v- STM32的通用定时器是一个通过可编程预分频器(PSC)驱动的16 位自动装载计数器(CNT)构成。STM32的通用定时器可以被用于:测量输入信号的脉冲长度(输入捕获)或者产生输出波形(输出比较和PWM)等。 使用定时器预分频器和RCC时钟控制器预分频器,脉冲长度和波形周期可以在几个微秒到几个毫秒间调整。STM32的每个通用定时器都是完全独立的,没有互相共享的任何资源。
5 Y/ [7 u' T8 R& [$ {+ B - 脉冲宽度调制(PWM),是英文“Pulse Width Modulation”的缩写,简称脉宽调制,是利用微处理器的数字输出来对模拟电路进行控制的一种非常有效的技术。简单一点,就是对脉冲宽度的控制。STM32的定时器除了TIM6和7。其他的定时器都可以用来产生PWM输出。其中高级定时器TIM1和TIM8可以同时产生多达7路的PWM输出。而通用定时器也能同时产生多达4路的PWM输出,这样,STM32最多可以同时产生30路PWM输出!
O% O8 _5 X1 z' C - 7 h+ B' `& P! b9 ]' R
- 要使STM32的通用定时器TIMx产生PWM输出,除了定时器介绍的寄存器外,我们还会用到3 个寄存器,来控制PWM 的。这三个寄存器分别是:捕获/比较模式寄存器(TIMx_CCMR1/2)、捕获/比较使能寄存器(TIMx_CCER)、捕获/比较寄存器(TIMx_CCR1~4)。
. @6 {0 G) V! o) h$ d - 9 Z) _- F; I; k6 W# O
- TIM3_CH2默认是接在PA7面的,而我们的DS0接在PB5上面,如果普通MCU,可能就只能用飞线把PA7飞到PB5上来实现了,不过,我们用的是STM32,它比较高级,可以通过重映射功能,把TIM3_CH2映射到PB5上。 STM32的重映射控制是由复用重映射和调试IO 配置寄存器(AFIO_MAPR)控制的。
' {5 l2 \1 w- P: N
4 _4 J% s- U* ?$ X/ Z1 l9 A- O: F3 W! y& W8 m2 l) V, D
- 1)开启TIM3时钟以及复用功能时钟,配置PB5为复用输出。
% c) i7 }4 |% d5 R% Q9 H- N
" g ?: X! [5 D* M1 T- Z! t+ |5 u- 要使用TIM3,我们必须先开启TIM3的时钟,还要配置PB5为复用输出,这是因为TIM3_CH2通道将重映射到PB5上,此时,PB5属于复用功能输出。
6 i5 R L) C' |& I$ _6 |
4 s5 `/ ^7 G, [* y/ w& P- 库函数使能TIM3时钟的方法是:
! W7 P4 r! X$ c$ j+ X - RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3, ENABLE); //使能定时器3时钟5 q: [! M& a3 z7 `( Q, x1 ?
- ) B x: P$ r# B G4 T6 Q5 [- g9 c
- 库函数设置AFIO时钟的方法是: . I' G5 y; H) q' T. B! w
- RCC_APB2PeriphClockCmd(RCC_APB2Periph_AFIO, ENABLE); //复用时钟使能( a7 m* |" h# h1 A7 s; s
! w- ?8 S# D* N0 i' h3 g" ?3 }- 设置PB5为复用功能输出的方法:GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP; //复用推挽输出
4 }/ @8 W$ K; T# R+ R+ B7 _ - ( m* J1 B1 m3 w2 j3 e/ f
- 2)设置TIM3_CH2重映射到PB5上。
R" N- p* `/ J0 K% G+ p1 ]# O - % o3 {/ i8 s: N/ U# T
- 因为TIM3_CH2默认是接在PA7上的,所以我们需要设置TIM3_REMAP为部分重映射(通过AFIO_MAPR配置),让TIM3_CH2重映射到PB5上面。在库函数函数里面设置重映射的函数是: ; k* ~# B$ B# r8 r
- void GPIO_PinRemapConfig(uint32_t GPIO_Remap, FunctionalState NewState);
B h. d$ r1 H2 a1 i - 3 ]. c6 F- _2 o. c8 M
- STM32重映射只能重映射到特定的端口。第一个入口参数可以理解为设置重映射的类型,比如TIM3部分重映射入口参数为 GPIO_PartialRemap_TIM3,这点可以顾名思义了。所以TIM3部分重映射的库函数实现方法是: GPIO_PinRemapConfig(GPIO_PartialRemap_TIM3, ENABLE);
; w# i" f5 I5 D9 o2 z" ^0 S3 t* ~+ Q
& h/ z: \' h6 H; }2 }- 3)初始化TIM3,设置TIM3的ARR和PSC。
; r; q6 y0 o" q$ G+ w
4 c% B8 j+ N1 n" N4 z' J- 在开启了TIM3的时钟之后,我们要设置ARR和PSC两个寄存器的值来控制输出PWM的周期。当PWM周期太慢(低于50Hz)的时候,我们就会明显感觉到闪烁了。因此,PWM周期在这里不宜设置的太小。这在库函数是通过TIM_TimeBaseInit函数实现的,在上一节定时器中断章节我们已经有讲解,这里就不详细讲解,调用的格式为:
C" e9 G y+ R h - TIM_TimeBaseStructure.TIM_Period = arr; //设置自动重装载值
# T- c& D" [* D( Y# X! |7 e - TIM_TimeBaseStructure.TIM_Prescaler =psc; //设置预分频值
- m: @( i/ q) `$ d6 l - TIM_TimeBaseStructure.TIM_ClockDivision = 0; //设置时钟分割:TDTS = Tck_tim 3 i" D# G$ C# Q) @/ k
- TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up; //向上计数模式 & [% m) ]* d8 O) d
- TIM_TimeBaseInit(TIM3, &TIM_TimeBaseStructure); //根据指定的参数初始化TIMx的7 n! L: O) u3 \' W' z/ {
- ! R+ a; j/ l/ U! K( w
- 4)设置TIM3_CH2的PWM模式,使能TIM3的CH2输出。; E" D! \' n- D9 J
- ( u9 s9 H, L# L) D
- 接下来,我们要设置TIM3_CH2为PWM模式(默认是冻结的),因为我们的DS0是低电平亮,而我们希望当CCR2的值小的时候,DS0就暗,CCR2值大的时候,DS0就亮,所以我们要通过配置TIM3_CCMR1的相关位来控制TIM3_CH2的模式。在库函数中,PWM通道设置是通过函数TIM_OC1Init()~TIM_OC4Init()来设置的,不同的通道的设置函数不一样,这里我们使用的是通道2,所以使用的函数是TIM_OC2Init()。
* S3 m0 y5 @5 s" j1 v - void TIM_OC2Init(TIM_TypeDef* TIMx, TIM_OCInitTypeDef* TIM_OCInitStruct);
" D5 _) Q3 F0 I - " t3 d% G5 I4 ]' y, V, j& D
- 5)使能TIM3。* i. T6 G- x/ h9 j/ T7 D
- 4 b, {- S. S- [5 H
- 在完成以上设置了之后,我们需要使能TIM3。使能TIM3的方法前面已经讲解过: TIM_Cmd(TIM3, ENABLE); //使能TIM3 1 L0 t% f6 o9 f6 V' o7 g
- 6)修改TIM3_CCR2来控制占空比。 % a2 l- Y- {& i R( _% G8 @
- 最后,在经过以上设置之后,PWM其实已经开始输出了,只是其占空比和频率都是固定的,而我们通过修改TIM3_CCR2则可以控制CH2的输出占空比。继而控制DS0的亮度。 在库函数中,修改TIM3_CCR2占空比的函数是:
% F" Z- [8 L4 v8 J - void TIM_SetCompare2(TIM_TypeDef* TIMx, uint16_t Compare2); 理所当然,对于其他通道,分别有一个函数名字,函数格式为TIM_SetComparex(x=1,2,3,4)。
S6 A, c1 e/ [& ` - 通过以上6个步骤,我们就可以控制TIM3的CH2输出PWM波了。
: E/ @4 ^7 E1 |9 Q/ V: |+ u - */
复制代码 / E F7 }% K3 a9 d
|