STM32主从模式实现两路同步PWM脉冲输出,频率、占空比可调

原理:定时器1为主模式,定时器8为从模式,TIM1的定时器使能操作作为触发输出[TRGO]触发TIM8并使能TIM8的计数器,同时输出两路频率、占空比以及脉冲数量(小于256个,高级定时器重复计数功能为8位)可调PWM波形。

关键代码:

定时器1(TIM1)设为主模式:

    TIM_SelectMasterSlaveMode(TIM1, TIM_MasterSlaveMode_Enable);
    TIM_SelectOutputTrigger(TIM1, TIM_TRGOSource_Enable);

定时器2(TIM8)设为从模式

    TIM_SelectSlaveMode(TIM8, TIM_SlaveMode_Trigger);
    TIM_SelectInputTrigger(TIM8, TIM_TS_ITR0);

STM32F103系列主从定时器Trigger设置详见STM32F103系列定时器通道对应IO汇总

定时器1开启计数功能、定时器2计数功能无需设定,即既不使能也不禁用

    TIM_Cmd(TIM1, ENABLE);  //启动定时器TIM1的计数功能  
	//TIM_Cmd(TIM8, ENABLE);  // 无需启动定时器TIM8的计数功能,该功能由trgo信号使能

测试结果:

① 橘色:定时器TIM1:频率8MHz;脉冲个数8个;占空比0.5;

② 蓝色:定时器TIM8:频率2MHz;脉冲个数2个;占空比0.5;

橘色和蓝色同步输出,误差仅4~5ns可忽略

 

 

主函数:

#include "sys.h"
#include "delay.h"
#include "pwm.h"

int main(void)
{
    delay_init();
    GPIO_Configuration();
    while(1)
    {
        PWM_TIM1_TIM8(8000E3,50,8, 2000E3,50,2); // Freq1, duty_cycle1,PulseNum1,Freq2, duty_cycle2,PulseNum2
        
        delay_ms(20);
        
    }
    
}

定时器配置 pwm.c:

#include "pwm.h"
void GPIO_Configuration(void)
{
    GPIO_InitTypeDef GPIO_InitStructure;
    
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_TIM8 | RCC_APB2Periph_TIM1, ENABLE);     //使能TIM1 TIM8
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC | RCC_APB2Periph_AFIO, ENABLE);    //使能GPIOC,使能复用
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOE | RCC_APB2Periph_AFIO, ENABLE);
    GPIO_PinRemapConfig(GPIO_FullRemap_TIM1,ENABLE);                               //串口重映射 PA11-->PE14
   /*-------- 主定时器 TIM1-CH4 -> PA11 (PE14)  --------*/    
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_14;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;
  //  GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_Init(GPIOE, &GPIO_InitStructure);
    /*--------从定时器 TIM8-CH2 -> PC7 --------*/
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_7;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_Init(GPIOC, &GPIO_InitStructure);
    GPIO_SetBits(GPIOC,GPIO_Pin_7);

  //  GPIO_SetBits(GPIOE,GPIO_Pin_14);
}

// Freq : 脉冲频率
// PulseNum : 脉冲数量
// duty_cycle : 占空比
void PWM_TIM1_TIM8(u32 Freq0, u8 duty_cycle0, u8 PulseNum0, u32 Freq1, u8 duty_cycle1, u8 PulseNum)
{
    u16 ARR0;
    u16 CCR0;
    u16 PSC0 = 1;
	
    u16 ARR1;
    u16 CCR1;
    u16 PSC1 = 1;
	
    TIM_TimeBaseInitTypeDef  TIM_TimeBaseStructure;
    TIM_OCInitTypeDef  TIM_OCInitStructure;
    while(SystemCoreClock/PSC0/Freq0>65535)
    {
        PSC0++;
    }
    ARR0 = SystemCoreClock/PSC0/Freq0;
    TIM_TimeBaseStructure.TIM_Period = (ARR0-1);		                               //  设置重装载数据										
    TIM_TimeBaseStructure.TIM_Prescaler = (PSC0-1);                                    //  定时器时钟分频											    
    TIM_TimeBaseStructure.TIM_ClockDivision = 0;                                       //  计数器时钟分频
    TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;                        //  向上计数模式		
    TIM_TimeBaseStructure.TIM_RepetitionCounter = (PulseNum0 -1);                      //  脉冲数量    
    TIM_TimeBaseInit(TIM1, &TIM_TimeBaseStructure);
    TIM_SelectOnePulseMode(TIM1, TIM_OPMode_Single);
    TIM_SelectMasterSlaveMode(TIM1, TIM_MasterSlaveMode_Enable);
    TIM_SelectOutputTrigger(TIM1, TIM_TRGOSource_Enable);
    
    CCR0 = ARR0*(100-duty_cycle0) /100;
    TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM2;
    TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable;;
    TIM_OCInitStructure.TIM_Pulse = CCR0;					                           //  比较值,当为PWM2模式且输出极性为High,表示低电平持续时间
    TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_High;
    TIM_OCInitStructure.TIM_OCIdleState = TIM_OCIdleState_Set;
    TIM_OC4Init(TIM1, &TIM_OCInitStructure);	
    TIM_OC4PreloadConfig(TIM1, TIM_OCPreload_Enable);
    TIM_ARRPreloadConfig(TIM1, ENABLE);	
    TIM_CtrlPWMOutputs(TIM1, ENABLE);

    while(SystemCoreClock/PSC1/Freq1>65535)
    {
        PSC1++;
    }
    ARR1 = SystemCoreClock/PSC1/Freq1;
    TIM_TimeBaseStructure.TIM_Period = (ARR1-1);		                               //  设置重装载数据										
    TIM_TimeBaseStructure.TIM_Prescaler = (PSC1-1);                                    //  定时器时钟分频											    
    TIM_TimeBaseStructure.TIM_ClockDivision = 0;                                       //  计数器时钟分频
    TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;                        //  向上计数模式		
    TIM_TimeBaseStructure.TIM_RepetitionCounter = (PulseNum -1);                       //  脉冲数量    
    TIM_TimeBaseInit(TIM8, &TIM_TimeBaseStructure);
    TIM_SelectOnePulseMode(TIM8, TIM_OPMode_Single);
    TIM_SelectSlaveMode(TIM8, TIM_SlaveMode_Trigger);
    TIM_SelectInputTrigger(TIM8, TIM_TS_ITR0);
    
    CCR1 = ARR1*(100-duty_cycle1) /100;
    TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM1;
    TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable;;
    TIM_OCInitStructure.TIM_Pulse = CCR1;					                           //  比较值,当为PWM2模式且输出极性为High,表示低电平持续时间
    TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_Low;
    TIM_OCInitStructure.TIM_OCIdleState = TIM_OCIdleState_Set;
    TIM_OC2Init(TIM8, &TIM_OCInitStructure);	
    TIM_OC2PreloadConfig(TIM8, TIM_OCPreload_Enable);
    TIM_ARRPreloadConfig(TIM8, ENABLE);	
    TIM_CtrlPWMOutputs(TIM8, ENABLE);

    TIM_Cmd(TIM1, ENABLE);  //启动定时器TIM1的计数功能  
	//TIM_Cmd(TIM8, ENABLE);  // 无需启动定时器TIM8的计数功能,该功能由trgo信号使能
}

  • 3
    点赞
  • 54
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
具体操作要求如下: 以给定频率输出脉冲脉冲数无限制 以给定频率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公司能够能够简化器件的使用。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值