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

利用STM32F103精确控制步进电机

[复制链接]
STMCU-管管 发布时间:2021-6-25 13:30
利用STM32F103精确控制步进电机
一、用到的元器件
STM32F103C8T6
42步进电机(42SHDC4040-17B)
TB6600电机驱动器升级版


二、42步进电机
1.步进电机的基本知识
(1)拍数——每一次循环所包含的通电状态数(电机转过一个齿距角所需脉冲数)
(2)单拍制分配方式——状态数=相数
(3)双拍制分配方式——状态数=相数的两倍
(4)步距角 ——步进机通过一个电脉冲转子转过的角度
11.png
N:一个周期的运行拍数
Zr:转子齿数
拍数:N=km
m:相数
k=1单拍制
k=2双拍制
(5)转速
12.png
(6)角度细分的原理


电磁力的大小与绕组通电电流的大小有关。
当通电相的电流并不马上升到位,而断电相的电流也非立即降为0时,它们所产生的磁场合力,会使转子有一个新的平衡位置,这个新的平衡位置是在原来的步距角范围内。
2. 42步进电机参数
无法查找到42SHDC4040-17B型号的详细资料,以通用42步进电机为例:
步距角 1.8°
步距角精度 ±5%
相数 2相
励磁方式 混合式
转子齿数 50
拍制 双拍制
其他参数:无
由步距角=1.8°推算出转子齿数为50,拍制为双拍制


3. 42步进电机速度与角度控制
电机的转速与脉冲频率成正比,电机转过的角度与脉冲数成正比。所以控制脉冲数和脉冲频率就可以精确调速。
理论上步进电机转速 = 频率 * 60 /((360/T)*x)
21.png
​       
转速单位: 转/ 分
频率单位:赫兹
x 细分倍数
T 步距角
例如,在本实验中,32细分;频率72000 赫兹;步距角1.8°;套用公式72000 ∗ 60 ( ( 360 / 1.8 ) ∗ 32 ) = 112.5 \frac{72000*60}{((360/1.8)*32)}=112.5
((360/1.8)∗32)
72000∗60
​       
=112.5rad/ min,即1.875 rad/s.
13.png
三、TB6600电机驱动器升级版参数
TB6600步进电机驱动器升级版是一款专业的两相步进电机驱动,可实现正反转控制。通过S1,S2,S3 3位拨码开关选择7档细分控制(1,2/A,2/B,4,8,16,32,),通过S4,S5,S6 3位拨码开关选择8 档电流控制(0.5A,1A,1.5A,2A,2.5A,2.8A,3.0A,3.5A)。适合驱动57,42 型两相、四相混合式步进电机。


1.信号输入端
PUL+:脉冲信号输入正。( CP+ )
PUL-:脉冲信号输入负。( CP- )
DIR+:电机正、反转控制正。
DIR-:电机正、反转控制负。
EN+:电机脱机控制正。
EN-:电机脱机控制负。


共阳极接法:分别将PUL+,DIR+,EN+连接到控制系统的电源上, 如果此电源是+5V则可直接接入,如果此电源大于+5V,则须外部另加限流电阻R,保证给驱动器内部光藕提供8—15mA 的驱动电流。
共阴极接法:分别将 PUL-,DIR-,EN-连接到控制系统的地端;脉冲输入信号通过PUL+接入,方向信号通过DIR+接入,使能信号通过EN+接入。若需限流电阻,限流电阻R的接法取值与共阳极接法相同。
注:EN端可不接,EN有效时电机转子处于自由状态(脱机状态),这时可以手动转动电机转轴。
2.电机绕组连接
A+:连接电机绕组A+相。
A-:连接电机绕组A-相。
B+:连接电机绕组B+相。
B-:连接电机绕组B-相。


3.电源电压连接
VCC:电源正端“+”
GND:电源负端“-”
注意:DC直流范围:9-32V。不可以超过此范围,否则会无法正常工作甚至损坏驱动器.


4.拨码开关
14.png
电流大小设定
15.png
三、STM32F103
------说明:引脚部分在文章末尾有解释--------


1.引脚连接
A 0——PUL+
A 3——KEY1——V3
A12——DIR+
A11——EAN+
GND——EAN- ——KEY0


