随言:

IAP应该是我唯一想写的文章,从创建这个账号开始。

但是不知不觉几年过去了,一直没去写这文章。

现在就随便写写吧~

曾做过4G模块UART协议与STM32通讯实现远程无线迭代升级,

一共2个APP,bootloader优先选择稳定高版本的APP启动。

下面文章就把这个简单实现大概,

由于我是之前使用无线模块透传+UART与服务器通讯的,功能比较多复杂。

为了简化,我不打算写个独立带协议的上位机,简单用UART实现大概双

APP的IAP模型和提及一些之前遇到的问题。简单修改一下可以用于网络IAP。

记录“美好”生活吧。

下面我不会使用Ymodem协议接收bin文件数据。

我只想以最简单的方式展示IAP的过程,不想过多复杂化IAP,微笑。

若需要Ymodem协议的,请自己移植,谢谢~!

1.什么是IAP?

英文名:in-application programming。

中文名:应用程序内编程。

作用:对于大多数基于闪存的系统,一个重要的要求是能够在最终产品中安装固件时进行更新。

           STM32微控制器可以运行用户特定的固件来对微控制器中嵌入的闪存执行IAP。

接口:此功能支持的任何通信接口。

由于不限制通信接口协议等,只要能通过任意通信接口拿到新版固件包数据(bin文件),就能自己升级固件。

这就能做到添加  外部无线模块(4G模块、wifi)做到OTA升级。

也可以使用U盘或TF卡等外部存储设备做到OTG升级。U盘升级的IAP官方有模板程序叫:FWupgrade_Standalone。

1、1  IAP适用场景:

1、设备附加功能卖点,固件新增功能,修复旧功能等,用户自己可以OTG升级固件。

2、设备笨重不利于返厂,OTA升级固件。

3、设备分布零散广泛不利人工去现场去维护固件,OTA升级。

1、2  参考官方例程:

..\STM32Cube\Repository\STM32Cube_FW_F4_V1.25.0\Projects\STM324xG_EVAL\Applications\IAP\

2、IAP执行原理:

2、1  正常程序上电执行流程:

STM32CUBEMX IWDG 配置 stm32cubeide iap_STM32CUBEMX IWDG 配置

1、从Flash相对0地址即STM32的0x8000000地址开始执行,存储的是栈顶地址。再偏移4个字节找到中断向量表起始地址,即复位中断向量,存储这复位中断程序入口,跳转到复位中断程序入口执行复位中断服务函数Reset_Handle(void)。

2、复位中断服务函数Reset_Handle(void)执行完会跳转到main函数,main函数循环运行。

3、main函数死循环时,发生中断请求,然后PC指针会跳转到中断向量表寻找对应的中断向量。

4、找到对应的中断向量后执行对应的中断服务函数xxx_Handler(void).

5、执行中断服务函数完成后,返回main函数循环运行。

2、2 加入IAP后的上电程序执行流程

STM32CUBEMX IWDG 配置 stm32cubeide iap_数据_02

不同点:

对比两图可以发现,加入IAP后其实就把芯片内的Flash分成了两个程序。IAP过程前是一个程序,IAP后是一个程序。

由于是两个程序,我们一般会把放在flash相对的0地址的程序叫bootloader引导启动程序,后面的都叫APP用户程序。

如上图的IAP程序就占用了flash的M个字节大小,然后通过跳转后APP程序的中断向量表也往后偏移了M个字节。

1、第2步的跳转是通过函数指针跳转。跳转前需要关闭所有的中断后取消外设功能,跳转后需要更新中断向量表。

特别是使用了RTOS系统,由于像FreeRTOS会用外设TIMx做系统时钟,跳转前一定要关闭系统滴答时钟中断,否则跑飞。

STM32F4xx系列更新中断向量表可以使用下面两种方法:

#define APPLICATION_ADDRESS   (uint32_t)0x08008000
SCB->VTOR = APPLICATION_ADDRESS;
#define APPLICATION_ADDRESS   (uint32_t)0x08008000
#define VECT_TAB_OFFSET  APPLICATION_ADDRESS

2、第5步中发送中断后PC指针会跳转到默认的向量表位置,然后默认的向量表会执行新的向量表对应的中断服务函数,最后返回main函数。

 

3、做个简单的UART IAP

3、1  框图

我使用的是STM32F429IGT6;环境是STM32CubeIDE 1.42版本。

由于实现IAP实际上写两个程序,分别是bootloader和APP.

首先先将芯片内部flash分区两个区域,分别存放bootloader和APP程序。

IAP功能就坐在bootloader程序上面,bootloader使用UART实现更新程序;

APP做一个1秒 LED闪烁的。


STM32CUBEMX IWDG 配置 stm32cubeide iap_数据_03

STM32F429IG内部flash分区

STM32CUBEMX IWDG 配置 stm32cubeide iap_双APP IAP_04

