打印
[STM32F1]

【转】STM32定时器捕获PWM脉冲频率

[复制链接]
1113|4
手机看帖
扫描二维码
随时随地手机跟帖
跳转到指定楼层
楼主
傲视群熊|  楼主 | 2016-8-29 23:12 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
一、概念理解
PWM输入捕获模式是输入捕获模式的特例,自己理解如下
1. 每个定时器有四个输入捕获通道IC1、IC2、IC3、IC4。且IC1 IC2一组,IC3 IC4一组。并且可是设置管脚和寄存器的对应关系。
2. 同一个TIx输入映射了两个ICx信号。
3. 这两个ICx信号分别在相反的极性边沿有效。
4. 两个边沿信号中的一个被选为触发信号,并且从模式控制器被设置成复位模式。
5. 当触发信号来临时,被设置成触发输入信号的捕获寄存器,捕获“一个PWM周期(即连续的两个上升沿或下降沿)”,它等于包含TIM时钟周期的个数(即捕获寄存器中捕获的为TIM的计数个数n)。
6. 同样另一个捕获通道捕获触发信号和下一个相反极性的边沿信号的计数个数m,即(即高电平的周期或低电平的周期)
7. 由此可以计算出PWM的时钟周期和占空比了
    frequency=f(TIM时钟频率)/n。
    duty cycle=(高电平计数个数/n),
    若m为高电平计数个数,则duty cycle=m/n
    若m为低电平计数个数,则duty cycle=(n-m)/n
注:因为计数器为16位,所以一个周期最多计数65535个,所以测得的 最小频率= TIM时钟频率/65535。
二、程序设计与分析
1. 程序概述:选择TIM3作为PWM输入捕获。IC2设置为上升沿,并设置为有效的触发输入信号。所以IC2的捕获寄存器捕获PWM周期,
                   IC1的捕获寄存器捕获PWM的高电平周期。
2.程序代码如下:

  RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3, ENABLE);      //时钟配置

  RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);

  GPIO_InitStructure.GPIO_Pin = GPIO_Pin_7;                               //GPIO配置
  PIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;
  GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  GPIO_Init(GPIOA, &GPIO_InitStructure);

  NVIC_InitStructure.NVIC_IRQChannel = TIM3_IRQn;                     //NVIC配置
  NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;
  NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1;
  NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
  NVIC_Init(&NVIC_InitStructure);
  TIM_ICInitStructure.TIM_Channel = TIM_Channel_2;                   //通道选择
  TIM_ICInitStructure.TIM_ICPolarity = TIM_ICPolarity_Rising;       //上升沿触发
  TIM_ICInitStructure.TIM_ICSelection = TIM_ICSelection_DirectTI;    //管脚与寄存器对应关系
  TIM_ICInitStructure.TIM_ICPrescaler = TIM_ICPSC_DIV1;           //输入预分频。意思是控制在多少个输入周期做一次捕获,如果
//输入的信号频率没有变,测得的周期也不会变。比如选择4分频,则每四个输入周期才做一次捕获,这样在输入信号变化不频繁的情况下,
//可以减少软件被不断中断的次数。
  TIM_ICInitStructure.TIM_ICFilter = 0x0;                            //滤波设置,经历几个周期跳变认定波形稳定0x0~0xF
  TIM_PWMIConfig(TIM3, &TIM_ICInitStructure);                 //根据参数配置TIM外设信息

  TIM_SelectInputTrigger(TIM3, TIM_TS_TI2FP2);                //选择IC2为始终触发源

  TIM_SelectSlaveMode(TIM3, TIM_SlaveMode_Reset);//TIM从模式:触发信号的上升沿重新初始化计数器和触发寄存器的更新事件

  TIM_SelectMasterSlaveMode(TIM3, TIM_MasterSlaveMode_Enable); //启动定时器的被动触发

  TIM_Cmd(TIM3, ENABLE);                                 //启动TIM2
           
  TIM_ITConfig(TIM3, TIM_IT_CC2, ENABLE);     //打开中断
中断处理函数
void TIM3_IRQHandler(void)
{

  TIM_ClearITPendingBit(TIM3, TIM_IT_CC2);                //清楚TIM的中断待处理位

  IC2Value = TIM_GetCapture2(TIM3);                         //读取IC2捕获寄存器的值,即为PWM周期的计数值
  if (IC2Value != 0)
  {
   
    DutyCycle = (TIM_GetCapture1(TIM3) * 100) / IC2Value;         //读取IC1捕获寄存器的值,并计算占空比
   
    Frequency = 72000000 / IC2Value;                                          //计算PWM频率。
  }
  else
  {
    DutyCycle = 0;
    Frequency = 0;
  }
}
注(一):若想改变测量的PWM频率范围,可将TIM时钟频率做分频处理
TIM_TimeBaseStructure.TIM_Period = 0xFFFF;     //周期0~FFFF
  TIM_TimeBaseStructure.TIM_Prescaler = 5;       //时钟分频,分频数为5+1即6分频
  TIM_TimeBaseStructure.TIM_ClockDivision = 0;   //时钟分割
  TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;//模式
  TIM_TimeBaseInit(TIM2, &TIM_TimeBaseStructure);//基本初始化
注注(二):定时器TIM的倍频器X1或X2。在APB分频为1时,倍频值为1,否则为2。


笔记:之前调试的时候,想检测低频,按照上面方法进行分频处理时,怎么都不行,后来才搞明白,由于检测时要进行计数,印记预装载值TIM_Period还是要设置成0xffff,而TIM_Prescaler是进行分频用的,也就是说,如果定时器时钟为72M,则将TIM_Prescaler设置成5后,TIM的时钟为14.4M,此时可以检测的最小频率=14.4M/0xffff,也就是说最小检测频率= 14.4M/TIM_Period;
    之前关于这个TIM_Period自动重载值没有理解好,怎么都调试不同,终于搞定了,记录一下,以便后期查阅

沙发
傲视群熊|  楼主 | 2016-8-29 23:15 | 只看该作者

使用特权

评论回复
板凳
木林森1| | 2016-9-19 16:41 | 只看该作者
我是参考下面这个链接的方法写出来的。楼主转的这个程序变量 IC2Value来的莫名其妙,但是这个变量又是最关键的!!!
呃呃呃  发不了链接,那直接搜“STM32F0xx_TIM输入捕获(计算频率)配置详细过程”——找strong-博客园。
我自己找了好久的,网上鱼龙混杂,心累

使用特权

评论回复
地板
木林森1| | 2016-9-19 16:42 | 只看该作者

使用特权

评论回复
5
傲视群熊|  楼主 | 2016-10-7 22:52 | 只看该作者
木林森1 发表于 2016-9-19 16:41
我是参考下面这个链接的方法写出来的。楼主转的这个程序变量 IC2Value来的莫名其妙,但是这个变量又是最关 ...


找资料确实让人头疼!

使用特权

评论回复
发新帖 我要提问
您需要登录后才可以回帖 登录 | 注册

本版积分规则

51

主题

84

帖子

1

粉丝