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

【经验分享】STM32实例-步进电机的速度控制

[复制链接]
STMCU小助手 发布时间:2022-6-1 21:01
项目需求
  O3 x# p2 n9 _
    在制作一个使用全向轮的机器人底盘,对于全向轮,电机的精度是影响效果的重要因素。所以使用了步进电机,使用步进电机的优点是可以不使用编码器,开环控制即可达到高精度的效果。

+ r& r. G* ?: R
调整占空比或者调整周期
    众所周知,PWM有两个重要参数,周期与占空比。6 v. V  m% D' y* a2 a
    步进电机的运动方式是,每收到一个脉冲,就旋转指定的角度。因此影响电机速度的唯一参数就是PWM的频率。以下附图两张来说明调整占空比与调整周期的区别。
* b3 X0 w1 J; N' N+ {
周期固定,调整占空比
640.png

+ m  h- ~& u9 f. Z1 }    上图中有4个PWM,它们的周期是一样的,不同之处是拥有从80%至20%的占空比。可以看出,无论占空比为多少,在1s内,它们产生的高电平的数量是一样的,即无论占空比为多少,PWM的频率都一致。因此也就无法调整步进电机的速度。因为步进电机的速度仅与PWM的频率有关。
8 V+ Y! h3 H# ]% F9 _
占空比固定,调整周期
640 (1).png

3 N7 P/ I6 F3 B& X: R" b* z    上图中有4个PWM,它们的占空比都固定为50%,却拥有不同的周期。可以看出,即使占空比固定,只要动态调整周期,PWM就能在相同时间内产生数量不同的脉冲。因此,固定占空比,动态调整PWM周期,即可达到控制步进电机速度的效果。
3 w8 K/ m6 v( d0 e9 v
基于STM32的实现
    在STM32F1中,定时器具有PWM模式,可以用来产生PWM。但是,STM32的PWM模式,只要确定了时基单元(即确定了PWM周期),改变输出比较寄存器,仅可改变PWM的占空比。
! m# x  l9 k  C% d  Q2 D3 ]* s    若需要改变PWM周期,需不停地改变定时器的时基单元。但时基单元与硬件相关,不适合频繁变更。因此,步进电机的调速,不适合使用STM32下定时器的PWM模式来控制。1 Q) o; n" w2 H9 l+ U( z- E
    在多方查找资料后,确定了一种利用输出比较精确控制PWM周期的方式。

* L9 r5 i" ?3 Y9 ?
利用输出比较产生频率可变的PWM
    利用输出比较产生频率可变的PWM,原理简单介绍如下:, c6 @) W' B8 V  x/ z
    首先配置定时器时基单元,确定脉冲最小长度单位CK。
2 Z% Z" r7 L  F. m6 z6 d    然后开启定时器的输出比较,设置模式为翻转模式,并开启输出比较中断。! G; i/ V$ F3 n. k: o1 W( E: {
    将定时器内计数器CNT当前值,加上脉冲长度X(单位为CK),写入输出比较寄存器。
& X( q/ H# M- G    在X个CK后,将会触发输出比较中断。同时电平翻转。2 k$ f7 f' c7 k* {, d9 y* P$ o
    在中断中再次将当前计数器CNT的值,加上脉冲长度X,写入输出比较寄存器。
! J$ b' Y( Y) J    在X个CK后,将会触发输出比较中断。同时电平翻转。
/ L: y. A% T6 }- J9 j; X    在中断中再次将当前计数器CNT的值,加上脉冲长度X,写入输出比较寄存器。: Y& n# E5 T( Q& m8 _( `9 L; ^) f
    如此往复……
! e! P' n3 W# f, g5 ^    以此即可得到一个占空比为50%,周期为2X个CK的PWM。

$ R3 J* i0 f) g4 o3 D. F
确定最小单位CK,配置时基单元
    首先确定一个最小的间隔CK,规定PWM的高电平长度和低电平长度的单位都是CK。
