当前位置:   article > 正文

【STM32篇】4988驱动步进电机

【STM32篇】4988驱动步进电机

本文介绍的步进电机驱动为毕设所用,学习时也借鉴了其他博主和商家的资料,介绍的都很详细。但对于刚入门的友友萌来说,可能还会存在的一些不解。而我就是被卡在驱动电源上。下文为我对4988驱动的一些理解,希望能帮助到刚学习步进电机的朋友。

4988驱动IC价格便宜驱动简单,但在驱动步进电机时,电机会存在的明显震动。下一篇文章将实现TMC2208驱动IC控制步进电机。

一、简介

A4988 是一款完全的微步电动机驱动器,带有内置转换器,易于操作。该产品可在全、半、1/4、1/8 及 1/16 步进模式时操作双极步进电动机,输出驱动性能可达 35 V 及 ±1 A。A4988 包括一个固定关断时间电流稳压器,该稳压器可在慢或混合衰减模式下工作。转换器是 A4988 易于实施的关键。只要在“步进”输入中输入一个脉冲,即可驱动电动机产生微步。无须进行相位顺序表、高频率控制行或复杂的界面编程。A4988 界面非常适合复杂的微处理器不可用或过载的应用。在微步运行时,A4988 内的斩波控制可自动选择电流衰减模式(慢或混合)。在混合衰减模式下,该器件初始设置为在部分固定停机时间内快速衰减,然后在余下的停机时间慢速衰减。混合衰减电流控制方案能减少可听到的电动机噪音、增加步进精确度并减少功耗。提供内部同步整流控制电路,以改善脉宽调制 (PWM) 操作时的功率消耗。内部电路保护包括:带滞后的过热关机、欠压锁定(UVLO) 及交叉电流保护。不需要特别的通电排序。

A4988 采用表面安装 QFN 封装 (ES),尺寸为 5 mm × 5mm, 标称整体封装高度为 0.90 mm ,并带有外露散热板以增强散热功能。该封装为无铅封装(后缀–T),采用 100% 雾锡电镀引脚框。

图1.HR4988

如图1所示为HR4988步进电机驱动模块非A4988,是一款国产pin to pin完全可替换A4988的驱动芯片。至于为什么选择HR4988而不用A4988?也就是懵懂无知,买了一款绿色的驱动模块,在学习过程中弄坏了,就想试试红色的。但二者的配置过程一致。

图2.4988模块和控制板

二、主要引脚介绍

图3.A4988典型应用示意图

图4.HR4988典型应用示意图

VMOT – 电机电源正极(A4988可用电源电压为8V ~ 35V,HR4988为8~32V)注:此引脚用于连接为电机供电的电源

GND – 电机电源接地

2B, 2A – 电机绕组2控制引脚

1A, 1B – 电机绕组1控制引脚

VDD – 逻辑电源正极(3 – 5.5伏)注:此引脚用于为4988电机驱动板供电

GND – 逻辑电源接地

ENABLE – 使能引脚(低电平有效)

引脚说明:此当此引脚为低电平时,A4988才能进行电机驱动工作,当该引脚为高电平,A4988将不会进行电机驱动工作。如果该引脚悬空,则A4988默认为使能状态。即该引脚没有连接任何电平时,A4988可以正常工作。

MS1, MS2, MS3 – 驱动模式引脚

引脚说明:这三个引脚控制A4988微步细分驱动模式。通过这三个引脚的逻辑电平,我们可以调整A4988驱动电机模式为全、半、1/4、1/8 及 1/16 步进模式。Table1图中右侧的表格里有具体如何调节这三个引脚电频以及A4988在不同的电平组合下的驱动模式。表格中“L”代表低电平,“H”代表高电平。“FULL”为全步进,“HALF”为半步进,“QUATER”为1/4步进, “EIGHTH”为1/8步进, “SIXTEENTH”为1/16步进。当MS1, MS2, MS3这几个引脚悬空时,A4988默认为全步进电机驱动模式。

RESET – 复位引脚

引脚说明:该引脚为低电平有效,即当该引脚为低电平时,A4988将复位。如果该引脚悬空,则A4988默认为高电平。即该引脚没有连接任何电平时,A4988可以正常工作。

SLEEP – 睡眠引脚

引脚说明:当该引脚连接电平为低电平时,A4988将进入低能耗睡眠状态,即消耗最小的电能。如果无需使用SLeep功能,则可以将SLEEP引脚与RESET引脚连接,则A4988将持续保持正常能耗状态而不会进入低能耗状态。

STEP – 步进引脚

引脚说明:此引脚用于通过Arduino等微控制器向A4988发送脉冲控制信号,A4988接收到此信号后,会根据 MS1, MS2 和 MS3引脚控制电机运转。占空比50%。

