【速成】蓝桥杯嵌入式省一教程:(七)定时器输入捕获中断与PWM频率占空比测量

前文已经讲述过定时器的两个用法:基本定时中断与PWM输出。本节接着介绍第三种用法:定时器输入捕获中断。

在此之前,需要解释一下前文一直出现过的与定时器有关的概念。

定时器(TIMER):所谓定时器,其基本功能就是定时,我们可以通过设置定时器的频率也就是周期,来帮助我们完成定时功能。

预分配系数(Prescaler):将时钟源的频率进行不同的分频,用于作为定时器的频率。若时钟源的频率为80MHz,预分频系数为80-1,那么得到的定时器频率就是80,000,000/80=1MHz,周期是1us。

计数周期(Counter Period):当定时器计数若干个周期以后,重置定时器的计数。假设计数周期设置为1000-1,那么当定时器完成1000个周期的计数后(按如上设置,也就是1ms后),定时器的计数值从999重置为0,重新开始计数。定时器中断的原理就是当计数值到达设定的计数周期后产生中断。计数周期又叫重装载值(AutoReload)。

比较值(Compare):在正脉冲输出模式下,若设置比较值为100,那么当计数值处于0~99时,端口输出高电平;当计数值处于100~999时,端口输出低电平,得到的效果就是前100us输出高电平,后900us输出低电平,产生一个频率为1kHz,占空比为10%的PWM波。在Cube中,比较值又称为Pulse。

在掌握了这几个概念之后,大家在阅读前几节时对定时器的疑问应该就可以尽数消除。这也就是定时器频率计算公式与PWM频率、占空比计算公式的由来。

所谓定时器输入捕获中断,就是将定时器的某个通道设置为直接输入捕获模式以后,将外部信号接到该通道所在引脚上,当外部信号满足一定条件(如边沿跳变)时,就会触发中断。因此,若我们设置在外部信号发生上升沿跳变时触发中断,在中断程序中读取定时器的计数值后手动将其清零,那么通过所得计数值,结合预先设置好的时钟源频率与预分频系数,就能通过公式计算得到输入信号的频率了。

下面我们就来看看如何测量输入到PA7引脚的信号频率。首先用Cube进行定时器的配置。

cb37cb9a5e2b47e8b3ba682a9cad6378.png

 可见PA7引脚同时作为多个定时器的通道,在这里我们选择TIM3的CH2通道。

 1edefc2972f346e8aca50b6f322bef85.png

同样,为了提高测量频率的精度,我们将预分频系数设置得尽可能小,计数周期设置得尽可能大。随后,打开定时器中断开关。

680a47a14bff4e6b99eab2296a3a9935.png

这样,我们就完成了在Cube中的设置。

与定时器中断类似,在程序初始化时,需要先开启定时器输入捕获中断:

HAL_TIM_IC_Start_IT(&htim3, TIM_CHANNEL_2);   //开启TIM3CH2的输入捕获(IC(Input Capture))中断

然后编写定时器输入捕获中断函数,同样要注意函数名和形参均不能改动!!!可参照下图查找:

cc1f8438ae3e4c5fbc2e8abe92a62cde.png24e2be45f3d141e3bab198c1f545f97b.png

uint16_t prescaler = 1-1;
uint32_t ccl_value;
uint32_t pa7_frq;

void HAL_TIM_IC_CaptureCallback(TIM_HandleTypeDef *htim)    //定时器输入捕获回调函数
{
    if (htim->Instance==TIM3 && htim->Channel==HAL_TIM_ACTIVE_CHANNEL_2) //TIM3CH2触发的中断
    {
        ccl_value = HAL_TIM_ReadCaptureValue(htim, TIM_CHANNEL_2);    //读取捕获值(计数值)
        __HAL_TIM_SetCounter(&htim3, 0);                              //重置计数值
        pa7_frq = 80000000/((prescaler+1)*ccl_value);                 //根据公式计算频率
        HAL_TIM_IC_Start(htim, TIM_CHANNEL_2);                        //重新开启输入捕获
    }
}

