STM32 IAP应用中的几个常见问题
我们经常会基于STM32芯片来自行编写用户升级程序,通过该升级程序对片内应用程序代码进行更新升级。我们通常称该过程为IAP(In Application Programming),称该升级程序为用户bootloader或IAP程序,用来被更新或升级的代码程序为APP程序。
一般来讲,IAP程序放在芯片启动后默认的起始地址空间。对STM32而言,多是在片内0x8000000起始的Flash地址空间,而用户APP程序地址空间由用户自行指定。
下面基于结合一个小实例来聊聊在做STM32 IAP应用中可能遇到的几类常见问题。
这里使用STM32F302_Nucleo板,模拟设计了一套IAP和一套APP代码,在IAP中使用了TIM2,在其更新中断里使用UART1打印输出目前所在工作区,完成三次打印输出动作后跳转到APP去执行程序。我在APP程序里使用了TIM1,同样在其更新中断里使用UART1打印输出当前所在工作区,也是三次动作后跳转回IAP程序区。【开发环境为MDK,基于stm32Cbue库组织代码】
基于ARM MDK环境的IAP相关配置如下:
基于ARM MDK环境的APP相关配置如下图所示:
根据上述配置,IAP放在片内0x8000000开始的0x3000大小的flash空间内,APP放在0x8003000开始的剩余空间内。最后的结果应该是下面这样子,程序在两个区间来回跳转,输出结果可控有规律,符合设计预期。【下面为Printf输出截图】
现在具体看看我们在做IAP应用中可能遇到的几类问题。
第一类的问题,就是指定APP程序空间的起始地址时出错。
虽然说,对于用户APP程序空间,用户可以自行安排,但我们在安排APP程序空间的起始地址时要注意两点。第一点,我们要给IAP程序代码留足空间,APP程序空间一般在IAP代码物理空间的后面,不能出现交叉重叠的情况。另外一点,安排APP的起始地址的时候,不能将地址放在跟IAP代码位于同一页或同一扇区的地方。否则在通过IAP做代码升级时,对APP区域擦除时就会连同IAP代码自身也一起擦掉。这些信息需要查看STM32各个系列参考手册中memory章节。【下面两幅截图分别来自STM32F30X系列和STM32F40x/41x系列的参考手册】
第二类问题,就是中断矢量表的地址给错或相关代码放错了地方。
STM32芯片里往往都有个中断矢量表,该表的起始地址除了STM32F0系列芯片外,都可以通过配置SCB_VTOR寄存器来指定。我们在给各个STM32工程配置中断矢量表地址时,要跟该工程所在地址空间的起始地址一致。具体到这里,IAP工程的中断矢量表的位置在0x8000000,APP工程的中断矢量表的位置就在0x8003000.
另外,在IAP应用中,关于中断矢量表地址的赋值代码还往往因为摆放位置不当而导致问题。对这个矢量表寄存器VTOR的赋值最好放在启动文件里的复位程序入口的系统初始化函数里,即.s文件里调用的SystemInit()函数。这样保证运行用户程序前已将正确的中断矢量表地址准备好了。
结果有些人将对VTOR寄存器的地址赋值操作不是在启动文件里完成,而是放到了用户main()函数里去执行。如果这样,若在该赋值语句执行之前没有任何中断发生倒也不会有什么问题。如果说在该赋值操作之前发生了中断,由于正确的中断矢量表地址又没准备好,这时硬件会到默认的而又不是我们预期的地方去提取中断向量地址并运行程序,此时不出问题才怪。
第三类问题,就是跳转过程中跟中断有关的处理不当导致异常。较为常见的情形就是在跳转前没有将当前工作区开启过的中断做关闭禁用处理。
我们结合上面的实例来看看。在IAP区我开启了TIM2的中断,如果在在跳转到APP之前不对当前区域的TIM2的中断相关配置做清除整理,看看会出现什么结果。将下面方框中有关TIM2外设及中断相关的代码屏蔽掉,即对开启过的TIM2中断配置不予理睬,就直接跳转到APP区。【此时APP代码不动】
这样编译运行后,发现输出结果跟我们预期的不一样了,输出结果明显有点乱了。实际的产品应用肯定不能接受类似这种程序不受控的状况。
还有一种跳转处理不当的情形,那就是让跳转动作发生在中断服务程序里。
还是结合上面实例来看,将从IAP跳转到APP的跳转执行代码放在TIM2的中断里完成,看看结果会怎么样?【此时APP代码不动】
当将跳转代码放到TIM2的中断服务程序里后,编译运行,结果发现这一跳就死机了。输出就变成下面的样子不动了。虽然执行了跳转动作,但APP程序没能得到正确运行,即跳转失败。所以,我们不要将跳转操作的代码放在中断里执行。
第四类的问题,也是这里要聊的最后一类问题。
我们做IAP应用自然要涉及到Flash编程,不同的STM32系列所支持的编程模式上可能有差异,有的仅支持半字编程模式,比方STM32F1系列,有的只支持双字编程模式,比方STM32G4系列,有的可能支持多种编程模式,比方STM32F4系列可以支持从字节到双字的编程模式。关于这点,我们在不同STM32系列间做移植时要特别注意。另外,虽然说有的系列,比方STM32F4系列支持多种种编程模式,但选择不同的编程模式时,芯片的供电一定要与所选择的编程模式相匹配,不然可能出现编程不可靠情况,这点常被人忽视,当出现问题时往往很难想到这里。当因为Flash编程导致的异常现象往往也很诡异,不太容易查找原因。【下图是STM32F4芯片有关编程模式与供电要求的表格】
好,上面重点分享了基于stm32IAP应用中比较常见的几类问题及注意事项。其实,产品开发过程中因为没有注意上面提到的几类问题,实际的异常现象往往跟具体应用有关,可能五花八门。这里只是抛砖引玉,权作提醒。祝君好运!
=============================
往期话题链接: