STM32F4的PWM脉冲数量精确输出

漫长的几场考试终于结束,也是开始继续之前因考试搁置的项目。

这是2022年第一篇博客,也希望自己更加精益求精、不轻视小问题、脚踏实地、无论事情简单与否都不浮躁地干完。

采用定时器主从模式来达到pwm脉冲的精确输出,主定时器用来控制从定时器的pwm脉冲输出数量,从定时器用来产生pwm脉冲。

TIMx内部触发连接
从定时器ITR0ITR1ITR2ITR3
TIM2TIM1TIM8TIM3TIM4
TIM3TIM1TIM2TIM5TIM4
TIM4TIM1TIM2TIM3TIM8
TIM5TIM2TIM3TIM4TIM8
通用定时器2~5引脚
CH1CH2CH3CH4
TIM2PA0/PA5/PA15PA1/PB3PA2/PB10PA3/PB11
TIM3PA6/PB4/PC6PA7/PB5/PC7PB0/PC8PB1/PC9
TIM4PB6/PD12PB7/PD13PB8/PD14PB9/PD15
TIM5PA0/PH10PA1/PH11PA2/PH12PA3/PI0

以TIM2和TIM3为例。引脚各取PA0和PC6。

将TIM2设置为主模式,把PWM的参考电平OC1REF作为TRGO触发输出。

将TIM3设置为从模式的门控模式,将TIM2的TRG0作为触发输入,控制定时器的开启和关闭。

只有当TIM2的PWM波为低电平时,TIM3才输出PWM波。这样当TIM3的PWM脉冲频率一定时,只要控制TIM2的PWM波的占空比就可以控制TIM3的PWM脉冲输出数量。

 

 程序如下:

TIM2主模式初始化:

void TIM2_Config(u32 cycle)
{
	GPIO_InitTypeDef GPIO_InitStructure;
	TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure;
	TIM_OCInitTypeDef TIM_OCInitStructure;
	
	RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2,ENABLE);//使能定时器时钟
	
	RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA, ENABLE);//使能IO口时钟
	
	GPIO_PinAFConfig(GPIOA,GPIO_PinSource0,GPIO_AF_TIM2);//PA0复用为定时器2
	
	
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0;           
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF;        //复用功能
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_100MHz;	//速度100MHz
	GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;      //推挽复用输出
	GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_UP;        //上拉
	GPIO_Init(GPIOA,&GPIO_InitStructure);               //初始化PA0
	
	
	TIM_TimeBaseStructure.TIM_Prescaler=8400-1;                  //定时器分频
	TIM_TimeBaseStructure.TIM_CounterMode=TIM_CounterMode_Up;//向上计数模式
	TIM_TimeBaseStructure.TIM_Period=cycle;                  //自动重装载值
	TIM_TimeBaseStructure.TIM_ClockDivision=TIM_CKD_DIV1; 	
	TIM_TimeBaseInit(TIM2,&TIM_TimeBaseStructure);           //初始化定时器

	
	TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM2;            //选择PWM模式
	TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable;//比较输出使能
	TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_High;    //输出极性低
	TIM_OC1Init(TIM2, &TIM_OCInitStructure);                     //初始化定时器通道
	
	/*定时器主模式设置*/
	TIM_SelectMasterSlaveMode(TIM2, TIM_MasterSlaveMode_Enable);
    TIM_SelectOutputTrigger(TIM2, TIM_TRGOSource_OC1Ref);
	
	TIM_Cmd(TIM2, ENABLE);

	TIM_OC1PreloadConfig(TIM2, TIM_OCPreload_Enable);//使能定时器在ccr上的预装载寄存器
	
	TIM_ARRPreloadConfig(TIM2,ENABLE);//ARPE使能
	TIM_SetCompare1(TIM2,1000);
}

TIM3从模式初始化:

void TIM3_Config(u32 PulseNum)
{
	GPIO_InitTypeDef GPIO_InitStructure;
	TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure;
	TIM_OCInitTypeDef TIM_OCInitStructure;
	
	RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3,ENABLE);//使能定时器时钟
	
	RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOC, ENABLE);//使能IO口时钟
	
	GPIO_PinAFConfig(GPIOC,GPIO_PinSource6,GPIO_AF_TIM3);//PC6复用为定时器3
	
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_6;           
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF;        //复用功能
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_100MHz;	//速度100MHz
	GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;      //推挽复用输出
	GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_UP;        //上拉
	GPIO_Init(GPIOC,&GPIO_InitStructure);               //初始化PC6
	
	TIM_TimeBaseStructure.TIM_Prescaler=8400-1;                  //定时器分频
	TIM_TimeBaseStructure.TIM_CounterMode=TIM_CounterMode_Up;//向上计数模式
	TIM_TimeBaseStructure.TIM_Period=PulseNum;               //自动重装载值
	TIM_TimeBaseStructure.TIM_ClockDivision=TIM_CKD_DIV1; 	
	TIM_TimeBaseInit(TIM3,&TIM_TimeBaseStructure);           //初始化定时器
	
	TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM2;            //选择PWM模式
	TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable;//比较输出使能
	TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_High;    //输出极性低
	TIM_OC1Init(TIM3, &TIM_OCInitStructure);                     //初始化定时器通道
	
	/*定时器从模式设置*/
	TIM_SelectInputTrigger(TIM3, TIM_TS_ITR1);
    TIM_SelectSlaveMode(TIM3,TIM_SlaveMode_Gated );

	TIM_Cmd(TIM3, ENABLE);
	
	TIM_OC1PreloadConfig(TIM3, TIM_OCPreload_Enable);//使能定时器在ccr上的预装载寄存器
	
	TIM_ARRPreloadConfig(TIM3,ENABLE);//ARPE使能
	
	TIM_SetCompare1(TIM3,20);
}

PWM控制输出:

void Pulse_output(u32 cycle,u32 PulseNum)
{
	TIM2_Config(cycle);
	TIM_Cmd(TIM2, ENABLE);
	TIM3_Config(PulseNum);
	TIM_Cmd(TIM3, ENABLE);
}

主函数:

int main(void)
{ 
	u16 led0pwmval=0;    
	u8 dir=1;
	NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);//设置系统中断优先级分组2
	delay_init(168);  //初始化延时函数
	uart_init(115200);//初始化串口波特率为115200
	//LED_Init();
	Pulse_output(2000-1,40-1);
    while(1);
}

TIM2产生周期为1S,占空比为50%的PWM波。

TIM3产生周期为0.1S,占空比为50%的PWM波。

理论是1S内TIM3产生5次PWM脉冲,示波器采集到的现象也是。

 拓展:根据此可行的理论,可通过TIM2的定时器中断来控制每次操作只输出想要的脉冲数,由于手上的示波器过于简单,因此实际实现会等后面专业示波器到手了再次实现。

具体操作要求如下: 以给定频率输出脉冲脉冲数无限制 以给定频率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公司能够能够简化器件的使用。
评论 6
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

逐渐掉头发的阿江

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值