简单的IAP程序流程图

APP就没什么流程图可言的了,毕竟只有一个LED闪烁。

3、1  关于Ymodem协议(纯主观)

官网例程都是用的Ymodem协议接收bin文件数据。

YModem协议是由XModem协议演变而来的,每包数据可以达到1024字节,是一个非常高效的文件传输协议

我就不用了,还是自由舒服。

3、3  Bootloader编程

STM32CubeIDE初始化了外部晶振时钟和UART(无中断)。

STM32CUBEMX IWDG 配置 stm32cubeide iap_STM32CUBEMX IWDG 配置_05

main代码:

int main(void)
{
  /* USER CODE BEGIN 1 */
	uint8_t key = 0;
	uint32_t temp = 0;
	uint32_t timeout = 0;
	uint32_t userapplen = 0;
  /* USER CODE END 1 */

  /* MCU Configuration--------------------------------------------------------*/

  /* Reset of all peripherals, Initializes the Flash interface and the Systick. */
  HAL_Init();

  /* USER CODE BEGIN Init */

  /* USER CODE END Init */

  /* Configure the system clock */
  SystemClock_Config();

  /* USER CODE BEGIN SysInit */

  /* USER CODE END SysInit */

  /* Initialize all configured peripherals */
  MX_GPIO_Init();
  MX_USART1_UART_Init();
  /* USER CODE BEGIN 2 */
  printf("\r\nSudaroot, This is Bootloader\r\n");

  /* USER CODE END 2 */

  /* Infinite loop */
  /* USER CODE BEGIN WHILE */
  while (1)
  {
	  printf("\r\n=================== Main Menu ============================\r\n\n");
	  printf("  Download user application to the internal Flash ------ 1\r\n\n");
	  printf("  Execute the loaded application ----------------------- 2\r\n\n");

	  /* Clean the input path */
	  __HAL_UART_FLUSH_DRREGISTER(&huart1);
	  /* Receive key */
	  HAL_UART_Receive(&huart1, &key, 1, HAL_MAX_DELAY);

	  switch(key)
	  {
	  case '1':
		  /* Download user application in the Flash */
		  /* 1. erase user application area */
		  printf("Wait for the internal Flash erase to complete\r\n");
		  if(FLASH_If_Erase(APPLICATION_ADDRESS, 10) == 1)
		  {
			  printf("Erase the internal Flash is fail\r\n");
			  Error_Handler();
		  }
		  printf("Erase the internal Flash is complete\r\n");

		  /* 2. download a file via serial port */
		  printf("Waiting for the file to be sent ... \r\n");
		  userapplen = 0;
		  timeout = HAL_MAX_DELAY;
		  /* Clean the input path */
		  __HAL_UART_FLUSH_DRREGISTER(&huart1);
		  while(1)
		  {
			  if(HAL_UART_Receive(&huart1, uart_buf, UART_BUF_SIZE, timeout) != HAL_OK)
			  {
				  temp = UART_BUF_SIZE - (huart1.RxXferCount + 1);
				  if(FLASH_If_Write(APPLICATION_ADDRESS + userapplen, (uint32_t*)uart_buf, temp / 4) != FLASHIF_OK)
				  {
					  printf("Write the internal Flash is fail\r\n");
					  Error_Handler();
				  }
				  userapplen = userapplen + temp;
				  break;
			  }
			  timeout = 1000;
			  if(FLASH_If_Write(APPLICATION_ADDRESS + userapplen, (uint32_t*)uart_buf, UART_BUF_SIZE / 4))
			  {
				  printf("Write the internal Flash is fail\r\n");
				  Error_Handler();
			  }
			  userapplen = userapplen + UART_BUF_SIZE;
		  }
		  printf("Programming Completed Successfully! %ldBtye\r\n", userapplen);
		  break;

	  case '2':
		  printf("Start program execution......\r\n\n");
		  /*
		   * 关闭或反初始化前面用到的外设和中断
		   * 1、反初始化UART
		   * 2、关闭系统滴答定时器中断
		   * */
		  HAL_UART_DeInit(&huart1);
		  HAL_SuspendTick();

	      /* execute the new program */
	      JumpAddress = *(__IO uint32_t*) (APPLICATION_ADDRESS + 4);
	      /* Jump to user application */
	      JumpToApplication = (pFunction) JumpAddress;
	      /* Initialize user application's Stack Pointer */
	      __set_MSP(*(__IO uint32_t*) APPLICATION_ADDRESS);
	      JumpToApplication();
		  break;

	  default:
		  printf("Invalid Number ! ==> The number should be either 1 or 2\r");
		  break;
	  }
    /* USER CODE END WHILE */

    /* USER CODE BEGIN 3 */
  }
  /* USER CODE END 3 */
}

1、上电后打印功能菜单。