在这里,我们用到了HAL_TIM_ReadCaptrueValue来读取当前捕获的计数值,其定义如下:

/**
  * @brief  Read the captured value from Capture Compare unit
  * @param  htim TIM handle.
  * @param  Channel TIM Channels to be enabled
  *          This parameter can be one of the following values:
  *            @arg TIM_CHANNEL_1: TIM Channel 1 selected
  *            @arg TIM_CHANNEL_2: TIM Channel 2 selected
  *            @arg TIM_CHANNEL_3: TIM Channel 3 selected
  *            @arg TIM_CHANNEL_4: TIM Channel 4 selected
  * @retval Captured value
  */
uint32_t HAL_TIM_ReadCapturedValue(TIM_HandleTypeDef *htim, uint32_t Channel)

若要测量占空比,则需要在测量频率的基础上,使用另一个通道作为间接输入,并设置为下降沿捕获。这样一来,每当上升沿捕获中断触发,定时器计数清零后,到达第一个下降沿处,间接输入捕获通道捕获从上升沿到下降沿之间的计数值ccl_value_1;到达第二个上升沿处,直接输入捕获通道捕获从上升沿到上升沿之间的计数值ccl_value_2,于是占空比就等于ccl_value_1/ccl_value_2。

在这里我们选择Channel1作为间接输入捕获通道,在Cube中的设置如下:

8690a3434cd745ec9e4d6d61c3c1b920.png

更改代码如下:

uint16_t prescaler = 1-1;
uint32_t ccl_value_1, ccl_value_2;
uint32_t pa7_frq;
float pa7_duty;

void HAL_TIM_IC_CaptureCallback(TIM_HandleTypeDef *htim)    //定时器输入捕获回调函数
{
    if (htim->Instance==TIM3 && htim->Channel==HAL_TIM_ACTIVE_CHANNEL_2) //TIM3CH2触发的中断
    {
        ccl_value_1 = HAL_TIM_ReadCaptureValue(htim, TIM_CHANNEL_1);  //间接
        ccl_value_2 = HAL_TIM_ReadCaptureValue(htim, TIM_CHANNEL_2);  //直接
        __HAL_TIM_SetCounter(&htim3, 0);                              //重置计数值
        pa7_frq = 80000000/((prescaler+1)*ccl_value);                 //根据公式计算频率
        pa7_duty = float(ccl_value_1/ccl_value_2);                    //计算占空比
        HAL_TIM_IC_Start(htim, TIM_CHANNEL_1);                        //重新开启间接输入捕获
        HAL_TIM_IC_Start(htim, TIM_CHANNEL_2);                        //重新开启直接输入捕获
    }
}

下面我们通过第十四届省赛题来总结本节所讲内容:

8ab43d9037144ecba881781035f92d09.png

/* 以下代码添加到task.c中 */

#define PI 3.14

uint16_t prescaler = 1-1;
uint32_t ccl_value;
uint32_t pa7_frq;

/* 测量频率 */
void HAL_TIM_IC_CaptureCallback(TIM_HandleTypeDef *htim)
{
    if (htim->Instance==TIM3 && htim->Channel==HAL_TIM_ACTIVE_CHANNEL_2)
    {
        ccl_value = HAL_TIM_ReadCaptureValue(htim, TIM_CHANNEL_2);
        __HAL_TIM_SetCounter(&htim3, 0);
        pa7_frq = 80000000/((prescaler+1)*ccl_value);
        V = (pa7_frq*2*PI*R)/(100*K);
        HAL_TIM_IC_Start(htim, TIM_CHANNEL_2);
    }
}

  • 10
    点赞
  • 85
    收藏
    觉得还不错? 一键收藏
  • 11
    评论
评论 11
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值