stm32f103c8t6控制多个步进电机

关于两相步进电机的详细内容,相信在网上一大堆可以找得到,这里我主要介绍的是实践部分。
首先,我们需要认清步进电机的四根线,一般来说四根线颜色基本都为红、蓝、绿、黑,对应的分别是A+,A-,B+,B-,一些步进电机上面会标哪根线对应哪个相,比如我用的步进电机是这款的
在这里插入图片描述
在这里插入图片描述

其实哪款步进电机都无所谓,原理都是差不多的,主要看电机的电流是否满足步进电机驱动器的要求。
我用的步进电机驱动器是以下这款,之所以用这款是因为这款步进电机驱动器体积相对比较小并且满足要求,从上面我们可以看到我上面的步进电机的工作电流是2.3A,而从步进电机驱动器的面板上面可以看出他最大可以工作在3A电流。
步进电机驱动器淘宝链接:步进电机驱动器
在这里插入图片描述
驱动器面板上面的STOP是指控制脉冲信号停止施加 0.5 秒左右,会自动进入半电流状态,这时电机相电流为运行时的 50%,以降低功耗减少电机发热,收到新的控制脉冲后驱动器自动退出半电流状态, 此功能 一直有效。一般都是设置50%,也就是半电流状态。
驱动器面板上面的Decay Setting是衰减设置,一般设置成0%即可。
驱动器面板上面的Excitation Mode是细分设置,原本发出一个脉冲步进电机转动1.8°,如果细分为2的话,步进电机转动0.9°。有些驱动板上面写着200、400、800、1600、3200等等,是指转动一圈所需要的脉冲数。
驱动器和两相混合式步进电机的连接采用四线制,电机的绕组有并联和串联接法,并联接法,高速性能好,但驱动电流大;串联接法适用于高力矩输出。一般的,高速输出的接线方式其输出电流设定在电机额定电流的 1.4 倍;高力矩输出的接线方式其输出电流设定在电机额定电流的 70%。
我采用的是共阴极接法,程序也会是共阴极的。
程序中stm32与步进电机驱动器接线如下

/*
步进电机1
Pluse±>PA.1,Pluse-接GND
DIR±>PA.8,DIR-接GND
EN±>PA.11,EN-接GND
步进电机2
Pluse±>PA.7,Pluse-接GND
DIR±>PA.6,DIR-接GND
EN±>PA.5,EN-接GND
*/

接下来附上相关程序,程序后面都有注释,个人认为这个注释已经很仔细了
主函数如下:

#include "delay.h"
#include "key.h"
#include "led.h"	 
#include "usmart.h"	
#include "driver.h"

/*
步进电机1,由按键PA.0控制
Pluse+->PA.1,Pluse-接GND
DIR+->PA.8,DIR-接GND
EN+->PA.11,EN-接GND
步进电机2,由按键PA.4控制
Pluse+->PA.7,Pluse-接GND
DIR+->PA.6,DIR-接GND
EN+->PA.5,EN-接GND
*/

int main(void)
{	 
	u8 keyval; 
	delay_init();	    	    //延时函数初始化	  
	NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2); 	 //设置NVIC中断分组2:2位抢占优先级,2位响应优先级
	uart_init(115200);	     	//串口初始化为115200
	KEY_Init();					//初始化按键
	Driver_Init();			    //驱动器初始化
	TIM2_Init(999,72-1);        //1MHz计数频率,第一个参数自动装载值在后面TIM2_Startup函数中会被改变
	TIM3_Init(999,72-1);
	while(1) 
	{	
		keyval=KEY_Scan(0);
		if(keyval==WKUP_PRES)
		{
			Locate_Rle2(500,CW);//500的意思你点进去看函数就明白了,大概就是重新设置定时器自动重装载值,让PWM一半高电平,一半低电平。
			Locate_Rle3(500,CW);
		}
		if(keyval==KEY0_PRES)
		{
			Locate_Rle2(500,CCW);
			Locate_Rle3(500,CCW);
		}

	}			
}



两个定时器初始化以及中断函数等等

#include "driver.h"
#include "delay.h"
#include "usart.h"