2、发送‘1’就会进入IAP模式;首先会擦除APP空间,然后等待用串口助手发送APP的bin文件写入flash,直至完成。

3、发送‘2’就会直接跳转到APP运行。

注意:编译完成后注意一下生成的bin文件是否超过了定义的32kB大小,超过后就要重新定义分区。

2021.10.25

关于temp = UART_BUF_SIZE - (huart1.RxXferCount + 1);这句代码获取一包数据剩余长度。

为什么加1呢?和我使用的“HAL库版本有关系”。我用的是当时写博客时的最新版本,但是现在肯定不是最新了。

主要是在 HAL_UART_Receive 这个库函数内部,huart1.RxXferCount先减1再等待超时,而新版本的已经变了,先等待超时再huart1.RxXferCount减1。具体情况具体分析,根据你自己的HAL库版本修改。

STM32CUBEMX IWDG 配置 stm32cubeide iap_数据_06

3、4 APP编程

STM32CubeIDE初始化了外部晶振时钟、UART(无中断)和一个LED的GPIO.

STM32CUBEMX IWDG 配置 stm32cubeide iap_数据_07

main函数代码:

int main(void)
{
  /* USER CODE BEGIN 1 */
	SCB->VTOR = APPLICATION_ADDRESS;
  /* USER CODE END 1 */

  /* MCU Configuration--------------------------------------------------------*/

  /* Reset of all peripherals, Initializes the Flash interface and the Systick. */
  HAL_Init();

  /* USER CODE BEGIN Init */

  /* USER CODE END Init */

  /* Configure the system clock */
  SystemClock_Config();

  /* USER CODE BEGIN SysInit */

  /* USER CODE END SysInit */

  /* Initialize all configured peripherals */
  MX_GPIO_Init();
  MX_USART1_UART_Init();
  /* USER CODE BEGIN 2 */
  printf("Sudaroot, This is IAP Application\r\n");
  /* USER CODE END 2 */

  /* Infinite loop */
  /* USER CODE BEGIN WHILE */
  while (1)
  {
	  printf("This is IAP Application running\r\n");
	  HAL_GPIO_TogglePin(RUN_LED_GPIO_Port, RUN_LED_Pin);
	  HAL_Delay(500);
    /* USER CODE END WHILE */

    /* USER CODE BEGIN 3 */
  }
  /* USER CODE END 3 */
}

 这行代码的时候是   SCB->VTOR = APPLICATION_ADDRESS; 上面提到的更新中断向量表偏移地址。

注意:编译完成后注意一下生成的bin文件是否超过了定义的分区大小,超过后就要重新定义分区。

还要修改一下程序flash链接地址。

1、打开下面这个文件

STM32CUBEMX IWDG 配置 stm32cubeide iap_STM32CUBEMX IWDG 配置_08

2、我用的是STM32F429IG,片内flash有1MB大小,前面32kB给了boot loader,即APP剩下992kB,起始地址是0x08008000.

STM32CUBEMX IWDG 配置 stm32cubeide iap_双APP IAP_09

修改成功后:

STM32CUBEMX IWDG 配置 stm32cubeide iap_STM32CUBEMX IWDG 配置_10

4、写UART双APP迭代升级的IAP

4、1 IAP单APP和双APP区别

IAP双APP和单APP相比之下就是更稳定。

我做的是网络升级的IAP,毕竟要确保设备软件的正常稳定工作,重中之重。

1、单APP的话,如果网络IAP升级失败(断网或断电)的话,基本可能变成了“砖头”,大忌(同事会很辛苦)。

2、双APP的话,首先启动肯定是一个稳定运行的版本,至于是不是最新版是次要,主要是稳定。

比如当运行稳定的是APP0,那么网络IAP升级的时候,肯定不会升级正在运行的APP0。会升级

另一个没有运行APP1,即使APP1网络IAP过程升级失败了,也没关系。毕竟稳定的APP0的没有

改动,还能稳定工作,等待下一次APP1正常升级完成后再尝试启动。只有升级后APP1能稳定工

作且版本号也比APP0高的时候才会去升级APP0。无论如何两个APP都能至少有一个稳定的版本

能运行,确保设备正常工作。

4、2 框图

我使用的是STM32F429IGT6;环境是STM32CubeIDE 1.42版本。

由于是bootloader + 双APP,所以UART IAP功能会放在APP处,且存放系统APP版本的参数十分重要,必须也需要一个备份区。

即bootloader主要负责选择启动哪APP,还有最重要的是负责对于系统APP版本参数检查及矫正修改。

APP功能:LED闪烁 + IAP功能。


STM32CUBEMX IWDG 配置 stm32cubeide iap_STM32CUBEMX IWDG 配置_11

STM32F429IGT6内部flash扇区表

先将芯片内部flash 1MB分成五个区域,分别存放bootloader,系统参数存储区、系统参数备份区域和两个APP程序。