DIR – 方向引脚

引脚说明:通过此引脚可以调整A4988控制电机运行方向。当此引脚为低电平,A4988将控制电机顺时针旋转。高电平则逆时针旋转。

驱动模式:

本次使用的二相步进电机步距角为1.8°,在全步模式下,给STEP一个脉冲信号,电机转动1.8°。半步模式下,一个脉冲信号电机的转动角度为1.8°/2,依次类推。

三、实验器材

  • stm32f103c8t6最小系统;

  • ST-Link烧录器;

  • 4988步进电机驱动模块;

  • 步进电机驱动控制板;

  • 42步进电机;

  • 12V开关电源;

  • 导线若干;

开发环境:Keil_5

图5.实验器材

硬件连接模拟图:

图6.硬件连接模拟图

如图6为MCU与驱动器和电机的连接示意图,可在面包板上完成接线,外部驱动电源(8-35V)需要在电源正极与负极之间接一个47uF电容保护驱动器。而我在选择电源的时候选择了24V2A开关电源,接线的时候需要讲开关电源的负极与开发板GND连接(共地)。也许是正负极接反了(猜测),导致上电后烧坏驱动器和开发板。

后续使用的驱动芯片为HR4988,将面包板换为控制板(如下图7),控制板的电压要求为9V,在理论上可以接8~32V,但问客服就说只能接9V。最后使用12V1A开关电源给驱动IC供电。

图7.4988控制板

控制板两个GND相连,SLEEP引脚也与RESET引脚连接,MS1、MS2、MS3采用拨动开关设置步进模式。使用控制板的好处就是只需将ENABLE、STEP 、DIR与MCU的IO相连,可节约IO资源(相较于面包板)。

四、程序设计

引脚连接:

ENABLE

PB3

DIR

PB5

STEP

PB8

控制板5V接开发板5V引脚,控制板GND接开关电源负极与开发板GND,控制板9V接12V开关电源正极。

初始化配置

将PB8设置为复用推挽输出模式,使用定时器4通道3输出PWM,控制电机运转。

  1. void MOTOR_Init(void)
  2. {
  3. //1.引脚初始化
  4. GPIO_InitTypeDef motor_gpio_init;
  5. motor_gpio_init.GPIO_Pin = GPIO_Pin_5| GPIO_Pin_3;
  6. motor_gpio_init.GPIO_Mode = GPIO_Mode_Out_PP;
  7. motor_gpio_init.GPIO_Speed = GPIO_Speed_50MHz;
  8. GPIO_Init(GPIOB,&motor_gpio_init); //DIR ENABLE 通用推挽输出模式
  9. motor_gpio_init.GPIO_Pin = GPIO_Pin_8;
  10. motor_gpio_init.GPIO_Mode = GPIO_Mode_AF_PP;
  11. GPIO_Init(GPIOB,&motor_gpio_init); //STEP 复用推挽输出
  12. //2.定时器初始化配置
  13. TIM_DeInit(TIM4);
  14. TIM_TimeBaseInitTypeDef motor_TimeInit;
  15. motor_TimeInit.TIM_Prescaler = 31; //预分频值
  16. motor_TimeInit.TIM_ClockDivision = TIM_CKD_DIV1;
  17. motor_TimeInit.TIM_CounterMode = TIM_CounterMode_Up;//向上计数模式
  18. motor_TimeInit.TIM_Period = 3999; //重装载值
  19. motor_TimeInit.TIM_RepetitionCounter = 0; //重复计数值
  20. TIM_TimeBaseInit(TIM4,&motor_TimeInit); //基本计数模式
  21. TIM_ITConfig(TIM4,TIM_IT_Update,ENABLE); //配置更新中断
  22. TIM_ClearFlag(TIM4,TIM_FLAG_Update); //清除更新中断位
  23. TIM_OCInitTypeDef motor_OCInit;
  24. TIM_OCStructInit(&motor_OCInit);
  25. motor_OCInit.TIM_OCMode = TIM_OCMode_PWM1;
  26. motor_OCInit.TIM_OutputState = TIM_OutputState_Enable;
  27. motor_OCInit.TIM_Pulse = 1499;
  28. TIM_OC3Init(TIM4,&motor_OCInit); //配置通道3PWM
  29. //失能motor
  30. MOTOR_STOP();
  31. MOTOR_ENABLE;
  32. MOTOR_DIR_CW; //顺时针
  33. //MOTOR_DIR_CCW;
  34. }

控制电机转动速度就是控制脉冲周期,一个脉冲电机步进一定角度,修改脉冲周期便可控制电机转动速度。

