[其他ST产品] STM32主从模式 精确脉冲数PWM (已实现)

[复制链接]
1579|15
 楼主| l63t89 发表于 2023-11-24 01:06 | 显示全部楼层 |阅读模式
定时器1作为主模式输出PWM,定时器4作为从模式
平台
STM32F407VGT6

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

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

 楼主| l63t89 发表于 2023-11-24 01:07 | 显示全部楼层
当然 这是使用了库函数的 寄存器的具体设置是:
44420655f86bb5846a.png
 楼主| l63t89 发表于 2023-11-24 01:07 | 显示全部楼层
越过这两个鸿沟以后 就能正常使用高级定时器输出PWM了

 楼主| l63t89 发表于 2023-11-24 01:07 | 显示全部楼层
进入正题 主从模式设置
34212655f86e21627b.png
 楼主| l63t89 发表于 2023-11-24 01:07 | 显示全部楼层
上图显示的是F4的主从配置关系
我的程序是使用的TIM1作为master TIM4 作为slave
所以 使用的是table59的第三行的ITR0
 楼主| l63t89 发表于 2023-11-24 01:08 | 显示全部楼层
具体程序是:


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

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

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

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

  22.     TIM_OC4Init(TIM1, &TIM_OCInitStructure);                                                         

  23.     TIM_SelectMasterSlaveMode(TIM1, TIM_MasterSlaveMode_Enable);
  24.     TIM_SelectOutputTrigger(TIM1, TIM_TRGOSource_Update);
  25.    

  26.     TIM_OC4PreloadConfig(TIM1, TIM_OCPreload_Enable);                              
  27.     TIM_ARRPreloadConfig(TIM1, ENABLE);                                                         
  28. }
  29. /***定时器4从模式***/
  30. void TIM4_config(u32 PulseNum)
  31. {
  32.     TIM_TimeBaseInitTypeDef  TIM_TimeBaseStructure;
  33.     NVIC_InitTypeDef NVIC_InitStructure;
  34.     RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM4, ENABLE);

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

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

  44.     //TIM_ARRPreloadConfig(TIM4, ENABLE);         
  45.     TIM_ITConfig(TIM4,TIM_IT_Update,DISABLE);

  46.    // NVIC_PriorityGroupConfig(NVIC_PriorityGroup_3);
  47.     NVIC_InitStructure.NVIC_IRQChannel = TIM4_IRQn;        
  48.     NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;
  49.     NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;     
  50.     NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
  51.     NVIC_Init(&NVIC_InitStructure);
  52. }
  53. void Pulse_output(u32 Cycle,u32 PulseNum)
  54. {
  55.     TIM4_config(PulseNum);
  56.     TIM_Cmd(TIM4, ENABLE);
  57.     TIM_ClearITPendingBit(TIM4,TIM_IT_Update);
  58.     TIM_ITConfig(TIM4,TIM_IT_Update,ENABLE);
  59.     TIM1_config(Cycle);
  60.    
  61.     TIM_Cmd(TIM1, ENABLE);
  62.     TIM_CtrlPWMOutputs(TIM1, ENABLE);   //高级定时器一定要加上,主输出使能
  63. }


  64. void TIM4_IRQHandler(void)
  65. {
  66.     if (TIM_GetITStatus(TIM4, TIM_IT_Update) != RESET)     // TIM_IT_CC1
  67.     {
  68.         TIM_ClearITPendingBit(TIM4, TIM_IT_Update); // 清除中断标志位
  69.         TIM_CtrlPWMOutputs(TIM1, DISABLE);  //主输出使能
  70.         TIM_Cmd(TIM1, DISABLE); // 关闭定时器
  71.         TIM_Cmd(TIM4, DISABLE); // 关闭定时器
  72.         TIM_ITConfig(TIM4, TIM_IT_Update, DISABLE);
  73.         
  74.     }
  75. }
 楼主| l63t89 发表于 2023-11-24 01:08 | 显示全部楼层
  1. #include "delay.h"
  2. #include "sys.h"
  3. #include "timer.h"


  4. int main(void)
  5. {      
  6.     NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);
  7.     delay_init();            //延时函数初始化   
  8.     uart_init(115200);   //串口初始化为115200
  9.     Pulse_output(1000,8000);//1KHZ,8000个脉冲
  10.     while(1)
  11.     {
  12.     }     
  13. }  
 楼主| l63t89 发表于 2023-11-24 01:08 | 显示全部楼层
 楼主| l63t89 发表于 2023-11-24 01:08 | 显示全部楼层
总结
输出一定数量PWM脉冲,
主要有几种方法:
1.单脉冲法,需要一个脉冲中断一次,中断次数多,影响效率,而且能保证每次的脉冲连续性么?
 楼主| l63t89 发表于 2023-11-24 01:08 | 显示全部楼层
2.另一定时器进行中断计数,与1一样,需要频繁中断;
 楼主| l63t89 发表于 2023-11-24 01:08 | 显示全部楼层
3.用主从定时器门控方式,比较繁琐 其实还好 就是有主从关系 需要看好;
 楼主| l63t89 发表于 2023-11-24 01:08 | 显示全部楼层
4.高级定时器T1、T8的重复计数方式,RCR计数中断,看手册好像这种方式最简单,能满足一部分人要求,缺点是寄存器只有8位,最多实现255个脉冲计数输出。只能设置0x00-0xFF
jf101 发表于 2023-11-30 12:24 | 显示全部楼层
其实库函数版还是比较经典的应用实简
Undshing 发表于 2023-11-30 16:31 来自手机 | 显示全部楼层
老型号库函数还是好用啊
linxi8693 发表于 2024-9-11 21:05 | 显示全部楼层
感谢楼主分享,F103直接到位
LOVEEVER 发表于 2024-9-12 14:59 | 显示全部楼层
非常不错的PWM避坑指南
您需要登录后才可以回帖 登录 | 注册

本版积分规则

101

主题

1077

帖子

1

粉丝
快速回复 在线客服 返回列表 返回顶部
0