前面32kB放bootloader;由于STM32F4擦除最小单位是扇区,后面两个16kB用于系统参数存储区、系统参数备份区域;

剩下的空间都用于存放APP,但是剩下1个64kB 和 7个128kB没法均分,故分成APP0 448kB 和 APP1 512kB.

STM32CUBEMX IWDG 配置 stm32cubeide iap_系统参数_12

 

4、3 bootloader编写

做双APP的话,最重要的就是保护系统参数的正确性,否则丢失的话可能导致启动APP失败。

看门狗的作用是当跳转APP后程序跑飞后能复位重新选择启动;只有当APP程序正常启动后能重新初始化看门狗及喂狗。

流程图:

STM32CUBEMX IWDG 配置 stm32cubeide iap_STM32 IAP_13

结构体定义:

typedef enum {
	APPLICATION_NORMAL = 0,		// APP能正常稳定运行
	APPLICATION_UPDATED,		// APP刚更新完成,等待测试启动
	APPLICATION_ERROR,			// APP错误,不能正常工作
}Application_Status_t;

typedef	__PACKED_STRUCT{
	uint32_t Device_id;					// 设备号
	uint32_t Hardware_Version;			// 硬件版本信息
	uint32_t Application0_Version;		// APP0软件版本
	uint32_t Application1_Version;		// APP1软件版本
	uint32_t Application0_Status;		// APP0的状态. @Application_Status_t
	uint32_t Application1_Status;		// APP1的状态. @Application_Status_t
	uint8_t  Server_Address[64];		// 服务器的地址
	uint32_t Server_Port;				// 服务器的端口号
	uint8_t  Server_Key[4];				// 服务器的登录密钥,随时更新
	uint32_t SystemParamCRC;
}SystemParamTypeDef;

这个函数System_ParamReadCheck();用于每次上电检测系统参数存放的两个区的参数正确性,

若不正确及时修正。  我用的是STM32内部CRC校验数据的正确性, 用到了CRC校验的话一定要注意

结构体的字节对齐,尽量内部参数用4字节对齐。我以前做这个吃过字节对齐亏,hahahahh~

1、读取主区和备份区的系统参数分别进行CRC校验。

2、   (1)系统参数主存储区的数据校验正确,
         (2)开始校验系统参数备份区的数据:
         (3)如果备份区的数据CRC校验正确,则进行对比数据一致性,不一致则将主区数据写入备份区
         (4)如果备份区的数据CRC校验错误,则将主区的系统参数据写入备份区;

3、   (1)系统参数主存储区的数据校验错误
         (2) 开始校验系统参数备份区的数据:
         (3)如果备份区的数据CRC校验正确,则将备份区数据写入主区
         (4)如果备份区的数据CRC校验错误,则将默认系统参数数据写入两个区域,强行启动APP0;

