黑龙江视窗_黑龙江最专业的门户网站
您当前的位置 : 黑龙江视窗  >  企业
STM32在线升级中断向量重定向深度剖析!
2020-11-19 15:35:21 来源:互联网 阅读:-

在做stm32 iap升级固件的时候通常需要多份中断向量表。比如bootloader的中断向量表在0x00000000位置,应用程序的中断向量表则会放在flash的另一个地方或者是放在RAM中运行。

要维护向量表位置就需要用VTOR这个东西,那么就要先从VTOR来聊聊, 先弄清这个东西又是干嘛的。

VTOR是arm内核的一个寄存器,叫做中断向量偏移量寄存器。当系统上电启动的时候CPU会从先找到中断向量表的位置,然后从表中找到复位中断Reset_Handler,而main函数的执行实际上就是在复位中断函数中的。如下汇编代码就是复位中断服务函数,该函数在启动文件中定义。在Reset_Handler 中会执行__main(),而main函数又是被__main()调用了。

Reset_Handler    PROC
EXPORT Reset_Handler [WEAK]
IMPORT __main
IMPORT SystemInit
LDR R0, =SystemInit
BLX R0
LDR R0, =__main
BX R0
ENDP

前面复习一下ARM处理器启动到执行main的一个大概流程。再回归正题继续探讨这个VTOR。当来一个中断时,cpu就会从0x0000000+VTOR位置找到中断向量表,然后再查表跳转到对应的中断服务函数去执行。所以你代码里面存在多份中断向量表是可以的,只需要动态改变VTOR的值就可以了。

STM32在线升级中断向量重定向深度剖析

APP的中断向量放在flash中:

如下图所示,在运行bootloader的时候用的是他自己的bootloader,当从bootloader跳转到app以后就需要用app的向量表了。所以进入APP的首要任务就是设置VTOR值为0x0x8001000,再产生中断的时候让cpu去app的向量表里找中断服务函数入口。

STM32在线升级中断向量重定向深度剖析

所以从在app中重定义向量表是不是很简单。但是有个前提,就是要有VTOR寄存器。实际上,并不是所有stm32都带有VTOR的。准确的说是因为ARM-CORTEX-M0内核的芯片没有,所以stm32F0系列所采用M0内核的芯片就没有VTOR。但是还有需要注意的就是L0系列使用的M0+内核和M0内核又是不同的,是有VTOR的。

这就意味着stm32F0系列没有办法使用这种简单的修改VTOR的方式重新定位APP的中断向量表。那肯定还有其他思路,就是稍微麻烦一点。

中断向量放在RAM中:

那就是把中断向量表放在RAM中去,并且针对F0没有VTOR的还必须把中断向量表拷贝到RAM的起始地址。

为什么这样做,因为前面我们提到过CPU默认会认为0x00000000处放的就是向量表。那么我们换种思路可以把RAM的地址映射到0x00000000处。这个通过配置SYSCFG寄存器的MEM_MODE就可以把RAM地址映射到0x00000000处。

STM32在线升级中断向量重定向深度剖析

这时候再进入到APP以后首要任务就是把APP的中断向量表拷贝到RAM的起始地址,并且修改MEM_MODE位。

在代码里面比较简单的做法就是在ram起始地址处定义一个数组,上电以后把向量表拷贝到这个数组中。如下代码是在MDK中定义的方法,在IAR或者GCC定义的方式会有区别。

__IO uint32_t vector_t[48] __attribute__((at(0x20000000)));

但是要编译通过还需要在mdk配置里面把ram的起始地址设置为0x200000c0,前面的0xc0的空间腾出来存放定义的这个数组。

有VTOR的系列向量表放在RAM中

stm32F0系列内核决定了即便把向量表放在RAM中也只能放在RAM的起始位置而不能任意放。但是有VTOR的型号就可以随便放。这样分配存放向量表的数组也可以不用定位在0x20000000位置了。可以如下方式定义:

__align(256) uint32_t vector_t[48];