long current_pos[2]={0,0}; //有符号方向
DIR_Type motor_dir2=0;
DIR_Type motor_dir3=0;
u8 count[2]={0,0};
/************** 驱动器控制信号线初始化 ****************/
void Driver_Init(void)
{
	GPIO_InitTypeDef  GPIO_InitStructure;

	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);	//使能PA端口时钟

	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_6|GPIO_Pin_5|GPIO_Pin_8|GPIO_Pin_11;	//PC0.2 端口配置
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; 		//推挽输出
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;		//IO口速度为50MHz
	GPIO_Init(GPIOA, &GPIO_InitStructure);					//根据设定参数初始化GPIOA
	GPIO_SetBits(GPIOA,GPIO_Pin_6);						 	//PA6输出高 顺时针方向  DRIVER_DIR
	GPIO_ResetBits(GPIOA,GPIO_Pin_5);						//PA5输出低 使能输出  DRIVER_OE
	GPIO_SetBits(GPIOA,GPIO_Pin_8);						 	//PA8输出高 顺时针方向  DRIVER_DIR
	GPIO_ResetBits(GPIOA,GPIO_Pin_11);						//PA11输出低 使能输出  DRIVER_OE
}

/***********************************************
//TIM2_CH2(PA1) 单脉冲输出+重复计数功能初始化
//TIM2 时钟频率 72MHz
//arr:自动重装值
//psc:时钟预分频数
************************************************/
void TIM2_Init(u16 arr,u16 psc)
{		 					 
	GPIO_InitTypeDef GPIO_InitStructure;
	TIM_TimeBaseInitTypeDef  TIM_TimeBaseStructure;
	TIM_OCInitTypeDef  TIM_OCInitStructure;
	NVIC_InitTypeDef NVIC_InitStructure;

	RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2, ENABLE); 	//TIM2时钟使能
 	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA , ENABLE);  //使能GPIOA外设时钟使能	                                                                     	

  //设置该引脚为复用输出功能,输出TIM2 CH2的PWM脉冲波形
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1;				//TIM2_CH2
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;			//复用推挽输出
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_Init(GPIOA, &GPIO_InitStructure);
	
	TIM_TimeBaseStructInit(&TIM_TimeBaseStructure);
	
	TIM_TimeBaseStructure.TIM_Period = arr;					//设置在下一个更新事件装入活动的自动重装载寄存器周期的值,设置计数到这个值就产生溢出	 
	TIM_TimeBaseStructure.TIM_Prescaler =psc;				//设置用来作为TIMx时钟频率除数的预分频值  
	TIM_TimeBaseStructure.TIM_ClockDivision = 0; 			//设置时钟分割:TDTS = Tck_tim,是改变作输入捕获时滤波用的并不是定时器的分频器
	TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up; //TIM向上计数模式
	TIM_TimeBaseInit(TIM2, &TIM_TimeBaseStructure); 		//根据TIM_TimeBaseInitStruct中指定的参数初始化TIMx的时间基数单位
	TIM_ClearITPendingBit(TIM2,TIM_IT_Update);				//清除的是一些中断标志位

	TIM_UpdateRequestConfig(TIM2,TIM_UpdateSource_Regular); //TIM_UpdateSource_Regular 生成单一的脉冲:计数器在下一个更新事件停止,即关闭    
	TIM_SelectOnePulseMode(TIM2,TIM_OPMode_Single);			//单脉冲模式 
 
	TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM2;		//选择定时器模式:TIM脉冲宽度调制模式2[1]
	TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable; //比较输出2使能
	TIM_OCInitStructure.TIM_Pulse = arr>>1; 				//设置待装入捕获比较寄存器的脉冲值
	TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_High; //输出极性:TIM输出比较极性高[2]
	//----[1][2]这两个配合一起然后查手册可以得到->当计时器值小于比较器设定值时则TIMX输出脚此时输出有效低电位。
	TIM_OC2Init(TIM2, &TIM_OCInitStructure); 				//根据TIM_OCInitStruct中指定的参数初始化外设TIMx

	TIM_OC2PreloadConfig(TIM2, TIM_OCPreload_Enable);  		//CH2预装载使能	 
	TIM_ARRPreloadConfig(TIM2, ENABLE); 					//使能TIMx在ARR上的预装载寄存器
	
	TIM_ITConfig(TIM2, TIM_IT_Update ,ENABLE);  			//TIM2使能或者失能指定的TIM中断
 
	NVIC_InitStructure.NVIC_IRQChannel = TIM2_IRQn;  		//TIM2中断
	NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;  //先占优先级0级
	NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0; 		//从优先级0级
	NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; 		//IRQ通道被使能
	NVIC_Init(&NVIC_InitStructure);  						//根据NVIC_InitStruct中指定的参数初始化外设NVIC寄存器
	
	TIM_ClearITPendingBit(TIM2, TIM_IT_Update);  			//清除TIMx的中断待处理位:TIM 中断源
	TIM_Cmd(TIM2, DISABLE);  								//使能TIM2									  
}
/******* TIM2更新中断服务程序 *********/
void TIM2_IRQHandler(void)
{
	if(TIM_GetITStatus(TIM2,TIM_FLAG_Update)!=RESET)//更新中断
	{
		TIM_ClearITPendingBit(TIM2,TIM_FLAG_Update);//清除更新中断标志位		
		count[0]++; 
		TIM_GenerateEvent(TIM2,TIM_EventSource_Update);//产生一个更新事件 重新初始化计数器,
		//这里产生一个脉冲送到步进电机让电机运转一个脉冲数,即不细分的情况下,一般步进电机转1.8°
		//定时器初始化中,设置单脉冲模式是让计数器在下一个更新事件停止,即关闭,防止这次中断还没结束,
		//下一个计数器时间已经到了使中断紊乱,也就是进入中断后,定时器自动关闭。			
		TIM_Cmd(TIM2, ENABLE);  					   //使能TIM2,让定时器开始下一个脉冲计数后输出	
		if(count[0]==200)
		{
			if(motor_dir2==CW) 						   //如果方向为顺时针   
				current_pos[0]+=count[0];
			else          							   //否则方向为逆时针
				current_pos[0]-=count[0];
			TIM_Cmd(TIM2, DISABLE);  				   //关闭TIM2			
			printf("motor2当前位置=%ld\r\n",current_pos[0]);//打印输出
			count[0]=0;
		}
	
	}
}