#define SET_MOTOR_SPEED(x) TIM_SetAutoreload(TIM4,x-1);TIM_SetCompare3(TIM2,x/2-1)

中断服务函数

通过拨动步进开关,设置为1/16步进模式。实现电机正转一圈反转一圈反复循环。

  1. void TIM4_IRQHandler(void)
  2. {
  3. if(RESET != TIM_GetITStatus(TIM4,TIM_IT_Update))
  4. {
  5. TIM_ClearITPendingBit(TIM4,TIM_IT_Update); //清除中断标志位
  6. motor_sleep++;
  7. if(motor_sleep==3200) //改变转向
  8. {
  9. MOTOR_STOP(); //停止运转
  10. motor_sleep=0;
  11. motor_dir=!motor_dir;
  12. if(motor_dir) //顺时针
  13. {
  14. MOTOR_DIR_CW;
  15. MOTOR_START();
  16. }
  17. else
  18. {
  19. MOTOR_DIR_CCW;
  20. MOTOR_START();
  21. }
  22. }
  23. }
  24. }

五、演示

4988驱动步进电机

六、程序代码

mian.c

  1. #include "motor.h"
  2. #include "systick.h"
  3. void clock_config(void);
  4. void nvic_config(void);
  5. int main(void)
  6. {
  7. clock_config(); //配置RCC时钟
  8. nvic_config(); //配置中断优先级
  9. MOTOR_Init(); //motor初始化
  10. MOTOR_START();
  11. while(1)
  12. {
  13. }
  14. }
  15. void clock_config(void)
  16. {
  17. RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB,ENABLE);
  18. RCC_APB2PeriphClockCmd(RCC_APB2Periph_AFIO,ENABLE);
  19. RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM4,ENABLE);
  20. }
  21. void NVIC_IQR_Confing(uint8_t nvic_IRQChannel,uint8_t nvic_PreemptionPriority, uint8_t nvic_SubPriority)
  22. {
  23. NVIC_InitTypeDef nvic_Init;
  24. nvic_Init.NVIC_IRQChannel = nvic_IRQChannel;//中断号
  25. nvic_Init.NVIC_IRQChannelPreemptionPriority = nvic_PreemptionPriority;//抢占优先级
  26. nvic_Init.NVIC_IRQChannelSubPriority = nvic_SubPriority;//子优先级
  27. nvic_Init.NVIC_IRQChannelCmd = ENABLE;//启用中断优先级
  28. NVIC_Init(&nvic_Init);
  29. }
  30. void nvic_config(void)
  31. {
  32. NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);//配置中断优先级分组
  33. NVIC_IQR_Confing(TIM4_IRQn,1,0);
  34. }

