
01引言 某客户设计需要启动定时器在 3ms 后产生中断,其后定时器不再运行,直至下一次软件要求再次启动定时器产生中断,实测代码后发现定时器启动后立即产生了超时中断。5 y$ ~3 [: p! @7 m " E+ n( g8 d1 w6 F3 U; Q 02调研 客户通过 STM32CubeMX 配置 TIM7 并生成工程,在主循环中添加定时器启动代码;在定时器中断处理函数处理 update event 回调时停止定时器,并翻转一个 I/O 脚指示定时器启停,代码如下: " M* o6 ]) | b N2 ] ![]() 图1.启动定时器 ; C, c \. y* J7 O; J' I ![]() 图2.定时器中断回调 客户期待的结果是启动定时器时 I/O 拉高并保持 3ms,当 3ms 到达后产生中断拉低I/O,但实测时发现,定时器启动后立即产生了超时中断(高电平持续时间约 2us): & F7 E/ @/ d9 q0 Q) b8 p ![]() 图3.实测结果 03分析 应用代码在初始化定时器时会调用 HAL_TIM_Base_Init( )接口,此接口会调用TIM_Base_SetConfig( ) 配置定时器,并产生更新事件(TIMx->EGR = TIM_EGR_UG)加载寄存器,此事件标志 UIF 会被置位,在调用 HAL_TIM_Base_Start_IT( ) 启动定时器,在此使能定时器中断时,由于 UIF 已经置位,所以会立即触发并进入中断处理函数,中断回调函数会停止定时器计数,并禁止定时器中断;但当从中断处理函数返回继续执行HAL_TIM_Base_Start_IT( )时,此接口会使能定时器开始计数,进而在下一次调用HAL_TIM_Base_Start_IT( )时又会立即产生中断,循环往复,详细时序和具体代码如下:/ M5 p: S7 r% q' {% u* U * G! d1 k5 D0 ` W# U ![]() 图4.问题产生时序描述 0 j( _8 @0 {* I$ [ ![]() 图5.定时器启动代码 7 V* e: j! Q' _& x* k- @+ C& |/ T* n% b M9 p 04处理6 ]/ r4 f' W# C; ~5 g2 G 修改代码,在启动定时器前强制停止定时器计数、清除中断位、清除 NVIC 挂起的中断后,再启动定时器,详见下图红框内代码;+ S8 U. P1 v- d ' q" g S# U0 Y! u ![]() 图6.问题修正代码 ![]() 图7.修正 BUG 后的运行结果 8 s' s0 J. l" u* t- E+ m. S 1 R# E0 [8 F( O. ~9 E0 y( v8 j 05小结 在碰到这类定时器异常问题时,可以利用 I/O 口指示运行状态,结合代码分析找到原因并加以解决。 3 k \" D, o* {- {4 k2 e/ W2 a2 q 转载自: STM32单片机 如有侵权请联系删除 / O6 n+ o- c+ w1 T + w( p, E; _9 M! F) ?' L2 i$ U |
狂欢三】STM32C031使用TIM定时器DMA方式实现WS2812彩灯输出(三)
【狂欢三】STM32C031使用TIM定时器DMA方式实现PWM输出(二)
【狂欢三】STM32C031使用TIM定时器PWM输出
stm32使用定时器触发dma传输,启动dma没反应的几种情况的解决方法
基于STM32串口中断之缓存区溢出卡死经验分享
基于STM32数组越界异常中断经验分享
基于STM32之数组越界异常中断经验分享
定时器剩余通道是否可以做PWM输出呢?
基于STM32连接参数更新进程后导致断连的问题分析
基于STM32连接参数更新进程后导致断连的问题分析