void TIM3_Init(u16 arr,u16 psc)
{		 					 
	GPIO_InitTypeDef GPIO_InitStructure;
	TIM_TimeBaseInitTypeDef  TIM_TimeBaseStructure;
	TIM_OCInitTypeDef  TIM_OCInitStructure;
	NVIC_InitTypeDef NVIC_InitStructure;

	RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3, ENABLE); 
 	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA , ENABLE);  	                                                                     	


	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_7; 
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;  
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_Init(GPIOA, &GPIO_InitStructure);
	
	TIM_TimeBaseStructInit(&TIM_TimeBaseStructure);
	
	TIM_TimeBaseStructure.TIM_Period = arr; 	 
	TIM_TimeBaseStructure.TIM_Prescaler =psc;  
	TIM_TimeBaseStructure.TIM_ClockDivision = 0; 
	TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up; 
	TIM_TimeBaseInit(TIM3, &TIM_TimeBaseStructure);
	TIM_ClearITPendingBit(TIM3,TIM_IT_Update);

	TIM_UpdateRequestConfig(TIM3,TIM_UpdateSource_Regular); 
	TIM_SelectOnePulseMode(TIM3,TIM_OPMode_Single);
 
	TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM2; 
	TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable;
	TIM_OCInitStructure.TIM_Pulse = arr>>1; 
	TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_High; 
	TIM_OC2Init(TIM3, &TIM_OCInitStructure);  

	TIM_OC2PreloadConfig(TIM3, TIM_OCPreload_Enable);  	 
	TIM_ARRPreloadConfig(TIM3, ENABLE); 
	
	TIM_ITConfig(TIM3, TIM_IT_Update ,ENABLE);  
 
	NVIC_InitStructure.NVIC_IRQChannel = TIM3_IRQn;  
	NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;  
	NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1; 
	NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; 
	NVIC_Init(&NVIC_InitStructure);  
	
	TIM_ClearITPendingBit(TIM3, TIM_IT_Update); 
	TIM_Cmd(TIM3, DISABLE);  							  
}

void TIM3_IRQHandler(void)
{
	if(TIM_GetITStatus(TIM3,TIM_FLAG_Update)!=RESET)
	{
		TIM_ClearITPendingBit(TIM3,TIM_FLAG_Update);	
		count[1]++; 
		TIM_GenerateEvent(TIM3,TIM_EventSource_Update);
		TIM_Cmd(TIM3, ENABLE);  		
		
		if(count[1]==200)
		{
			if(motor_dir3==CW) 						//如果方向为顺时针   
				current_pos[1]+=count[1];
			else          							//否则方向为逆时针
				current_pos[1]-=count[1];
			TIM_Cmd(TIM3, DISABLE);  				
			printf("motor3当前位置=%ld\r\n",current_pos[1]);
			count[1]=0;
		}	
	}
}