# T2 e2 `' V, f! u. f( E    即高电平的长度一定是CK的整数倍。低电平亦然。
% E$ `- B' K( A: z  A5 m% S    然后配置定时器的时基单元,通过CK的长度确定预分频系数。
/ X( r; B& y) _3 Q    已知STM32F103的主频为72MHZ,则时基单元中预分频系数为
  1. PSC = 72M / (1/CK)
复制代码
. e7 c6 X+ e# n0 }  I* A
    例如,rtz所确定的CK长度为10us(0.00001s),即可得出方程。
+ ~) ?; q  x1 C. ]
CT%}9N_KML5GYYJXKS`V31I.png
   预分频系数确定为720后,由高速晶振产生的72MHZ的时钟信号被720分频,得到100000HZ的时钟信号。即时钟信号每秒变动100000次。每次10us。同时可将重装载值设定为0XFFFF(16位定时器的最大值)。因为本次使用的输出比较模式不使用更新中断,该值可随意设置。
  1. TIM_TimeBaseStructure.TIM_Period = 0XFFFF;# v/ z7 s  O4 x7 x) G
  2. TIM_TimeBaseStructure.TIM_Prescaler =720;
    : |# k  @0 w9 L8 e9 I& g
  3. TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1;6 w5 _" P. a8 n) k. o/ b7 W
  4. TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;$ M( Q9 F7 h% C6 f
  5. TIM_TimeBaseInit(TIM2, &TIM_TimeBaseStructure);
复制代码
6 W9 O& h. |4 z: W+ }8 @  N
- E4 Q9 M4 O) W# z" @
开启输出比较通道,设置输出比较模式为翻转模式,并配置NVIC,开启输出比较中断,配置输出比较通道:* f9 @$ a! M, T; z  e
  1. TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_Toggle;' W: R% E0 C9 q9 S# T
  2. TIM_OCInitStructure.TIM_Pulse = 500;  O6 [* q; M) }1 j- e! g9 t1 c. a
  3. TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable;
    5 s: ^  o3 x# `3 a8 T; v, k
  4. TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_High;" ]. K, Y6 V: V. N* _; V
  5. TIM_OCInitStructure.TIM_OCIdleState = TIM_OCIdleState_Set;
    $ |/ U2 `4 k# A7 W" h! }( N' o
  6. TIM_OCInitStructure.TIM_OCNIdleState = TIM_OCIdleState_Reset;
    ) [+ l* k3 c2 u
  7. TIM_OC1Init(TIM2, &TIM_OCInitStructure);
复制代码
7 |$ i) g# a% \/ k( k
    开启输出比较中断,配置NVIC优先级:

( G/ R; i2 [3 a, U
  1. TIM_ITConfig(TIM2, TIM_IT_CC1, ENABLE);
    $ @% ]3 {! @) a3 C0 @1 X7 o
  2. NVIC_InitStructure.NVIC_IRQChannel = TIM2_IRQn;
    0 Z* Q4 I& b! v* V
  3. NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;6 ]/ X; L0 P( c, G! N8 e& _$ c
  4. NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;) e4 d8 M- A. T& x0 T) H6 ]8 b
  5. NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;+ n3 W+ q) [+ K6 X& X
  6. NVIC_Init(&NVIC_InitStructure);
复制代码

* K/ J# ^8 `  B" S' Y( |$ T编写中断服务函数
8 ]' a! z+ R3 H3 @, H- S) l: [
    在输出比较中断中,唯一要做的事情就是把当前CNT的值取出,加上脉冲长度X,写入输出比较寄存器。
5 n8 f+ ?1 h- B% |1 \/ M5 l    当计数器达到0XFFFF(之前设定的重装载值)后,再加一会自动变为0.
8 E4 `2 P1 c+ q+ b+ K7 k    例如,当前CNT值为0XFFFF,脉冲长度为5,很明显,输出比较寄存器应设置为0X0004才可触发下一次中断,而不是0X10004,这样会造成溢出。  ~# F) E" B5 g. N( L3 d5 A
    因此将CNT的值与脉冲长度相加后,需要取0XFFFF的余数后,再写入输出比较寄存器。
  1. int t_m=5;//低电平和高电平的长度
    ) Y" {  F" e: _& o( `5 ^6 p# x
  2. void TIM2_IRQHandler(void)
      D! v7 u* j, t4 D0 V( l2 f" o
  3. {/ V3 C. T  x. V. H/ X+ U+ k
  4. if (TIM_GetITStatus(TIM2, TIM_IT_CC1) != RESET) {
    1 d' [( ^: ?" p8 ]4 }* j, N0 \! K
  5.     TIM_SetCompare1(TIM2,(TIM2->CNT+t_m)%0XFFFF);, }. |% @6 Q* ?
  6.     TIM_ClearITPendingBit(TIM2, TIM_IT_CC1);" x2 u3 J" N! d* p) _. U
  7.   }6 x& `3 k6 `8 R1 w4 J2 S
  8. }
复制代码
! O0 T0 e6 g" f# N
    动态调整中断服务函数中的变量t_m,即可达到修改PWM频率的效果。

- t7 F% c6 h6 {) b8 w/ \" N3 U* j+ Q& ^" x  [% o
收藏 评论0 发布时间:2022-6-1 21:01

举报

0个回答

所属标签

相似分享

官网相关资源

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