void System_ParamReadCheck(SystemParamTypeDef *pData)
{
	uint32_t crcdatalen = 0;
	SystemParamTypeDef sysparam, sysparambackup;

	/* 读取主区和备份区的系统参数 */
	FLASH_If_Read(SYSTEMPARAM_ADDRESS, (uint32_t*) &sysparam, sizeof(SystemParamTypeDef) / 4);
	FLASH_If_Read(SYSTEMPARAM_BACKUPADDRESS, (uint32_t*) &sysparambackup, sizeof(SystemParamTypeDef) / 4);

	// 去掉系统参数CRC的数据校验长度
	crcdatalen = sizeof(SystemParamTypeDef) / 4 - 1;
	/* 主系统参数区放的的数据CRC校验  */
	if (HAL_CRC_Calculate(&hcrc, (uint32_t*) &sysparam, crcdatalen)	== sysparam.SystemParamCRC)
	{
		/*
		 *  @ 系统参数主存储区的数据校验正确,
		 *  @ 开始校验系统参数备份区的数据:
		 *    1、如果备份区的数据CRC校验正确,则进行对比数据一致性,不一致则将主区数据写入备份区
		 *    2、如果备份区的数据CRC校验错误,则将主区的系统参数据写入备份区;
		 * */
		printf("System parameter main sector data checked OK\r\n");
		if (HAL_CRC_Calculate(&hcrc, (uint32_t*) &sysparambackup, crcdatalen) == sysparambackup.SystemParamCRC)
		{
			printf("System parameter backup sector data checked OK\r\n");
			if (memcmp(&sysparam, &sysparambackup, sizeof(SystemParamTypeDef)) != 0)
			{
				printf("System parameter main sector and backup sector data are different, update backup sector data\r\n");
				memcpy(&sysparambackup, &sysparam, sizeof(SystemParamTypeDef));
				System_ParamUpdate(SYSTEMPARAM_BACKUPADDRESS, &sysparambackup);
			}
			else printf("System parameter main sector and backup sector data are the same\r\n");
		}
		else
		{
			printf("System parameter backup sector data checked Fail, update backup sector data\r\n");
			memcpy(&sysparambackup, &sysparam, sizeof(SystemParamTypeDef));
			System_ParamUpdate(SYSTEMPARAM_BACKUPADDRESS, &sysparambackup);
		}
	}
	else
	{
		/*
		 *  @ 系统参数主存储区的数据校验错误
		 *  @ 开始校验系统参数备份区的数据:
		 *    1、如果备份区的数据CRC校验正确,则将备份区数据写入主区
		 *    2、如果备份区的数据CRC校验错误,则将默认系统参数数据写入两个区域,强行启动APP0;
		 * */
		printf("System parameter main sector data checked Fail\r\n");
		if (HAL_CRC_Calculate(&hcrc, (uint32_t*) &sysparambackup, crcdatalen) == sysparambackup.SystemParamCRC)
		{
			printf("System parameter backup sector data checked OK, update master sector data\r\n");
			memcpy(&sysparam, &sysparambackup, sizeof(SystemParamTypeDef));
			System_ParamUpdate(SYSTEMPARAM_ADDRESS, &sysparam);
		}
		else
		{
			printf("System parameter main sector and backup sector data checked Fail, Restore defaults\r\n");
			SystemParam_default.SystemParamCRC = HAL_CRC_Calculate(&hcrc, (uint32_t*) &SystemParam_default, crcdatalen);
			memcpy(&sysparam, &SystemParam_default, sizeof(SystemParamTypeDef));
			memcpy(&sysparambackup, &SystemParam_default, sizeof(SystemParamTypeDef));
			System_ParamUpdate(SYSTEMPARAM_ADDRESS, &sysparam);
			System_ParamUpdate(SYSTEMPARAM_BACKUPADDRESS, &sysparambackup);
		}
	}

	memcpy(pData, &sysparam, sizeof(SystemParamTypeDef));
	printf("Hardware_Version = 0x%08lX\r\n", pData->Hardware_Version);
	printf("Application0_Version = 0x%08lX\r\n", pData->Application0_Version);
	printf("Application0_Status  = 0x%08lX\r\n", pData->Application0_Status);
	printf("Application1_Version = 0x%08lX\r\n", pData->Application1_Version);
	printf("Application1_Status  = 0x%08lX\r\n", pData->Application1_Status);
}

这个System_SelectBootAddress()函数用于选择启动哪个APP,返回APP地址进行调整。

1、选择哪个APP版本高。

2、判读APP的状态标志位。

      (1)APPLICATION_NORMAL:表示APP可以正常稳定工作。

       (2)APPLICATION_UPDATED:表示刚IAP更新的程序固件,把APP的status改成

                错误标志位 APPLICATION_ERROR。这么做的是为了如果首次启动成功后APP会

                修正这个标志位为APPLICATION_NORMAL,表示正常稳定工作,下次可以启动。

                如果首次启动失败后,则看门狗复位后再次选择启动的时候这个APP的状态标志位

                是错误就不会启动这个APP了。

       (3)APPLICATION_ERROR:表示这个APP是不能启动的。只能强行启动另一个APP,无论结果。

uint32_t System_SelectBootAddress(SystemParamTypeDef *pData)
{
	uint32_t BootAddress = 0;

	if (pData->Application0_Version >= pData->Application1_Version)
	{
		if (pData->Application0_Status == APPLICATION_NORMAL)
		{
			/* 正常启动 */
			printf("APP0 APPLICATION_NORMAL. Run the APP0.\r\n");
			BootAddress = APPLICATION0_ADDRESS;
		}
		else if (pData->Application0_Status == APPLICATION_UPDATED)
		{
			/*
			 * 刚IAP更新的程序固件,把APP的status改成 错误标志位 @APPLICATION_ERROR,
			 * 1、若APP跳转运行正常后会修改成正常启动标志 @APPLICATION_NORMAL。
			 * 2、若APP跳转运行失败(跑飞),那么看门狗会复位,重启后该APP是错误标志不会启动。
			 * 3、清空版本信息,待启动修正。
			 * 4、更新系统参数,写入flash
			 * */
			printf("APP0 APPLICATION_UPDATED. Clear APP0 system parameters, Run the APP0.\r\n");
			pData->Application0_Version = 0;
			pData->Application0_Status = APPLICATION_ERROR;
			System_ParamUpdate(SYSTEMPARAM_ADDRESS, pData);
			System_ParamUpdate(SYSTEMPARAM_BACKUPADDRESS, pData);
			BootAddress = APPLICATION0_ADDRESS;
		}
		else
		{
			/* 由于APP0信息错误,强制启动APP1 */
			printf("APP0 APPLICATION_ERROR. Forced the APP1 to run.\r\n");
			BootAddress = APPLICATION1_ADDRESS;
		}
	}
	else
	{
		if (pData->Application1_Status == APPLICATION_NORMAL)
		{
			printf("APP1 APPLICATION_NORMAL. Run the APP1.\r\n");
			BootAddress = APPLICATION1_ADDRESS;
		}
		else if (pData->Application1_Status == APPLICATION_UPDATED)
		{
			printf("APP1 APPLICATION_UPDATED. Clear APP1 system parameters, Run the APP1.\r\n");
			pData->Application1_Version = 0;
			pData->Application1_Status = APPLICATION_ERROR;
			System_ParamUpdate(SYSTEMPARAM_ADDRESS, pData);
			System_ParamUpdate(SYSTEMPARAM_BACKUPADDRESS, pData);
			BootAddress = APPLICATION1_ADDRESS;
		}
		else
		{
			/* 由于APP1信息错误,强制启动APP0 */
			printf("APP1 APPLICATION_ERROR. Forced the APP0 to run.\r\n");
			BootAddress = APPLICATION0_ADDRESS;
		}
	}

	return BootAddress;
}

