stm32F4 IAP实现原理讲解以及中断向量表的偏移

本文深入解析STM32的IAP(In-Application Programming)原理,介绍如何通过IAP实现在应用编程,更新固件而不需重新烧录芯片。文章详细说明IAP与正常程序运行流程的区别,阐述IAP程序的运行机制,包括中断向量表的偏移设置,以及如何在实际编程中正确配置中断向量表偏移量。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

一、IAP原理

IAP即是在应用编程, IAP 是用户自己的程序在运行过程中对User Flash 的部分区域进行烧写,目的是为了在产品发布后可以方便地通过预留的通信口对产
品中的固件程序进行更新升级。 通常实现IAP 功能时,即用户程序运行中作自身的更新操作,所以需要在设计固件程序时编写两个项目代码,第一个项目程序不执行正常的功能操作,而只是通过某种通信方式(如 USB、USART)接收程序或数据,执行对第二部分代码的更新;第二个项目代码才是真正的功能代码。这两部分项目代码都同时烧录在 User Flash 中,当芯片上电后,首
先是第一个项目代码开始运行,它作如下操作:
1)检查是否需要对第二部分代码进行更新
2)如果不需要更新则转到 第4个步骤
3)执行更新操作
4)跳转到第二部分代码执行

二、stm32正常的程序运行流程

STM32 的内部闪存( FLASH)地址起始于 0x08000000,一般情况下,程序文件就从此地址开始写入。此外STM32其内部通过一张“中断向量表”来响应中断,程序启动后,将首先从“中断向量表”取出复位中断向量执行复位中断程序完成启动,而这张“中断向量表”的起始地址是0x08000004,当中断来临, STM32 的内部硬件机制亦会自动将 PC 指针定位到“中断向量表”处,并根据中断源取出对应的中断向量执行中断服务程序。
在图1中, STM32 在复位后,先从 0X08000004 地址取出复位中断向量的地址,并跳转到复位中断服务程序,如图标号①所示;在复位中断服务程序执行完之后,会跳转到我们的main 函数,如图标号②所示;而我们的main 函数一般都是一个死循环,在 main 函数执行过程中,如果收到中断请求(发生重中断),此时STM32 强制将 PC 指针指回中断向量表处,如图标号③所示;然后,根据中断源进入相应的中断服务程序,如图标号④所示;在执行完中断服务程序以后,程序再次返回main 函数执行,如图标号⑤所示。
图1:stm32程序运行流程

三、IAP程序运行流程

在下图2 所示流程中, STM32 复位后,还是从 0X08000004 地址取出复位中断向量的地址,并跳转到复位中断服务程序,在运行完复位中断服务程序之后跳转到IAP 的 main 函数,如图标号①所示,此部分同图1 一样;在执行完 IAP 以后(即将新的 APP 代码写入 STM32的FLASH,灰底部分。新程序的复位中断向量起始地址为 0X08000004+N+M),跳转至新写入程序的复位向量表,取出新程序的复位中断向量的地址,并跳转执行新程序的复位中断服务程序,随后跳转至新程序的 main 函数,如图标号②和③所示,同样 main 函数为一个死循环,并且注意到此时 STM32 的FLASH,在不同位置上,共有两个中断向量表。在 main 函数执行过程中,**如果 CPU 得到一个中断请求, PC 指针仍强制跳转到地址0X08000004 中断向量表处,而不是新程序的中断向量表,如图标号④所示;程序再根据我们设置的中断向量表偏移量,跳转到对应中断源新的中断服务程序中,如图标号⑤所示;在执行完中断服务程序后,程序返回 main 函数继续运行,如图标号⑥所示。**通过以上两个过程的分析,我们知道 IAP 程序必须满足两个要求:

1) 新程序必须在 IAP 程序之后的某个偏移量为 x 的地址开始;
2) 必须将新程序的中断向量表相应的移动,移动的偏移量为 x;

图2:IAP程序运行流程

四、程序起始地址的设置

如下图,在keil设置程序要烧录的地址,点击 Options for Targe-> Target选项卡
图3:keil设置烧录地址
默认的条件下,图中 IROM1 的起始地址( Start)一般为 0X08000000,大小(Size)为0X80000,即从0X08000000 开始的512K 空间为我们的程序存储(假设使用的STM32F103ZET6,其 FLASH大小是 512K)。而图中,我们设置起始地址( Start)为 0X08010000,即偏移量为 0X10000( 64K字节),因而,留给 APP 用的 FLASH 空间( Size)只有0X80000-0X10000=0X70000( 448K 字节)大小了。设置好 Start 和Szie,就完成APP 程序的起始地址设置。

五.程序中如何设置中断向量表的偏移量

看完三IAP程序运行流程的讲解,想必知道了第二部分程序得设置中断向量表的偏移量,否则如果第二部分程序存在中断功能,当发生中断时,程序不能正确的找到对应的中断函数。后面会做实验验证结果,接下来请看如何改中断向量表的偏移地址,我使用的是stm32f407,打开System_stm32f4xx.c这个文件,在
void SystemInit(void)这个函数中有如下代码:
#ifdef VECT_TAB_SRAM
SCB->VTOR = SRAM_BASE | VECT_TAB_OFFSET;
#else
SCB->VTOR = FLASH_BASE | VECT_TAB_OFFSET;
#endif
我们没有定义VECT_TAB_SRAM这个宏,所以执行的是
SCB->VTOR = FLASH_BASE | VECT_TAB_OFFSET;
其中VECT_TAB_OFFSET就是偏移量,代码的定义是
#define VECT_TAB_OFFSET 0x00
把它修改成你需要的偏移量即可,比如第二部分代码要放在0x08000100,那么偏移量就是0x100,即改成
#define VECT_TAB_OFFSET 0x100即可。

