一个跟STM32中断优先级有关的话题

近日有人咨询问题如下:

“我用STM32F7 MCU开发产品,用到STemwin。在桌面有一个图标,点击图标后创建一个窗口,窗口中3个按钮,和1listbox。点击窗口中的CANCEL按钮窗口关闭。当我多次打开关闭这个窗口时,程序就会死!经调试,程序是死在了硬件I2Cwhile循环中,如下代码中:

static
HAL_StatusTypeDef I2C_WaitOnFlagUntilTimeout(I2C_HandleTypeDef *hi2c, uint32_t
Flag, FlagStatus Status, uint32_t Timeout) 

  uint32_t tickstart = HAL_GetTick();

    /* Wait until flag is set */

  if(Status == RESET)

  {   

    while(__HAL_I2C_GET_FLAG(hi2c, Flag) ==
RESET)

    {

      /* Check for the Timeout */

      if(Timeout != HAL_MAX_DELAY)

      {

        if((Timeout == 0)||((HAL_GetTick() -
tickstart ) > Timeout))

        {

          hi2c->State= HAL_I2C_STATE_READY;

          /* Process Unlocked */

          __HAL_UNLOCK(hi2c);

          return HAL_TIMEOUT;

        }

      }

    }

  }...........省略 

或者

static
HAL_StatusTypeDef I2C_IsAcknowledgeFailed(I2C_HandleTypeDef *hi2c, uint32_t
Timeout)

{

  uint32_t tickstart = 0x00;

  tickstart = HAL_GetTick(); 

  if(__HAL_I2C_GET_FLAG(hi2c, I2C_FLAG_AF) ==
SET)

  {

    /* Wait until STOP Flag is reset */

    /* AutoEnd should be initiate after AF */

    while(__HAL_I2C_GET_FLAG(hi2c,
I2C_FLAG_STOPF) == RESET)

    {

      /* Check for the Timeout */

      if(Timeout != HAL_MAX_DELAY)

      {

        if((Timeout == 0)||((HAL_GetTick() -
tickstart ) > Timeout))

        {

          hi2c->State= HAL_I2C_STATE_READY;

          /* Process Unlocked */

          __HAL_UNLOCK(hi2c);

          return HAL_TIMEOUT;

        }

      }

    }...........省略

I2C程序是操作电阻触摸屏芯片的,由TIMER3定时器定时调用它。timer3中断优先级是[0,1]I2C程序中超时检测是用的SYSTICK,其中断优先级是[0x0f,0]第一个数字为抢占优先级,第二个数字为响应优先级)。

调试发现,程序死掉后,HAL_GetTick() 返回值始终不变,这样才进入死循环!

怀疑是systick优先级低,将其改为[0,0]最高优先级后,依然会进入上面的死循环!” 

很明显,上面用户代码是基于STM32cube实现的。根据上面用户反馈的问题描述,他是说程序有时会死掉,而且总死在I2C通信代码里。这个I2C通信代码是通过TIM3的中断调用的,其中I2C通信中的TIMEOUT超时检测又是依据SYSTICK中断计时实现。【注:用户设计的合理性,这里暂且不谈】 

应该说用户已经意识到问题的原因了。如果SYSTICK优先级低于TIM3中断优先级,而I2C通信又是在TIM3中断里执行,那在I2C程序运行过程中SYSTICK中断就没法响应,无法进行TICK计数的加减。如果此时I2C通信出现故障,TIMOUT超时检测条件就永远不会成立。程序当然就卡死在那里面了。 

后来用户将SYSTICK的优先级从之前的【0x0f,0】调整为【0,0】后现象仍未改善。他认为【0,0】是最高了。其实,此时SYSTICK的抢占优先级与TIM3抢占优先级是一样的,只是二者响应优先级不一样。如果这样的话,在TIM3 中断服务程序里,SYSTICK仍然无法响应中断进行计数,也就无法通过TIMEOUT检测退出死循环。 

如果反过来,将SYSTICK的抢占优先级设置得比TIM3的抢占优先级高,情况就不同了。比方将SYSTICK的优先级配置为【0,0】,TIM3的优先级配置为【1,0。如果这样,TIM3中断里调用I2C通信程序,如果I2C通信出现故障,TIMEOUT超时检测就不会受阻卡死。因为此时SYSTICK的抢占优先级高于TIM3,它可以打断TIM3中断程序进行TICK的计数计时,当I2C程序检测到TIMEOUT成立时就可以全身而退了。后来建议用户如此调整的确改善。

小结下,问题源于用户对中断优先级的理解不到位。在MCU开发应用中因为中断优先级处理不当而导致困扰其实还挺多的,而且问题往往还比较隐蔽。 

顺便提醒下:对于STM32F7/F4/F3/F1/L1/L4系列芯片的中断优先级往往分为抢占优先级和响应优先级。只有抢占优先级不同时才会发生中断的打断和嵌套。如果抢占优先级一样的话,在同时发生中断事件时,响应优先级高的中断源优先得到响应。如果抢占优先级和响应优先级也一样,同时发生中断时根据其中断向量表的序号决定。

对于基于CORTEX M0/M0+STM32F0/L0系列,它们的中断优先级只有抢占优先级,不分响应优先级。

永不止步步 发表于10-20 09:16 浏览65535次
分享到:

已有0条评论

暂时还没有回复哟,快来抢沙发吧

添加一条新评论

只有登录用户才能评论,请先登录注册哦!

话题作者

永不止步步
金币:67410个|学分:345327个
立即注册
畅学电子网,带你进入电子开发学习世界
专业电子工程技术学习交流社区,加入畅学一起充电加油吧!

x

畅学电子网订阅号