bootloader 主函数:

其中跳转前要做的是bootloader用到什么外设和中断一定要关闭。

原因就是如果在APP端如果外设和中断复用基本没问题。

比如APP用freeRTOS,bootloader不带实时操作系统,那么freeRTOS一般会用外设定时器TIMx,而不用滴答时钟作为系统时钟。在bootloader调整前如果不关闭滴答时钟的定时器的话,则在APP端会跑飞;原因就是APP由于flash链接地址变了,APP会进行中断向量表的偏移,而新的中断向量表滴答时钟中断为空。

int main(void)
{
  /* USER CODE BEGIN 1 */

  /* USER CODE END 1 */

  /* MCU Configuration--------------------------------------------------------*/

  /* Reset of all peripherals, Initializes the Flash interface and the Systick. */
  HAL_Init();

  /* USER CODE BEGIN Init */

  /* USER CODE END Init */

  /* Configure the system clock */
  SystemClock_Config();

  /* USER CODE BEGIN SysInit */

  /* USER CODE END SysInit */

  /* Initialize all configured peripherals */
  MX_GPIO_Init();
  MX_USART1_UART_Init();
  MX_CRC_Init();
  MX_IWDG_Init();
  /* USER CODE BEGIN 2 */
  printf("\r\nSudaroot, This is a advanced bootloader\r\n");
  /* system param read check */
  System_ParamReadCheck(&SystemParam);
  /* execute the new program */
  HAL_IWDG_Refresh(&hiwdg);
  JumpAddress = System_SelectBootAddress(&SystemParam);
  JumpAddress = *(__IO uint32_t*) (JumpAddress + 4);
  printf("Start program execution......\r\n\n\n");
  /*
   * 关闭或反初始化前面用到的外设和中�?
   * 1、反初始化UART
   * 2、关闭系统滴答定时器中断
   * */
  HAL_UART_DeInit(&huart1);
  HAL_SuspendTick();
  /* Refresh the IWDG. */
  HAL_IWDG_Refresh(&hiwdg);
  /* Jump to user application */
  JumpToApplication = (pFunction) JumpAddress;
  /* Initialize user application's Stack Pointer */
  __set_MSP(*(__IO uint32_t*) (JumpAddress - 4));
  JumpToApplication();
  /* USER CODE END 2 */

  /* Infinite loop */
  /* USER CODE BEGIN WHILE */
  while (1)
  {
    /* USER CODE END WHILE */

    /* USER CODE BEGIN 3 */
  }
  /* USER CODE END 3 */
}

4、4 APP程序编写

流程图:

STM32CUBEMX IWDG 配置 stm32cubeide iap_系统参数_14

首先说明一点:APP的版本号是由程序写死的,不能说由外部指定。外部可以告知有没有新版更新,

即使告知新版本的具体版本号也只是仅供参考,不能当做实际版本号。如下程序定义了APP0的版本号。

/* Define APP0 software version */
#define	APPLICATION0_VERSION		0x00000010U

这个函数System_ParamReadCheckUpdate()作用就是检查APP的软件版本号和状态标志位,

如果需要修改标志,则修改完成后需要重启后进入bootloader重新进行APP选择,确保启动的是版本稳定且最新。

这个的作用就和bootloader修改首次启动的APP相关联,APP启动成功后会自己修正。