2.引脚功能
A0控制电机转速
A3控制按键
A9
A11
A11控制电机是否为锁死状态
A12控制电机正反转


3.定时器
1.本实验利用定时器TIM2和定时器TIM3构造一个主从定时器,TIM2作为主定时器控制电机的转速,TIM3作为从定时器控制电机的转动角度。
2.电机的转速和转角还与驱动器自身的细分数有关,但是驱动器细分数是通过影响电机的步距角来影响转速和转角,而TIM2和TIM3是控制步进电机的频率和脉冲数来控制转速转角
3.电机的转速和角度与定时器的关系(在不考虑电机自身的细分数下)


设TIM2的定时周期(即重装值)为nPDTemp2,预分频值为OCPolarity2
 TIM3的定时周期(即重装值)为nPDTemp3,预分频值为OCPolarity3,
则单片机产生一个脉冲所需要的时间为:
23.png
​       
  本实验中设TIM2的定时周期nPDTemp2=72000/5000-1,预分频值OCPolarity2=999,TIM3的定时周期nPDTemp3=6399,预分频值OCPolarity3为0。即
22.png
  定时器共产生nPDTemp3+1=6400个脉冲,电机转过的角度为6400*1.8°=11520°,即电机转了32圈。
  转动速度
24.png


4.在32细分的情况下电机1rad/s和转1°需要的重装值为nPDTemp2=11.25,nPDTemp3=17.7778。


