打印
[应用相关]

STM32输入捕获详解

[复制链接]
850|2
手机看帖
扫描二维码
随时随地手机跟帖
跳转到指定楼层
楼主
八层楼|  楼主 | 2024-12-16 15:32 | 只看该作者 |只看大图 回帖奖励 |倒序浏览 |阅读模式
一、引言
        在嵌入式系统开发中,STM32 系列微控制器因其强大的性能和丰富的功能被广泛应用。其中,输入捕获功能是 STM32 定时器的一个重要特性,它可以用来测量外部信号的周期、频率和占空比等参数,在电机控制、速度测量、通信同步等众多领域都有着重要的应用。本文将详细介绍 STM32 的输入捕获功能,包括其原理、配置步骤以及程序示例。

二、输入捕获原理
STM32 的输入捕获,简单来说就是通过检测 TIMx_CHx 上的边沿信号,在边沿信号发生跳变(比如上升沿/下降沿/上升沿和下降沿均触发)的时候,将当前定时器计数器的值(TIMx_CNT)存放到对应的通道的捕获/比较寄存器(TIMx_CCRx)里面,完成一次捕获。同时还可以配置捕获时是否触发中断或 DMA 等操作。

例如,我们要测量一个高电平脉冲的宽度,首先设置定时器通道为上升沿捕获,在上升沿到来时,会捕获到当前的 TIMx_CNT 值,然后立即清零 TIMx_CNT,并设置通道为下降沿捕获,当下降沿到来时,又会发生捕获事件,得到此时的 TIMx_CNT 值。这样,前后两次 TIMx_CNT 的差值,就是高电平的脉宽。如果脉宽比较长,定时器可能会产生多次溢出,这就需要我们对定时器溢出进行处理,以保证测量结果的准确性。

三、寄存器介绍
TIMx_ARR:自动重装载寄存器,用来设置定时器的计数周期。当定时器计数器的值达到 ARR 的值时,会重新从 0 开始计数。
TIMx_PSC:预分频寄存器,用于对定时器的时钟源进行分频,以得到合适的计数频率。
TIMx_CCMR1:捕获/比较模式寄存器 1,用于配置输入捕获的通道方向、触发极性、输入分频等参数。
TIMx_CCER:捕获/比较使能寄存器,用来使能或禁止捕获/比较通道的输入捕获功能。
TIMx_DIER:DMA/中断使能寄存器,用于开启或关闭捕获/比较通道的中断或 DMA 请求。
TIMx_CR1:控制寄存器 1,用于控制定时器的启动、停止、计数模式等。
TIMx_CCR1:捕获/比较寄存器 1,用来存储捕获发生时的 TIMx_CNT 值。
四、配置步骤
以下是以 STM32F4 为例,配置输入捕获的一般步骤:

1.开启时钟
开启定时器的时钟。例如,如果要使用 TIM5,需要调用 RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM5, ENABLE) 来使能 TIM5 的时钟。
开启对应 GPIO 引脚的时钟。例如,如果捕获信号连接到 PA0 引脚,需要调用 RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE) 来使能 GPIOA 的时钟。
2.GPIO 初始化
将捕获信号对应的 GPIO 引脚配置为复用功能。例如,如果使用 TIM5 的通道 1 捕获信号,且信号连接到 PA0 引脚,需要将 PA0 引脚配置为复用功能,并设置复用功能为 AF2(具体的复用功能值根据芯片手册确定)。
根据需要配置 GPIO 引脚的上拉、下拉电阻等。


3.初始化定时器
        设置 TIM_TimeBaseInitTypeDef 结构体的参数,包括自动重装载值 TIM_Period、预分频值 TIM_Prescaler、计数模式 TIM_CounterMode 等。例如,如果要设置定时器的计数周期为 1000,预分频值为 7199(使得定时器的计数频率为 10kHz),计数模式为向上计数,可以这样设置:



如果需要,可以设置定时器的重复计数器值 TIM_RepetitionCounter,该参数在高级定时器中使用较多。

4.配置输入捕获模式
        设置 TIM_ICInitTypeDef 结构体的参数,包括捕获通道 TIM_Channel、触发极性 TIM_ICPolarity、输入选择 TIM_ICSelection、输入分频 TIM_ICPrescaler、输入滤波 TIM_ICFilter 等。例如,如果要配置 TIM5 的通道 1 为上升沿捕获,不分频,不滤波,可以这样设置:

IM_ICInitTypeDef TIM5_ICInitStructure;
TIM5_ICInitStructure.TIM_Channel = TIM_Channel_1;
TIM5_ICInitStructure.TIM_ICPolarity = TIM_ICPolarity_Rising;
TIM5_ICInitStructure.TIM_ICSelection = TIM_ICSelection_DirectTI;
TIM5_ICInitStructure.TIM_ICPrescaler = TIM_ICPSC_DIV1;
TIM5_ICInitStructure.TIM_ICFilter = 0x00;
TIM_ICInit(TIM5, &TIM5_ICInitStructure);
5.使能捕获和更新中断
如果需要在捕获事件发生时产生中断,需要调用 TIM_ITConfig 函数开启捕获中断和更新中断。例如:

TIM_ITConfig(TIMx, TIM_IT_Update | TIM_IT_CC1, ENABLE);
6.设置中断分组并编写中断服务函数
设置中断分组,通常使用 NVIC_Init 函数来完成。中断分组的设置决定了中断的优先级。

在中断服务函数中,需要判断中断类型,并进行相应的数据处理。例如,如果是捕获中断,需要读取捕获寄存器的值;如果是更新中断,可能需要处理定时器溢出等情况。以下是一个简单的中断服务函数示例:

void TIMx_IRQHandler(void)
{
    if (TIM_GetITStatus(TIMx, TIM_IT_Update)!= RESET)
    {
        // 处理更新中断
    }
    if (TIM_GetITStatus(TIMx, TIM_IT_CC1)!= RESET)
    {
        // 处理捕获中断
        // 读取捕获寄存器的值,进行数据处理
        uint16_t captureValue = TIM_GetCapture1(TIMx);
        // 清除中断标志位
        TIM_ClearITPendingBit(TIMx, TIM_IT_CC1 | TIM_IT_Update);
    }
}
7.使能定时器
调用 TIM_Cmd 函数使能定时器,开始输入捕获。例如:

TIM_Cmd(TIMx, ENABLE);


五、程序示例
以下是一个完整的 STM32 输入捕获程序示例,用于测量 PA0 引脚上输入信号的高电平脉宽,并通过串口打印结果:

#include "stm32f4xx.h"
#include <stdio.h>

// 定义捕获状态和捕获值变量
volatile uint8_t TIM5CH1_CAPTURE_STA = 0;
volatile uint16_t TIM5CH1_CAPTURE_VAL = 0;

// 定时器 5 中断服务函数
void TIM5_IRQHandler(void)
{
    if ((TIM5CH1_CAPTURE_STA & 0X80) == 0) // 还未成功捕获
    {
        if (TIM_GetITStatus(TIM5, TIM_IT_Update)!= RESET) // 检测是否发生更新中断
        {
            if (TIM5CH1_CAPTURE_STA & 0X40) // 已经捕获到高电平
            {
                if ((TIM5CH1_CAPTURE_STA & 0X3F) == 0X3F) // 定时器溢出
                {
                    TIM5CH1_CAPTURE_STA |= 0X80; // 标记为成功捕获
                    TIM5CH1_CAPTURE_VAL = 0XFFFF;
                }
                else
                {
                    TIM5CH1_CAPTURE_STA++; // 对溢出次数进行计数
                }
            }
        }
        if (TIM_GetITStatus(TIM5, TIM_IT_CC1)!= RESET) // 判断是否发生捕获事件
        {
            if (TIM5CH1_CAPTURE_STA & 0X40) // 如果已经捕获到上升沿
            {
                // 捕获到下降沿,计算脉宽
                TIM5CH1_CAPTURE_VAL = TIM_GetCapture1(TIM5);
                TIM5CH1_CAPTURE_STA |= 0X80; // 标记为成功捕获
            }
            else
            {
                // 捕获到上升沿,记录当前定时器值,并设置为下降沿捕获
                TIM_SetCounter(TIM5, 0);
                TIM5CH1_CAPTURE_STA = 0X40;
                TIM_OC1PolarityConfig(TIM5, TIM_ICPolarity_Falling);
            }
        }
    }
    // 清除中断标志位
    TIM_ClearITPendingBit(TIM5, TIM_IT_CC1 | TIM_IT_Update);
}