void System_ParamReadCheckUpdate(SystemParamTypeDef *pData)
{
	uint32_t flash_update = 0;

	FLASH_If_Read(SYSTEMPARAM_ADDRESS, (uint32_t*)pData, sizeof(SystemParamTypeDef) / 4);

	/* APP0的软件版本号与固件内部软件版本号不一致,更新版本号并写入Flash */
	if(pData->Application0_Version != APPLICATION0_VERSION)
	{
		flash_update = 1;
		printf("Update APP0 Software Version : 0x%08X\r\n", APPLICATION0_VERSION);
		pData->Application0_Version = APPLICATION0_VERSION;
	}

	/* APP0 启动了,但是APP0的状态是错误标志.
	 * 可能是由于更新完成后,第一次启动标记的,
	 * 修正为APPLICATION_NORMAL,重新写入flash
	 * */
	if(pData->Application0_Status != APPLICATION_NORMAL)
	{
		flash_update = 1;
		printf("Update APP0 Status, APPLICATION_NORMAL\r\n");
		pData->Application0_Status = APPLICATION_NORMAL;
	}

	/* 如果flash_update 等于 1, 即flash需要更新主存储区和备份区的系统参数数据
	 * 更新数据后,需要重启
	 * 注意:更新数据时,必须擦除一个扇区后写入完成后,才能更新另一个区域数据,否则有概率丢失两个扇区数据
	 * */
	if(flash_update == 1)
	{
		pData->SystemParamCRC = HAL_CRC_Calculate(&hcrc, (uint32_t *)pData, sizeof(SystemParamTypeDef) / 4 - 1);
		System_ParamUpdate(SYSTEMPARAM_ADDRESS, pData);
		System_ParamUpdate(SYSTEMPARAM_BACKUPADDRESS, pData);
		printf("device reboot\r\n");
		HAL_NVIC_SystemReset();
	}
}

这个函数SystemParam_IAP主要作用主要升级APP。记住一点就行了:升级的永远是没有运行的那个APP区域。

写入bin文件完成后把版本号改成0xFFFFFFFF,上面说过了,这个版本号无所谓。升级完成后重启,进入bootloader。

void SystemParam_IAP(SystemParamTypeDef *pData)
{
	uint32_t temp = 0;
	uint32_t file_length= 0;
	uint32_t timeout = HAL_MAX_DELAY;

	printf("System IAP start, update APP1\r\n");
	/* Download user application in the Flash */
	/* 1. erase user application1 area */
	printf("Wait for the internal Flash erase to complete\r\n");
	if(FLASH_If_Erase(APPLICATION1_ADDRESS, APPLICATION1_SECTOR_NUM) == 1)
	{
		printf("Erase the internal Flash is fail\r\n");
		Error_Handler();
	}
	printf("Erase the internal Flash is complete\r\n");

	/* 2. download a file via serial port */
	/* Clean the input path */
	__HAL_UART_FLUSH_DRREGISTER(&huart1);
	printf("Waiting for the file to be sent ... \r\n");

	while(1)
	{
		if(HAL_UART_Receive(&huart1, uart_buf, UART_BUF_SIZE, timeout) != HAL_OK)
		{
			temp = UART_BUF_SIZE - (huart1.RxXferCount + 1);
			FLASH_If_Write(APPLICATION1_ADDRESS + file_length, (uint32_t*)uart_buf, file_length / 4);
			file_length = file_length + temp;
			break;
		}
		timeout = 1000;
		FLASH_If_Write(APPLICATION1_ADDRESS + file_length, (uint32_t*)uart_buf, UART_BUF_SIZE / 4);
		file_length = file_length + UART_BUF_SIZE;
	}
	printf("Programming Completed Successfully! %ldBtye\r\n", file_length);
	printf("Update system parameters\r\n");
	pData->Application1_Version = 0xFFFFFFFF;
	pData->Application1_Status  = APPLICATION_UPDATED;
	pData->SystemParamCRC = HAL_CRC_Calculate(&hcrc, (uint32_t *)pData, sizeof(SystemParamTypeDef) / 4 - 1);
	System_ParamUpdate(SYSTEMPARAM_ADDRESS, pData);
	System_ParamUpdate(SYSTEMPARAM_BACKUPADDRESS, pData);

	printf("device reboot\r\n");
	HAL_NVIC_SystemReset();
}

APP的主函数:

APP运行后发送‘1’即可进入IAP模式。

int main(void)
{
  /* USER CODE BEGIN 1 */
	uint8_t key = 0;

	SCB->VTOR = APPLICATION0_ADDRESS;
  /* USER CODE END 1 */

  /* MCU Configuration--------------------------------------------------------*/

  /* Reset of all peripherals, Initializes the Flash interface and the Systick. */
  HAL_Init();

  /* USER CODE BEGIN Init */

  /* USER CODE END Init */

  /* Configure the system clock */
  SystemClock_Config();

  /* USER CODE BEGIN SysInit */

  /* USER CODE END SysInit */

  /* Initialize all configured peripherals */
  MX_GPIO_Init();
  MX_USART1_UART_Init();
  MX_IWDG_Init();
  MX_CRC_Init();
  MX_TIM2_Init();
  /* USER CODE BEGIN 2 */
  HAL_TIM_Base_Start_IT(&htim2);
  printf("\r\nSudaroot, This is IAP Application0\r\n");
  System_ParamReadCheckUpdate(&SystemParam);
  /* USER CODE END 2 */

  /* Infinite loop */
  /* USER CODE BEGIN WHILE */
  while (1)
  {
	  printf("\r\n=================== Main Menu ============================\r\n\n");
	  printf("  IAP:Download user application to the internal Flash ----- 1\r\n\n");
	  printf("  Current System APP0 Version: 0x%08lX\r\n", SystemParam.Application0_Version);
	  /* Clean the input path */
	  __HAL_UART_FLUSH_DRREGISTER(&huart1);
	  if(HAL_UART_Receive(&huart1, &key, 1, HAL_MAX_DELAY) == HAL_OK)
	  {
		  if(key == '1')
		  {
			  SystemParam_IAP(&SystemParam);
		  }
	  }
    /* USER CODE END WHILE */

    /* USER CODE BEGIN 3 */
  }
  /* USER CODE END 3 */
}

