STM32主从模式 精确脉冲数PWM (已实现)

定时器1作为主模式输出PWM,定时器4作为从模式

平台

STM32F407VGT6

特别记录两个问题:

移植通用定时器的程序 没有看手册 结果pwm移植无法输出 磨蹭了一两天
后来发现 是因为 高级定时器 有几个特殊的设置:

  • 重复计数设置:
TIM_TimeBaseStructure.TIM_RepetitionCounter = 0;            //重复计数,一定要=0!!!
  • 输出使能:
TIM_CtrlPWMOutputs(TIM1, ENABLE);   //高级定时器一定要加上,主输出使能

当然 这是使用了库函数的 寄存器的具体设置是:
在这里插入图片描述
越过这两个鸿沟以后 就能正常使用高级定时器输出PWM了

进入正题 主从模式设置

在这里插入图片描述
上图显示的是F4的主从配置关系
我的程序是使用的TIM1作为master TIM4 作为slave
所以 使用的是table59的第三行的ITR0
具体程序是:

/***定时器1主模式***/
void TIM1_config(u32 Cycle)
{
    GPIO_InitTypeDef GPIO_InitStructure;
    TIM_TimeBaseInitTypeDef  TIM_TimeBaseStructure;
    TIM_OCInitTypeDef  TIM_OCInitStructure;
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA|RCC_APB2Periph_TIM1 , ENABLE); 

    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_11;                   //TIM1_CH4 PA11
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;             //复用推挽输出
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_Init(GPIOA, &GPIO_InitStructure);

    TIM_TimeBaseStructure.TIM_Period = Cycle-1;                                                   
    TIM_TimeBaseStructure.TIM_Prescaler =71;                    //设置用来作为TIMx时钟频率除数的预分频值                                                     
    TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1;     //设置时钟分割:TDTS= Tck_tim            
    TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up; //TIM向上计数模式
    TIM_TimeBaseStructure.TIM_RepetitionCounter = 0;            //重复计数,一定要=0!!!
    TIM_TimeBaseInit(TIM1, &TIM_TimeBaseStructure);                                       

    TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM1;          //选择定时器模式:TIM脉冲宽度调制模式1       
    TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable; //比较输出使能
    TIM_OCInitStructure.TIM_Pulse = Cycle/2-1;                    //设置待装入捕获寄存器的脉冲值                                   
    TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_Low;      //输出极性       

    TIM_OC4Init(TIM1, &TIM_OCInitStructure);                                                         

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

    TIM_OC4PreloadConfig(TIM1, TIM_OCPreload_Enable);                              
    TIM_ARRPreloadConfig(TIM1, ENABLE);                                                          
}
/***定时器4从模式***/
void TIM4_config(u32 PulseNum)
{
    TIM_TimeBaseInitTypeDef  TIM_TimeBaseStructure;
    NVIC_InitTypeDef NVIC_InitStructure; 
    RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM4, ENABLE);

    TIM_TimeBaseStructure.TIM_Period = PulseNum-1;   
    TIM_TimeBaseStructure.TIM_Prescaler =0;    
    TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1;     
    TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;  
    TIM_TimeBaseInit(TIM4, &TIM_TimeBaseStructure);  

    TIM_SelectInputTrigger(TIM4, TIM_TS_ITR0);
    //TIM_InternalClockConfig(TIM4);
    TIM4->SMCR|=0x07;                                  //设置从模式寄存器 
    //TIM_ITRxExternalClockConfig(TIM4, TIM_TS_ITR0);

    //TIM_ARRPreloadConfig(TIM4, ENABLE);         
    TIM_ITConfig(TIM4,TIM_IT_Update,DISABLE);

   // NVIC_PriorityGroupConfig(NVIC_PriorityGroup_3);
    NVIC_InitStructure.NVIC_IRQChannel = TIM4_IRQn;        
    NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;
    NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;     
    NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; 
    NVIC_Init(&NVIC_InitStructure);
}
void Pulse_output(u32 Cycle,u32 PulseNum)
{
    TIM4_config(PulseNum);
    TIM_Cmd(TIM4, ENABLE);
    TIM_ClearITPendingBit(TIM4,TIM_IT_Update);
    TIM_ITConfig(TIM4,TIM_IT_Update,ENABLE);
    TIM1_config(Cycle);
    
    TIM_Cmd(TIM1, ENABLE);
    TIM_CtrlPWMOutputs(TIM1, ENABLE);   //高级定时器一定要加上,主输出使能
}


