[STM32L1] STM32F103步进电机驱动简单控制

[复制链接]
1847|19
 楼主| 狗啃模拟 发表于 2022-8-22 23:29 | 显示全部楼层 |阅读模式
stm32f103步进电机驱动控制 一、硬件接线方法 本文采用混合式二相步进电机-J8HB2401-野火42步进电机,驱动器为野火EBF-MSD4805,下面是常用接线方式:
934706303a07c335a9.png



评论

———————————————— 版权声明:本文为CSDN博主「小昌快跑」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。 原文链接:https://blog.csdn.net/weixin_44690557/article/details/125607583  发表于 2022-8-22 23:49
 楼主| 狗啃模拟 发表于 2022-8-22 23:30 | 显示全部楼层
 楼主| 狗啃模拟 发表于 2022-8-22 23:32 | 显示全部楼层
如上图所示通常采用共阴接线方式,具体接线按照自己需求进行完成。
另外二相电机步距角1.8°,步进驱动器侧面SW1-SW8分别为细分设定、电流设定和驱动模式选择开关。
 楼主| 狗啃模拟 发表于 2022-8-22 23:40 | 显示全部楼层
 楼主| 狗啃模拟 发表于 2022-8-22 23:44 | 显示全部楼层
二、程序

    代码如下(示例):

1.main.c
  1. #include "stm32f10x.h"
  2. #include "tim.h"
  3. #include "bsp_usart.h"
  4. #include "stm32f10x_it.h"
  5. #include "stm32f10x_conf.h"
  6. /* 共阴 */
  7. /*
  8. PUL+->PA.1  PUL-接GND
  9. DIR+->PA.8  DIR-接GND
  10. */
  11. extern int motor_dir2;
  12. extern int motor_speed;

  13. int main(void)
  14. {       
  15.         NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);        
  16.         USART_Config();                    
  17.         Driver_Init();/* GPIO_Init */                       
  18.         TIM2_Init(999,72-1);      
  19.         while(1)
  20.         {       
  21.                 Locate_Rle2(motor_speed,motor_dir2);/* 变速以及方向 */       
  22.         }
  23. }
 楼主| 狗啃模拟 发表于 2022-8-22 23:46 | 显示全部楼层