至于为什么要在SystemInit(void)里面改这个,请查看启动文件,相关部分代码如下:
Reset_Handler PRO
EXPORT Reset_Handler [WEAK]
IMPORT SystemInit
IMPORT __main

             LDR     R0, =SystemInit
             BLX     R0
             LDR     R0, =__main
             BX      R0
             ENDP

程序在进入复位中断后,先执行SystemInit这个函数后,才跳到main函数去执行,所以中断向量表的地址在执行main函数前就已经设置了

六、实验验证

1.我写了第一部分的程序,支持跳转到第二部分程序
2.第二部分程序,定义了一个中断函数,当按下按键1,则进入外部中断0点亮LED1
3.第二部分程序我先不修改中断偏移量,烧录完两个程序,当我按下按键,LED1不亮,证明程序没有跳转到中断函数执行
4.第二部分程序我修改中断偏移量,烧录完两个程序,当我按下按键,LED1点亮,证明修改了中断偏移量后程序跑入的是第二部分程序的中断函数

<think>好的,我现在需要帮助用户找到关于STM32F4芯片的IAP实现方法或示例代码。首先,我要回忆一下IAP的基本概念。IAP是In-Application Programming,即在应用中编程,允许微控制器通过某种通信接口(如串口、USB、以太网等)更新自身的程序,而不需要外部编程器。这对于现场固件升级非常有用。 用户提到了正点原子的实验50,这可能是一个重要的参考资料。我需要确认这个实验是否提供了示例代码或者具体的实现步骤。根据引用[2],实验50涉及到设置APP程序的起始地址和存储空间大小,这说明在实现IAP时,需要将Flash分为两部分:IAP引导程序和用户应用程序。引导程序负责接收新固件并写入到应用程序区域,然后跳转到应用程序执行。 接下来,用户需要示例代码或实现指南。我应该考虑推荐官方资源,比如STMicroelectronics的官方应用笔记和示例代码。例如,AN2557和AN3965可能包含IAP实现方法,这些文档通常会提供详细的步骤和代码片段。同时,社区资源如GitHub、STM32论坛和电子工程博客可能会有开源项目或教程,用户可以参考这些资源。 在实现步骤方面,用户需要了解如何配置Flash的分区,设置中断向量表偏移,以及如何通过串口或其他接口进行固件传输。可能需要使用标准库或HAL库中的Flash操作函数,比如HAL_FLASH_Program()来写入数据。此外,在应用程序中需要调整链接脚本,指定正确的起始地址,并确保中断向量表正确重定位。 还要注意可能的问题,比如在跳转到应用程序之前禁用中断,避免Flash写入时的中断干扰,以及校验固件的完整性(如CRC校验)以确保数据传输正确。这些细节都是实现可靠IAP功能的关键。 最后,我应该整理这些信息,分步骤说明实现方法,并推荐相关的示例代码来源,同时附上相关引用,确保用户能够根据指引找到所需的资源。</think>关于STM32F4系列芯片的IAP(在应用编程)实现方法及示例代码,以下是分步骤说明: 1. **基本原理** IAP通过将Flash划分为引导程序(Bootloader)和用户程序(APP)两部分。引导程序通过串口/USB/网络等接口接收新固件,写入APP区域,最后跳转执行[^2]。关键点包括: - Flash地址空间划分(如0x08000000为Bootloader,0x08010000为APP) - 中断向量表重定向 - 固件传输协议设计(如YModem) 2. **实现步骤** (1) **Bootloader开发** ```c // 跳转到APP的代码示例 void jump_to_app(uint32_t app_addr) { typedef void (*pFunction)(void); pFunction Jump_To_Application; uint32_t stack_pointer = *(volatile uint32_t*)app_addr; app_addr += 4; Jump_To_Application = (pFunction)(*(volatile uint32_t*)app_addr); __disable_irq(); HAL_RCC_DeInit(); HAL_DeInit(); __set_MSP(stack_pointer); Jump_To_Application(); } ``` (2) **APP工程配置** 修改链接脚本(.ld文件)设置Flash起始地址: ```ld MEMORY { FLASH (rx) : ORIGIN = 0x08010000, LENGTH = 512K RAM (rwx) : ORIGIN = 0x20000000, LENGTH = 128K } ``` (3) **系统初始化** 在APP的main()开头添加中断向量表偏移设置: ```c SCB->VTOR = FLASH_BASE | 0x10000; // 与链接脚本一致 ``` 3. **参考资源** - **官方文档**:ST应用笔记AN2557(USART IAP实现)、AN3965(DFU模式)[^2] - **开源示例**: - STM32CubeF4软件包中的IAP例程(路径:STM32Cube_FW_F4_Vxx/Projects/STM32F4-Discovery/Applications/IAP) - 正点原子实验50完整代码(包含YModem协议实现) - **关键API**: ```c HAL_FLASH_Unlock(); HAL_FLASH_Program(FLASH_TYPEPROGRAM_WORD, Address, Data); HAL_FLASH_Lock(); ```
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值