/***************** 启动TIM8 *****************/
void TIM2_Startup(u32 frequency)   //启动定时器2
{
	u16 temp_arr=1000000/frequency-1; 
	TIM_SetAutoreload(TIM2,temp_arr);//设定自动重装值	
	TIM_SetCompare2(TIM2,temp_arr>>1); //匹配值2等于重装值一半,是以占空比为50%,这个是一半高电平,一半低电平,
	//定时器计数到最高或最低就是产生一个脉冲,脉冲从这里发送到PA1(TIM2_CH2	),占空比越大,输出的电压越大。
	TIM_SetCounter(TIM2,0);//计数器清零
	TIM_Cmd(TIM2, ENABLE);  //使能TIM2
}
void TIM3_Startup(u32 frequency)   
{
	u16 temp_arr=1000000/frequency-1; 
	TIM_SetAutoreload(TIM3,temp_arr);
	TIM_SetCompare2(TIM3,temp_arr>>1); 
	TIM_SetCounter(TIM3,0);
	TIM_Cmd(TIM3, ENABLE);  
}
/********************************************
//相对定位函数 
//num 0~2147483647
//frequency: 20Hz~100KHz
//dir: CW(顺时针方向)  CCW(逆时针方向)
*********************************************/
void Locate_Rle2(u32 frequency,DIR_Type dir) //相对定位函数
{
	
	if(TIM2->CR1&0x01)//上一次脉冲还未发送完成  直接返回,TIM2->CR1=0x01是使能定时器,
		//上面中断2函数进入后,产生一个脉冲,然后定时器自动关闭,再程序中开启,即TIM2->CR1=0x01
	{
		printf("\r\nThe last time pulses is not send finished,wait please!\r\n");
		return;
	}
	if((frequency<20)||(frequency>100000))//脉冲频率不在范围内 直接返回
	{
		printf("\r\nThe frequency is out of range! please reset it!!(range:20Hz~100KHz)\r\n");
		return;
	}
	motor_dir2=dir;//将枚举类型为DIR_Type的dir的值赋值给另一个枚举类型motor_dir2,方便在中断中判断motor_dir2是否为正
	//然后在中断中的current_pos[0]进行总和计算出当前的位置
	DRIVER_DIR2=motor_dir2;				 //将旋转方向赋值给DRIVER_DIR2所定义的接口
	TIM2_Startup(frequency);				 //开启TIM2
}
void Locate_Rle3(u32 frequency,DIR_Type dir) //相对定位函数
{
	
	if(TIM3->CR1&0x01)
	{
		printf("\r\nThe last time pulses is not send finished,wait please!\r\n");
		return;
	}
	if((frequency<20)||(frequency>100000))
	{
		printf("\r\nThe frequency is out of range! please reset it!!(range:20Hz~100KHz)\r\n");
		return;
	}
	motor_dir3=dir;
	DRIVER_DIR3=motor_dir3;
	TIM3_Startup(frequency);
}

下面我简述一下这个程序的效果,就是通过按键1和按键2来操作步进电机,通过按下按键1后,两个步进电机同时向同一个方向旋转一圈,按下按键2后,两个步进电机同时向另一个方向旋转一圈。
最后我附上stm32f103控制步进电机的程序:
stm32控制步进电机程序
代码要的下方留言邮箱我会第一时间发给你们的,有什么问题也可以下面评论!

版权声明:本文为CSDN博主「i土豆」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/weixin_44069765/article/details/109823841

生成海报
点赞 0

i土豆

我还没有学会写个人说明!

暂无评论

发表评论

相关推荐

STM32 C++编程系列一:STM32 C++编程介绍

一、STM32及其他单片机开发现状 在目前绝大部分的单片机开发当中,C语言占据着主流的地位,但由于C语言本身是一种面向过程的语言,因此在当前利用面向对象思想构建可复用代码为主流的今天显得比较麻烦&#x

六种电平转换的优缺点

作为一名电子设计的硬件工程师,电平转换是每个人都必须面对的的话题,主芯片引脚使用的1.2V、1.8V、3.3V等,连接外部接口芯片使用的1.8V、3.3V、5V等,由于电平不匹配就必须进行

OV7670摄像头模块资料

OV7670摄像头模块资料 一、实物图和原理图 二、模块简介 OV7670 是 OV( OmniVision)公司生产的 CMOS VGA 图像传感器。该传感器体积小、工作电压低,提供单片 VGA