深入STM32F103定时器:用TIM2输入捕获精准测量脉冲宽度与频率
深入STM32F103定时器用TIM2输入捕获精准测量脉冲宽度与频率在嵌入式开发中精确测量外部信号的脉冲宽度和频率是一项常见但极具挑战性的任务。无论是工业控制中的旋转编码器、消费电子中的红外遥控信号还是无人机领域的PPM控制信号都需要可靠且精确的测量方法。STM32F103系列微控制器内置的通用定时器TIM2提供了强大的输入捕获功能能够帮助我们实现这一目标而无需依赖外部专用芯片。本文将从一个实际工程角度出发详细介绍如何利用TIM2的输入捕获功能构建高精度的脉冲测量系统。不同于简单的功能介绍我们会深入探讨配置细节、误差处理机制并通过一个完整的旋转编码器测速案例展示从硬件连接到软件滤波的全过程。无论您是正在开发电机控制系统还是需要处理各种数字传感器信号这些技术都能直接应用于您的项目。1. TIM2输入捕获基础与配置1.1 输入捕获工作原理STM32F103的TIM2定时器作为通用定时器提供了4个独立的输入捕获通道。输入捕获的基本原理是当检测到输入引脚上的特定边沿上升沿或下降沿时定时器会立即将当前计数器的值捕获到专门的寄存器中并可以触发中断。通过记录两个连续边沿的捕获值我们就能计算出脉冲的宽度或周期。TIM2的输入捕获功能具有以下关键特性16位计数器最大计数值65535配合预分频器可适应不同频率范围的信号可编程数字滤波器有效消除输入信号抖动边沿检测可独立配置为上升沿、下降沿或双边沿触发直接内存访问(DMA)支持减轻CPU负担适合高频信号采集1.2 硬件连接与引脚配置在使用TIM2的输入捕获功能前首先需要正确配置硬件连接。以STM32F103ZET6为例TIM2的四个通道对应引脚如下通道引脚复用功能CH1PA0TIM2_CH1_ETRCH2PA1TIM2_CH2CH3PA2TIM2_CH3CH4PA3TIM2_CH4对于旋转编码器或红外接收头等常见信号源通常只需使用一个通道如PA0。在硬件设计时需注意确保信号电压在3.3V范围内否则需要电平转换对于长距离传输或噪声环境建议在输入端添加RC滤波避免将定时器输入引脚与其他高速切换的数字信号走线平行布置1.3 软件初始化步骤配置TIM2输入捕获需要按照特定顺序初始化相关寄存器。以下是关键步骤的代码实现void TIM2_InputCapture_Init(void) { GPIO_InitTypeDef GPIO_InitStruct {0}; TIM_ICInitTypeDef TIM_ICInitStruct {0}; TIM_TimeBaseInitTypeDef TIM_TimeBaseInitStruct {0}; // 1. 使能时钟 RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2, ENABLE); RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE); // 2. 配置GPIO为输入 GPIO_InitStruct.GPIO_Pin GPIO_Pin_0; GPIO_InitStruct.GPIO_Mode GPIO_Mode_IPD; // 下拉输入 GPIO_InitStruct.GPIO_Speed GPIO_Speed_50MHz; GPIO_Init(GPIOA, GPIO_InitStruct); // 3. 配置时基 TIM_TimeBaseInitStruct.TIM_Period 0xFFFF; // 最大计数值 TIM_TimeBaseInitStruct.TIM_Prescaler 72 - 1; // 72MHz/72 1MHz TIM_TimeBaseInitStruct.TIM_ClockDivision TIM_CKD_DIV1; TIM_TimeBaseInitStruct.TIM_CounterMode TIM_CounterMode_Up; TIM_TimeBaseInit(TIM2, TIM_TimeBaseInitStruct); // 4. 配置输入捕获 TIM_ICInitStruct.TIM_Channel TIM_Channel_1; TIM_ICInitStruct.TIM_ICPolarity TIM_ICPolarity_Rising; // 上升沿触发 TIM_ICInitStruct.TIM_ICSelection TIM_ICSelection_DirectTI; TIM_ICInitStruct.TIM_ICPrescaler TIM_ICPSC_DIV1; // 不分频 TIM_ICInitStruct.TIM_ICFilter 0x0; // 不滤波 TIM_ICInit(TIM2, TIM_ICInitStruct); // 5. 使能捕获中断 TIM_ITConfig(TIM2, TIM_IT_CC1, ENABLE); // 6. 配置NVIC NVIC_InitTypeDef NVIC_InitStruct; NVIC_InitStruct.NVIC_IRQChannel TIM2_IRQn; NVIC_InitStruct.NVIC_IRQChannelPreemptionPriority 0; NVIC_InitStruct.NVIC_IRQChannelSubPriority 1; NVIC_InitStruct.NVIC_IRQChannelCmd ENABLE; NVIC_Init(NVIC_InitStruct); // 7. 启动定时器 TIM_Cmd(TIM2, ENABLE); }提示预分频器值的选择需要根据被测信号频率范围确定。1MHz的计数频率(72MHz/72)适合测量1us-65ms范围内的脉冲如需测量更窄或更宽的脉冲应相应调整预分频值。2. 脉冲测量算法实现2.1 基本测量方法最简单的脉冲测量方法是记录两个连续边沿的捕获值之差。对于频率测量可以记录两个相同极性边沿之间的时间差对于占空比测量则需要分别捕获上升沿和下降沿。以下是一个基本的中断服务程序框架volatile uint32_t lastCapture 0; volatile uint32_t pulseWidth 0; volatile uint32_t period 0; void TIM2_IRQHandler(void) { if(TIM_GetITStatus(TIM2, TIM_IT_CC1) ! RESET) { uint32_t currentCapture TIM_GetCapture1(TIM2); // 计算脉冲宽度或周期 if(currentCapture lastCapture) { pulseWidth currentCapture - lastCapture; } else { // 处理计数器溢出 pulseWidth (0xFFFF - lastCapture) currentCapture; } lastCapture currentCapture; // 清除中断标志 TIM_ClearITPendingBit(TIM2, TIM_IT_CC1); } }这种方法虽然简单但存在两个主要问题无法处理计数器溢出的情况当脉冲宽度超过计数器周期时测量会出错2.2 溢出计数增强算法为了克服上述限制我们需要引入溢出计数器来扩展测量范围。具体实现如下volatile uint32_t overflows 0; volatile uint32_t lastCapture 0; volatile uint32_t pulseWidth 0; void TIM2_IRQHandler(void) { // 处理更新事件(溢出) if(TIM_GetITStatus(TIM2, TIM_IT_Update) ! RESET) { overflows; TIM_ClearITPendingBit(TIM2, TIM_IT_Update); } // 处理捕获事件 if(TIM_GetITStatus(TIM2, TIM_IT_CC1) ! RESET) { uint32_t currentCapture TIM_GetCapture1(TIM2); uint32_t captureDifference; if(currentCapture lastCapture) { captureDifference currentCapture - lastCapture; } else { captureDifference (0xFFFF - lastCapture) currentCapture; } // 计算实际时间考虑溢出 pulseWidth (overflows * 0x10000) captureDifference; overflows 0; lastCapture currentCapture; TIM_ClearITPendingBit(TIM2, TIM_IT_CC1); } }注意使用溢出计数时必须同时使能更新中断TIM_ITConfig(TIM2, TIM_IT_Update, ENABLE);2.3 双边沿捕获与占空比测量要测量脉冲的占空比需要同时捕获上升沿和下降沿。TIM2支持在每个通道上独立配置极性我们可以利用这一特性void TIM2_DualEdgeCapture_Init(void) { TIM_ICInitTypeDef TIM_ICInitStruct; // 配置通道1为上升沿触发 TIM_ICInitStruct.TIM_Channel TIM_Channel_1; TIM_ICInitStruct.TIM_ICPolarity TIM_ICPolarity_Rising; TIM_ICInitStruct.TIM_ICSelection TIM_ICSelection_DirectTI; TIM_ICInitStruct.TIM_ICPrescaler TIM_ICPSC_DIV1; TIM_ICInitStruct.TIM_ICFilter 0x04; // 适当滤波 TIM_ICInit(TIM2, TIM_ICInitStruct); // 配置通道2为下降沿触发(同一引脚) TIM_ICInitStruct.TIM_Channel TIM_Channel_2; TIM_ICInitStruct.TIM_ICPolarity TIM_ICPolarity_Falling; TIM_ICInitStruct.TIM_ICSelection TIM_ICSelection_IndirectTI; TIM_ICInit(TIM2, TIM_ICInitStruct); // 使能两个通道的中断 TIM_ITConfig(TIM2, TIM_IT_CC1 | TIM_IT_CC2, ENABLE); }对应的中断处理程序需要区分两种边沿事件void TIM2_IRQHandler(void) { static uint32_t riseTime 0; static uint32_t fallTime 0; if(TIM_GetITStatus(TIM2, TIM_IT_CC1) ! RESET) // 上升沿 { riseTime TIM_GetCapture1(TIM2); // 计算周期和占空比... TIM_ClearITPendingBit(TIM2, TIM_IT_CC1); } if(TIM_GetITStatus(TIM2, TIM_IT_CC2) ! RESET) // 下降沿 { fallTime TIM_GetCapture2(TIM2); // 计算脉冲宽度... TIM_ClearITPendingBit(TIM2, TIM_IT_CC2); } // 处理溢出事件... }3. 旋转编码器测速实战3.1 硬件系统设计旋转编码器是工业控制中常用的位置和速度传感器。以常见的增量式编码器为例它通常输出两路正交信号(A相和B相)和一个索引信号(Z相)。我们可以使用TIM2的输入捕获功能测量A相信号的频率或周期从而计算出转速。典型连接方式如下编码器A相 → PA0(TIM2_CH1)编码器B相 → PA1(可选用于方向判断)编码器Z相 → 不使用或连接至其他GPIOVCC → 3.3V(注意编码器电压等级)GND → 共地提示对于高分辨率编码器(如每转1000脉冲以上)应考虑使用TIM2的编码器接口模式而非输入捕获以获得更好的性能。3.2 软件实现与滤波旋转编码器测速的核心是准确测量脉冲周期并转换为转速。以下是关键实现步骤配置输入捕获如前面章节所述设置TIM2通道1为上升沿触发周期测量使用溢出计数增强算法获取精确的脉冲周期转速计算根据编码器分辨率和测量周期计算转速软件滤波对测量结果进行平滑处理转速计算公式为 [ \text{RPM} \frac{60 \times 10^6}{\text{PPR} \times \text{PeriodInUs}} ] 其中PPR是编码器每转脉冲数。考虑到机械振动和电气噪声可能导致测量波动我们需要添加软件滤波。一个简单有效的方法是移动平均滤波#define FILTER_WINDOW_SIZE 5 typedef struct { uint32_t buffer[FILTER_WINDOW_SIZE]; uint8_t index; uint32_t sum; } MovingAverageFilter; void Filter_Init(MovingAverageFilter* filter) { memset(filter, 0, sizeof(MovingAverageFilter)); } uint32_t Filter_Update(MovingAverageFilter* filter, uint32_t newValue) { // 减去最旧的值 filter-sum - filter-buffer[filter-index]; // 添加新值 filter-sum newValue; filter-buffer[filter-index] newValue; // 更新索引 filter-index (filter-index 1) % FILTER_WINDOW_SIZE; // 返回平均值 return filter-sum / FILTER_WINDOW_SIZE; }3.3 性能优化技巧在实际应用中可能需要进一步优化测量系统动态调整预分频器根据当前转速自动调整TIM2预分频值保持测量精度双重缓冲使用DMA将捕获值直接传输到内存减少中断延迟硬件滤波适当配置TIM2的输入滤波器参数(ICFilter)平衡响应速度和抗噪能力空闲检测当编码器停止时自动进入低功耗模式以下是动态调整预分频器的示例代码void Adjust_TIM2_Prescaler(uint32_t measuredPeriod) { uint16_t newPrescaler TIM_GetPrescaler(TIM2); if(measuredPeriod 60000) { // 接近溢出 newPrescaler newPrescaler / 2; if(newPrescaler 1) newPrescaler 1; } else if(measuredPeriod 1000) { // 分辨率不足 newPrescaler newPrescaler * 2; if(newPrescaler 7200) newPrescaler 7200; // 最小10kHz计数 } if(newPrescaler ! TIM_GetPrescaler(TIM2)) { TIM_SetPrescaler(TIM2, newPrescaler); // 需要重新校准测量参数 } }4. 高级应用与调试技巧4.1 多通道同步测量TIM2的四个输入捕获通道可以独立工作这使得同时测量多个信号成为可能。例如在电机控制系统中我们可能需要同时测量电机编码器信号电流保护信号的脉冲宽度温度传感器的PWM输出用户控制输入配置多通道输入捕获时需注意每个通道可以独立设置极性和滤波器中断服务程序需要正确区分不同通道的事件高频率信号应分配到不同定时器避免中断冲突以下是三通道配置示例void TIM2_MultiChannel_Init(void) { // 初始化GPIO(PA0, PA1, PA2)... // 通道1: 上升沿触发8周期滤波 TIM_ICInitStructure.TIM_Channel TIM_Channel_1; TIM_ICInitStructure.TIM_ICPolarity TIM_ICPolarity_Rising; TIM_ICInitStructure.TIM_ICFilter 0x08; TIM_ICInit(TIM2, TIM_ICInitStructure); // 通道2: 下降沿触发4周期滤波 TIM_ICInitStructure.TIM_Channel TIM_Channel_2; TIM_ICInitStructure.TIM_ICPolarity TIM_ICPolarity_Falling; TIM_ICInitStructure.TIM_ICFilter 0x04; TIM_ICInit(TIM2, TIM_ICInitStructure); // 通道3: 双边沿触发无滤波 TIM_ICInitStructure.TIM_Channel TIM_Channel_3; TIM_ICInitStructure.TIM_ICPolarity TIM_ICPolarity_BothEdge; TIM_ICInitStructure.TIM_ICFilter 0x00; TIM_ICInit(TIM2, TIM_ICInitStructure); // 使能所有通道中断 TIM_ITConfig(TIM2, TIM_IT_CC1 | TIM_IT_CC2 | TIM_IT_CC3, ENABLE); }4.2 输入捕获与PWM输出联合应用TIM2的输入捕获和PWM输出功能可以协同工作实现更复杂的应用。例如可以构建一个频率-电压转换器使用输入捕获测量输入信号频率根据频率计算所需的PWM占空比使用TIM2的另一通道输出相应PWM信号这种技术常用于模拟转速表、亮度调节等场景。关键优势是所有操作都在同一定时器内完成无需CPU频繁干预。4.3 调试与问题排查输入捕获系统常见的调试挑战包括无中断触发检查GPIO配置、定时器使能、中断优先级设置测量值不稳定调整输入滤波器参数检查硬件连接和接地计数器溢出处理错误验证溢出中断是否使能检查计算逻辑高频率信号丢失减少中断处理时间考虑使用DMA一个实用的调试技巧是利用定时器的从模式功能将输入信号同时路由到输出比较通道用示波器对比原始信号和捕获事件// 配置通道4为PWM输出反映通道1的输入状态 TIM_OCInitTypeDef TIM_OCInitStructure; TIM_OCInitStructure.TIM_OCMode TIM_OCMode_PWM1; TIM_OCInitStructure.TIM_OutputState TIM_OutputState_Enable; TIM_OCInitStructure.TIM_Pulse 10; // 固定脉冲宽度 TIM_OCInitStructure.TIM_OCPolarity TIM_OCPolarity_High; TIM_OC4Init(TIM2, TIM_OCInitStructure); TIM_OC4PreloadConfig(TIM2, TIM_OCPreload_Enable); // 配置从模式触发输出 TIM_SelectSlaveMode(TIM2, TIM_SlaveMode_Trigger); TIM_SelectInputTrigger(TIM2, TIM_TS_TI1FP1); TIM_SelectMasterSlaveMode(TIM2, TIM_MasterSlaveMode_Enable);4.4 低功耗优化对于电池供电设备输入捕获系统的功耗优化至关重要动态时钟调整当信号频率较低时降低定时器时钟频率间歇工作模式仅在预期有信号时使能定时器利用唤醒中断配置输入捕获事件从低功耗模式唤醒MCU关闭未用功能禁用不需要的定时器功能和外设时钟以下是低功耗配置示例void Enter_LowPowerMode(void) { // 降低定时器时钟 RCC_PCLK1Config(RCC_HCLK_Div8); // APB1时钟9MHz // 重新配置预分频器保持相同计数频率 uint32_t newPrescaler (TIM_GetPrescaler(TIM2) 1) * 8 - 1; TIM_SetPrescaler(TIM2, newPrescaler); // 配置唤醒中断 EXTI_InitTypeDef EXTI_InitStructure; EXTI_InitStructure.EXTI_Line EXTI_Line0; // PA0 EXTI_InitStructure.EXTI_Mode EXTI_Mode_Interrupt; EXTI_InitStructure.EXTI_Trigger EXTI_Trigger_Rising; EXTI_InitStructure.EXTI_LineCmd ENABLE; EXTI_Init(EXTI_InitStructure); // 进入STOP模式 PWR_EnterSTOPMode(PWR_Regulator_LowPower, PWR_STOPEntry_WFI); // 唤醒后恢复时钟配置 SystemInit(); // 重置时钟 TIM2_InputCapture_Init(); // 重新初始化定时器 }
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2629557.html
如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!