你的浏览器版本过低,可能导致网站不能正常访问!
为了你能正常使用网站功能,请使用这些浏览器。

【经验分享】利用STM32F103单片机输出SPWM波

[复制链接]
STMCU小助手 发布时间:2022-3-20 20:46
最近需要用到单片机输出SPWM波功能,在网上找了好多资料,发现都不完整,有算法的没有代码,有代码的看不懂算法。于是只好自己摸索,现将方法整理如下。
, Y. ~1 x1 f! B! u6 d$ k0 G2 ?4 C' n关于什么是SPWM波,为什么要用SPWM波,网上的介绍有很多,就不多说了。主要说一下在STM32F103C8T6上是如何实现的。
6 y* _: Q4 e5 [7 b5 o! |要产生SPWM波,核心就是调节PWM波的占空比,在一定时间段内使输出PWM波所占的面积和对应的正弦波面积相等。占空比的调节需要根据正弦波上对应的点来调节。那么首先就是要生成一组正弦波数据。生成正弦波的数据代码如下:
6 C' u5 ~+ b4 j* b& ?+ S" O
% {4 M4 R* M! \
  1. //point 一个周期内的点数& I  F4 q( K% H* k
  2. //maxnum 最大值
    ( O6 ^. ^" U. L& i) N
  3. void get_sin_tab1( u16 point, u16 maxnum )7 v+ o( |+ m% F/ V; x, G( W
  4. {6 P, ]- |0 n) M7 `* Z
  5.     u16 i = 0, j = 0, k = 0;0 I* w: o9 b$ M& D
  6.     float hd = 0.0;                //弧度' P* z6 \: L$ ~. K$ Y1 t; u
  7.     float fz = 0.0;               //峰值
    $ U! j, y$ `8 |& |0 f
  8.     u16 tem = 0;) a- M4 V# g( v% p9 T1 C' a# q' b
  9.     j = point / 2;                        //水平线位置 单片机没有负电压
    ( r6 z- j( E3 f3 a' G( j( m
  10.     hd = PI / j;                // π/2 内每一个点对应的弧度值
    % ~" I6 O7 `1 d; z
  11.     k = maxnum / 2;              //最大值一半7 m0 P& z  z, a) w# Z' f: H# J
  12.     for( i = 0; i < point; i++ )
    1 {% V5 y' @) T9 P" F8 O* X
  13.     {     * o( U% j8 e& P+ m$ x  P7 A
  14.                 fz = k * sin( hd * i ) + k;                                        4 @" Y1 S# M5 r3 [
  15.         tem = ( u16 )fz ;                                                
    - a* V7 D! U' m$ q/ z, I
  16.                 printf("%d\r\n",tem);( u# l+ n% @3 A% v# a4 y8 i
  17.         sinData = tem;
    $ r& F  l) p! M6 R3 D) s) I) ^. A
  18.     }
复制代码
0 d% `+ l% I6 P4 b

9 i' ?* v8 g# S3 m; z通过串口波形软件可以看到生成的数据和波形如下:
4 z0 e+ Z5 B7 t/ }# H
  a- t* F1 @) r, b6 v' d 9KJ@O%~{`5BB3_4$Q}Z897K.png
' U4 I9 p6 H" ?9 d0 {$ R  {" S
6 q; I5 Y  E8 p# s* W9 U, J数据生成好之后,下来设置定时器输出PWM,/ h, w$ s+ V7 S. K% h

' U6 w7 \, x* b0 Z, j0 P
  1. void TIM1_PWM_Init(u16 arr, u16 psc)
    # x9 |7 ]! t' {% M8 W8 T/ Z
  2. {, j8 z3 J9 W5 K* y+ q& n
  3.     TIM_TimeBaseInitTypeDef TIM_TimeBaseInitStructure;# ~/ v* m! ?# h' {+ T9 [! A
  4.     TIM_OCInitTypeDef TIM_OCInitStructure;& N7 M' a' i" m! D# g  |
  5.     GPIO_InitTypeDef GPIO_InitStructure;6 ^( C; u3 g9 p6 _) ?) U

  6. + Q/ D3 c' C5 j1 T
  7.     RCC_APB2PeriphClockCmd(RCC_APB2Periph_TIM1, ENABLE);
    / D- O0 e+ P' V- h
  8.     RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE); //复用时钟
    0 Q3 j+ V( q- e' O9 |' e

  9. # S% s( @9 |' |
  10.     GPIO_InitStructure.GPIO_Pin = GPIO_Pin_8;* T: U  y- K# l0 N9 Z
  11.     GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;
    ' u5 Q8 w# }7 r7 a
  12.     GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    . U- U, _- M/ Q  l4 S$ f0 Z+ D: C
  13.     GPIO_Init(GPIOA, &GPIO_InitStructure);
    9 X5 y1 R! y3 A/ w% Q/ \
  14. ' w0 s/ @6 S1 z. |4 z" Z/ {( j2 H
  15.     TIM_TimeBaseInitStructure.TIM_Period = arr;1 c5 \4 v: I2 W4 L9 B0 I* H9 j
  16.     TIM_TimeBaseInitStructure.TIM_Prescaler = psc;
    $ K4 [" A: Q; a! p# D. [
  17.     TIM_TimeBaseInitStructure.TIM_RepetitionCounter = 0;                                //只有高级定时器需要设置,其他定时器可以不设置。0 C$ O( R: {9 T& X
  18.     TIM_TimeBaseInitStructure.TIM_ClockDivision = TIM_CKD_DIV1;' y1 A4 A0 a1 ?/ Y+ V( Y* J
  19.     TIM_TimeBaseInitStructure.TIM_CounterMode = TIM_CounterMode_Up;
    * Q, `/ X- W5 J1 p" a+ u
  20.     TIM_TimeBaseInit(TIM1, &TIM_TimeBaseInitStructure);& A0 u: K, ^! t# }0 C+ d# \/ z/ s- o
  21. 3 @3 J; q, n- g% E7 P3 G7 Z
  22.     TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM1;
    ) Z1 C( I6 L5 O/ S" a  p
  23.     TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable;4 E2 m# V* i3 P# q  g" `# `
  24.     TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_High;
      J7 p4 y3 I, x; M$ N

  25. ) t( h, q/ n) T; s3 |3 R; w
  26.     TIM_OC1Init(TIM1, &TIM_OCInitStructure);                                                        //TIM1_OC1   ) z) D. R  |% x2 S( y) E0 @1 z

  27. 3 y0 @! n! L6 E0 _
  28.     TIM_OC1PreloadConfig(TIM1, TIM_OCPreload_Enable);$ e" R; @0 f- c; }2 \% F
  29. . u) T1 `% q8 Q; `
  30.     TIM_Cmd(TIM1, ENABLE);# \- ~3 @8 P' C. @
  31.     TIM_CtrlPWMOutputs(TIM1, ENABLE);                                                                        //高级定时器才有  必须打开# X) U/ y: `# W' ]& y
  32. }
复制代码
+ A) s; t( c- X' K8 N; L
设置定时器1输出PWM波,定时器初始化为:( Q- w4 a' G* k* W

( Y& m) D- ]& F3 w 定时器 TIM1_PWM_Init(1000 - 1, 3);                5 ?. ~2 J1 a, z

* S5 _% J' |) i# L5 H: t定时器1输出18K的PWM波,此时输出的PWM波占空比是固定的,还不能随着正弦规律变化。下来利用定时器2的定时中断,在中断中改变PWM的占空比。) [4 Q6 o) ]7 w+ N. o& x
$ e0 c$ g0 n8 k) h, x, {
  1. //APB1时钟分频为2  TIM2-7 时钟数为APB1 2倍0 F8 G6 v4 u7 J5 a
  2. // Tout= (arr+1)*(psc+1) / Tclk* E4 K2 R1 w; N6 Y8 i
  3. void TIM2_Init(u16 arr, u16 psc)! Z+ P" z: b) c2 [' s
  4. {  z" n( l/ w6 ^
  5.     TIM_TimeBaseInitTypeDef TIM_TimeBaseInitStructure;. T4 U, Z+ y. Q0 b7 r
  6.     NVIC_InitTypeDef NVIC_InitStructure;% l& W- @" [/ }& f6 G

  7. . |. k" l8 |8 M$ Q) o, R4 M
  8.     RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2, ENABLE);
    3 Z: Q2 x0 F7 O- |; [! ?) z

  9. " ?* t1 x2 p! I- \
  10.     TIM_TimeBaseInitStructure.TIM_Period = arr;- D" i6 v) y4 d' i- m; z
  11.     TIM_TimeBaseInitStructure.TIM_Prescaler = psc;( a6 J' B, P! G3 n3 U  W
  12.     TIM_TimeBaseInitStructure.TIM_RepetitionCounter = 0;                        //只有高级定时器需要设置,其他定时器可以不设置。) f) {% g9 y+ g' x! I
  13.     TIM_TimeBaseInitStructure.TIM_ClockDivision = TIM_CKD_DIV1;- g# K4 W% Z4 t3 H0 k' q  O+ H3 H8 F
  14.     TIM_TimeBaseInitStructure.TIM_CounterMode = TIM_CounterMode_Up;
    # I' r! c$ [6 `
  15.     TIM_TimeBaseInit(TIM2, &TIM_TimeBaseInitStructure);
    5 v6 ?. a3 g4 Q

  16. & ~/ a+ @! g1 m6 Y2 J4 F) L- ~3 \- @
  17.     TIM_ITConfig(TIM2, TIM_IT_Update, ENABLE);
    ; c3 x# S  @. {: V! Q7 w5 U

  18. $ w3 x  B4 c8 A: \0 U9 q6 d( p
  19.     NVIC_InitStructure.NVIC_IRQChannel = TIM2_IRQn;4 T7 L: M* f  @# Q+ Q6 a
  20.     NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0x01;
    . b9 `; J1 g" |, s0 h2 V% f
  21.     NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0x01;$ M; B) [  k# }9 b+ G& Z5 o5 W
  22.     NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
    ' v5 E3 g* g1 h9 ~- [
  23.     NVIC_Init(&NVIC_InitStructure);
    ; C( K; x2 `- y, v. m: M
  24.     TIM_Cmd(TIM2, ENABLE);
    / _( w4 N. O. b2 w7 c% Q( o: k6 d: d% T, R
  25. }
    ! s9 Z2 q, b" E; X, i1 ?* ]2 _0 I

  26. " L8 h. F+ E& S& \2 @* Z
  27. u8 dc_cnt = 0;                                                        //占空比计数
    8 U% k. B% C+ ]6 V5 S. H
  28. //定时器2中断中改变 定时器1的占空比值0 z; k: ?/ F0 q: J2 }/ A! V3 ]
  29. void TIM2_IRQHandler(void)/ t0 N  a. p/ t1 l; }- }# ?
  30. {
    1 u) V) A& H' ]) `% a& M4 O
  31.     if(TIM_GetITStatus(TIM2, TIM_IT_Update) != RESET)8 ]  I8 _( U. x  h" L4 T
  32.     {
    3 h0 g8 X7 e1 H: \1 ^1 U' [+ {
  33.         TIM_SetCompare1(TIM1, sinData[dc_cnt]);0 I3 y" x9 {0 F. _
  34.         dc_cnt++;  _: m; `9 n& Z# C5 @! s
  35.         if(dc_cnt >= PointMax)
    # z2 g$ `, o7 c' t% B
  36.             dc_cnt = 0;
    9 N4 t8 V% u, F8 m" @2 f
  37.         TIM_ClearITPendingBit(TIM2, TIM_IT_Update);+ L9 I+ v+ Q5 [5 T" B$ @# W2 a
  38.     }5 @* ~; L, S% b7 b/ z, x
  39. }
复制代码
8 f3 U+ Y0 ]9 V4 @  g
在定时器2中断中,依次将sinData数组中存放的正弦数据做为PWM的占空比值。由于生成的正弦数据最大值是1000,输出PWM的ARR值也是1000,那么生成的正弦值就可以直接做为占空比用。如果生成的正弦波数据和ARR值不一样,在此处需要一个转换,用比例系数调整正弦波中的最大值刚好是占空比最大值就行。
. K* y0 L; o% x/ }  g这样在定时器1输出PWM比的时候,在定时器2中断中调整PWM的占空比,这样输出的PWM波经过外部电容后,就是一个正弦波。在PA8引脚接105电容。生成的波形如下:" ^7 ?& v' @8 ], B
E]3`1G4L%[T25O2G~~7_A9I.png
. [' p! k1 C9 @6 }- L1 s& P2 f+ ~" [6 g' {' z
生成了正弦波之后,初步目的算是达到了。那么如何用生成的正弦波,和外部市电波形同步呢。' v4 o9 W6 j+ H! ~% t
可以用过零检测电路将外部波形测出来。在每个零过点的时候输出一个脉冲。然后用中断检测这个脉冲,每次过零点到来时,在中断中将正弦波数组下标设置为0,这样每个过零点时,输出的正弦波也从0点开始输出。6 d1 s* v9 b4 Q0 I
外部中断代码如下:# P2 ^8 X* M- I# g& E% a8 H  s

) C; U9 p3 i1 z# X
  1. void IO_Exti_Init(void)
    " W: N9 ]. k$ z  l: G
  2. {
    1 G- V/ `8 r& F! _1 n: J7 e: ]/ r5 ~, ^
  3.     EXTI_InitTypeDef EXTI_InitStructure;
    0 s( Q; @1 B  }- X) ~; v9 L
  4.     NVIC_InitTypeDef NVIC_InitStructure;" `- M, x8 N2 [' C* H
  5.     GPIO_InitTypeDef GPIO_InitStructure;
    . t3 R& _  k9 `+ |' g9 u# ]
  6. / R6 B) C+ H+ u. F0 u* S; I, N# X
  7.     RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE);
    : q- h1 J; i. ^4 P( N
  8.     RCC_APB2PeriphClockCmd(RCC_APB2Periph_AFIO, ENABLE);
    6 z! f  f+ O6 X: V. O, F
  9. ; I. s9 N% `1 |! w4 c6 o% c; O
  10.     GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0;
    ) n' U; F* K. P* i, I
  11.     GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPD;/ K2 n2 {/ u( t, `) h3 g
  12.     GPIO_InitStructure.GPIO_Speed = GPIO_Speed_2MHz;
    + S- P2 h, f) z* |
  13.     GPIO_Init(GPIOB, &GPIO_InitStructure);
    8 V4 z1 m7 ^7 D5 y! n

  14. / D& S  R  Q  b  Z) m( }7 b) [
  15.     GPIO_EXTILineConfig(GPIO_PortSourceGPIOB, GPIO_PinSource0);; A( J5 i" Q' l  Q8 Z3 a
  16. 2 F6 C' ]# y2 u9 g
  17.     EXTI_InitStructure.EXTI_Line = EXTI_Line0;
    ( j0 ?* S% U" f# r$ n
  18.     EXTI_InitStructure.EXTI_Mode = EXTI_Mode_Interrupt;- i- M! x& b( O3 e3 [- C' X
  19.     EXTI_InitStructure.EXTI_Trigger = EXTI_Trigger_Rising;  j1 o% S) L' Q+ r
  20.     EXTI_InitStructure.EXTI_LineCmd = ENABLE;
    2 D! f9 W, _! A/ @7 P8 H
  21.     EXTI_Init(&EXTI_InitStructure);" j; W: D+ @! I, W+ x: L6 z
  22. 3 N$ L. @5 @7 N' Z" z# I
  23.     NVIC_InitStructure.NVIC_IRQChannel = EXTI0_IRQn;
    ! G$ S4 V8 d% Y: X
  24.     NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0x02;0 F3 K, W  `; s7 E4 Y
  25.     NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0x01;4 W5 B7 c. B3 ^5 D# m( g
  26.     NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;! Z+ r3 P9 l, `1 {
  27.     NVIC_Init(&NVIC_InitStructure);
    $ @% G4 W2 ?- k1 a' r
  28. }
    9 o' h# M7 K2 j0 v" l

  29. 2 D8 ^  R5 W2 e8 g& g
  30. extern u8 dc_cnt;                                                        //占空比计数
    + G# Q" q' A& C' m; ~" P! {& G
  31. //10ms中断一次9 E1 B0 k. C$ G. g
  32. void EXTI0_IRQHandler(void)
    8 W) |7 w! p3 r* d
  33. {2 U5 Z6 m* n7 g. C+ o
  34.     if(EXTI_GetITStatus(EXTI_Line0) != RESET)* v# _! O$ {  o. ?
  35.     {! S' D  o& d# i; {7 d4 G3 J6 G1 M
  36.         dc_cnt = 0;
    ) H# H2 }3 t9 A
  37.         EXTI_ClearITPendingBit(EXTI_Line0);
    ) C+ M3 Y$ g8 l& f
  38.     }  T- }' C( A* L! B) d
  39. }
复制代码

5 ^1 y/ n, P0 O7 n- H3 g在外部中断中将正弦波数组的下标控制值dc_cnt 清零,这样每个过零点时,输出的正弦波数据正好是第一个。
& l; T  R0 V1 l8 n- {+ t这样输出的SPWM波就可以通过中断和外部波形进行同步了。
" L/ f# P- h+ k3 M2 [7 g; {* N+ h. w: x- @! c, N3 \  O
@K]5DH~30](6AD%IIFNY8J7.png
' e+ ~+ F" C# N5 P% l% s
& ^0 F9 m/ ]8 u6 t通过波形可以看出,在过零点位置,输出正弦波也刚好在零点。
5 L& P& f: p; v" ^如果要调整输出正弦波数据的初始位置怎么办呢?只需要在生成正弦波数据时,设置初始位置就行。5 F3 I2 \& b# [6 g) N$ z8 R
2 `4 F1 y! d' ^7 U
  1. void get_sin_tab1( u16 point, u16 maxnum )
      u" W1 h4 g8 H/ B$ h0 O; }
  2. {! M- w- ?/ Y! W) F- B0 q4 }  T
  3.     u16 i = 0, j = 0, k = 0;" `2 O6 r& C0 F# P
  4.     float hd = 0.0;                //弧度) ~: u7 U+ p/ s' N' D& _
  5.     float fz = 0.0;               //峰值+ t& n2 _7 x& Z: b6 V+ M/ w: K: b
  6.     u16 tem = 0;! d5 l# B0 |1 h2 s" Y  c4 F
  7.     j = point / 2;                        //水平线位置 单片机没有负电压 水平线为点值数量的一半
    4 k- I3 y* K! U5 T+ y
  8.     hd = PI / j;                // π/2 内每一个点对应的弧度值2 B' ]/ B- @; E
  9.     k = maxnum / 2;              //最大值一半
    3 G; Q# _0 J$ V# E' u
  10.     for( i = 0; i < point; i++ )- b' Y7 q6 ?' w8 v! ~8 [0 m
  11.     {, m+ D- A) }( f9 p6 M
  12.                 fz = k * sin( hd * i ) + k;                                                //起点为中间值3 c  G% L2 Z' l) ~* }' C3 w
  13.         //fz = k * sin( hd * i + PI / 2) + k;                        //起点位置偏移到 π/2  起点为最大值5 u# |6 ]) c$ k- }, i2 ~
  14.                 //fz = k * sin( hd * i + PI / 2 + PI ) + k;                //起点位置偏移到 π/2 + π 起点为最小值% n. w$ Q* K. I5 T

  15. 1 z" ~: U, F7 X% v; g
  16.         tem = ( u16 )fz ;                                                
    / F  q) F2 Y! V+ X" M0 o1 }
  17.         sinData = tem;% ]" K8 Z" y% o. B
  18.     }( L( d' b0 i6 t" j& Y
  19. }
复制代码

4 J, k2 c1 i8 R9 M在生成数据是 sin( hd * i ) ,的值就是初始相位,如果要调整初始位置,只需要将括号中的值加上偏移量就行,默认的起始位置在正弦波的0度位置,如果要让初始位置在90度,那么给括号中的值在加上 π/2就可以了。* M5 j" o- I/ V/ V( @/ B+ u! M! t; K

8 o: L0 M( y) [& j6 o6 w* R ](`2(GOO1_@@Y8$MFY_3K)A.png * ?: g7 l! R* h) u
  v! I. m. J. i6 `; @3 S2 ]' D
如果要让起始位置在270度时,在括号中加上 3/2π就可以了。
2 n% O# D6 ~$ K( {- P
9 e/ i  F% g- e7 {  a5 U& w! Z BQ3R6SHM5213R_M}D{E710J.png % A) J" l% m( H+ V
4 i; R/ m2 B/ l7 S

* g0 R# [; p8 ]6 d
收藏 评论0 发布时间:2022-3-20 20:46

举报

0个回答

所属标签

相似分享

官网相关资源

关于
我们是谁
投资者关系
意法半导体可持续发展举措
创新与技术
意法半导体官网
联系我们
联系ST分支机构
寻找销售人员和分销渠道
社区
媒体中心
活动与培训
隐私策略
隐私策略
Cookies管理
行使您的权利
官方最新发布
STM32N6 AI生态系统
STM32MCU,MPU高性能GUI
ST ACEPACK电源模块
意法半导体生物传感器
STM32Cube扩展软件包
关注我们
st-img 微信公众号
st-img 手机版