四、程序实现
1.main.c程序
  1. #include "main.h"
  2. #include "sys.h"
  3. #include "usart1.h"
  4. #include "delay.h"
  5. #include "math.h"

  6. u16 t;  
  7. u16 len;                    //接收到的数据长度
  8. u16 times=0;
  9. char receive_data[60];        //接收到的字符
  10. char state;                                //电机正反转标志位0正转,1反转
  11. int speed,angle;                //旋转速度,角度
  12. int a=0;                        //判断是否接收到数据
  13. int r,data=0;                          //用于数据转换
  14. int type;                                //转换后的数据   
  15. extern u16 USART_RX_STA;

  16. /**************************
  17. * 函数名:delay
  18. * 描述  :延时函数
  19. * 输入  :无
  20. * 输出  :无
  21. * 返回值:无
  22. ****************************/
  23. void delay()//延时
  24. {
  25.          int i,j;
  26.          for(i=0;i<2000;i++)
  27.          for(j=0;j<1000;j++);
  28. }

  29. /**************************
  30. * 函数名:Waiting_receptio
  31. * 描述  :等待串口接收数据
  32. * 输入  :无
  33. * 输出  :无
  34. * 返回值:无
  35. ****************************/
  36. void Waiting_reception(void)
  37. {
  38.         while(a==0)//等待数据发送完成
  39.         {
  40.                 delay_ms(100);
  41.                 if(a==1)//接收到数据
  42.                 {
  43.                         if(USART_RX_STA&0x8000)//(串口接收用到了正点原子的例程)
  44.                         {
  45.                                 len=USART_RX_STA&0x3fff;//得到此次接收到的数据长度
  46.                                 r=len;                               
  47.                                 type=0;
  48.                                 for(t=0;t<len;t++)//拷贝数据,将字符转换为十进制数
  49.                                 {
  50.                                         receive_data[t]=USART_RX_BUF[t];
  51.                                         data=(int)receive_data[t]-48;
  52.                                         r=r-1;
  53.                                         type=type+data*(pow(10,r));
  54.                                 }
  55.                                 USART_RX_STA=0;
  56.                                 a=0;
  57.                                 delay_ms(500);
  58.                                 break;
  59.                         }
  60.                 }
  61.         }
  62. }
  63. /**************************
  64. * 函数名:KeyStates
  65. * 描述  :监测按键状态
  66. * 输入  :无
  67. * 输出  :无
  68. * 返回值:0/1
  69. ****************************/
  70. u8 KeyStates()//按键状态
  71. {
  72.         static u8 i = 0;
  73.         if(GPIO_ReadInputDataBit(GPIOA, GPIO_Pin_3)==0)
  74.         {
  75.                 delay();
  76.                 if(GPIO_ReadInputDataBit(GPIOA, GPIO_Pin_3)==0)
  77.                         while(GPIO_ReadInputDataBit(GPIOA, GPIO_Pin_3)==0);
  78.                         i = ~i;
  79.         }
  80.         return i;
  81. }
  82. /****************
  83. * 函数名:main
  84. * 描述  :主函数
  85. * 输入  :无
  86. * 输出  :无
  87. ******************/
  88. int main(void)
  89. {
  90.         NVIC_Configuration();                        //中断初始化
  91.         GPIO_Config();                                        //IO口初始化
  92.         USART1_Config();                                //串口初始化
  93.     delay_init();                                   //延时函数初始化       
  94.         GPIO_ResetBits(GPIOA, GPIO_Pin_11);//A11置零  A11——EAN+
  95.         GPIO_ResetBits(GPIOA, GPIO_Pin_12);//A12置零  A12——DIR+
  96.         while(1)
  97.         {
  98.                 delay_ms(100);
  99.                 Initial_state:                printf("\r\n 请选择正反转,正转输入0,反转输入1 (以新行作为结束标志)\r\n");       
  100.                 Waiting_reception();
  101.                 state=type;//将接收到的数据给type
  102.                 if(type==0)//电机正转
  103.                 {
  104.                         GPIO_SetBits(GPIOA, GPIO_Pin_12);//电机正转
  105.                         printf("\r\n 电机为正转模式,请输入旋转速度(rad/s),输入0返回初始模式 \r\n");
  106.                        
  107. /*********************************************此模块用于配置速度参数********************************************************/

  108.                         part1:Waiting_reception();
  109.                         speed =type;//将接收到的数据给speed
  110.                         if(speed==0)goto Initial_state;//如果是0则返回初始模式
  111.                                 else{
  112.                                         if(speed>=15)
  113.                                         {
  114.                                                 printf("\r\n 旋转速度>15rad/s,请重新选择旋转速度。\r\n");
  115.                                                 goto part1;
  116.                                         }
  117.                                                 else printf("\r\n 电机旋转速度为%d rad/s,请输入旋转角度,输入0返回初始模式 \r\n",speed);
  118.                                 }
  119.        
  120. /*********************************************此模块用于配置角度参数********************************************************/               

  121.                         Waiting_reception();
  122.                         angle =type;//将接收到的数据给type
  123.                         for(;;)
  124.                         {
  125.                                 if(angle>0)//接收到的数据不是0
  126.                                 {
  127.                                         TIM2_Master__TIM3_Slave_Configuration(speed,angle);        // 配置TIM2和TIM3的重装值 ,改变电机转动速度和角度
  128.                                         delay_ms(20000);//电机保护
  129.                                         printf("\r\n 电机已旋转%d °,请输入下一次旋转角度,输入0返回初始模式; \r\n",angle);
  130.                                         angle=0;
  131.                                         Waiting_reception();               
  132.                                         angle =type;                                       
  133.                                 }else{
  134.                                         if(angle==0)goto Initial_state;//返回初始状态       
  135.                                         else {
  136.                                                 printf("\r\n 角度错误,已返回初始模式 \r\n");               
  137.                                                 goto Initial_state;
  138.                                         }                                               
  139.                                 }
  140.                         }
  141.                 }               
  142. /*********************************************反转模式********************************************************/               
  143.                 else{
  144.                         if(type==1)
  145.                         {
  146.                                 GPIO_ResetBits(GPIOA, GPIO_Pin_12);//电机反转
  147.                                 printf("\r\n 电机为正转模式,请输入旋转速度(rad/s),输入0返回初始模式 \r\n");
  148.                                
  149. /*********************************************此模块用于配置速度参数********************************************************/
  150.                                 part2:                                Waiting_reception();
  151.                                 speed =type;//将接收到的数据给speed
  152.                                 if(speed==0)goto Initial_state;//如果是0则返回初始模式
  153.                                         else{
  154.                                                 if(speed>=15)
  155.                                                 {
  156.                                                         printf("\r\n旋转速度>15rad/s,请重新选择旋转速度。\r\n");
  157.                                                         goto part2;
  158.                                                 }else printf("\r\n 电机旋转速度为%d rad/s,请输入旋转角度,输入0返回初始模式  \r\n",speed);
  159.                                         }       
  160. /*********************************************此模块用于配置角度参数********************************************************/               
  161.                                 Waiting_reception();
  162.                                 angle =type;//将接收到的数据给type
  163.                                 for(;;)
  164.                                 {
  165.                                         if(angle>0)//接收到的数据不是0
  166.                                         {
  167.                                                 TIM2_Master__TIM3_Slave_Configuration(speed,angle);        // 配置TIM2和TIM3的重装值 ,改变电机转动速度和角度
  168.                                                 delay_ms(20000);//电机保护
  169.                                                 printf("\r\n 电机已旋转%d °,请输入下一次旋转角度,输入0返回初始模式;\r\n",angle);
  170.                                                 angle=0;
  171.                                                 Waiting_reception();               
  172.                                                 angle =type;                                       
  173.                                         }else{
  174.                                                 if(angle==0)goto Initial_state;//返回初始状态       
  175.                                                 else {
  176.                                                         printf("\r\n 角度错误,已返回初始模式 \r\n");               
  177.                                                         goto Initial_state;
  178.                                                 }                                                               
  179.                                         }
  180.                                 }
  181. /****************************************************************************************************************************/                                       
  182.                 }else{//if(a!=0)&(a!=1)
  183.                         type=NULL;
  184.                         printf("\r\n 输入无效 \r\n");
  185.                         goto Initial_state;//返回初始状态
  186.                         }
  187.                 }
  188.         }
  189. }