motor.c

  1. #include "motor.h"
  2. /*
  3. 引脚连接:
  4. PB5 - DIR 方向引脚
  5. PB8 - STEP TIM4_CH3 步进引脚
  6. PB3 - ENABLE 使能引脚
  7. */
  8. uint16_t motor_sleep;
  9. uint8_t motor_dir=1;
  10. void MOTOR_Init(void)
  11. {
  12. //1.引脚初始化
  13. GPIO_InitTypeDef motor_gpio_init;
  14. motor_gpio_init.GPIO_Pin = GPIO_Pin_5| GPIO_Pin_3;
  15. motor_gpio_init.GPIO_Mode = GPIO_Mode_Out_PP;
  16. motor_gpio_init.GPIO_Speed = GPIO_Speed_50MHz;
  17. GPIO_Init(GPIOB,&motor_gpio_init); //DIR ENABLE 通用推挽输出模式
  18. motor_gpio_init.GPIO_Pin = GPIO_Pin_8;
  19. motor_gpio_init.GPIO_Mode = GPIO_Mode_AF_PP;
  20. GPIO_Init(GPIOB,&motor_gpio_init); //STEP 复用推挽输出
  21. //2.定时器初始化配置
  22. TIM_DeInit(TIM4);
  23. TIM_TimeBaseInitTypeDef motor_TimeInit;
  24. motor_TimeInit.TIM_Prescaler = 31; //预分频值
  25. motor_TimeInit.TIM_ClockDivision = TIM_CKD_DIV1;
  26. motor_TimeInit.TIM_CounterMode = TIM_CounterMode_Up;//向上计数模式
  27. motor_TimeInit.TIM_Period = 3999; //重装载值
  28. motor_TimeInit.TIM_RepetitionCounter = 0; //重复计数值
  29. TIM_TimeBaseInit(TIM4,&motor_TimeInit); //基本计数模式
  30. TIM_ITConfig(TIM4,TIM_IT_Update,ENABLE); //配置更新中断
  31. TIM_ClearFlag(TIM4,TIM_FLAG_Update); //清除更新中断位
  32. TIM_OCInitTypeDef motor_OCInit;
  33. TIM_OCStructInit(&motor_OCInit);
  34. motor_OCInit.TIM_OCMode = TIM_OCMode_PWM1;
  35. motor_OCInit.TIM_OutputState = TIM_OutputState_Enable;
  36. motor_OCInit.TIM_Pulse = 1499;
  37. TIM_OC3Init(TIM4,&motor_OCInit); //配置通道3PWM
  38. //失能motor
  39. MOTOR_STOP();
  40. MOTOR_ENABLE;
  41. MOTOR_DIR_CW; //顺时针
  42. //MOTOR_DIR_CCW;
  43. }
  44. //中断服务函数
  45. void TIM4_IRQHandler(void)
  46. {
  47. if(RESET != TIM_GetITStatus(TIM4,TIM_IT_Update))
  48. {
  49. TIM_ClearITPendingBit(TIM4,TIM_IT_Update); //清除中断标志位
  50. motor_sleep++;
  51. if(motor_sleep==3200) //改变转向
  52. {
  53. MOTOR_STOP(); //停止运转
  54. motor_sleep=0;
  55. motor_dir=!motor_dir;
  56. if(motor_dir) //顺时针
  57. {
  58. MOTOR_DIR_CW;
  59. MOTOR_START();
  60. }
  61. else
  62. {
  63. MOTOR_DIR_CCW;
  64. MOTOR_START();
  65. }
  66. }
  67. }
  68. }
  69. /*
  70. \brief: 设置motor运转速度
  71. \param: speed: 速度
  72. 速度=(1/2M) * speed * (1.8/16) 度/秒
  73. \retval: none
  74. */
  75. void MOTOR_Set_Speed(uint16_t speed)
  76. {
  77. TIM_SetAutoreload(TIM4,speed-1);
  78. TIM_SetCompare3(TIM4,speed/2-1);
  79. }
  80. //设置步进模式 (用于MS1、MS2、MS3与单片机IO连接时使用)
  81. #if 0
  82. void setMotorStepMod(uint8_t stepMod)
  83. {
  84. switch(StepMode)
  85. {
  86. case MOTOR_FULL_STEP:
  87. MOTOR_MS1(0);
  88. MOTOR_MS2(0);
  89. MOTOR_MS3(0);
  90. break;
  91. case MOTOR_HALF_STEP:
  92. MOTOR_MS1(1);
  93. MOTOR_MS2(0);
  94. MOTOR_MS3(0);
  95. break;
  96. case MOTOR_QUARTER_STEP:
  97. MOTOR_MS1(0);
  98. MOTOR_MS2(1);
  99. MOTOR_MS3(0);
  100. break;
  101. case MOTOR_EIGHTH_STEP:
  102. MOTOR_MS1(1);
  103. MOTOR_MS2(1);
  104. MOTOR_MS3(0);
  105. break;
  106. case MOTOR_SIXTEENTH_STEP:
  107. MOTOR_MS1(1);
  108. MOTOR_MS2(1);
  109. MOTOR_MS3(1);
  110. break;
  111. default :
  112. MOTOR_MS1(1);
  113. MOTOR_MS2(1);
  114. MOTOR_MS3(1);
  115. break;
  116. }
  117. }
  118. #endif

motor.h

  1. #ifndef _MOTOR_H_
  2. #define _MOTOR_H_
  3. #include "stm32f10x.h"
  4. #define MOTOR_DIR_CW GPIO_ResetBits(GPIOB,GPIO_Pin_5) //顺时针
  5. #define MOTOR_DIR_CCW GPIO_SetBits(GPIOB,GPIO_Pin_5) //逆时针
  6. #define MOTOR_ENABLE GPIO_ResetBits(GPIOB,GPIO_Pin_3)
  7. #define MOTOR_STOP() TIM_Cmd(TIM4,DISABLE) //停止转动
  8. #define MOTOR_START() TIM_Cmd(TIM4,ENABLE) //开始转动
  9. #define MOTOR_FULL_STEP 0 //满步
  10. #define MOTOR_HALF_STEP 1 //二分之一步
  11. #define MOTOR_QUARTER_STEP 2 //四分之一步
  12. #define MOTOR_EIGHTH_STEP 3 //八分之一步
  13. #define MOTOR_SIXTEENTH_STEP 4 //十六分之一步
  14. extern uint8_t motor_dir;
  15. void MOTOR_Init(void);
  16. void MOTOR_Set_Speed(uint16_t speed);
  17. #endif
本文内容由网友自发贡献,转载请注明出处:https://www.wpsshop.cn/w/小小林熬夜学编程/article/detail/351455
推荐阅读
相关标签
  

闽ICP备14008679号