STM32自学笔记17-步进电机驱动项目-磁编码器的正常使用

文章讲述了步进电机编码器的非线性校准过程,通过采集数据点拟合角度与编码器输出的关系,建立转换公式,确保更准确的角度测量。代码示例展示了如何检测阶跃并计算校正值。之后,文章介绍了系统的工作流程,包括正反转测量和数据记录,最终将校准数据存储于Flash,并用于FOC算法的电机控制。
摘要由CSDN通过智能技术生成

上节有这样一句话:

步进电机旋转角度和编码器输出数据之间的关系通常是非线性的。在校准过程中,可以通过采集一系列已知角度位置的数据点,并拟合出角度与编码器数据之间的关系。这个拟合可以使用曲线拟合算法或其他数学方法来实现。通过拟合,可以建立编码器输出数据与实际相位角之间的非线性转换公式,从而实现更准确的角度测量。

代码是这样实现的,在步进电机旋转的每一步都计算下一步和这一步的传感器读取数值的差,如果这个差值和预期不一致,则认为发生了阶跃,同时把阶跃差值确定。

	uint32_t step_num = 0;  //阶跃次数
	if(encode_cali.dir){    //电机正转
		for(count=0; count<200; count++){
			sub_data = (int32_t)encode_cali.coder_data_f[CycleRem(count+1, 200)] - (int32_t)encode_cali.coder_data_f[CycleRem(count, 200)];
			if(sub_data < 0){   //这个差值是应该大于0的,如果小于0则代表发生了阶跃,即转了超过1圈
				step_num++;  //阶跃次数加1
				encode_cali.rcd_x = count;//使用区间前标
				encode_cali.rcd_y = (2^14-1) - encode_cali.coder_data_f[CycleRem(encode_cali.rcd_x, 200)]; //阶跃差值
			}
		}
		if(step_num != 1){
			encode_cali.error_code = CALI_Error_PhaseStep;  //如果阶跃次数不为1,则报错
			return;
		}
	}
	else{   //反转也是类似的,差值应该小于0,如果大于0则代表发生了阶跃
		for(count=0; count<200; count++){
			sub_data = (int32_t)encode_cali.coder_data_f[CycleRem(count+1, 200)] - (int32_t)encode_cali.coder_data_f[CycleRem(count, 200)];
			if(sub_data > 0){
				step_num++;
				encode_cali.rcd_x = count;//使用区间前标
				encode_cali.rcd_y = (2^14-1) - encode_cali.coder_data_f[CycleRem(encode_cali.rcd_x+1, 200)]; //这里要注意,反转需要把步数+1,即下一步的差值
			}
		}
		if(step_num != 1){
			encode_cali.error_code = CALI_Error_PhaseStep;
			return;
		}
	}

到这里为止,整个编码器的校准过程就完成了。

接下去就是正常使用的过程。

  1. 首先要定义一个编码器状态的枚举类型
typedef enum{
	CALI_Disable = 0x00,						//不校准
	CALI_Forward_Encoder_AutoCali,				//编码器正转自动校准
	CALI_Forward_Measure,						//正向测量
	CALI_Reverse_Ret,							//反向回退
	CALI_Reverse_Gap,							//反向消差
	CALI_Reverse_Measure,						//反向测量
	CALI_Operation,								//解算
}CALI_State;

会把它分为中断过程的回调使用正常回调两个函数来执行。

  1. 首先是中断回调,一般是用户在正常工作时想要进行校准时的操作。来看下代码:
switch(encode_cali.state)  //看现时的编码器状态
	{
		case CALI_Disable:
			if(encode_cali.trigger)  //如果用户开启了校准请求,初始化也会使trigger置1
			{
				REIN_HW_Elec_SetDivideElec(encode_cali.out_location, Current_Cali_Current);   //这一句是和FOC算法相关的,后面再学习
				encode_cali.out_location = Move_Pulse_NUM;	//步进电机转一圈,这个宏定义的值是单圈脉冲数,=200*256,256是单步细分数
				encode_cali.gather_count = 0;	//采集清零
				encode_cali.state = CALI_Forward_Encoder_AutoCali;	  //--->编码器正转自动校准
				encode_cali.error_code = CALI_No_Error; //初始化
				encode_cali.error_data = 0;
			}
		break;
		//编码器正转自动校准
		case CALI_Forward_Encoder_AutoCali:
			encode_cali.out_location += 2;  //每次走2个脉冲,一直走到第二圈
			REIN_HW_Elec_SetDivideElec(encode_cali.out_location, Current_Cali_Current);
			if(encode_cali.out_location == 2 * Move_Pulse_NUM)
			{
				encode_cali.out_location = Move_Pulse_NUM;  //重置到一圈
				encode_cali.state = CALI_Forward_Measure;  //走到第二圈后到下一个状态,正向测量
			}
		break;
		//正向测量
		case CALI_Forward_Measure:
			if((encode_cali.out_location % Move_Divide_NUM) == 0)//每到达采集细分量点采集一次数据
			{
				//开始采集传感器的角度数据
				encode_cali.coder_data_gather[encode_cali.gather_count++] = mt6816.angle_data;
				if(encode_cali.gather_count == Gather_Quantity){
					//记录数据
					encode_cali.coder_data_f[(encode_cali.out_location - Move_Pulse_NUM) / Move_Divide_NUM]
						= CycleDataAverage(encode_cali.coder_data_gather, Gather_Quantity, CALI_Encode_Res);
					//采集计数清零
					encode_cali.gather_count = 0;
					//移动位置
					encode_cali.out_location += 1;  //每次移动1个脉冲
				}
			}
			else{
				//移动位置
				encode_cali.out_location += 1;
			}	
			REIN_HW_Elec_SetDivideElec(encode_cali.out_location, Current_Cali_Current);
		
			if(encode_cali.out_location > (2 * Move_Pulse_NUM)) //如果走的脉冲数超过2圈了,进入下一个状态
			{
				encode_cali.state = CALI_Reverse_Ret;//--->反向回退
			}
		break;
		//反向回退
		case CALI_Reverse_Ret: 
			encode_cali.out_location += 1;
			REIN_HW_Elec_SetDivideElec(encode_cali.out_location, Current_Cali_Current);
			
			if(encode_cali.out_location == (2 * Move_Pulse_NUM + Move_Divide_NUM * 20))  //从第二圈再走20步,到下一个状态
			{
				encode_cali.state = CALI_Reverse_Gap;//--->反向消差
			}
		break;
		//反向消差
		case CALI_Reverse_Gap:
			encode_cali.out_location -= 1;  //每次往回退一个脉冲
			REIN_HW_Elec_SetDivideElec(encode_cali.out_location, Current_Cali_Current);
			
			if(encode_cali.out_location == (2 * Move_Pulse_NUM))  //退回到第二圈
			{
				encode_cali.state = CALI_Reverse_Measure;//--->反向测量
			}
		break;
		//反向测量,和正向测量类似
		case CALI_Reverse_Measure:
			if((encode_cali.out_location % Move_Divide_NUM) == 0)//每到达采集细分量点采集一次数据
			{
				//采集
				encode_cali.coder_data_gather[encode_cali.gather_count++] = mt6816.angle_data;
				if(encode_cali.gather_count == Gather_Quantity){
					//记录数据
					encode_cali.coder_data_r[(encode_cali.out_location - Move_Pulse_NUM) / Move_Divide_NUM]
						= CycleDataAverage(encode_cali.coder_data_gather, Gather_Quantity, CALI_Encode_Res);
					//采集计数清零
					encode_cali.gather_count = 0;
					//移动位置
					encode_cali.out_location -= 1;  回退一个脉冲
				}
			}
			else{
				//移动位置
				encode_cali.out_location -= 1;
			}	
			REIN_HW_Elec_SetDivideElec(encode_cali.out_location, Current_Cali_Current);
			
			if(encode_cali.out_location < Move_Pulse_NUM)
			{
				encode_cali.state = CALI_Operation;//如果退回1圈以内,则进入下一个状态
			}
		break;
		//计算
		case CALI_Operation:
			//进行校准计算中
			REIN_HW_Elec_SetDivideElec(0, 0);
			
		break;
		default:
		break;
	}
}

可以看到,进中断时的操作是走一个流程,把每个状态都走一遍,并把采集的数据记录到coder_data_fcoder_data_r两个数组中。
下面看看主循环中的调用函数,下面代码中Move_Step_NUM是步进电机总步数200,Move_Divide_NUM是每步的细分数256,Move_Pulse_NUM是总的脉冲数,即256*200

void Calibration_Loop_Callback(void)
{
	int32_t		data_i32;  //32位有符号
	uint16_t	data_u16;  //16位无符号数
	
	//必须要是校准计算状态才进入主循环,否则退出
	if(encode_cali.state != CALI_Operation)
		return;
	
	//给电机的4线低电平
	REIN_HW_Elec_SetSleep();

	//传感器数据检查
	Calibration_Data_Check();
	

	if(encode_cali.error_code == CALI_No_Error)
	{
		int32_t step_x, step_y;
		encode_cali.result_num = 0;
		Stockpile_Flash_Data_Empty(&stockpile_quick_cali);		//Flash擦除数据区
		Stockpile_Flash_Data_Begin(&stockpile_quick_cali);		//开始写数据区
		if(encode_cali.dir){  //正转的情况
			for(step_x = encode_cali.rcd_x; step_x < encode_cali.rcd_x + Move_Step_NUM + 1; step_x++)  //从rcd_x的位置整一圈
			{  																						   //可以认为rcd_x是步数,rcd_y是该步数对应的传感器读数				
				data_i32 = CycleSub(	encode_cali.coder_data_f[CycleRem(step_x+1, Move_Step_NUM)],   
															encode_cali.coder_data_f[CycleRem(step_x, Move_Step_NUM)],  
															CALI_Encode_Res);                          //data_i32的值存的是两步之间的传感器的读取数值差值
				if(step_x == encode_cali.rcd_x){//开始边缘
					for(step_y = encode_cali.rcd_y; step_y < data_i32; step_y++){ 
						data_u16 = CycleRem(	Move_Divide_NUM * step_x + Move_Divide_NUM * step_y / data_i32,
																	Move_Pulse_NUM);   //data_u16存值:step_x这一步各个脉冲的传感器读取值
						Stockpile_Flash_Data_Write_Data16(&stockpile_quick_cali, &data_u16, 1);  //分为开始、中间、结尾3个区间获取并存入flash
						encode_cali.result_num++;
					}
				}
				else if(step_x == encode_cali.rcd_x + Move_Step_NUM){//结束边缘
					for(step_y = 0; step_y < encode_cali.rcd_y; step_y++){
						data_u16 = CycleRem(	Move_Divide_NUM * step_x + Move_Divide_NUM * step_y / data_i32,
																	Move_Pulse_NUM);
						Stockpile_Flash_Data_Write_Data16(&stockpile_quick_cali, &data_u16, 1);
						encode_cali.result_num++;
					}
				}
				else{//中间
					for(step_y = 0; step_y < data_i32; step_y++){
						data_u16 = CycleRem(	Move_Divide_NUM * step_x + Move_Divide_NUM * step_y / data_i32,
																	Move_Pulse_NUM);
						Stockpile_Flash_Data_Write_Data16(&stockpile_quick_cali, &data_u16, 1);
						encode_cali.result_num++;
					}
				}
			}
		}
		else  //以下是反转的情况,同样也是把每一步的每个脉冲的传感器数值存入flash
		{
			for(step_x = encode_cali.rcd_x + Move_Step_NUM; step_x > encode_cali.rcd_x - 1; step_x--)   
			{
				data_i32 = CycleSub(	encode_cali.coder_data_f[CycleRem(step_x, Move_Step_NUM)],
															encode_cali.coder_data_f[CycleRem(step_x+1, Move_Step_NUM)],
															CALI_Encode_Res);
				if(step_x == encode_cali.rcd_x+Move_Step_NUM){//开始边缘
					for(step_y = encode_cali.rcd_y; step_y < data_i32; step_y++){
						data_u16 = CycleRem(	Move_Divide_NUM * (step_x+1) - Move_Divide_NUM * step_y / data_i32,
																	Move_Pulse_NUM);
						Stockpile_Flash_Data_Write_Data16(&stockpile_quick_cali, &data_u16, 1);
						encode_cali.result_num++;
					}
				}
				else if(step_x == encode_cali.rcd_x){//结束边缘
					for(step_y = 0; step_y < encode_cali.rcd_y; step_y++){
						data_u16 = CycleRem(	Move_Divide_NUM * (step_x+1) - Move_Divide_NUM * step_y / data_i32,
																	Move_Pulse_NUM);
						Stockpile_Flash_Data_Write_Data16(&stockpile_quick_cali, &data_u16, 1);
						encode_cali.result_num++;
					}
				}
				else{//中间
					for(step_y = 0; step_y < data_i32; step_y++){
						data_u16 = CycleRem(	Move_Divide_NUM * (step_x+1) - Move_Divide_NUM * step_y / data_i32,
																	Move_Pulse_NUM);
						Stockpile_Flash_Data_Write_Data16(&stockpile_quick_cali, &data_u16, 1);
						encode_cali.result_num++;
					}
				}
			}
		}
		Stockpile_Flash_Data_End(&stockpile_quick_cali);	//结束写数据区
		
		if(encode_cali.result_num != CALI_Encode_Res)   //result_num应该等于2^14
			encode_cali.error_code = CALI_Error_Analysis_Quantity; //报解析数据错误
	}

	//确认校准结果
	if(encode_cali.error_code == CALI_No_Error){
		mt6816.rectify_valid = true;  //磁编码器数据确认
	}
	else{
		mt6816.rectify_valid = false;  //不进行数据的存储
		Stockpile_Flash_Data_Empty(&stockpile_quick_cali);	//清除校准区数据
	}
	
	//运动配置覆盖
	motor_control.stall_flag = true;	//这是电机相关的,可以先不看,意思是堵转保护,即校准后禁用运动控制
	
	//清理校准信号
	encode_cali.state = CALI_Disable;
	encode_cali.trigger = false;			//清除校准触发
}