复制代码
2.main.h程序
  1. #ifndef _MAIN_H
  2. #define        _MAIN_H
  3. #include <stm32f10x.h>
  4. #include <usart1.h>
  5. #include <misc.h>
  6. #include <nvic.h>
  7. #include <stdio.h>
  8. #include "stm32f10x_tim.h"
  9. #include "timer.h"
  10. #endif
复制代码
3.time.c程序
  1. #include "timer.h"
  2. /**************************
  3. * 函数名:GPIO_Config
  4. * 描述  :无
  5. * 输入  :无
  6. * 输出  :无
  7. * 调用  :主函数
  8. * 返回值:无
  9. ****************************/
  10. void GPIO_Config(void)
  11. {
  12.                 GPIO_InitTypeDef GPIO_InitStructure;
  13.                 RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA , ENABLE); //使能IOA
  14.                 RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2 | RCC_APB1Periph_TIM3, ENABLE); //使能TIM2,TIM3
  15.                
  16.             /* Timer2 Channel 1, PA0 */
  17.             GPIO_InitStructure.GPIO_Pin =  GPIO_Pin_0;
  18.             GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP; //复用输出
  19.             GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  20.             GPIO_Init(GPIOA, &GPIO_InitStructure);
  21.                  
  22.                 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1|GPIO_Pin_2|GPIO_Pin_11|GPIO_Pin_12;
  23.                 GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;         //通用推挽输出模式
  24.                 GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;        //指定GPIO引脚可输出的最高频率为50MHZ
  25.                 GPIO_Init(GPIOA, &GPIO_InitStructure);                   //
  26.                 GPIO_ResetBits(GPIOA, GPIO_Pin_1);//指定引脚输出低电平,此时灯全灭,方向
  27.                 GPIO_ResetBits(GPIOA, GPIO_Pin_2);//指定引脚输出低电平,此时灯全灭        使能
  28.                 GPIO_SetBits(GPIOA, GPIO_Pin_11);//指定引脚输出低电平,此时灯全灭,方向
  29.                 GPIO_SetBits(GPIOA, GPIO_Pin_12);//指定引脚输出低电平,此时灯全灭        使能
  30.        
  31.          
  32.                 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_3;
  33.                 GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;         //通用推挽输出模式
  34.                 GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;        //指定GPIO引脚可输出的最高频率为50MHZ
  35.                 GPIO_Init(GPIOA, &GPIO_InitStructure);                   //
  36.                
  37.                 RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE);
  38.                 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_12;
  39.                 GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;         //通用推挽输出模式
  40.                 GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;        //指定GPIO引脚可输出的最高频率为50MHZ
  41.                 GPIO_Init(GPIOB, &GPIO_InitStructure);                   //
  42.                 GPIO_ResetBits(GPIOB, GPIO_Pin_12);//指定引脚输出低电平,此时灯全灭        使能
  43.                
  44.                 //GPIO_ResetBits  GPIO_SetBits
  45. }

  46. //================================================================================
  47. /**************************
  48. * 函数名:TIM2_Master__TIM3_Slave_Configuration
  49. * 描述  :主从定时器配置
  50. * 输入  :电机转速speed,转角angle
  51. * 输出  :无
  52. * 调用  :主函数
  53. * 返回值:无
  54. ****************************/
  55. void TIM2_Master__TIM3_Slave_Configuration(u32 PulseFrequency, u32 pulse)
  56. {
  57.         TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure;
  58.         TIM_OCInitTypeDef TIM_OCInitStructure;
  59.        
  60.         u16 nPDTemp ;
  61.         u16 pulse_number;
  62.         float p=PulseFrequency;
  63.         TIM_Cmd(TIM2, DISABLE);
  64.         nPDTemp = (11.25/p);                            //TIM2重装值是11.25时1s转一圈(电机32细分下)
  65.         pulse_number = (16.7778*pulse);//TIM3重装值是16.7778时转1°(电机32细分下)
  66.        
  67.         // 时基配置:配置PWM输出定时器——TIM2
  68.         /* Time base configuration */
  69.         TIM_TimeBaseStructure.TIM_Period = nPDTemp; //定时周期为nPDTemp
  70.         TIM_TimeBaseStructure.TIM_Prescaler = 999; //预分频值1000,即f=72khz
  71.         TIM_TimeBaseStructure.TIM_ClockDivision = 0; //时钟分频因子,会影响滤波器采样频率,与本实验无影响
  72.         TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up; //向上计数模式
  73.         TIM_TimeBaseStructure.TIM_RepetitionCounter = 0; //指定重复计数器值
  74.         TIM_TimeBaseInit(TIM2, &TIM_TimeBaseStructure);
  75.        
  76.         // 输出配置:配置PWM输出定时器——TIM2
  77.         /* PWM1 Mode configuration: Channel1 */   
  78.         TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM1; //TIM 脉冲宽度调制模式 1
  79.         TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_High; //高电平有效
  80.         TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable; //使能输出
  81.         TIM_OCInitStructure.TIM_Pulse = nPDTemp>>1;//50% //比较tim_ccr的值,输出脉冲发生跳变
  82.         TIM_OC1Init(TIM2, &TIM_OCInitStructure); //初始化
  83.         TIM_OC1PreloadConfig(TIM2, TIM_OCPreload_Enable); //使能 TIMx 在 CCR1 上的预装载寄存器
  84.         TIM_ARRPreloadConfig(TIM2, ENABLE); //使能或者失能 TIMx 在 ARR 上的预装载寄存器
  85.        
  86.         // 时基配置:配置脉冲计数寄存器——TIM3
  87.         TIM_TimeBaseStructure.TIM_Period = pulse_number;      //0x1900是360°;//改变给电机的脉冲个数                     
  88.         TIM_TimeBaseStructure.TIM_Prescaler = 0;
  89.         TIM_TimeBaseStructure.TIM_ClockDivision = 0;
  90.         TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;
  91.         TIM_TimeBaseStructure.TIM_RepetitionCounter = 0;
  92.         TIM_TimeBaseInit(TIM3, &TIM_TimeBaseStructure);
  93.         // 输出配置:配置输出比较非主动模式定时器——TIM3
  94.         // Output Compare Active Mode configuration: Channel1
  95.         TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_Inactive; //输出比较非主动模式,(匹配时设置输出引脚为无效        电平,当计数值为比较/捕获寄存器值相同时,强制输出为低电平)      
  96.         TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_High;
  97.         TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable;
  98.         TIM_OCInitStructure.TIM_Pulse = 0xFFFF; // 这里的配置值意义不大   
  99.         TIM_OC1Init(TIM3, &TIM_OCInitStructure);

  100.         // 配置TIM2为主定时器
  101.         // Select the Master Slave Mode
  102.         TIM_SelectMasterSlaveMode(TIM2, TIM_MasterSlaveMode_Enable); //设置 TIM2 主/从模式并使能
  103.         // Master Mode selection  
  104.         TIM_SelectOutputTrigger(TIM2, TIM_TRGOSource_Update); //使用更新事件作为触发输出
  105.        
  106.         // 配置TIM3为从定时器
  107.         // Slave Mode selection: TIM3
  108.         TIM_SelectSlaveMode(TIM3, TIM_SlaveMode_Gated); //选择 TIM3为从模式   TIM_SlaveMode_Gated-当触发信号(TRGI)为高电平时计数器时钟使能
  109.         TIM_SelectInputTrigger(TIM3, TIM_TS_ITR1); //选择 TIM3 输入触发源    TIM_TS_ITR1-TIM 内部触发 1
  110.         TIM_ITRxExternalClockConfig(TIM3, TIM_TS_ITR1);//设置 TIM3 内部触发为外部时钟模式   TIM_TS_ITR1-TIM 内部触发 1
  111.         TIM_ITConfig(TIM3, TIM_IT_CC1, ENABLE); //使能TIM3     TIM 捕获/比较 1 中断源          
  112.        
  113.         TIM_Cmd(TIM2, ENABLE);
  114.         TIM_Cmd(TIM3, ENABLE);
  115. }

  116. /****************************************************
  117. * 函数名:Output_Pulse
  118. * 描述  :无
  119. * 输入  :无
  120. * 输出  :无
  121. * 返回值:无
  122. ******************************************************/
  123. void Output_Pulse(u16 Num)
  124. {
  125.                 GPIO_ResetBits(GPIOA, GPIO_Pin_2);//指定引脚输出低电平,此时灯全灭        使能
  126.                 TIM3->CCR1 = Num;
  127.                 TIM3->CNT = 0;
  128.                 TIM_Cmd(TIM3, ENABLE);
  129.                 TIM_ITConfig(TIM3, TIM_IT_CC1, ENABLE);
  130.                 TIM_Cmd(TIM2, ENABLE);
  131. }

  132. /****************************************************
  133. * 函数名:angle_set
  134. * 描述  :无
  135. * 输入  :无
  136. * 输出  :无
  137. * 返回值:无
  138. ******************************************************/
  139. void angle_set(u8 dir,u8 angle)
  140. {
  141.         if(dir==0)
  142.                                 GPIO_ResetBits(GPIOA, GPIO_Pin_1);//指定引脚输出低电平,此时灯全灭,方向
  143.         else
  144.                                 GPIO_SetBits(GPIOA, GPIO_Pin_1);//指定引脚输出低电平,此时灯全灭,方向
  145.        
  146.         Output_Pulse(angle*6400);
  147. }
