如何使用STM32F103系列指定脉冲数输出控制步进电机

如何使用STM32F103系列指定脉冲数输出控制步进电机

1.定时器的初始化&配置

我们采用的是HAL库来实现初始化,以下附上代码,但注意的一点是必须调用,
我看有一个点很多博主的文章都没有写清楚,必须调用HAL_TIM_Base_Start_IT
否则将无法实现回调函数指定输出脉冲!

此处以高级定时器1为例,使用通道2和通道3

void TIM1_PWM_Init(u16 arr,u16 psc)
{   
	TIM1_Handler.Instance=TIM1;          	//定时器1
    TIM1_Handler.Init.Prescaler=psc;       //定时器分频 72000000/72=1000000
    TIM1_Handler.Init.CounterMode=TIM_COUNTERMODE_UP;//向上计数模式
    TIM1_Handler.Init.Period=arr;          //自动重装载值 1M/500=2khz 1M/625=1600Hz	
    TIM1_Handler.Init.ClockDivision=TIM_CLOCKDIVISION_DIV1;
    HAL_TIM_PWM_Init(&TIM1_Handler);       //初始化PWM
   
	HAL_NVIC_SetPriority(TIM1_UP_IRQn,1,3);    //设置中断优先级,抢占优先级1,子优先级3
	HAL_NVIC_EnableIRQ(TIM1_UP_IRQn);          //开启ITM1中断 
		
    TIM1_CH23Handler.OCMode=TIM_OCMODE_PWM1; //模式选择PWM1
    TIM1_CH23Handler.Pulse=arr/2;            //设置比较值,此值用来确定占空比,默认比较值为自动重装载值的一半,即占空比为50%
    TIM1_CH23Handler.OCPolarity=TIM_OCPOLARITY_HIGH; //输出比较极性为高 
		
	HAL_TIM_PWM_ConfigChannel(&TIM1_Handler,&TIM1_CH23Handler,TIM_CHANNEL_2);//配置TIM3通道2
	HAL_TIM_PWM_ConfigChannel(&TIM1_Handler,&TIM1_CH23Handler,TIM_CHANNEL_3);//配置TIM3通道3
	
	HAL_TIM_Base_Init(&TIM1_Handler);
	HAL_TIM_Base_Start_IT(&TIM1_Handler); //使能定时器3和定时器3更新中断:TIM_IT_UPDATE
}


void HAL_TIM_PWM_MspInit(TIM_HandleTypeDef *htim)
{
	GPIO_InitTypeDef GPIO_Initure;
	if(htim->Instance==TIM1){
		__HAL_RCC_TIM1_CLK_ENABLE();			 	//使能定时器1
		__HAL_RCC_GPIOE_CLK_ENABLE();				//开启GPIOE时钟
		__HAL_AFIO_REMAP_TIM1_ENABLE();
			
		GPIO_Initure.Pin=GPIO_PIN_11|GPIO_PIN_13;  	//PE11 PE13
		GPIO_Initure.Mode=GPIO_MODE_AF_PP;  		//复用推挽输出
		GPIO_Initure.Pull=GPIO_PULLUP;          	//上拉
		GPIO_Initure.Speed=GPIO_SPEED_FREQ_HIGH;	//高速
		HAL_GPIO_Init(GPIOE,&GPIO_Initure); 
	}
}

2.使用回调函数实现精准输出每一个脉冲

定义一个PWM的计数值,用来计算每一次的脉冲
这里使用pwm_val_L,pwm_val_R,我测试的时候刚好是左右两个电机
pwm_val 和 cnt_ch作比较,就能根据指定的pwm_val输出脉冲

若你正常初始化,后面输出时,通道3将输出20个脉冲,通道2输出30个脉冲