void TIM4_IRQHandler(void) 
{ 
    if (TIM_GetITStatus(TIM4, TIM_IT_Update) != RESET)     // TIM_IT_CC1
    { 
        TIM_ClearITPendingBit(TIM4, TIM_IT_Update); // 清除中断标志位 
        TIM_CtrlPWMOutputs(TIM1, DISABLE);  //主输出使能
        TIM_Cmd(TIM1, DISABLE); // 关闭定时器 
        TIM_Cmd(TIM4, DISABLE); // 关闭定时器 
        TIM_ITConfig(TIM4, TIM_IT_Update, DISABLE); 
        
    } 
}
#include "delay.h"
#include "sys.h"
#include "timer.h"
 

 int main(void)
 {      
    NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2); 
    delay_init();            //延时函数初始化    
    uart_init(115200);   //串口初始化为115200
    Pulse_output(1000,8000);//1KHZ,8000个脉冲
    while(1)
    {
    }     
}  

效果

在这里插入图片描述

总结

输出一定数量PWM脉冲,
主要有几种方法:
1.单脉冲法,需要一个脉冲中断一次,中断次数多,影响效率,而且能保证每次的脉冲连续性么?
2.另一定时器进行中断计数,与1一样,需要频繁中断;
3.用主从定时器门控方式,比较繁琐 其实还好 就是有主从关系 需要看好
4.高级定时器T1、T8的重复计数方式,RCR计数中断,看手册好像这种方式最简单,能满足一部分人要求,缺点是寄存器只有8位,最多实现255个脉冲计数输出。只能设置0x00-0xFF

(((((((((((っ•ω•)っ Σ(σ`•ω•´)σ 起飞!

  • 21
    点赞
  • 177
    收藏
    觉得还不错? 一键收藏
  • 27
    评论
### 回答1: STM32主从定时器是一种嵌入式系统的定时器,可以用于生成固定脉冲数量的输出。 主从定时器由一个主定时器和一个从定时器组成,主定时器负责计数,从定时器负责输出。工作原理是主定时器开始计数后,当计数值达到预设的固定脉冲数量时,从定时器触发输出脉冲。 首先,需要配置主定时器的计数模式和计数器的加载值,以确定计数范围。例如,如果要生成100个脉冲,可以将计数器加载值设置为100-1,即99。 其次,需要配置从定时器的触发模式和输出模式。触发模式确定从定时器何时触发,可以选择主定时器计数值等于预设值时触发。输出模式确定输出脉冲的参数,如脉冲频率、占空比等。 最后,开启主定时器和从定时器,并等待主定时器计数值达到预设的固定脉冲数量。一旦达到,从定时器将触发输出脉冲,完成固定脉冲数量的生成。 需要注意的是,具体的配置和控制方式可能因不同的STM32型号而有所差异。因此,在使用STM32主从定时器生成固定脉冲数量之前,应该仔细阅读所使用的芯片的相关文档,并根据具体的需求进行配置和编程。 ### 回答2: STM32主从定时器可用于输出固定脉冲数量的功能。主从定时器在STM32微控制器中是指由一个主定时器和一个从定时器组成的定时器组。主从定时器之间通过一个信号线相连,主定时器控制从定时器的工作模式和计数脉冲。 首先,我们需要配置主定时器和从定时器的参数。通过设置主定时器的计数方向和频率,以及从定时器的计数方向和分频系数,我们可以实现不同频率和方向的计数。然后,我们需要设置主定时器的计数周期,即所需输出的脉冲数量。 接下来,我们设置主定时器的工作模式为触发模式,并选择一个触发源用于触发从定时器的计数。当主定时器的计数达到设定的周期后,触发源会触发从定时器开始计数。 在主从定时器开始工作后,从定时器会根据主定时器的计数触发信号来进行计数,直至达到所设定的脉冲数量。一旦从定时器计数完成,我们可以设置一个中断标志位来表示脉冲数量已达到预期,并进行相应的处理。 总结起来,使用STM32主从定时器输出固定脉冲数量的功能,我们需要配置主定时器和从定时器的参数,并设置主定时器的计数周期和触发源。通过触发源的触发信号,从定时器开始计数并输出固定数量的脉冲。 ### 回答3: STM32主从定时器可以通过编程实现输出固定脉冲数量。具体步骤如下: 1. 首先,选择一个主定时器和一个从定时器来实现功能。主定时器负责计数,从定时器负责输出脉冲信号。 2. 设置主定时器的计数器模式为向上计数,并设置计数周期(脉冲数量)。 3. 配置主定时器的输入时钟源和预分频系数,使计数器按照指定的频率进行计数。 4. 配置从定时器的输出模式PWM输出。 5. 设置从定时器的输出比较值,使脉冲的高电平和低电平持续时间达到期望的占空比。 6. 启动主定时器和从定时器的计数。 7. 在主定时器的中断服务函数中,检查计数值是否达到预设的脉冲数量。一旦达到,关闭主定时器和从定时器的计数功能。 8. 关闭从定时器的PWM输出功能。 通过以上步骤,就可以实现输出固定脉冲数量的功能。需要注意的是,根据具体的需求,还可以对输出信号的占空比、频率等参数进行进一步调整。
评论 27
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值