探究STM32F103定时器初始化或程序运行中重配后会立即进入一次中断的原因和解决办法

工作中开发某项目程序,需要使用定时器中断的方式产生设定的脉冲宽度(非PWM模式)。调试过程中发现,定时器中断进入的次数并不是程序中所控制的次数导致输出的脉宽不对,针对这一个BUG展开了深入的探究。

调试中发现当单片机上电给定时器初始化的时候还有刚刚开始产生脉冲的时候会莫名其妙的进入一次中断。由于所需要的脉冲为单个脉冲并非连续脉冲,脉冲宽度从us到ms不等,所以每一次产生单脉冲的时候需要对定时器的预分频器(PSC)和自动重装载寄存器(ARR)重新配置。

尝试在使能定时器之前清除TIMX->SR寄存器相关flag,不能解决问题。

进一步debug。定时器初始化官方库代码如下:

 

 定时器初始化程序最后一步调用了TIMx->EGR = TIM_PSCReloadMode_Immediate; 语句字面意思是将PSC装载模式设置成立即装载模式。

修改定时预分频器的官方库函数如下

函数的最后依旧有一个配置PSC寄存器装载模式的过程。

查找宏定义,#define TIM_PSCReloadMode_Immediate        ((uint16_t)0x0001),,

查阅手册,查看TIMX->EGR寄存器

TIMx->EGR = TIM_PSCReloadMode;这段语句就是把EGR寄存器的UG位配置为1,UG位说明如下

  

看到了对TIMX->EGR寄存器UG位的解释,就明白了,官方库中让PSC预分频器立即更新的方式竟然是粗暴的强行产生一次事件更新,程序在定时器配置之后会被强行进入一次中断。好了,到此整个问题的诊断结束,接下来探究一下解决办法。

通过上网搜索,有一种较广泛的说法是在使能定时器之前调用TIM_ClearITPendingBit(TIM1, TIM_IT_Update);但是在我的工程中使用这种办法并没有效果。

我坚信羊毛都是出在羊身上~~~~~

果然在手册上查到了TIMX->CR1寄存器

该寄存器的URS位如下解释:

TIMX->CR1寄存器的URS位特别提到了设置EGR寄存器UG位产生的更新事件。URS位配置为1时则可以筛选掉除了计数上溢、下溢以外的更新事件。

最后在定时器配置过程中,修改预分频器PSC之前的位置调用一下TIM4->CR1 |= TIM_CR1_URS;语句可完美解决定时器配置后默认进入一次中断的问题。

 

 

  • 9
    点赞
  • 21
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
以下是STM32f103C8定时器2中断初始化的示例代码: ```c #include "stm32f10x.h" void TIM2_IRQHandler(void) { if (TIM_GetITStatus(TIM2, TIM_IT_Update) != RESET) { // 中断处理代码 // ... TIM_ClearITPendingBit(TIM2, TIM_IT_Update); } } void TIM2_Init(void) { TIM_TimeBaseInitTypeDef TIM_InitStruct; // 使能定时器2时钟 RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2, ENABLE); // 定时器2初始化 TIM_InitStruct.TIM_Prescaler = 7199; // 分频系数 TIM_InitStruct.TIM_CounterMode = TIM_CounterMode_Up; // 向上计数 TIM_InitStruct.TIM_Period = 999; // 计数器自动重装值 TIM_InitStruct.TIM_ClockDivision = TIM_CKD_DIV1; // 时钟分割 TIM_TimeBaseInit(TIM2, &TIM_InitStruct); // 定时器2中断初始化 NVIC_InitTypeDef NVIC_InitStruct; NVIC_InitStruct.NVIC_IRQChannel = TIM2_IRQn; // 定时器2中断 NVIC_InitStruct.NVIC_IRQChannelPreemptionPriority = 0; // 抢占优先级 NVIC_InitStruct.NVIC_IRQChannelSubPriority = 1; // 子优先级 NVIC_InitStruct.NVIC_IRQChannelCmd = ENABLE; NVIC_Init(&NVIC_InitStruct); TIM_ITConfig(TIM2, TIM_IT_Update, ENABLE); // 开启定时器2更新中断 TIM_Cmd(TIM2, ENABLE); // 开启定时器2 } ``` 在上述代码,我们首先定义了一个中断处理函数 `TIM2_IRQHandler()`,当定时器2更新中断发生时,该函数被调用。 然后我们定义了一个初始化函数 `TIM2_Init()`,该函数首先使能了定时器2时钟,然后初始化定时器2的相关参数。接着,我们初始化定时器2的中断,并开启了定时器2更新中断定时器2本身。
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值