2.tim.c
  1. #include "tim.h"
  2. #include "bsp_usart.h"

  3. long current_pos[2] = {0,0};
  4. int motor_dir2 = 0;
  5. u8 count[2] = {0,0};
  6. u32 motor_speed = 0;
  7. u8 rcr_remainder;     //重复计数余数部分
  8. u8 is_rcr_finish = 1;   //重复计数器是否设置完成
  9. long rcr_integer;          //重复计数整数部分
  10. long target_pos = 0;    //有符号方向

  11. /* 电机转速与方向函数 */
  12. void Locate_Rle2(u32 frequency,DIR_Type dir)
  13. {
  14.         motor_dir2 = dir;
  15.        
  16.         DRIVER_DIR = motor_dir2;
  17.         if(motor_dir2 == CW){motor_speed = 3000;}/* motor_dir2高电平时转速3000Hz */
  18.         else if(motor_dir2 == CCW){motor_speed = 10000;}/* motor_dir2低电平时转速10000Hz */
  19.         frequency = motor_speed;
  20.        
  21.         if(TIM2->CR1&0x01)
  22.         {
  23.                 printf("\r\nThe last time pulses is not send finished,wait please!\r\n");
  24.                 return;
  25.         }
  26.         if((frequency<20)||(frequency>100000))
  27.         {
  28.                 printf("\r\nThe frequency is out of range! please reset it!!(range:20Hz~100KHz)\r\n");
  29.                 return;
  30.         }
  31.         TIM2_Startup(frequency);                       
  32. }

  33. /* PA8 - DIR+*/
  34. void Driver_Init(void)
  35. {
  36.         GPIO_InitTypeDef  GPIO_InitStructure;
  37.         RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);       
  38.         GPIO_InitStructure.GPIO_Pin = GPIO_Pin_8;       
  39.         GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;                
  40.         GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;       
  41.         GPIO_Init(GPIOA, &GPIO_InitStructure);                                       
  42.         GPIO_SetBits(GPIOA,GPIO_Pin_8);                                               
  43. }

  44. /* PA1- PUL+ TIM2_CH2 */
  45. void TIM2_Init(u16 arr,u16 psc)
  46. {                                                          
  47.         GPIO_InitTypeDef GPIO_InitStructure;
  48.         TIM_TimeBaseInitTypeDef  TIM_TimeBaseStructure;
  49.         TIM_OCInitTypeDef  TIM_OCInitStructure;
  50.         NVIC_InitTypeDef NVIC_InitStructure;

  51.         RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2, ENABLE);        
  52.         RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA , ENABLE);                                                                             

  53.         GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1;                               
  54.         GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;                       
  55.         GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  56.         GPIO_Init(GPIOA, &GPIO_InitStructure);
  57.        
  58.         TIM_TimeBaseStructInit(&TIM_TimeBaseStructure);
  59.        
  60.         TIM_TimeBaseStructure.TIM_Period = arr;                                         
  61.         TIM_TimeBaseStructure.TIM_Prescaler =psc;                               
  62.         TIM_TimeBaseStructure.TIM_ClockDivision = 0;                        
  63.         TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;
  64.         TIM_TimeBaseInit(TIM2, &TIM_TimeBaseStructure);                
  65.         TIM_ClearITPendingBit(TIM2,TIM_IT_Update);                               

  66.         TIM_UpdateRequestConfig(TIM2,TIM_UpdateSource_Regular);
  67.         TIM_SelectOnePulseMode(TIM2,TIM_OPMode_Single);                       
  68.         TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM2;               
  69.         TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable;
  70.         TIM_OCInitStructure.TIM_Pulse = arr>>1;                                
  71.         TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_High;
  72.         TIM_OC2Init(TIM2, &TIM_OCInitStructure);                                

  73.         TIM_OC2PreloadConfig(TIM2, TIM_OCPreload_Enable);                 
  74.         TIM_ARRPreloadConfig(TIM2, ENABLE);                                        
  75.        
  76.         TIM_ITConfig(TIM2, TIM_IT_Update ,ENABLE);                         

  77.         NVIC_InitStructure.NVIC_IRQChannel = TIM2_IRQn;                 
  78.         NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;  
  79.         NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;                
  80.         NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;        
  81.         NVIC_Init(&NVIC_InitStructure);                                                 
  82.        
  83.         TIM_ClearITPendingBit(TIM2, TIM_IT_Update);                         
  84.         TIM_Cmd(TIM2, DISABLE);                                                                                                                            
  85. }


  86. void TIM2_Startup(u32 frequency)   
  87. {
  88.         u16 temp_arr=1000000/frequency-1;
  89.         TIM_SetAutoreload(TIM2,temp_arr);
  90.         TIM_SetCompare2(TIM2,temp_arr>>1);
  91.        
  92.         TIM_SetCounter(TIM2,0);
  93.         TIM_Cmd(TIM2, ENABLE);  
  94. }
 楼主| 狗啃模拟 发表于 2022-8-22 23:47 | 显示全部楼层
3.tim.h
  1. #include "stm32f10x.h"
  2. #include "stdlib.h"

  3. #define BITBAND(addr, bitnum) ((addr & 0xF0000000)+0x02000000+((addr & 0x00FFFFFF)<<5)+(bitnum<<2))
  4. #define MEM_ADDR(addr)  *((volatile unsigned long  *)(addr))
  5. #define BIT_ADDR(addr, bitnum)   MEM_ADDR(BITBAND(addr, bitnum))
  6. #define PAout(n)          BIT_ADDR(GPIOA_ODR_Addr,n)
  7. #define GPIOA_ODR_Addr    (GPIOA_BASE+12)
  8. #define DRIVER_DIR        PAout(8)

  9. typedef enum
  10. {
  11.         CW = 1,//高电平顺时针
  12.         CCW = 0,//低电平逆时针
  13. }DIR_Type;

  14. void Driver_Init(void);
  15. void TIM2_Init(u16 arr,u16 psc);
  16. void TIM2_Startup(u32 frequency);
  17. void Locate_Rle2(u32 frequency,DIR_Type dir);
 楼主| 狗啃模拟 发表于 2022-8-22 23:48 | 显示全部楼层
