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

STM32驱动步进电机

[复制链接]
STMCU-管管 发布时间:2020-12-4 15:56
STM32驱动步进电机

3 H! S) i, V9 A6 X& \2 z! Y' X

$ P* A, i4 w- A) T" x# \' y步进电机是将电脉冲信号转变为角位移或线位移,可以通过控制脉冲来控制,因此比一般的电机更精准。这篇文章将会讲述我对四线两项步进电机的调试经验。: O2 I- i) t* l; J

' F: l& }' ?/ H( ^  M- k
1 ]; c, @; f; A/ w" X
1.步进电机以及THB7128的接线
* H! k1 J3 S" Y9 hTHB7128步进电机驱动有两个接线口,J4口与stm32的IO口相接,J2口与步进电机和电源相接。
, v3 A% s' P2 @. \. p6 S7 R5 ?3 M6 ~* Z2 t0 d( V( O
+ b! U$ c! o% i8 Z0 T/ x
J2口有6个端子,从两个端口旁的接口定义可以看出,接口分别为B-,B+,A-,A+,V-,V+。其中V+与9~32V的电源相接,V-接电源地,其余的接口B+,B-一组,A+,A-一组与步进电机的两组同相线相接(注:想要判断步进电机哪两根线同相,只需用万用表测量,短接的即为同相)。5 j1 z! }9 e& ?. R8 A$ \! x0 ~
7 Q5 k* c& _' I3 c" q( z