4、5 演示效果:


STM32 UART双APP的IAP

5 过程经验分享

1、芯片内部Flash擦写的时候是无法响应中断的,故在发送需要写入flash的数据 间隔时间 要相应拉长。非常重要~~~@@@!!!

STM32CUBEMX IWDG 配置 stm32cubeide iap_STM32CUBEMX IWDG 配置_15

2、CRC校验问题与字节对齐问题,CRC校验与字节数相关的。

当时我在裸板上写bootloader和APP带RTOS的校验结果不一样,导致数据一直不通过,最后发现是裸板默认是4字节对齐,RTOS是单字节对齐。最后我全改成4字节对齐了。

3、芯片内部Flash擦写次数是十分有限的,大概是10K次,故如果某些数据是频繁写入内部芯片flash,建议最好改用外部flash。

4、关于生产的问题,由于是多个程序烧写,为了生产的高效性只能想办法合成一个程序下发生产。

有两种选择:(1)使用bin文件合成工具。

                        (2)自己用系统编程的方式编写一个程序用于合成。

                     (3)把所有程序都烧录进ST芯片,再整个芯片程序读出,这样就合成一个程序了,且十分稳定。

5、程序跳转前必须关闭用到的外设和中断,关键是中断,否则跑飞。

再次提醒特别是使用了RTOS系统,由于像FreeRTOS会用外设TIMx做系统时钟,跳转前一定要关闭系统滴答时钟中断,否则跑飞。

        特别注意:HAL_Init()会重新初始化systick时钟,故在APP跳转后的初始化需注释掉。


STM32CUBEMX IWDG 配置 stm32cubeide iap_STM32 IAP_16

 

6、文章中APP程序分别使用了两个程序,有人提出疑问说是否需要维护两个APP程序?

其实我在实际项目中也只是维护一个程序代码。原因是APP0和APP1的代码差异非常小,用宏定义#define区分即可,但是每次编译必须检查编译器设置的flash地址是否与宏定义一致。 文章中使用两个APP是我想两个APP代码表现的更加简单明了。

7、调试APP程序的时候,不要带bootloader,把APP的flash地址改回0x08000000,待功能调试完成后再修改设置编译下发测试。

8、判断下发的bin文件是不是所需的bin文件?比如说在APP0中升级APP1,怎么知道下方的bin文件是APP0还是APP1呢?

其实很简单,如果你对STM32启动流程熟悉的话,就会发现bin文件的第2个(4字节)复位地址,这个地址会与APP的flash链接地址有关。下面可以看图,APP0的flash地址是0x08010000,APP1的flash地址是0x08080000。接收第一包bin文件后判断复位地址即可知道下发的APP bin文件是否是所需的。

STM32CUBEMX IWDG 配置 stm32cubeide iap_系统参数_17

STM32CUBEMX IWDG 配置 stm32cubeide iap_STM32 IAP_18

9、如果测试我的代码时发现用串口助手下发APP bin文件,然后启动下发APP,启动不了。原因就是APP下发丢包了。

那么你肯定没认真理解第1点经验分享的重要性。解决如下:把下发的延时适当拉长。

STM32CUBEMX IWDG 配置 stm32cubeide iap_双APP IAP_19

10、如果是用于项目生产,无线升级过程中,必须考虑   断电断网  和   网络接收到的数据错误请求重发  的问题。

断电和断网是一类问题,归APP固件数据接收不完全。解决:加入下发APP固件前,先发APP固件的长度和CRC校验  ,还有加上  数据接收超时就行,超时就中止升级。最后才更新芯片内部flash对应的标志位。

网络接收到的数据错误请求重发。解决:重发咯。重发可以是包(整个APP固件肯定分包下发的)重发,并不一定是整个APP固件重发。

我上面的垃圾代码没有实现这几个功能,给点思路参考,最后自己动手动脑。

11、APP的版本号是由固件程序写死的,不能说由外部指定。外部可以告知有没有新版更新,即使告知新版本的具体版本号也只是仅供参考,不能当做实际版本号。

  全篇完。