可以看到虽然没有了必须放在ram起始地址的限制,但是还是要遵守一定的规则。就是对齐有要求的。比如L0系列有48个中断向量,一共占用内存为48*4=192bytes。扩大为2的整次方为256:28=256。所以就要求向量存储的地址必须是256对齐的。存放在0x0或者0x20000100都可以满足要求。

另外这种方式虽然更灵活了一些,但是却造成了一定RAM空间的浪费。我在这样定义编译以后查看MAP文件如下,红色的区域空出来的一段内存空间就会被浪费掉:

STM32在线升级中断向量重定向深度剖析

关于0X00000000地址

前面说过默认cpu会从0x00000000地址来取向量表,那么stm32的flash都是从0x8000000地址开始的,这时候不设置VTOR偏移为什么也可以正常运行。实际上当从用户flash启动的时候,从0x8000000或者0x00000000访问的都是同一片区域。所以cpu从0x0取向量表相当于取的就是0x8000000地址处的数据。

更进一步理解,stm32通过配置boot脚(或者SYSCFG中MEM_MODE位)有三种启动模式选择,这三种模式的本质就是看把那个区域重映射为0x00000000地址。这样CPU取第一条指令就是从哪里取:

  1. 从用户flash区启动,用户flash起始地址被映射到0x00000000上
  2. 从系统flash区启动,系统flash起始地址被映射到0x00000000上,上电就执行内置的bootloader
  3. 从RAM区启动,RAM起始地址被映射到0x00000000上,这时候你再回过去看我刚才说的F0无VTOR寄存器实现IAP就是这样的思路。向量表放在了RAM区域的起始地址。

推荐阅读:浙江企业新闻网

频道推荐
  • 智能家居引进蓝牙5.0技术有何不一样?
    智能家居引进蓝牙5.0技术有何不一样?

    蓝牙5.0技术的发布,对每个行业都产生了巨大影响,尤为突出的是智能家居行业。现时绝大多数的普通家用电器仍然采用红外传输,最常见的就是家里各种各样的遥控器,空调的...

    2020-03-05
  • 卡萨帝高端空调:智能服务推动高端智慧生活
    卡萨帝高端空调:智能服务推动高端智慧生活

    卡萨帝高端空调,用智能服务推动高端智慧生活。近年来,各大家电品牌不断加深在高端智能领域的布局,推出不同种类的高端家电产品。在空调市场,虽然久居高端前列的日系产品...

    2020-03-05
  • 所有空调1个网关!中弘智能空调网关的市场运营之道
    所有空调1个网关!中弘智能空调网关的市场运营

      随着近几年智能建筑的发展,智能家居、中央空调的集控也变得越来越智能化,方便用户或物业控制管理。此外在节能减排大背景下,建筑节能尤其是中央空调的节能是建筑节能...

    2020-03-05
  • 飞利浦诠释智能,让你的生活有“锁”不同
    飞利浦诠释智能,让你的生活有“锁”不同

    随着智能家居产品的普及,指纹锁逐渐得到了用户的认可。特别是80、90后逐渐成为市场消费的主体之后,智能锁等智能设备更是广泛应用。飞利浦电子作为世界上最大的电子品...

    2020-03-05
  • 当指纹锁遇上HomeKit是什么体验——小燕全自动智能门锁使用测评
    当指纹锁遇上HomeKit是什么体验——小燕

    作为一个果粉,对于苹果的智能家居生态HomeKit已经小有研究,家中也有不少支持HomeKit的设备,不过大多都只是开关灯泡之类的小打小闹,玩个乐趣。对于真正实...

    2020-03-05
  • 打好家庭防疫保卫战!商汤携手TCL打造“无接触”式智能家居门锁
    打好家庭防疫保卫战!商汤携手TCL打造“无接

    全民同心抗击新冠肺炎疫情的战斗仍在继续中。预防新冠肺炎,从我做起!这段时间,各种“无接触”式操作已广泛融入到大家的工作和生活行为中。小到戴手套和口罩出入家门、办...

    2020-03-05