4.stm32f10x_it.c
  1. /* Includes ------------------------------------------------------------------*/
  2. #include "stm32f10x_it.h"
  3. #include "tim.h"
  4. #include "bsp_usart.h"
  5. #include "systick.h"

  6. extern u8 count[2];
  7. extern long current_pos[2]; //有符号方向
  8. extern int motor_dir2;
  9. int counter = 0;

  10. /******* TIM2 - 1ms*********/
  11. void TIM2_IRQHandler(void)
  12. {
  13.         if(TIM_GetITStatus(TIM2,TIM_FLAG_Update)!=RESET)
  14.         {
  15.                 TIM_ClearITPendingBit(TIM2,TIM_FLAG_Update);       
  16.                 count[0]++;
  17.                 counter ++;
  18.                 TIM_GenerateEvent(TIM2,TIM_EventSource_Update);
  19.        
  20.                 TIM_Cmd(TIM2, ENABLE);                                            

  21.                 if(count[0]==200)
  22.                 {
  23.                         if(motor_dir2==CW)                                                   
  24.                                 current_pos[0]+=count[0];
  25.                         else                                                                    
  26.                                 current_pos[0]-=count[0];
  27.                         TIM_Cmd(TIM2, DISABLE);                                                   
  28.                         printf("\r\n motor2=%ld\r\n",current_pos[0]);
  29.                         count[0]=0;
  30.                 }
  31.                
  32.                 if(counter <= 5000){motor_dir2=CW;}/* 定时器前5s拉高motor_dir2电平 */
  33.                 else if((counter > 5000) && (counter <= 10000)){motor_dir2=CCW;}/* 拉低电平 */
  34.                 else if(counter > 10000){counter=0;}/* 复位 */
  35.         }
  36. }
 楼主| 狗啃模拟 发表于 2022-8-22 23:49 | 显示全部楼层
a.中断函数可以从保准库中进行拷贝,具体采用哪一个串口自己需要对照原理图;
b.PAout(n)函数采用GPIO-位带操作, 把“位带地址+ 位序号”转换成别名地址宏统一公式如下,详见野火指南:
#define BITBAND(addr, bitnum) ((addr & 0xF0000000)+0x02000000+((addr & 0x00FFFFFF)<<5)+(bitnum<<2))
童雨竹 发表于 2022-9-29 19:24 | 显示全部楼层

需要设定一个阈值来对像素点进行设置
Uriah 发表于 2022-9-30 11:30 | 显示全部楼层

一种了解状态变化的简单方法
Bblythe 发表于 2022-9-30 14:29 | 显示全部楼层

单片机一般都有内部程序区和数据区
Wordsworth 发表于 2022-12-25 07:28 | 显示全部楼层

代码量小的时候用来做条件判断
万图 发表于 2022-12-25 10:27 | 显示全部楼层

但是如果不懂程序设计的思想的话,会给你做项目的过程中带来很多很多的困惑。
帛灿灿 发表于 2022-12-25 13:26 | 显示全部楼层

STM32CUBEMX配置生成初始化代码
Bblythe 发表于 2022-12-25 14:29 | 显示全部楼层

单片机一般都有内部程序区和数据区
周半梅 发表于 2022-12-25 16:25 | 显示全部楼层

光绘出零件间联机的PCB设计布线
Pulitzer 发表于 2022-12-25 17:28 | 显示全部楼层

只要内存占用量不超过 256.0 就可以用 small 模式编译
huangcunxiake 发表于 2022-12-25 17:45 | 显示全部楼层
这是步进电机吗
童雨竹 发表于 2022-12-25 19:24 | 显示全部楼层

二值化就是让图像的像素点矩阵中的每个像素点的灰度值为0(黑色)或者255(白色
您需要登录后才可以回帖 登录 | 注册

本版积分规则

68

主题

1054

帖子

2

粉丝
快速回复 在线客服 返回列表 返回顶部