4 m4 [/ y3 t8 j- v6 u5 \& g) d0 Y6 lJ4口同样可以从端口定义看出,分别有CP+,CP-,DIR+,DIR-,EN+,EN-6个端子。CP为脉冲信号输入端,DIR为电机正反转控制端,EN为电机脱机控制端。在J4端口有共阳极和共阴极两种接线方法。1 Z2 ?" w4 x0 w6 p
共阳极接法:分别将CP+,DIR+,EN+连接到STM32板子的输出电压上,脉冲输入信号通过CP-接入;此时,DIR,EN-在低电平有效。
8 \9 `( |0 r# m共阴极接法:分别将CP-,DIR-,EN-连接到控制系统的地端(与电源地隔离);此时脉冲输入信号通过CP+加入,U/D+,EN+在高电平有效。
1 V3 Z5 B) o9 g1 b: z(注:我采用的是共阴极接法,所以之后只介绍共阴极的程序)" z0 j7 N8 ]' Z2 z
J4端口共阴极接线
* e5 [6 ^5 |" y* C
11.png
2. THB7128设置8 z( ?* w# ?' E  E
TBH7128有6个拨盘开关(S1~S6),前3个开关控制细分设置,即控制电机每一步转动多少度,步距角=电机固有步距角/细分数。后三个S3~S6设置电流大小,设置的电流不要超过电机的额定电流。具体如何设置根据自己实际情况参考驱动芯片背后的说明表进行调节。说明表所示:
. }  z- s9 `6 S+ r' p8 F# e
12.png
3.程序设计5 z4 g* b( e  d) B: Q) I& R
(1)使步进电机稳定转动7 B' `( k) [/ R
驱动接收到一个脉冲步进电机就动一步,所以我们可以用定时器中断将脉冲按一定的频率从STM32的IO口输入到驱动中,这样就可以使步进电机稳定转动起来
9 N( x. c2 j3 n  Z) M(2)步进电机的开启、截止以及变向& C$ l* o/ v% U) }4 E$ _3 V
当THB7128使能端EN+接收到高电平时步进电机停止运转,接收到低电平时正常运转。因此改变与EN+相接的IO口的电平高低就可以控制电机的开启与截止。
* U2 D- n3 k5 A8 ^& H- I4 H当正反控制端DIR+接收到高电平时步进电机改变转动方向,改变相应的IO口电平即可让步进电机变向。) }) D  r6 n5 v. S- D
(3)步进电机变速
# G% h0 r& l" T! o( @我用的定时器中断来驱动电机转动,所以只要想办法改变定时器中断的参数就可以让电机改变速度: v8 O3 J; |9 G8 I( n0 q$ r
" r. W  q0 [! f  N% Z6 Z/ ~' u
8 m% J( J" F# Y' p+ l
具体程序如下:+ z' l2 J7 {1 V+ V
% A. X2 X9 R' O+ w6 r$ E

2 }! `& a. p7 ~# v$ W. k; n2 I0 ~4 ^5 Q% n) [9 ~5 ~6 Y
①定时器设置及IO口初始化+ K$ C& [& Y7 a. A7 O9 s  H2 t
  1. #include "dianji.h"* U6 j5 T  u8 M  k
  2. #include "led.h"
    1 e" R6 j! ~3 k+ K$ j& M
  3. * X0 h2 f5 [. j# @1 A3 a8 _( i5 _
  4. void TIM3_Int_Init(u16 arr,u16 psc)2 t! R  B' u: g5 a( c0 M# Q
  5. {
    ' k* T- E9 t+ Y2 ~1 ~( w6 u2 s
  6.   TIM_TimeBaseInitTypeDef  TIM_TimeBaseStructure;" i% V; K3 ]# _  V+ N( }1 q
  7.     NVIC_InitTypeDef NVIC_InitStructure;
    ' U. Q  m+ q) }& \8 S

  8. $ y& Y6 w& `1 [! T% I
  9.     RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3, ENABLE); //时钟使能4 l* L: h# d/ q7 K  q% _: p

  10. ) n  Y9 r1 o) v' s! k
  11.     //定时器TIM3初始化
    # S! C% ^( _$ o
  12.     TIM_TimeBaseStructure.TIM_Period = arr;//自动重装载寄存器周期的值
    : f: ?/ y6 W  L4 I* C* Z7 |
  13.     TIM_TimeBaseStructure.TIM_Prescaler = psc;//预分频值4 Y5 r6 {2 i, h6 v
  14.     TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1;//设置时钟分割
    : L% J, p; ^5 j3 p' i
  15.     TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;//向上计数模式
    1 @; _3 S' P8 O7 `5 s
  16.     TIM_TimeBaseInit(TIM3,&TIM_TimeBaseStructure);. {1 V7 L, P: i! s' [6 A  r4 Q( g

  17. - u* }- Y3 j4 r) I$ t
  18.     TIM_ITConfig(TIM3,TIM_IT_Update,ENABLE); //使能指定的TIM3中断,允许更新中断
    % g& ~8 a( ?. C1 C4 L2 A8 N

  19. : Z7 S& S3 X: Q$ a
  20.     //中断优先级NVIC设置
    : y# A. i' Z( w* Z" R" ?0 A
  21.     NVIC_InitStructure.NVIC_IRQChannel = TIM3_IRQn;//TIM3中断
      r# b$ h( [/ @9 |6 o( n" C% ?
  22.     NVIC_InitStructure.NVIC_IRQChannelCmd =ENABLE;
    # G" }% H: n+ e3 r
  23.     NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0; //抢占优先级7 _) @; i1 p' Y5 K1 R3 w
  24.     NVIC_InitStructure.NVIC_IRQChannelSubPriority = 3;//响应优先级: @, P* Y$ A! r4 a" ^) X
  25.     NVIC_Init(&NVIC_InitStructure);
    ' \" t9 K1 A( f+ W3 ]6 T7 Z6 a) |

  26. - ]" n' ~; C$ o% g
  27.     TIM_Cmd(TIM3,ENABLE);//使能TIMx
    3 |; I! P, ~6 V0 T+ B  l) g, L
  28. }. s7 r% ~2 c- T5 w. L
  29. , q  p  a: ?0 M+ V- E% C, L
  30. void TIM3_IRQHandler(void)//TIM3中断函数2 }  o% u/ K( z/ {2 x% p
  31. {* r: N9 a$ T9 W4 k
  32.     if(TIM_GetITStatus(TIM3,TIM_IT_Update) != RESET)//检查TIM3更新中断发生与否. x4 H, t; z: }0 L& [
  33.     {
    ( c9 i/ M  P4 c% o# _& j+ m! X" e1 ?
  34.         TIM_ClearITPendingBit(TIM3,TIM_IT_Update);//清除TIMx更新中断标志% g& U- [- e. U+ v0 \
  35.         LED0=!LED0;* [0 a8 V& [! o) \( |9 H' k
  36.     }
    9 A6 t7 P; o6 P/ w6 p7 p
  37. }
    / t5 W- D2 Q3 T( M

  38. ! e! ]" W1 y4 Z( ~7 n8 v: ^9 l% T

  39. " I6 s: L' p$ T* r
  40. void DJ_Init(void), i! u; {! q+ U0 d+ ^! j
  41. {1 m9 x1 V6 h* F) P
  42.     GPIO_InitTypeDef GPIO_InitStruct;
    4 q, S$ O, j8 [# L! [( m
  43. 4 m: {5 k8 A7 {, ]2 a
  44.     RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB,ENABLE);//使能GPIOB
    / H: U5 N3 N" F% g* E1 ~
  45.     RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOG,ENABLE);//使能GPIOG; I. ?: Y8 A+ |# }( R/ I1 @

  46. / A0 f1 {1 A% v
  47.     GPIO_InitStruct.GPIO_Mode = GPIO_Mode_Out_PP;//推挽输出 6 Y1 ]. Q8 ?6 i9 v0 |9 h: S1 f
  48.     GPIO_InitStruct.GPIO_Pin = GPIO_Pin_6|GPIO_Pin_4;1 [3 S3 p/ \  A1 h( x) L( g) z
  49.     GPIO_InitStruct.GPIO_Speed = GPIO_Speed_50MHz;2 n/ i$ Y6 O0 W2 Q! R5 Y
  50.     GPIO_Init(GPIOB,&GPIO_InitStruct);) F$ ~- z: K" \& y4 i& ]1 @
  51.     GPIO_SetBits(GPIOB,GPIO_Pin_6);1 _: c: ?! I( j4 e" M
  52. % r3 f; |3 i0 J/ {4 c
  53.     GPIO_InitStruct.GPIO_Mode = GPIO_Mode_Out_PP;//推挽输出
    3 Y" A; V% N  A  ]
  54.     GPIO_InitStruct.GPIO_Pin = GPIO_Pin_9;
    ' ]6 l% K8 l- z( T" Z. [
  55.     GPIO_InitStruct.GPIO_Speed = GPIO_Speed_50MHz;, p  `1 E1 j  Z3 H. x) ?3 I
  56.     GPIO_Init(GPIOG,&GPIO_InitStruct);
    1 `# d& i( b8 j3 p7 F0 [
  57.     GPIO_SetBits(GPIOG,GPIO_Pin_9);   D, t4 g2 z% I
  58. }
复制代码
②主函数(具体操作控制)4 J/ [5 Y4 C3 a( |8 v* e  ~
  1. #include "stm32f10x.h"5 T. E: {8 j- R3 q7 e8 z
  2. #include "usart.h"( U- d8 r8 N2 h& n$ M
  3. #include "led.h"7 e! x. ?6 z7 ^  p9 b
  4. #include "delay.h", i" i9 Q! a3 x
  5. #include "sys.h"8 f9 ~6 y' Z8 p! Z! \
  6. #include "timer.h"
    + `: M" C2 n; r* H4 Y4 U6 R. y1 H* D% k
  7. #include "key.h"8 R# s( U! U$ o6 ]' Q: h
  8. #include "dianji.h"; @* ~0 ~# N8 [0 I8 o# c4 f

  9. 1 u$ N! B! q1 o: i1 D; l
  10. #define up 1
    & N/ v* w3 k) j! f, x* D6 y/ v
  11. #define down 07 q( K6 \9 l  b0 W" S
  12. . b* ^; h0 N7 w0 s
  13. * m& I8 S; I1 y5 ~- {
  14. int main(void)1 E4 i& U/ a5 O7 Q8 m& k- J
  15. {9 b" P9 N! j& o" N* d4 ?( c
  16.     vu8 key = 0;
    0 \" [  S) w8 _
  17.     vu8 key1 = 0;9 C  Y* @# y( U4 G, Y/ v, J
  18.     vu8 num0 = 0;0 Q% y* f: E( F; g! x
  19.     vu8 num1 = 0;8 s) s# m: ~6 I2 |) a2 y
  20.     vu8 num2 = 0;
    6 I& R7 z8 b8 \4 n% j
  21.     vu8 speed = 1;
    5 c5 {6 P- G9 b( K% a
  22.     vu16 arrLowest = 189;
    0 p3 s. m% e. j$ z$ v( v
  23.     vu16  arrMax= 409;: z3 l) q0 y$ [0 V$ G# u- O) o/ X) b
  24.     vu16 arr =299;& u, d7 D- B# {9 \9 T
  25.     vu16 psc = 359;7 @7 E1 l- X! R) h8 b
  26.     LED_Init();$ `9 g6 U& c+ K8 O6 F  W. Y
  27.     delay_init();4 }1 F+ ~1 Y; z5 v8 S# G& t
  28.     KEY_Init();
    : s* \. O! H5 D! }, J
  29.     DJ_Init();" E/ z5 ?" \$ Z! e5 ]
  30.     NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);, U0 v' B; C" _+ o4 ^3 a0 J
  31.     uart_init(115200);   //串口初始化为115200
    ! k6 Z* ~4 T3 \* w1 L

  32. , L) O% ]7 }0 |+ n
  33.     TIM3_Int_Init(arr,psc);//10Khz的计数频率,计数到100为10ms (arr+1)*(psc+1)/72  $ S3 I9 x/ z* A$ N+ J  U6 X- p
  34.     while(1)
    # w* o! N# a; w! _* y
  35.     {   
    7 }3 f9 N% N! ^9 L+ J) N
  36.         key = KEY_Scan(0);
    $ Z. q, }1 Q5 T4 R+ x
  37.         if(key == KEY0_PRES)+ T+ o" \: K1 L' x
  38.         {
    , |4 C- f0 R$ G1 _0 w# C
  39.             if(num0 == 0)( d9 o* o* ^3 ?- J6 E! q
  40.             {
    : b% @: H9 R' O5 i7 ?9 u/ K( Y5 E7 V
  41.                 num0 = 1;1 G8 T" Q+ [" c: C  D: n! I: d& _" y3 }
  42.                 LED1=0;
    , T/ v  s7 Y# z0 @8 J
  43.                 GPIO_ResetBits(GPIOB,GPIO_Pin_6);
    - m- [+ g) ^) u
  44.             }
    6 m3 y2 Q) h( n; s+ t2 B( x. y
  45.             else, {! u& X4 m; f& W9 C
  46.             {. q" t( j: x5 l, k0 ~
  47.                 num0 = 0;7 E# T- o0 g4 b) y2 L& A5 B
  48.                 LED1=1;
    9 L. ~* d# b9 }/ z, {4 {
  49.                 GPIO_SetBits(GPIOB,GPIO_Pin_6);9 p6 k- W4 r  o' L
  50.             }+ [* o* c0 |! l
  51.         }           
    . c6 Y& C5 j, ^5 z2 W& ]
  52.         if(key == KEY1_PRES)
    4 U$ v, L4 l2 V4 e
  53.         {
    % J! t0 `4 n4 x
  54.             if(num2 == 0)3 B$ _+ L1 A4 m5 b. e# v! n2 G
  55.             {
    # H( G/ r( q! O* e0 y' R
  56.                 GPIO_SetBits(GPIOG,GPIO_Pin_9);% _" `3 N* z: r
  57.                 num2 = 1;
    + v2 m5 @! m" Q4 S  E  ]  e
  58.             }
    3 V, E; l- |7 u3 g& c
  59.             else
    : }% p4 n5 u% T" m2 i: d
  60.             {
    5 U: a- N4 D" B. ]2 g  r
  61.                 GPIO_ResetBits(GPIOG,GPIO_Pin_9);
    % T3 g* x2 ?( S3 U
  62.                 num2 = 0;9 m$ c5 w  n- P  U
  63.             }
    ( e( e1 F( e6 B. k0 v; {& @
  64.         }
    # ]. z/ d& D- f+ u2 W
  65.         while(key == WKUP_PRES)
    4 w- h* r0 a3 M" R, U
  66.         {                       
    # Z4 A5 O) O8 v; s! }( Y, t6 Z
  67.             if(speed == down)
    4 _8 n8 ^& z. a
  68.             {               / G* T7 o6 j  |' [* I" m
  69.                 arr+=2;% {% U$ g# k- r
  70.                 TIM3_Int_Init(arr,psc);                 : \1 s! A3 ?3 C! q
  71.                 delay_ms(50);
    " h" Q- U' b# L
  72.                 key1 = KEY_Scan(0);/ `6 i! S4 a# w; ?5 {( x% S
  73.                 if(arr > arrMax)& @1 M- J+ P* `& }4 R& {% ?
  74.                 {
    0 c/ u+ \* |$ H! ^3 A
  75.                     speed = up;& y: o, \8 |' _5 s2 m4 i7 X
  76.                 }6 G. }6 R. ?) D/ n6 n1 f
  77.                 if(key1 == WKUP_PRES)//关闭变速% o1 J5 r/ ]; S* |& `
  78.                 {9 |; \7 f, G/ P
  79.                     LED1 = 0;
    5 X  v( ~6 }- |" L9 R* b( t
  80.                     key = 0;
    . o! W8 V% {" J& _2 w4 S! M7 }
  81.                 }
    0 N% J3 w& U' @9 A$ `$ n. Z
  82.             }
    7 {. U% l3 K# G+ S" S9 R$ ~
  83.             if(speed == up)
    7 J% L) j2 L1 W0 l& S; l4 d
  84.             {                   ( R/ U" D0 o1 P# O/ R$ }
  85.                 arr-=2;             % X1 _/ m+ ~* }8 ?4 d& ~
  86.                 TIM3_Int_Init(arr,psc);                 
    7 o) p; s$ s4 b6 }3 W
  87.                 delay_ms(50);
    , z% W" [% G4 R! Q0 }: U( ]& Q
  88.                 if(arr < arrLowest)
    . F' D; f  h7 r1 X! s6 h: ~
  89.                 {   
    / c6 u( G( w8 [8 h  T& M
  90.                     speed = down;" H' }# t4 U* _- v$ o) D6 S' Y( ]
  91.                 }6 J; J+ `# i8 s
  92.                 key1 = KEY_Scan(0);
    , V: Y- P* H9 u4 [2 x
  93.                 if(key1 == WKUP_PRES)
    0 }5 O3 i: P9 ^; {* [6 m2 _' |+ H
  94.                 {0 w* D+ H2 l( k* v
  95.                     LED1 = 0;
    ! V! I5 w5 @) o5 L1 s5 Y
  96.                     key = 0;
    , i7 A; ]# y8 ]2 A" f
  97.                 }   
    & X7 r- }4 D8 \$ m* x, |5 o
  98.             }1 e- D2 [; s. h9 b6 a' X
  99.         }       : a4 F5 G$ F1 d& y/ n8 _7 Y; H
  100.     }
    & O( G( o( r9 \) u/ U& j/ w+ F
  101. }
复制代码
4.总结7 X' X. J, {7 J- R, o2 |
通过以上过程的调试,我实现了用按键控制步进电机的启动、截止、变向以及加减速这些功能。
9 n$ M1 ]5 }6 N可能我的电机驱动和大家的不一样, 希望能让读者起到参考作用。
; @! j, c8 y: W( p' I8 I
. \6 p; S( u* f( W7 o
收藏 1 评论0 发布时间:2020-12-4 15:56

举报

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