原子哥的教程是不是有个bug,还是我理解不对?

2019-07-21 06:01发布

本帖最后由 xuande 于 2016-5-30 10:48 编辑

最近在弄bootloader,由于使用的不是F1系列的,用的是F0的芯片,跟F1的略有不同,如下


STM32F103cortex-m3/m4内核的单片机上可以通过设置SCB->VTOR = FLASH_BASE | VECT_TAB_OFFSET;该寄存器的值来实现中断向量表的重定义。但在STM32F0xx系列以cortex-m0为内核的单片机中却怎么也找不到这个设置中断向量表的寄存器,可以通过以下方法来实现中断向量表重定义。
基本思想:1、将中断向量表放入到RAM的起始地址(只需要在应用程序中保留RAM其实地址的0x100大小不使用即可)。2、在bootload中将应用程序的中断向量表从Flash中拷贝到RAM中。3、设置STM32F0xx中断向量表位于RAM中,重定向中断向量表程序如下。


[mw_shl_code=c,true]void IAP_Set()
{
   uint32_t i = 0;

/* Relocate by software the vector table to the internal SRAM at 0x20000000 ***/  

  /* Copy the vector table from the Flash (mapped at the base of the application
     load address 0x08003000) to the base address of the SRAM at 0x20000000. */
        
  for(i = 0; i < 48; i++)//先将”应用程序“的中断向量表拷贝至SRAM中
  {
    *((uint32_t*)(0x20000000 + (i << 2)))=*(__IO uint32_t*)(APPLICATION_ADDRESS + (i<<2));
        }

  /* Enable the SYSCFG peripheral clock*/
  RCC_APB2PeriphClockCmd(RCC_APB2Periph_SYSCFG, ENABLE);
        
        /* Remap SRAM at 0x00000000 */
        
        /*然后在配置SYSCFG寄存器的最低两位,将SRMA映射至地址0,这样,在发生中断后,CPU从地址0取中断向量,
        这样实际上就是从SRAM中取的中断向量,而SRAM中的中断向量表又是之前从”应用程序“的中断向量表拷贝过
        来的,所以,最终其实是取的”应用程序“的中断向量表。*/
  SYSCFG_MemoryRemapConfig(SYSCFG_MemoryRemap_SRAM);
}        [/mw_shl_code]

说明了F0和其他系列的芯片区别后,回到正题,就是关于原子哥手册上的bug,不知道有没有人提过,还是自己理解有误,如下,原子哥手册中是这样说道的
7D6B.tmp.jpg

不太理解的是,如果真是每次产生一个中断就跳回bootloader的中断向量表处去,然后再根据提到的"程序再根据我们设置的中断向量表偏移量,跳转到对应
中断源新的中断服务程序中"这句话来说,似乎只看到在main函数里面有对中断向量表进行重新设置,并没有看到每次中断后去取bootloader的向量表
->做向量表的偏移(这个过程没有在程序中体现吧?)


所以我认为每次发生中断都是从新的中断向量表取的,而不会像原子哥画的图示一样,回到bootloader去取向量表
ED0D.tmp.jpg

那么问题来了,每次都从新的中断向量表取的,为什么会出现复位后是从bootloader的main启动的呢?从新的表里面取的不应该是直接运行
APP的main了吗?复位也是个中断,并且有中断服务函数吧?这个问题我也困扰了很久,后面查F0的手册得知,复位是个特例,如下
2 (2).png
2 (1).png

当复位后对SYSCFG这个寄存器进行了重新配置,最后两位取决于
3.png

也就是说复位后又从FLASH的最低地址开始去读取向量表了,而此时的向量表正是bootloader中的向量表,其他的中断里面
没有重新配置这个寄存器,所以一直用的是APP中的新的向量表,
至此我觉得原子哥的手册中可能这样写会更合适一些,如下
4.jpg

也不知道自己这个理论成不成立,如果我误解了,麻烦指正,谢谢!


友情提示: 此问题已得到解决,问题已经关闭,关闭后问题禁止继续编辑,回答。
该问题目前已经被作者或者管理员关闭, 无法添加新回复
14条回答
一叶扁舟V5
1楼-- · 2019-07-22 05:53
豁然开朗
kokoromi
2楼-- · 2019-07-22 06:08
楼主,你说的这个是对的,因为在进入APP程序后会设置中断向量表寄存器重新定位向量表,所以除非发生了复位中断,否则其他中断都是会跳到新的向量表中的,再说了,如果不是这样,每次中断都得跳到IAP的向量表,那设置中断向量表寄存器还有啥意义,是吧?
WM_CH
3楼-- · 2019-07-22 10:33
顶一下子                  
shangwei4
4楼-- · 2019-07-22 15:31
我的理解原子大哥是对的:M0/M3都是根据查表的方法寻找中断程序入口地址的,那么就需要先找到表头,否则从哪里开始查表?
对于F1芯片,APP程序运行时,每产生一个中断就跳回IAP(bootloader)的中断向量表复位向量处,然后再根据我们设置的中断向量表偏移量,找到APP中新的复位向量,即向量表首地址,再继续查询对应的中断地址。使用中断向量重映射直接实现了向量表的偏移,不需要再有复制向量表的程序(在下载程序时那里本身就存储了一个APP向量表)。
对于F0芯片,不能直接设置中断向量表的偏移地址,APP程序开始运行时在SRAM那里抄录一遍中断向量表,再将SRAM重映射。这样,APP程序运行时,每产生一个中断就会跳转到SRAM首地址,再继续查询对应的中断地址。
zcz0426
5楼-- · 2019-07-22 20:21
 精彩回答 2  元偷偷看……
genius
6楼-- · 2019-07-23 00:07
帮顶                     

一周热门 更多>