uint32_t pwm_val_L=20,pwm_val_R=30;
void HAL_TIM_PWM_PulseFinishedCallback(TIM_HandleTypeDef *htim)
{
 	static uint32_t cnt_ch3 = 0;
	static uint32_t cnt_ch2 = 0;
	
	if(htim==(&TIM1_Handler))
    {
	  if(TIM1_Handler.Channel == HAL_TIM_ACTIVE_CHANNEL_3){
			cnt_ch3++;
			if(cnt_ch3 == pwm_val_L){
				cnt_ch3 = 0;
				HAL_TIM_PWM_Stop_IT(&TIM1_Handler,TIM_CHANNEL_3);
			}
		}
		if(TIM1_Handler.Channel == HAL_TIM_ACTIVE_CHANNEL_2){
			cnt_ch2++;
			if(cnt_ch2 == pwm_val_R){
				cnt_ch2 = 0;
				HAL_TIM_PWM_Stop_IT(&TIM1_Handler,TIM_CHANNEL_2);
			}
		}
		
	}
}

3.输出脉冲

//1.按设置的脉冲数输出(触发回调函数)-脉冲输出完自动停止
HAL_TIM_PWM_Start_IT(&TIM1_Handler,TIM_CHANNEL_3);
HAL_TIM_PWM_Start_IT(&TIM1_Handler,TIM_CHANNEL_2);

//2.持续输出,直到关闭,中间可以加延时(不触发回调函数)
HAL_TIM_PWM_Start(&TIM1_Handler,TIM_CHANNEL_3);
HAL_TIM_PWM_Start(&TIM1_Handler,TIM_CHANNEL_2);
delay_s(5);
HAL_TIM_PWM_Stop(&TIM1_Handler,TIM_CHANNEL_3);
HAL_TIM_PWM_Stop(&TIM1_Handler,TIM_CHANNEL_2);

最后
这也是我查阅了不少资料,踩过一些坑,总结出来的点,制作不易,希望看到这里,对有帮助的朋友们顺手点个赞,谢谢啦!

  • 5
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
具体操作要求如下: 以给定频率输出脉冲,脉冲数无限制 以给定频率f、输出n个脉冲 相对定位 相对定位+绝对定位 脉冲输出PORTA.0 方向信号输出PORTB.5 模仿PLC定位指令 可以作为简易运动控制控制伺服电机 发脉冲两种目的 1)速度控制 2)位置控制 速度控制目的和模拟量一样,没有什么需要关注的地方 发送脉冲方式为PWM,速率稳定而且资源占用少 stm32位置控制需要获得发送的脉冲数,有下面4种手段 1)每发送一个脉冲,做一次中断计数 2)根据发送的频率×发送的时间,获得脉冲数量,对于变速的脉冲,可以累计积分的方法来获得总脉冲 3)一个定时器作为主发送脉冲,另外一个定时器作为从,对发送的脉冲计数 4)使用DMA方式,例如共发送1000个脉冲,那么定义u16 per[1001],每发送一个脉冲,dma会从数组中更新下一个占空比字,数组最后一个字为0,表示停发脉冲 上面4种方法的用途和特点 1)对于低速率脉冲比较好,可以说低速发脉冲的首选,例如10Khz以下的,否则中断占用太多的cpu,这种方法要注意将中断优先级提高,否则会丢计数, 2)用作定时的计时精确高,可以允许有脉冲计数丢失的情况 3)主从方式,需额外的定时器来计数,例如tim1发脉冲 tim2计数,最方便的方式,无论高速低速即可,同时占用cpu最低,只是要占用多一个定时器 4)DMA方式也算是一个很确定的方式,不会丢失脉冲,但是高速的时候,会较多的占用内部总线同时会使用一个多余的DMA控制器,而且有个缺点,就是使用起来比较复杂,没有达到KISS原则 个人推荐方式,低速时中断方式,如果不知高速还是低速,则使用主从方式。具体的方式需要根据资源和需求来确定。 stm32定时器算是比较复杂的器件,而且用户要较多的介入底层,希望将来st公司能够能够简化器件的使用

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值