在实际使用时,当有外部事件要求(例如实体按键,或者用户界面按钮),会进入中断回调函数。
主loop程序会调用Calibration_Loop_Callback()

至此为止,磁编码器和电机驱动芯片的驱动基本完成了,接下去就是最核心的FOC算法实现对步进电机的控制。

未完待续

  • 7
    点赞
  • 29
    收藏
    觉得还不错? 一键收藏
  • 10
    评论
STM32F103C8T6是一款非常常用的Cortex-M3内核的微控制器,而A4988是一款常用的步进电机驱动芯片,两者可以结合使用来驱动42步进电机STM32F103C8T6具有丰富的外设,适合用于控制步进电机。作为一个Cortex-M3内核的微控制器,它具有高性能和低功耗的特点。它提供了足够的GPIO引脚用于与A4988进行通讯,同时也能够通过SPI、I2C以及USART等接口与其他设备进行通信。 A4988是一款单片MOSFET式高流细分步进电机驱动器。它可以通过3级高效的电流控制来实现细分,从而提高步进电机的精度。此外,它还具有过温保护、过流保护和欠压锁定等功能,确保安全稳定地驱动步进电机。 通过将STM32F103C8T6与A4988连接,我们可以通过STM32的GPIO控制A4988的引脚,从而实现对步进电机控制。常规的接线方式是将STM32的GPIO与A4988的STEP、DIR和EN引脚连接起来,STEP引脚用于控制步进电机的步进脉冲,DIR引脚用于设置步进电机的方向,EN引脚用于使能或禁用A4988驱动器。通过控制这些引脚,我们可以控制步进电机的转动速度和方向。 在编程方面,我们需要使用STM32的开发环境,如Keil或CubeMX,来编写控制步进电机的代码。编写代码时,我们可以利用STM32的GPIO库函数或底层寄存器操作来控制对应的引脚状态。我们还可以通过调整A4988的细分设置来实现不同的旋转精度。 综上所述,通过将STM32F103C8T6与A4988步进电机驱动器连接,我们可以利用STM32的GPIO控制塑电机的转速和方向,实现对42步进电机的精确控制
评论 10
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值