STM32+IAP方案 实现网络升级应用固件

摘要:
我已经关注这个概念好几天了。在此期间,我终于有机会练习网络升级应用程序固件。这里我记录了遇到的问题和解决方案==TFTP客户端的应用。参考:IAR环境下STM32+IAP解决方案的实施STM32 IAP简介.pdf基于IAP和KeilMDK的远程升级设计如何生成*概述。KeilMDK中的bin文件:什么是IAP,为什么IAP是可实现的,可以从中找到实现过程的详细信息和上述基本信息。IAP过程描述:1。IAP的引导程序IAP在应用程序中编程,可以扩展到远程网络以更新应用程序固件。

关注了这个概念有些日子了,这段时间总算有机会实战==网络升级应用固件,这里记录下遇到的问题,及解决方案。
原理与网上流传的串口作为传输手段 一致;不同之处,无非我这里使用了网络设备传输。==(lwip)TFTP客户端的应用.
参考:
IAR环境下STM32+IAP方案的实现
STM32浅谈之IAP.pdf
基于IAP和Keil MDK的远程升级设计
keil MDK中如何生成*.bin格式的文件


概况:
  • 什么是IAP,为什么要IAP
  • 可实现的原理
  • 实现过程
  • 细节及实现
    以上基本都可以从【IAR环境下STM32+IAP方案的实现】中找到答案。这里只是贴图,醒目:
    IAP框架布局:
    这里写图片描述
    STM32F103ZET6的启动方式有三种:内置FLASH启动、内置SRAM启动、系统存储器ROM启动,通过BOOT0和BOOT1引脚的设置可以选择从哪中方式启动,这里选择内置的FLASH启动。其FLASH的地址为0x08000000—0x0807ffff,共512KB,这些都能从芯片数据手册中直接得到。而这里首要的一个问题是中断的问题。正常情况下发生中断的过程为:发生中断(中断请求),到中断向量表查找中断函数入口地址,跳转到中断函数,执行中断函数,中断返回。也就是说在STM32的内置的Flash中有一个中断向量表来存放各个中断服务函数的入口地址,内置Flash的分配情况大致如下图2-1。
    这里写图片描述
    这里写图片描述
    这里写图片描述
    这里写图片描述
    在内置的Flash里面添加一个BootLoader程序,BootLoader程序和user application各有一个中断向量表,假设BootLoader程序占用的空间为N+M字节,则程序的走向应该如图2-2所示(借用网友的原图并做改动,其中虚线部分为原图步骤④⑤的走向,本人改为指向灰色部分)。
    这里写图片描述
    上电初始程序依然从0x08000004处取出复位中断向量地址,执行复位中断函数后跳转到IAP的main(标号①所示),在IAP的main函数执行完成后强制跳转到0x08000004+N+M处(标号②所示),最后跳转到新的main函数中来(标号③所示),当发生中断请求后,程序跳转到新的中断向量表中取出新的中断函数入口地址,再跳转到新的中断服务函数中执行(标号④⑤所示),执行完中断函数后再返回到main函数中来(标号⑥所示)。
    对于步骤④⑤,网友认为是:“在main执行的过程中,如果CPU得到一个中断请求,PC指针仍强制跳转到地址0x08000004中断向量表处,而不是新的中断向量表,如图标号④所示,程序再根据我们设置的中断向量表偏移量,跳转到对应中断源新的中断服务程序中,如图标号⑤所示”。我对此的理解是:“当发生中断后,程序从0x08000004(旧)处的中断向量表中得到相应的中断服务函数入口地址,继而跳转到相应的中断服务程序”。但是旧的中断向量列表里边存放的是IAP程序中断函数的入口地址,它是如何得到user程序中断函数的入口地址呢?所以我觉得此种说法是错误的。“当发生中断时PC指针强制会跳转到0x08000004处”这种说法并没有错,只是忽略了后续的一些知识要点而导致这个说法出现矛盾。
    对于步骤④⑤我认为的是,在main函数的执行过程中,如果CPU得到一个中断请求,PC指针本来应该跳转到0x08000004处的中断向量表,由于我们设置了中断向量表偏移量为N+M,因此PC指针被强制跳转到0x08000004+N+M处的中断向量表中得到相应的中断函数地址(待求证),再跳转到相应新的中断服务函数,执行结束后返回到main函数中来。

IAP流程描述:

1、IAP的bootloader引导程序

IAP在应用中编程,可以拓展成远程网络更新应用固件。
片内的flash,至少划分成2个分区,对应至少两个完整的程序;
低地址分区端推荐放入IAP程序==bootloader引导程序(这里边的手段可以是串口、网络等不同的方式),高地址分区端推荐烧写app固件。
关键点提及:
IAP程序中,当满足跳转条件(被触发)时,执行跳转代码到app应用固件程序,跳转代码流程:
至少需要设定跳转目的地的app应用固件 栈顶指针,:

/* Initialize user application's Stack Pointer */
__set_MSP(*(__IO uint32_t*) USER_FLASH_FIRST_PAGE_ADDRESS);
  • 1
  • 2

其中,app应用固件的分区地址:

#define USER_FLASH_FIRST_PAGE_ADDRESS 0x08009000
  • 1

2、app应用固件

需要两处的更改,不然错误未知
IROM设置如图:
这里写图片描述
中断向量表偏移:
这里写图片描述

NVIC_SetVectorTable(NVIC_VectTab_FLASH,VectorTable_Offset);
  • 1

其中:

#define NVIC_VectTab_FLASH           ((uint32_t)0x08000000)
#define VectorTable_Offset  0x9000 
  • 1
  • 2

查错:

如果做了上边的工作,IAP依然无法顺利执行跳转至app应用程序,可以查看.map和.bin文件,确定是否如实的改变的中断向量表的偏移和栈顶指针,如图:
这里写图片描述
.bin文件:
这里写图片描述
可以看到,主栈顶MSP地址=0x2000C8C8、reset_handler地址=0x08009189
如此,才能生效,否则,可能原因:
修改后的向量表偏移,在之后的程序中,又再次被还原,通过如下的函数:

void NVIC_SetVectorTable(uint32_t NVIC_VectTab, uint32_t Offset);
void SystemInit (void);
SCB->VTOR = FLASH_BASE | VECT_TAB_OFFSET;
  • 1
  • 2
  • 3

附:

1、如需要.hex文件转.bin,参见上边的文章
当然,就算使用.hex文件,同样可以升级,只是需要修改IAP中判定已经升级的文件是否有效,文件条件部分的代码,

if(((*(__IO uint32_t*)USER_FLASH_FIRST_PAGE_ADDRESS) & 0x2FFE0000 ) == 0x20000000)
  • 1

2、地址偏移后的app应用程序,是否能够独立的运行?
不能,理由:
这里写图片描述
可知,开机上电并不能够找到我们指定的偏移后的向量表。

免责声明:文章转载自《STM32+IAP方案 实现网络升级应用固件》仅用于学习参考。如对内容有疑问,请及时联系本站处理。

上篇Graphviz 画图的一些总结MySQL查询截取分析下篇

宿迁高防,2C2G15M,22元/月;香港BGP,2C5G5M,25元/月 雨云优惠码:MjYwNzM=

相关文章

stm32的DFU使用方法

stm32的dfu看上去是个很高级的东西,似乎可以通过USB给内部flash、外部spi flash、外部nor等东西刷写数据、把数据读出来,但是用了一下感觉确实有点麻烦。 先不管原理是怎样的,使用方法是这样: 1、先下载这个Dfuse,然后安装。 2、用Jlink之类的东西把这个hex的bootloader刷进stm32的内部flash 3、改成usb直...

VS调试技巧

下面有从浅入深的6个问题,您可以尝试回答一下 一个如下的语句for (int i = 0; i < 10; i++){if (i == 5)j = 5;},什么都写在一行,你怎么在j=5前面插入断点 在一个1000次的循环体内部设置断点,你希望当循环进行到900次后中断,怎么才能做到呢? 你有一个表达式在上面循环的某一次发生了变化,你想知道是哪一次...

第十二章 内中断

12.1 内中断的产生 中断是CPU处理外部突发事件的一个重要技术。 它能使CPU在运行过程中对外部事件发出的中断请求及时地进行处理,处理完成后又立即返回断点,继续进行CPU原来的工作。 引起中断的原因或者说发出中断请求的来源叫做中断源。根据中断源的不同,可以把中断分为硬件中断和软件中断两大类,而硬件中断又可以分为外部中断和内部中断两类。 外部中断一般是指...

Freertos学习:08-信号量

--- title: rtos-freertos-08-ipc-semaphore date: 2020-06-23 11:01:12 categories: tags: - freertos - ipc --- 章节概述: 介绍 FreeRTOS中的任务间通讯机制:信号量的使用(尽管FreeRTOS中没有进程的概念,但为了统一,我们还是以进程间通讯(IP...

线程中断总结

1、线程中断 每个线程都有一个与之相关联的 Boolean 属性,用于表示线程的中断状态(interrupted status)。中断状态初始时为 false;当另一个线程通过调用Thread.interrupt()中断一个线程时,会出现以下两种情况之一。如果那个线程在执行一个低级可中断阻塞方法,例如Thread.sleep()、Thread.join()...

基于STM32之UART串口通信协议(一)详解

一、前言 1、简介   写的这篇博客,是为了简单讲解一下UART通信协议,以及UART能够实现的一些功能,还有有关使用STM32CubeMX来配置芯片的一些操作,在后面我会以我使用的STM32F429开发板来举例讲解(其他STM32系列芯片大多数都可以按照这些步骤来操作的),如有不足请多多指教。 2、UART简介   嵌入式开发中,UART串口通信协议是我...