int main(void)
{
    // 开启 TIM5 和 GPIOA 时钟
    RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM5, ENABLE);
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);

    // GPIOA 引脚初始化
    GPIO_InitTypeDef GPIO_InitStructure;
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_DOWN;
    GPIO_Init(GPIOA, &GPIO_InitStructure);

    // 连接 PA0 到 TIM5 的通道 1
    GPIO_PinAFConfig(GPIOA, GPIO_PinSource0, GPIO_AF_TIM5);

    // 定时器 5 初始化
    TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure;
    TIM_TimeBaseStructure.TIM_Period = 0xFFFF;
    TIM_TimeBaseStructure.TIM_Prescaler = 7199;
    TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1;
    TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;
    TIM_TimeBaseInit(TIM5, &TIM_TimeBaseStructure);

    // 输入捕获配置
    TIM_ICInitTypeDef TIM5_ICInitStructure;
    TIM5_ICInitStructure.TIM_Channel = TIM_Channel_1;
    TIM5_ICInitStructure.TIM_ICPolarity = TIM_ICPolarity_Rising;
    TIM5_ICInitStructure.TIM_ICSelection = TIM_ICSelection_DirectTI;
    TIM5_ICInitStructure.TIM_ICPrescaler = TIM_ICPSC_DIV1;
    TIM5_ICInitStructure.TIM_ICFilter = 0x00;
    TIM_ICInit(TIM5, &TIM5_ICInitStructure);

    // 使能捕获和更新中断
    TIM_ITConfig(TIM5, TIM_IT_Update | TIM_IT_CC1, ENABLE);

    // 设置中断分组
    NVIC_InitTypeDef NVIC_InitStructure;
    NVIC_InitStructure.NVIC_IRQChannel = TIM5_IRQn;
    NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;
    NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;
    NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
    NVIC_Init(&NVIC_InitStructure);

    // 使能定时器
    TIM_Cmd(TIM5, ENABLE);

    while (1)
    {
        if (TIM5CH1_CAPTURE_STA & 0X80)
        {
            // 计算高电平脉宽
            uint32_t pulseWidth = (TIM5CH1_CAPTURE_STA & 0X3F) * 65536 + TIM5CH1_CAPTURE_VAL;
            // 打印结果
            printf("高电平脉宽:%d us\n", pulseWidth);
            // 清除捕获状态
            TIM5CH1_CAPTURE_STA = 0;
        }
    }
}

在上述程序中,首先初始化了定时器和 GPIO 引脚,配置了输入捕获模式和中断。在中断服务函数中,根据上升沿和下降沿的捕获情况,记录定时器的值并计算高电平脉宽。在主循环中,不断检查是否成功捕获到高电平脉宽,并打印结果。

六、总结
        STM32 的输入捕获功能是一个非常强大的工具,可以帮助我们准确地测量外部信号的参数。通过合理地配置定时器和相关寄存器,以及编写中断服务函数,我们可以轻松地实现输入捕获功能。在实际应用中,我们需要根据具体的需求选择合适的定时器和捕获通道,并注意处理定时器溢出等情况,以保证测量结果的准确性。希望本文对大家理解和使用 STM32 的输入捕获功能有所帮助。
————————————————

                            版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。

原文链接:https://blog.csdn.net/qq_38072731/article/details/144412293

使用特权

评论回复
沙发
发顺丰更大nc| | 2024-12-23 16:59 | 只看该作者
为了完成输入捕获功能,我们需要将捕获信号对应的 GPIO 引脚配置为输入模式,并关联到定时器的捕获通道。

使用特权

评论回复
板凳
一路向北lm| | 2024-12-24 09:14 | 只看该作者
要使用STM32的输入捕捉功能,通常需要进行以下步骤:

选择合适的定时器并配置其参数,如时钟源、预分频器、计数模式等。
配置定时器的输入捕捉通道和触发条件。这通常涉及到设置输入捕获中断和捕获触发事件的类型(如边沿触发或水平触发)。
启动定时器并开始捕获外部事件。
在中断服务程序中处理输入捕捉事件,获取事件发生的时刻或其他相关信息。

使用特权

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

本版积分规则

91

主题

4166

帖子

2

粉丝