复制代码
4.time.h程序
  1. #ifndef __TIMER_H
  2. #define        __TIMER_H
  3. #include "main.h"
  4. extern unsigned char Flag;
  5. extern unsigned char TIM2_Pulse_TIM3_Counter_OK;
  6. void GPIO_Config(void);
  7. void TIM2_Master__TIM3_Slave_Configuration(u32 PulseFrequency,u32 pulse);
  8. void Frequence_Setting(u32 PulseFrequency);
  9. void Output_Pulse(u16 Num);
  10. void angle_set(u8 dir,u8 angle);
  11. #endif
复制代码
5.usart1.c程序
  1. #include <main.h>
  2. #include <usart1.h>
  3. #include <string.h>
  4. #include <math.h>

  5. /******************************************************
  6. * 函数名:USART1_Config
  7. * 描述  :USART1 GPIO 配置,工作模式配置
  8. * 输入  :无
  9. * 输出  : 无
  10. * 调用  :外部调用
  11. ***************************************************** */
  12. void USART1_Config(void)
  13. {
  14.         GPIO_InitTypeDef GPIO_InitStructure;
  15.         USART_InitTypeDef USART_InitStructure;
  16.        
  17.         /* config USART1 clock */
  18.         RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1, ENABLE);
  19.         RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);

  20.         /* USART1 GPIO config */
  21.         /* Configure USART1 Tx (PA.09) as alternate function push-pull */
  22.         GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9;
  23.         GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;
  24.         GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  25.         GPIO_Init(GPIOA, &GPIO_InitStructure);   
  26.         /* Configure USART1 Rx (PA.10) as input floating */
  27.         GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;
  28.         GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;
  29.         GPIO_Init(GPIOA, &GPIO_InitStructure);
  30.        
  31.         /* USART1 mode config */
  32.         USART_InitStructure.USART_BaudRate = 115200;
  33.         USART_InitStructure.USART_WordLength = USART_WordLength_8b;
  34.         USART_InitStructure.USART_StopBits = USART_StopBits_1;
  35.         USART_InitStructure.USART_Parity = USART_Parity_No ;
  36.         USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;
  37.         USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;
  38.         USART_Init(USART1, &USART_InitStructure);
  39.         USART_ITConfig(USART1, USART_IT_RXNE, ENABLE);
  40.        
  41.         USART_Cmd(USART1, ENABLE);
  42. }

  43. /******************************************************
  44. * 函数名:fputc
  45. * 描述  :重定向c库函数printf到USART1
  46. * 输入  :无
  47. * 输出  :无
  48. * 调用  :由printf调用
  49. ***************************************************** */
  50. int fputc(int ch, FILE *f)
  51. {
  52.         /* 将Printf内容发往串口 */
  53.         USART_SendData(USART1, (unsigned char) ch);
  54.         while (!(USART1->SR & USART_FLAG_TXE));
  55.        
  56.         return (ch);
  57. }

  58. /*-------------------------------------------------------------------------------*/
  59. /******************************************************
  60. * 函数名:USART1_IRQHandler
  61. * 描述  :USART1中断服务函数
  62. * 输入  :无
  63. * 输出  :无
  64. * 调用  :中断调用
  65. ***************************************************** */
  66. u8 Res;
  67. extern int a;
  68. u16 USART_RX_STA=0;       //接收状态标记         
  69. u8 USART_RX_BUF[USART_REC_LEN];     //接收缓冲,最大USART_REC_LEN个字节.
  70. void USART1_IRQHandler(void)                        //串口1中断服务程序
  71.         {
  72.                
  73.                 Res =USART_ReceiveData(USART1);        //读取接收到的数据
  74.                
  75.                 if((USART_RX_STA&0x8000)==0)//接收未完成
  76.                         {
  77.                         if(USART_RX_STA&0x4000)//接收到了0x0d
  78.                                 {
  79.                                 if(Res!=0x0a)USART_RX_STA=0;//接收错误,重新开始
  80.                                 else USART_RX_STA|=0x8000;        //接收完成了
  81.                                 }
  82.                         else //还没收到0X0D
  83.                                 {       
  84.                                 if(Res==0x0d)USART_RX_STA|=0x4000;
  85.                                 else
  86.                                         {
  87.                                         USART_RX_BUF[USART_RX_STA&0X3FFF]=Res ;
  88.                                         USART_RX_STA++;
  89.                                         if(USART_RX_STA>(USART_REC_LEN-1))USART_RX_STA=0;//接收数据错误,重新开始接收          
  90.                                         }                 
  91.                                 }
  92.                         }               
  93.                                 a=1;         
  94.         }
  95. #if SYSTEM_SUPPORT_OS         //如果SYSTEM_SUPPORT_OS为真,则需要支持OS.
  96.         OSIntExit();                                                                                           
  97. #endif

  98. /******************* (C) COPYRIGHT 2012 WildFire Team *****END OF FILE************/
复制代码
6.usart1.h程序
  1. #ifndef __USART1_H
  2. #define        __USART1_H
  3. #include <main.h>
  4. #include <stdio.h>

  5. #define USART_REC_LEN                          200          //定义最大接收字节数 200
  6. #define EN_USART1_RX                         1                //使能(1)/禁止(0)串口1接收
  7. extern u8  USART_RX_BUF[USART_REC_LEN]; //接收缓冲,最大USART_REC_LEN个字节.末字节为换行符
  8. extern u16 USART_RX_STA;                         //接收状态标记
  9. int simple_atoi(char *source);
  10. void USART1_Config(void);
  11. #endif /* __USART1_H */
复制代码
源码:
步进电机程序—串口控制速度角度.zip (4.75 MB, 下载次数: 116)
收藏 1 评论0 发布时间:2021-6-25 13:30

举报

0个回答

所属标签

相似分享

官网相关资源

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