1454 lines
		
	
	
		
			58 KiB
		
	
	
	
		
			C
		
	
	
	
			
		
		
	
	
			1454 lines
		
	
	
		
			58 KiB
		
	
	
	
		
			C
		
	
	
	
| /*
 | ||
| * Copyright (c) 2020 AIIT XUOS Lab
 | ||
| * XiUOS is licensed under Mulan PSL v2.
 | ||
| * You can use this software according to the terms and conditions of the Mulan PSL v2.
 | ||
| * You may obtain a copy of Mulan PSL v2 at:
 | ||
| *        http://license.coscl.org.cn/MulanPSL2
 | ||
| * THIS SOFTWARE IS PROVIDED ON AN "AS IS" BASIS, WITHOUT WARRANTIES OF ANY KIND,
 | ||
| * EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO NON-INFRINGEMENT,
 | ||
| * MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE.
 | ||
| * See the Mulan PSL v2 for more details.
 | ||
| */
 | ||
| 
 | ||
| /**
 | ||
| * @file:    ota.c
 | ||
| * @brief:   file ota.c
 | ||
| * @version: 1.0
 | ||
| * @author:  AIIT XUOS Lab
 | ||
| * @date:    2023/4/23
 | ||
| *
 | ||
| */
 | ||
| #include <stdio.h>
 | ||
| #include <stdlib.h>
 | ||
| #include <stdbool.h>
 | ||
| #include <transform.h>
 | ||
| #include "shell.h"
 | ||
| #include "boot_for_ota.h"
 | ||
| #include "ymodem.h"
 | ||
| #include "ota.h"
 | ||
| 
 | ||
| #ifdef CONNECTION_ADAPTER_4G
 | ||
| #include <adapter.h>
 | ||
| #endif
 | ||
| 
 | ||
| #ifdef OTA_BY_PLATFORM
 | ||
| #include "platform_mqtt.h"
 | ||
| #endif
 | ||
| 
 | ||
| #ifdef USING_DOWNLOAD_JSON
 | ||
| #include "cJSON.h"
 | ||
| #endif
 | ||
| 
 | ||
| /****************************************************************************
 | ||
|  * Private Function Prototypes
 | ||
|  ****************************************************************************/
 | ||
| static uint32_t calculate_crc32(uint32_t addr, uint32_t len);
 | ||
| static int create_version(uint8_t* cur_version, uint8_t* new_version);
 | ||
| static status_t UpdateOTAFlag(ota_info_t *ptr);
 | ||
| static void InitialVersion(void);
 | ||
| static void BackupVersion(void);
 | ||
| static bool UpdateNewApplication(void);
 | ||
| static void Update(void);
 | ||
| static void BootLoaderJumpApp(void);
 | ||
| 
 | ||
| /****************************************************************************
 | ||
|  * Private Data
 | ||
|  ****************************************************************************/
 | ||
| static const mcuboot_t mcuboot =
 | ||
| {
 | ||
|     mcuboot_bord_init,
 | ||
|     UartConfig,
 | ||
|     Serial_PutString,
 | ||
|     FLASH_Init,
 | ||
|     FLASH_DeInit,
 | ||
|     Flash_Erase,
 | ||
|     Flash_Write,
 | ||
|     Flash_Read,
 | ||
|     Flash_Copy,
 | ||
|     SerialDownload,
 | ||
|     mcuboot_reset,
 | ||
|     mcuboot_jump,
 | ||
|     mcuboot_delay
 | ||
| };
 | ||
| static const uint32_t crc32tab[] = {
 | ||
|     0x00000000, 0x77073096, 0xee0e612c, 0x990951ba, 0x076dc419, 0x706af48f, 0xe963a535, 0x9e6495a3,
 | ||
|     0x0edb8832, 0x79dcb8a4, 0xe0d5e91e, 0x97d2d988, 0x09b64c2b, 0x7eb17cbd, 0xe7b82d07, 0x90bf1d91,
 | ||
|     0x1db71064, 0x6ab020f2, 0xf3b97148, 0x84be41de, 0x1adad47d, 0x6ddde4eb, 0xf4d4b551, 0x83d385c7,
 | ||
|     0x136c9856, 0x646ba8c0, 0xfd62f97a, 0x8a65c9ec, 0x14015c4f, 0x63066cd9, 0xfa0f3d63, 0x8d080df5,
 | ||
|     0x3b6e20c8, 0x4c69105e, 0xd56041e4, 0xa2677172, 0x3c03e4d1, 0x4b04d447, 0xd20d85fd, 0xa50ab56b,
 | ||
|     0x35b5a8fa, 0x42b2986c, 0xdbbbc9d6, 0xacbcf940, 0x32d86ce3, 0x45df5c75, 0xdcd60dcf, 0xabd13d59,
 | ||
|     0x26d930ac, 0x51de003a, 0xc8d75180, 0xbfd06116, 0x21b4f4b5, 0x56b3c423, 0xcfba9599, 0xb8bda50f,
 | ||
|     0x2802b89e, 0x5f058808, 0xc60cd9b2, 0xb10be924, 0x2f6f7c87, 0x58684c11, 0xc1611dab, 0xb6662d3d,
 | ||
|     0x76dc4190, 0x01db7106, 0x98d220bc, 0xefd5102a, 0x71b18589, 0x06b6b51f, 0x9fbfe4a5, 0xe8b8d433,
 | ||
|     0x7807c9a2, 0x0f00f934, 0x9609a88e, 0xe10e9818, 0x7f6a0dbb, 0x086d3d2d, 0x91646c97, 0xe6635c01,
 | ||
|     0x6b6b51f4, 0x1c6c6162, 0x856530d8, 0xf262004e, 0x6c0695ed, 0x1b01a57b, 0x8208f4c1, 0xf50fc457,
 | ||
|     0x65b0d9c6, 0x12b7e950, 0x8bbeb8ea, 0xfcb9887c, 0x62dd1ddf, 0x15da2d49, 0x8cd37cf3, 0xfbd44c65,
 | ||
|     0x4db26158, 0x3ab551ce, 0xa3bc0074, 0xd4bb30e2, 0x4adfa541, 0x3dd895d7, 0xa4d1c46d, 0xd3d6f4fb,
 | ||
|     0x4369e96a, 0x346ed9fc, 0xad678846, 0xda60b8d0, 0x44042d73, 0x33031de5, 0xaa0a4c5f, 0xdd0d7cc9,
 | ||
|     0x5005713c, 0x270241aa, 0xbe0b1010, 0xc90c2086, 0x5768b525, 0x206f85b3, 0xb966d409, 0xce61e49f,
 | ||
|     0x5edef90e, 0x29d9c998, 0xb0d09822, 0xc7d7a8b4, 0x59b33d17, 0x2eb40d81, 0xb7bd5c3b, 0xc0ba6cad,
 | ||
|     0xedb88320, 0x9abfb3b6, 0x03b6e20c, 0x74b1d29a, 0xead54739, 0x9dd277af, 0x04db2615, 0x73dc1683,
 | ||
|     0xe3630b12, 0x94643b84, 0x0d6d6a3e, 0x7a6a5aa8, 0xe40ecf0b, 0x9309ff9d, 0x0a00ae27, 0x7d079eb1,
 | ||
|     0xf00f9344, 0x8708a3d2, 0x1e01f268, 0x6906c2fe, 0xf762575d, 0x806567cb, 0x196c3671, 0x6e6b06e7,
 | ||
|     0xfed41b76, 0x89d32be0, 0x10da7a5a, 0x67dd4acc, 0xf9b9df6f, 0x8ebeeff9, 0x17b7be43, 0x60b08ed5,
 | ||
|     0xd6d6a3e8, 0xa1d1937e, 0x38d8c2c4, 0x4fdff252, 0xd1bb67f1, 0xa6bc5767, 0x3fb506dd, 0x48b2364b,
 | ||
|     0xd80d2bda, 0xaf0a1b4c, 0x36034af6, 0x41047a60, 0xdf60efc3, 0xa867df55, 0x316e8eef, 0x4669be79,
 | ||
|     0xcb61b38c, 0xbc66831a, 0x256fd2a0, 0x5268e236, 0xcc0c7795, 0xbb0b4703, 0x220216b9, 0x5505262f,
 | ||
|     0xc5ba3bbe, 0xb2bd0b28, 0x2bb45a92, 0x5cb36a04, 0xc2d7ffa7, 0xb5d0cf31, 0x2cd99e8b, 0x5bdeae1d,
 | ||
|     0x9b64c2b0, 0xec63f226, 0x756aa39c, 0x026d930a, 0x9c0906a9, 0xeb0e363f, 0x72076785, 0x05005713,
 | ||
|     0x95bf4a82, 0xe2b87a14, 0x7bb12bae, 0x0cb61b38, 0x92d28e9b, 0xe5d5be0d, 0x7cdcefb7, 0x0bdbdf21,
 | ||
|     0x86d3d2d4, 0xf1d4e242, 0x68ddb3f8, 0x1fda836e, 0x81be16cd, 0xf6b9265b, 0x6fb077e1, 0x18b74777,
 | ||
|     0x88085ae6, 0xff0f6a70, 0x66063bca, 0x11010b5c, 0x8f659eff, 0xf862ae69, 0x616bffd3, 0x166ccf45,
 | ||
|     0xa00ae278, 0xd70dd2ee, 0x4e048354, 0x3903b3c2, 0xa7672661, 0xd06016f7, 0x4969474d, 0x3e6e77db,
 | ||
|     0xaed16a4a, 0xd9d65adc, 0x40df0b66, 0x37d83bf0, 0xa9bcae53, 0xdebb9ec5, 0x47b2cf7f, 0x30b5ffe9,
 | ||
|     0xbdbdf21c, 0xcabac28a, 0x53b39330, 0x24b4a3a6, 0xbad03605, 0xcdd70693, 0x54de5729, 0x23d967bf,
 | ||
|     0xb3667a2e, 0xc4614ab8, 0x5d681b02, 0x2a6f2b94, 0xb40bbe37, 0xc30c8ea1, 0x5a05df1b, 0x2d02ef8d
 | ||
| };
 | ||
| 
 | ||
| 
 | ||
| /*******************************************************************************
 | ||
| * 函 数 名: calculate_crc32
 | ||
| * 功能描述: 计算给定Flash内存地址范围中数据的CRC32校验和
 | ||
| * 形    参: addr:表示Flash地址的起始位置
 | ||
|             len:表示需要计算CRC32的数据长度
 | ||
| * 返 回 值: 计算得到的CRC32值
 | ||
| *******************************************************************************/
 | ||
| static uint32_t calculate_crc32(uint32_t addr, uint32_t len)
 | ||
| {
 | ||
|     uint32_t crc = 0xFFFFFFFF;
 | ||
|     uint8_t byte = 0xFF;
 | ||
| 
 | ||
|     for(uint32_t i = 0; i < len; i++) 
 | ||
|     {
 | ||
|         byte = *((volatile uint8_t *)(addr + i));
 | ||
|         crc = crc32tab[(crc ^ byte) & 0xff] ^ (crc >> 8);
 | ||
|     }
 | ||
|     return crc^0xFFFFFFFF;
 | ||
| }
 | ||
| 
 | ||
| 
 | ||
| /*******************************************************************************
 | ||
| * 函 数 名: create_version
 | ||
| * 功能描述: 根据当前版本号生成新的三段式版本号,适用于iap方式和TCPSERVER
 | ||
| * 形    参: cur_version:当前版本号,new_version:生成的新版本号
 | ||
| * 返 回 值: 0:生成成功,-1:生成失败
 | ||
| * 说    明: 为保持一致,平台通过OTA传输而来的版本号也要保持这样三段式的形式
 | ||
|             版本形式为major.minor.patch,如001.002.003
 | ||
| *******************************************************************************/
 | ||
| static int create_version(uint8_t* cur_version, uint8_t* new_version) 
 | ||
| {
 | ||
|     int major, minor, patch; //三段式版本号的各个部分
 | ||
| 
 | ||
|     //解析当前版本号,版本号格式不对直接返回
 | ||
|     if (sscanf(cur_version, "%03d.%03d.%03d", &major, &minor, &patch) != 3) {
 | ||
|         return -1;
 | ||
|     }
 | ||
| 
 | ||
|     //将当前版本号加1
 | ||
|     patch++;
 | ||
|     if(patch > 999)
 | ||
|     {
 | ||
|         minor++;
 | ||
|         patch = 0;
 | ||
|         if(minor > 999) 
 | ||
|         {
 | ||
|             major++;
 | ||
|             minor = 0;
 | ||
|             patch = 0;
 | ||
|             if(major > 999) 
 | ||
|             {
 | ||
|                 return -1;
 | ||
|             }
 | ||
|         }
 | ||
|     }
 | ||
| 
 | ||
|     //更新版本号
 | ||
|     sprintf(new_version, "%03d.%03d.%03d", major, minor, patch);
 | ||
|     return 0;
 | ||
| }
 | ||
| 
 | ||
| 
 | ||
| /*******************************************************************************
 | ||
| * 函 数 名: UpdateOTAFlag
 | ||
| * 功能描述: 更新OTA Flag区域的信息,版本完成下载后在app里进行调用
 | ||
| * 形    参: ptr:ota_info_t结构体指针,描述OTA升级相关信息
 | ||
| * 返 回 值: 如果函数执行成功,状态值为 kStatus_Success,否则状态值为其他错误码
 | ||
| *******************************************************************************/
 | ||
| static status_t UpdateOTAFlag(ota_info_t *ptr)
 | ||
| {
 | ||
|     status_t status;
 | ||
| 
 | ||
|     status = mcuboot.op_flash_erase(FLAG_FLAH_ADDRESS,sizeof(ota_info_t));
 | ||
|     if(status != kStatus_Success)
 | ||
|     {
 | ||
|         return status;
 | ||
|     }
 | ||
|     status = mcuboot.op_flash_write(FLAG_FLAH_ADDRESS,(void *)ptr,sizeof(ota_info_t));
 | ||
| 
 | ||
|     return status;
 | ||
| }
 | ||
| 
 | ||
| 
 | ||
| /*******************************************************************************
 | ||
| * 函 数 名: InitialVersion
 | ||
| * 功能描述: 该函数可以烧写APP分区的初始化版本,初始化版本的版本号为0x1
 | ||
| * 形    参: 无
 | ||
| * 返 回 值: 无
 | ||
| *******************************************************************************/
 | ||
| static void InitialVersion(void)
 | ||
| {
 | ||
|     int32_t size;
 | ||
|     ota_info_t ota_info;
 | ||
| 
 | ||
|     memset(&ota_info, 0, sizeof(ota_info_t));
 | ||
|     size = mcuboot.download_by_serial(XIUOS_FLAH_ADDRESS);
 | ||
|     if(size > 0)
 | ||
|     {
 | ||
|         ota_info.os.size = size;
 | ||
|         ota_info.os.crc32 = calculate_crc32(XIUOS_FLAH_ADDRESS, size);
 | ||
| 
 | ||
|         strncpy(ota_info.os.version,"001.000.000",sizeof(ota_info.os.version));
 | ||
|         strncpy(ota_info.os.description, "The initial firmware.", sizeof(ota_info.os.description));
 | ||
| 
 | ||
|         UpdateOTAFlag(&ota_info);
 | ||
|     }
 | ||
| }
 | ||
| 
 | ||
| 
 | ||
| /*******************************************************************************
 | ||
| * 函 数 名: BackupVersion
 | ||
| * 功能描述: 版本回退函数,如果升级的APP存在bug导致无法跳转需调用此函数进行版本回退
 | ||
| * 形    参: 无
 | ||
| * 返 回 值: 无
 | ||
| *******************************************************************************/
 | ||
| static void BackupVersion(void)
 | ||
| {
 | ||
|     status_t status;
 | ||
| 
 | ||
|     ota_info_t ota_info;
 | ||
|     memset(&ota_info, 0, sizeof(ota_info_t));
 | ||
|     mcuboot.op_flash_read(FLAG_FLAH_ADDRESS, (void*)&ota_info, sizeof(ota_info_t));
 | ||
| 
 | ||
|     ota_info.status = OTA_STATUS_BACKUP;
 | ||
|     UpdateOTAFlag(&ota_info);
 | ||
|     // 1.先清空XiUOS System分区
 | ||
|     status = mcuboot.op_flash_erase(XIUOS_FLAH_ADDRESS, ota_info.os.size);
 | ||
|     if(status == kStatus_Success)
 | ||
|     {
 | ||
|         memset(&ota_info.os,0,sizeof(ota_info.os));
 | ||
|         UpdateOTAFlag(&ota_info);
 | ||
|         mcuboot.print_string("\r\n------Clear app partition success!------\r\n");
 | ||
|     }
 | ||
|     else
 | ||
|     {
 | ||
|         mcuboot.print_string("\r\n------Clear app partition failed!------\r\n");
 | ||
|         return;
 | ||
|     }
 | ||
|     // 2.拷贝backup分区到XiUOS System分区
 | ||
|     status = mcuboot.op_flash_copy(BAKUP_FLAH_ADDRESS, XIUOS_FLAH_ADDRESS, ota_info.bak.size);
 | ||
|     if(status == kStatus_Success)
 | ||
|     {
 | ||
|         ota_info.os.size = ota_info.bak.size;
 | ||
|         ota_info.os.crc32 = ota_info.bak.crc32;
 | ||
|         memset(ota_info.os.version,0,sizeof(ota_info.os.version)); 
 | ||
|         strncpy(ota_info.os.version, ota_info.bak.version, sizeof(ota_info.bak.version));
 | ||
|         memset(ota_info.os.description,0,sizeof(ota_info.os.description)); 
 | ||
|         strncpy(ota_info.os.description, ota_info.bak.description, sizeof(ota_info.bak.description));
 | ||
|         UpdateOTAFlag(&ota_info);
 | ||
|         mcuboot.print_string("\r\n------Version rollback successful!------\r\n");
 | ||
|     }
 | ||
|     else
 | ||
|     {
 | ||
|         ota_info.status = OTA_STATUS_ERROR;
 | ||
|         memset(ota_info.error_message,0,sizeof(ota_info.error_message)); 
 | ||
|         strncpy(ota_info.error_message, "Version rollback failed!",sizeof(ota_info.error_message));
 | ||
|         UpdateOTAFlag(&ota_info);
 | ||
|         mcuboot.print_string("\r\n------Version rollback failed!------\r\n");
 | ||
|     }
 | ||
| }
 | ||
| 
 | ||
| 
 | ||
| /*******************************************************************************
 | ||
| * 函 数 名: UpdateNewApplication
 | ||
| * 功能描述: 在bootloader里进行调用,根据Flash中Flag分区中的信息决定是否进行版本更新
 | ||
| * 形    参: 无
 | ||
| * 返 回 值: true:需要升级,发生了flash搬移的操作,不代表升级的结果
 | ||
|             false:不需要升级,未发生flash搬移
 | ||
| * 注    释: 该函数调用后如果不需要升级APP分区保持不变,否则APP分区的版本为新版本
 | ||
| *******************************************************************************/
 | ||
| static bool UpdateNewApplication(void)
 | ||
| {
 | ||
|     status_t status;
 | ||
|     ota_info_t ota_info;  // 定义OTA信息结构体
 | ||
| 
 | ||
|     memset(&ota_info, 0, sizeof(ota_info_t));
 | ||
|     // 从Flash中读取OTA信息
 | ||
|     mcuboot.op_flash_read(FLAG_FLAH_ADDRESS, (void*)&ota_info, sizeof(ota_info_t));
 | ||
| 
 | ||
|     // 如果OTA升级状态为准备状态,且APP分区与download分区版本不同,才需要进行升级
 | ||
|     if((ota_info.status == OTA_STATUS_READY) && (ota_info.os.crc32 != ota_info.down.crc32)) 
 | ||
|     {
 | ||
|         mcuboot.print_string("\r\n------Start upgrading to new version!------\r\n");
 | ||
|         // 校验downlad分区固件CRC
 | ||
|         if(calculate_crc32(DOWN_FLAH_ADDRESS, ota_info.down.size) == ota_info.down.crc32) 
 | ||
|         {
 | ||
|             ota_info.status = OTA_STATUS_UPDATING;
 | ||
|             UpdateOTAFlag(&ota_info);
 | ||
| 
 | ||
|             // 1.如果CRC校验通过,开始升级,先清空Backup分区
 | ||
|             status = mcuboot.op_flash_erase(BAKUP_FLAH_ADDRESS, ota_info.bak.size);
 | ||
|             if(status == kStatus_Success)
 | ||
|             {
 | ||
|                 memset(&ota_info.bak,0,sizeof(ota_info.bak));
 | ||
|                 UpdateOTAFlag(&ota_info);
 | ||
|                 mcuboot.print_string("\r\n------Clear backup partition success!------\r\n");
 | ||
|             }
 | ||
|             else
 | ||
|             {
 | ||
|                 mcuboot.print_string("\r\n------Clear backup partition failed!------\r\n");
 | ||
|                 goto finish;
 | ||
|             }
 | ||
| 
 | ||
|             // 2.逐字节拷贝Flash,备份当前XiUOS System分区内容到Backup分区
 | ||
|             status = mcuboot.op_flash_copy(XIUOS_FLAH_ADDRESS, BAKUP_FLAH_ADDRESS, ota_info.os.size);
 | ||
|             if((status == kStatus_Success) &&(calculate_crc32(BAKUP_FLAH_ADDRESS, ota_info.os.size) == ota_info.os.crc32))
 | ||
|             {
 | ||
|                 ota_info.bak.size = ota_info.os.size;
 | ||
|                 ota_info.bak.crc32 = ota_info.os.crc32;
 | ||
|                 memset(ota_info.bak.version,0,sizeof(ota_info.bak.version)); 
 | ||
|                 strncpy(ota_info.bak.version, ota_info.os.version, sizeof(ota_info.os.version));
 | ||
|                 memset(ota_info.bak.description,0,sizeof(ota_info.bak.description)); 
 | ||
|                 strncpy(ota_info.bak.description, ota_info.os.description, sizeof(ota_info.os.description));
 | ||
|                 UpdateOTAFlag(&ota_info);
 | ||
|                 mcuboot.print_string("\r\n------Backup app success!------\r\n");
 | ||
|             }
 | ||
|             else
 | ||
|             {
 | ||
|                 ota_info.status = OTA_STATUS_ERROR;
 | ||
|                 memset(ota_info.error_message,0,sizeof(ota_info.error_message)); 
 | ||
|                 strncpy(ota_info.error_message, "Backup app failed!",sizeof(ota_info.error_message));
 | ||
|                 UpdateOTAFlag(&ota_info);
 | ||
|                 mcuboot.print_string("\r\n------Backup app failed!------\r\n");
 | ||
|                 goto finish;
 | ||
|             }
 | ||
| 
 | ||
|             // 3.清空XiUOS System分区
 | ||
|             status = mcuboot.op_flash_erase(XIUOS_FLAH_ADDRESS, ota_info.os.size);
 | ||
|             if(status == kStatus_Success)
 | ||
|             {
 | ||
|                 memset(&ota_info.os,0,sizeof(ota_info.os));
 | ||
|                 UpdateOTAFlag(&ota_info);
 | ||
|                 mcuboot.print_string("\r\n------Clear app partition success!------\r\n");
 | ||
|             }
 | ||
|             else
 | ||
|             {
 | ||
|                 mcuboot.print_string("\r\n------Clear app partition failed!------\r\n");
 | ||
|                 goto finish;
 | ||
|             }
 | ||
| 
 | ||
|             // 4.拷贝download分区到XiUOS System分区
 | ||
|             status = mcuboot.op_flash_copy(DOWN_FLAH_ADDRESS, XIUOS_FLAH_ADDRESS, ota_info.down.size);
 | ||
|             if((status == kStatus_Success) &&(calculate_crc32(XIUOS_FLAH_ADDRESS, ota_info.down.size) == ota_info.down.crc32))
 | ||
|             {
 | ||
|                 ota_info.os.size = ota_info.down.size;
 | ||
|                 ota_info.os.crc32 = ota_info.down.crc32;
 | ||
|                 memset(ota_info.os.version,0,sizeof(ota_info.os.version)); 
 | ||
|                 strncpy(ota_info.os.version, ota_info.down.version, sizeof(ota_info.down.version));
 | ||
|                 memset(ota_info.os.description,0,sizeof(ota_info.os.description)); 
 | ||
|                 strncpy(ota_info.os.description, ota_info.down.description, sizeof(ota_info.down.description));
 | ||
|                 UpdateOTAFlag(&ota_info);
 | ||
|                 mcuboot.print_string("\r\n------The download partition is copied successfully!------\r\n");
 | ||
|             }
 | ||
|             else
 | ||
|             {
 | ||
|                 ota_info.status = OTA_STATUS_ERROR;
 | ||
|                 memset(ota_info.error_message,0,sizeof(ota_info.error_message)); 
 | ||
|                 strncpy(ota_info.error_message, "The download partition copy failed!",sizeof(ota_info.error_message));
 | ||
|                 UpdateOTAFlag(&ota_info);
 | ||
|                 mcuboot.print_string("\r\n------The download partition copy failed!------\r\n");
 | ||
|                 goto finish;
 | ||
|             }
 | ||
| 
 | ||
|             // 5.清空download分区
 | ||
|             status = mcuboot.op_flash_erase(DOWN_FLAH_ADDRESS, ota_info.down.size);
 | ||
|             if(status == kStatus_Success)
 | ||
|             {
 | ||
|                 memset(&ota_info.down,0,sizeof(ota_info.down));
 | ||
|                 UpdateOTAFlag(&ota_info);
 | ||
|                 mcuboot.print_string("\r\n------Clear download partition success!------\r\n");
 | ||
|             }
 | ||
|             else
 | ||
|             {
 | ||
|                 mcuboot.print_string("\r\n------Clear download partition failed!------\r\n");
 | ||
|                 goto finish;
 | ||
|             }
 | ||
| 
 | ||
|             ota_info.status == OTA_STATUS_IDLE; //将OTA升级状态设置为IDLE
 | ||
|             UpdateOTAFlag(&ota_info);
 | ||
|             mcuboot.print_string("\r\n------New version upgrade completed,reboot again!------\r\n");
 | ||
|             goto finish;
 | ||
|         }
 | ||
|         else
 | ||
|         {
 | ||
|             // 如果download分区CRC校验失败,升级失败
 | ||
|             ota_info.status = OTA_STATUS_ERROR;
 | ||
|             memset(ota_info.error_message,0,sizeof(ota_info.error_message)); 
 | ||
|             strncpy(ota_info.error_message, "Download Firmware CRC check failed!",sizeof(ota_info.error_message));
 | ||
|             UpdateOTAFlag(&ota_info);
 | ||
|             mcuboot.print_string("\r\n------Download Firmware CRC check failed!------\r\n");
 | ||
|             goto finish;  
 | ||
|         }  
 | ||
|     }
 | ||
|     else
 | ||
|     {
 | ||
|         // 如果OTA升级状态为准备状态但APP分区与download分区版本相同,不需要进行升级,重置status,打印提示信息
 | ||
|         if((ota_info.status == OTA_STATUS_READY) && (ota_info.os.crc32 == ota_info.down.crc32)) 
 | ||
|         {
 | ||
|             ota_info.status == OTA_STATUS_IDLE;
 | ||
|             UpdateOTAFlag(&ota_info);
 | ||
|             mcuboot.print_string("\r\n------The app partition is the same as the download partition, no need to upgrade!------\r\n");
 | ||
|         }
 | ||
|         return false;
 | ||
|     }
 | ||
| finish:
 | ||
|     return true;
 | ||
| }
 | ||
| 
 | ||
| 
 | ||
| /*******************************************************************************
 | ||
| * 函 数 名: Update
 | ||
| * 功能描述: 根据实际情况进行初始化版本的烧录或者新版本的升级
 | ||
| * 形    参: 无
 | ||
| * 返 回 值: 无
 | ||
| *******************************************************************************/
 | ||
| static void Update(void)
 | ||
| {
 | ||
|     ota_info_t ota_info;
 | ||
|     mcuboot.flash_init();
 | ||
|     memset(&ota_info, 0, sizeof(ota_info_t));
 | ||
|     mcuboot.op_flash_read(FLAG_FLAH_ADDRESS, (void*)&ota_info, sizeof(ota_info_t));
 | ||
|     /* APP分区无有效固件时,需在bootloader下烧写初始固件, os.size大于分区大小视为无效固件*/
 | ||
|     if(ota_info.os.size > APP_FLASH_SIZE)
 | ||
|     {
 | ||
|         mcuboot.print_string("\r\nNeed to flash initial firmware!\r\n");
 | ||
|         InitialVersion();
 | ||
|         mcuboot.flash_deinit();
 | ||
|     }
 | ||
|     /*进行新版本的升级*/
 | ||
|     else 
 | ||
|     {
 | ||
|         if(UpdateNewApplication() == true) /*如果实际发生了flash搬移,操作完成后再重启一次*/
 | ||
|         {
 | ||
|             mcuboot.flash_deinit();
 | ||
|             mcuboot.op_delay(2000);
 | ||
|             mcuboot.op_reset();
 | ||
|         }
 | ||
|         else /*如果实际未发生flash搬移,说明未发生实际的升级,操作完成后不必再重启*/
 | ||
|         {
 | ||
|             mcuboot.flash_deinit();
 | ||
|         }
 | ||
|     }
 | ||
| }
 | ||
| 
 | ||
| 
 | ||
| /*******************************************************************************
 | ||
| * 函 数 名: BootLoaderJumpApp
 | ||
| * 功能描述: 上次跳转若是失败的,先从BAKUP分区进行恢复,然后再进行跳转
 | ||
| * 形    参: 无
 | ||
| * 返 回 值: 无
 | ||
| *******************************************************************************/
 | ||
| static void BootLoaderJumpApp(void)
 | ||
| {
 | ||
|     ota_info_t ota_info;
 | ||
| 
 | ||
|     mcuboot.flash_init();
 | ||
|     memset(&ota_info, 0, sizeof(ota_info_t));
 | ||
|     mcuboot.op_flash_read(FLAG_FLAH_ADDRESS, (void*)&ota_info, sizeof(ota_info_t));
 | ||
| 
 | ||
|     if(ota_info.lastjumpflag == JUMP_FAILED_FLAG)
 | ||
|     {
 | ||
|         mcuboot.print_string("\r\n------Jump to app partition failed,start version rollback!------\r\n");
 | ||
|         BackupVersion();   
 | ||
|     }
 | ||
|     else
 | ||
|     {
 | ||
|         ota_info.lastjumpflag = JUMP_FAILED_FLAG;
 | ||
|         UpdateOTAFlag(&ota_info);
 | ||
|     }
 | ||
|     mcuboot.flash_deinit();
 | ||
|     mcuboot.op_jump();
 | ||
| }
 | ||
| 
 | ||
| 
 | ||
| /*********************************************************************************
 | ||
| * 函 数 名: app_ota_by_iap
 | ||
| * 功能描述: 通过命令来进行ota升级,该函数与升级的命令关联,通过串口iap方式传输bin文件
 | ||
| * 形    参: 无
 | ||
| * 返 回 值: 无
 | ||
| *********************************************************************************/
 | ||
| static void app_ota_by_iap(void)
 | ||
| {
 | ||
|     int32_t size;
 | ||
|     ota_info_t ota_info;
 | ||
| 
 | ||
|     mcuboot.flash_init();
 | ||
|     mcuboot.serial_init();
 | ||
|     memset(&ota_info, 0, sizeof(ota_info_t));
 | ||
|     mcuboot.op_flash_read(FLAG_FLAH_ADDRESS, (void*)&ota_info, sizeof(ota_info_t));
 | ||
|     ota_info.status = OTA_STATUS_DOWNLOADING;
 | ||
|     UpdateOTAFlag(&ota_info);
 | ||
|     size = mcuboot.download_by_serial(DOWN_FLAH_ADDRESS);
 | ||
|     ota_info.status = OTA_STATUS_DOWNLOADED;
 | ||
|     UpdateOTAFlag(&ota_info);
 | ||
|     if(size > 0)
 | ||
|     {
 | ||
|         ota_info.down.size = size;
 | ||
|         ota_info.down.crc32= calculate_crc32(DOWN_FLAH_ADDRESS, size);
 | ||
|     
 | ||
|         memset(ota_info.down.version,0,sizeof(ota_info.down.version)); 
 | ||
|         create_version(ota_info.os.version, ota_info.down.version);
 | ||
|      
 | ||
|         memset(ota_info.down.description,0,sizeof(ota_info.down.description)); 
 | ||
|         strncpy(ota_info.down.description, "OTA Test bin.",sizeof(ota_info.down.description));
 | ||
| 
 | ||
|         ota_info.status = OTA_STATUS_READY;
 | ||
|     
 | ||
|         memset(ota_info.error_message,0,sizeof(ota_info.error_message));
 | ||
|         strncpy(ota_info.error_message, "No error message!",sizeof(ota_info.error_message));
 | ||
|         UpdateOTAFlag(&ota_info);
 | ||
|     }
 | ||
|     else
 | ||
|     {
 | ||
|         ota_info.status = OTA_STATUS_ERROR;
 | ||
| 
 | ||
|         memset(ota_info.error_message,0,sizeof(ota_info.error_message));
 | ||
|         strncpy(ota_info.error_message, "Failed to download firmware to download partition!",sizeof(ota_info.error_message));
 | ||
| 
 | ||
|         UpdateOTAFlag(&ota_info);
 | ||
|     }
 | ||
|     mcuboot.flash_deinit();
 | ||
|     MdelayKTask(2000);
 | ||
|     mcuboot.op_reset();
 | ||
| }
 | ||
| SHELL_EXPORT_CMD(SHELL_CMD_PERMISSION(0)|SHELL_CMD_TYPE(SHELL_TYPE_CMD_FUNC)|SHELL_CMD_PARAM_NUM(0),iap, app_ota_by_iap, ota by iap function);
 | ||
| 
 | ||
| 
 | ||
| #ifdef OTA_BY_TCPSERVER
 | ||
| static uint16_t calculate_crc16(uint8_t * data, uint32_t len);
 | ||
| static void get_start_signal(struct Adapter* adapter);
 | ||
| static int ota_data_recv(struct Adapter* adapter);
 | ||
| static ota_data start_msg;
 | ||
| static ota_data recv_msg;
 | ||
| 
 | ||
| /*******************************************************************************
 | ||
| * 函 数 名: calculate_crc16
 | ||
| * 功能描述: 计算给定长度的数据的crc16的值,用于OTA传输过程中数据帧的校验
 | ||
| * 形    参: data:数据buffer
 | ||
|             len:表示需要计算CRC16的数据长度
 | ||
| * 返 回 值: 计算得到的CRC16值
 | ||
| *******************************************************************************/
 | ||
| static uint16_t calculate_crc16(uint8_t * data, uint32_t len)
 | ||
| {
 | ||
|     uint16_t reg_crc=0xFFFF;
 | ||
|     while(len--) 
 | ||
|     {
 | ||
|         reg_crc ^= *data++;
 | ||
|         for(int j=0;j<8;j++) 
 | ||
|         {
 | ||
|             if(reg_crc & 0x01)
 | ||
|                 reg_crc=reg_crc >>1 ^ 0xA001;
 | ||
|             else
 | ||
|                 reg_crc=reg_crc >>1;
 | ||
|         }
 | ||
|     }
 | ||
|     KPrintf("crc = [0x%x]\n",reg_crc);
 | ||
|     return reg_crc;
 | ||
| }
 | ||
| 
 | ||
| 
 | ||
| /*******************************************************************************
 | ||
| * 函 数 名: get_start_signal
 | ||
| * 功能描述: 通过4G方式从服务端接收开始信号
 | ||
| * 形    参: adapter:Adapter指针,指向注册的4G设备
 | ||
| * 返 回 值: 0:传输成功,-1:传输失败
 | ||
| *******************************************************************************/
 | ||
| static void get_start_signal(struct Adapter* adapter)
 | ||
| {
 | ||
|     ota_info_t ota_info;
 | ||
|     uint8_t reply[16] = {0};  
 | ||
|     uint32_t flashdestination = DOWN_FLAH_ADDRESS;
 | ||
| 
 | ||
|     memset(&ota_info, 0, sizeof(ota_info_t));
 | ||
|     mcuboot.op_flash_read(FLAG_FLAH_ADDRESS, (void*)&ota_info, sizeof(ota_info_t));
 | ||
|     ota_info.status = OTA_STATUS_DOWNLOADING;
 | ||
|     UpdateOTAFlag(&ota_info);
 | ||
|     while(1)
 | ||
|     {
 | ||
|         memset(&start_msg, 0, sizeof(ota_data));
 | ||
|         /* step1:Confirm the start signal of transmission. */
 | ||
|         KPrintf("waiting for start msg...\n");
 | ||
|         if(AdapterDeviceRecv(adapter, &start_msg, sizeof(start_msg)) >= 0 && start_msg.header.frame_flag == STARTFLAG) 
 | ||
|         {
 | ||
|             if(start_msg.header.total_len > APP_FLASH_SIZE)
 | ||
|             {
 | ||
|                 KPrintf("File size is larger than partition size,the partition size is %dk.\n",APP_FLASH_SIZE/1024);
 | ||
|                 break;
 | ||
|             }   
 | ||
|             if(mcuboot.op_flash_erase(DOWN_FLAH_ADDRESS,start_msg.header.total_len) != kStatus_Success)
 | ||
|             {
 | ||
|                 KPrintf("Failed to erase target fash!\n");
 | ||
|                 break;
 | ||
|             } 
 | ||
|             else 
 | ||
|             {
 | ||
|                 KPrintf("Erase flash successful,erase length is %d bytes.\n",start_msg.header.total_len);
 | ||
|             }
 | ||
|             memset(reply, 0, sizeof(reply));
 | ||
|             memcpy(reply, "ready", strlen("ready"));
 | ||
|             KPrintf("receive start signal,send [ready] signal to server\n");
 | ||
|             while(AdapterDeviceSend(adapter, reply, strlen(reply)) < 0);
 | ||
|             break;
 | ||
|         }
 | ||
|         else
 | ||
|         {
 | ||
|             memset(reply, 0, sizeof(reply));
 | ||
|             memcpy(reply, "notready", strlen("notready"));
 | ||
|             KPrintf("not receive start signal,send [notready] signal to server\n");
 | ||
|             while(AdapterDeviceSend(adapter, reply, strlen(reply)) < 0);
 | ||
|             continue;
 | ||
|         }
 | ||
|     }
 | ||
| }
 | ||
| 
 | ||
| 
 | ||
| /*******************************************************************************
 | ||
| * 函 数 名: ota_data_recv
 | ||
| * 功能描述: 通过4G方式从服务端接收数据
 | ||
| * 形    参: adapter:Adapter指针,指向注册的4G设备
 | ||
| * 返 回 值: 0:传输成功,-1:传输失败
 | ||
| *******************************************************************************/
 | ||
| static int ota_data_recv(struct Adapter* adapter)
 | ||
| {
 | ||
|     ota_info_t ota_info;
 | ||
|     uint8_t reply[16] = {0};
 | ||
|     int ret = 0, frame_cnt = 0, try_times = 5;
 | ||
|     uint32_t file_size = 0;  
 | ||
|     uint32_t flashdestination = DOWN_FLAH_ADDRESS;
 | ||
| 
 | ||
|     memset(&ota_info, 0, sizeof(ota_info_t));
 | ||
|     mcuboot.op_flash_read(FLAG_FLAH_ADDRESS, (void*)&ota_info, sizeof(ota_info_t));
 | ||
|     ota_info.status = OTA_STATUS_DOWNLOADING;
 | ||
|     UpdateOTAFlag(&ota_info);
 | ||
|     
 | ||
|     while(1)
 | ||
|     {
 | ||
|         memset(&recv_msg, 0, sizeof(ota_data));
 | ||
|         if(AdapterDeviceRecv(adapter, &recv_msg, sizeof(recv_msg)) >= 0) 
 | ||
|         {
 | ||
|             if(recv_msg.header.frame_flag == STARTFLAG) //这里不应该再出现开始帧,丢弃当前数据继续接收
 | ||
|             {
 | ||
|                 continue;
 | ||
|             }  
 | ||
|             else if(recv_msg.header.frame_flag == DATAFLAG)  //说明当前是bin包里数据封装成的数据帧
 | ||
|             {
 | ||
|                 frame_cnt = recv_msg.frame.frame_id;
 | ||
|                 if(recv_msg.frame.crc == calculate_crc16(recv_msg.frame.frame_data,recv_msg.frame.frame_len))
 | ||
|                 {
 | ||
|                     KPrintf("current frame[%d],length %d bytes.\n",frame_cnt,recv_msg.frame.frame_len);
 | ||
|                     if(mcuboot.op_flash_write(flashdestination, recv_msg.frame.frame_data, recv_msg.frame.frame_len) != kStatus_Success)
 | ||
|                     {
 | ||
|                         KPrintf("current frame[%d] flash failed.\n",frame_cnt);
 | ||
|                         ret = -1;
 | ||
|                         break;
 | ||
|                     }
 | ||
|                     else
 | ||
|                     {
 | ||
|                         KPrintf("current frame[%d] is written to flash 0x%x address successful.\n", frame_cnt, flashdestination);
 | ||
|                         flashdestination += recv_msg.frame.frame_len;
 | ||
|                     } 
 | ||
|                 }  
 | ||
|                 else 
 | ||
|                 {
 | ||
|                     KPrintf("current frame[%d] crc check failed,try again!\n",frame_cnt);
 | ||
|                     goto try_again;
 | ||
|                 }
 | ||
|             }
 | ||
|             else if(recv_msg.header.frame_flag == ENDTFLAG)  //说明当前是结束帧
 | ||
|             {
 | ||
|                 KPrintf("total %d frames %d bytes crc[0x%x],receive successful.\n",frame_cnt,recv_msg.header.total_len,recv_msg.frame.crc);
 | ||
|                 memset(reply, 0, sizeof(reply));
 | ||
|                 memcpy(reply, "ok", strlen("ok"));
 | ||
|                 AdapterDeviceSend(adapter, reply, strlen(reply));
 | ||
|     
 | ||
|                 ota_info.status = OTA_STATUS_DOWNLOADED;
 | ||
|                 UpdateOTAFlag(&ota_info);
 | ||
| 
 | ||
|                 file_size = recv_msg.header.total_len;
 | ||
|                 ret = 0;
 | ||
|                 break;
 | ||
|             }
 | ||
|             else //说明当前接收的数据帧不是上述三种数据帧的任意一种
 | ||
|             {
 | ||
|                 goto try_again;
 | ||
|             }
 | ||
|             
 | ||
| send_ok_again:
 | ||
|             memset(reply, 0, sizeof(reply));
 | ||
|             memcpy(reply, "ok", strlen("ok"));
 | ||
| 
 | ||
|             ret = AdapterDeviceSend(adapter, reply, strlen(reply));
 | ||
|             if(ret < 0)
 | ||
|             {
 | ||
|                 KPrintf("send ok failed.\n");
 | ||
|                 goto send_ok_again;
 | ||
|             }
 | ||
|             KPrintf("send reply[%s] done.\n",reply);
 | ||
|             //send ok后把try_times重置为5
 | ||
|             try_times = 5;
 | ||
|             continue;
 | ||
|         }
 | ||
| 
 | ||
|         //没有接收到数据或者接收到的数据帧不满足条件,需要发个retry的命令告诉服务器需要重传
 | ||
|         else 
 | ||
|         {
 | ||
| try_again:
 | ||
|             if(try_times == 0)
 | ||
|             {
 | ||
|                 KPrintf("current frame[%d] try 5 times failed,break out!\n",frame_cnt);
 | ||
|                 ret = -1;
 | ||
|                 break;
 | ||
|             }
 | ||
|             memset(reply, 0, sizeof(reply));
 | ||
|             memcpy(reply, "retry", strlen("retry"));
 | ||
|             KPrintf("current frame[%d] receive failed. retry\n",frame_cnt);
 | ||
|             AdapterDeviceSend(adapter, reply, strlen(reply));
 | ||
|             try_times--;
 | ||
|             continue;
 | ||
|         }
 | ||
|     }
 | ||
| 
 | ||
|     //download分区固件下载成功,更新Flag分区
 | ||
|     if(0 == ret) 
 | ||
|     {
 | ||
|         ota_info.down.size = file_size;
 | ||
|         ota_info.down.crc32= calculate_crc32(DOWN_FLAH_ADDRESS, file_size);
 | ||
| 
 | ||
|         memset(ota_info.down.version,0,sizeof(ota_info.down.version)); 
 | ||
|         create_version(ota_info.os.version, ota_info.down.version);
 | ||
| 
 | ||
|         memset(ota_info.down.description,0,sizeof(ota_info.down.description)); 
 | ||
|         strncpy(ota_info.down.description, "4G OTA bin.",sizeof(ota_info.down.description));
 | ||
| 
 | ||
|         ota_info.status = OTA_STATUS_READY;
 | ||
|     
 | ||
|         memset(ota_info.error_message,0,sizeof(ota_info.error_message)); 
 | ||
|         strncpy(ota_info.error_message, "No error message!",sizeof(ota_info.error_message));
 | ||
| 
 | ||
|         UpdateOTAFlag(&ota_info);
 | ||
|     } 
 | ||
|     else 
 | ||
|     {
 | ||
|         ota_info.status = OTA_STATUS_ERROR;
 | ||
| 
 | ||
|         memset(ota_info.error_message,0,sizeof(ota_info.error_message));
 | ||
|         strncpy(ota_info.error_message, "Failed to download firmware to download partition!",sizeof(ota_info.error_message));
 | ||
| 
 | ||
|         UpdateOTAFlag(&ota_info);
 | ||
|     }
 | ||
|     return ret;
 | ||
| }
 | ||
| 
 | ||
| 
 | ||
| /*******************************************************************************
 | ||
| * 函 数 名: app_ota_by_4g
 | ||
| * 功能描述: 通过命令来进行ota升级,该函数与升级的命令关联,通过4g方式传输bin文件
 | ||
| * 形    参: 无
 | ||
| * 返 回 值: 无
 | ||
| *******************************************************************************/
 | ||
| static void app_ota_by_4g(void)
 | ||
| {
 | ||
|     uint32_t baud_rate = BAUD_RATE_115200;
 | ||
|     uint8_t server_addr[16] = "115.238.53.60";
 | ||
|     uint8_t server_port[8] = "7777";
 | ||
| 
 | ||
|     mcuboot.flash_init();
 | ||
| 
 | ||
|     struct Adapter* adapter =  AdapterDeviceFindByName(ADAPTER_4G_NAME);
 | ||
|     adapter->socket.socket_id = 0;
 | ||
| 
 | ||
|     AdapterDeviceOpen(adapter);
 | ||
|     AdapterDeviceControl(adapter, OPE_INT, &baud_rate);
 | ||
|     AdapterDeviceConnect(adapter, CLIENT, server_addr, server_port, IPV4);
 | ||
|     MdelayKTask(100);
 | ||
|     while(1)
 | ||
|     {
 | ||
|         /* step1:Confirm the start signal of transmission. */
 | ||
|         get_start_signal(adapter);
 | ||
|         KPrintf("start receive ota bin file.\n");
 | ||
|         /* step2:start receive bin file,first wait for 4s. */
 | ||
|         MdelayKTask(4000);
 | ||
|         if(0 == ota_data_recv(adapter))
 | ||
|         {
 | ||
|             break;
 | ||
|         } 
 | ||
|     }
 | ||
|     mcuboot.flash_deinit();
 | ||
|     KPrintf("firmware file transfer successful,start reboot!\n");
 | ||
|     MdelayKTask(2000);
 | ||
|     mcuboot.op_reset();
 | ||
| }
 | ||
| 
 | ||
| SHELL_EXPORT_CMD(SHELL_CMD_PERMISSION(0)|SHELL_CMD_TYPE(SHELL_TYPE_CMD_FUNC)|SHELL_CMD_PARAM_NUM(0),ota, app_ota_by_4g, ota by 4g function);
 | ||
| #endif
 | ||
| 
 | ||
| #ifdef OTA_BY_PLATFORM
 | ||
| 
 | ||
| #if (OTA_FRAME_SIZE <= MQTT_FRAME_SIZE)
 | ||
|     #define FRAME_LEN  OTA_FRAME_SIZE
 | ||
| #else
 | ||
|     #error "The value of FRAME_LEN should not be greater than MQTT_FRAME_SIZE!!"
 | ||
| #endif
 | ||
| static uint8_t MqttRxbuf[FRAME_LEN + 1024];
 | ||
| static uint8_t FrameBuf[FRAME_LEN];
 | ||
| static OTA_TCB platform_ota;
 | ||
| 
 | ||
| 
 | ||
| /*******************************************************************************
 | ||
| * 函 数 名: displayProgress
 | ||
| * 功能描述: 显示当前ota下载进度的百分比
 | ||
| * 形    参: progress:当前下载的进度,0-100之间
 | ||
| * 返 回 值: 无
 | ||
| *******************************************************************************/
 | ||
| void displayProgress(uint8_t progress) 
 | ||
| {
 | ||
|     uint8_t buf[101];
 | ||
|     if((progress >= 0) && (progress <= 100))
 | ||
|     {
 | ||
|         memset(buf, '=', progress);
 | ||
|         buf[progress] = '\0';
 | ||
|         KPrintf("[%-100s][%d%%]\r",buf,progress);
 | ||
|         if(progress == 100)
 | ||
|             KPrintf("\r\n---------------------------------------------\r\n");
 | ||
|     }
 | ||
| }
 | ||
| 
 | ||
| 
 | ||
| #ifdef XIUOS_PLATFORM
 | ||
| /*******************************************************************************
 | ||
| * 函 数 名: PropertyVersion
 | ||
| * 功能描述: 向服务器上传当前设备版本信息
 | ||
| * 形    参: 无
 | ||
| * 返 回 值: 无
 | ||
| *******************************************************************************/
 | ||
| static void PropertyVersion(void)
 | ||
| {
 | ||
|     uint8_t tempdatabuff[128];
 | ||
|     ota_info_t ota_info;
 | ||
| 
 | ||
|     memset(&ota_info, 0, sizeof(ota_info_t));
 | ||
|     mcuboot.op_flash_read(FLAG_FLAH_ADDRESS, (void*)&ota_info, sizeof(ota_info_t)); 
 | ||
|     
 | ||
|     memset(tempdatabuff,0,128);
 | ||
|     sprintf(tempdatabuff,"{\"clientId\":\"%s\",\"version\":\"%s\"}",CLIENTID,ota_info.os.version);
 | ||
|     KPrintf("------current version is:%s------\r\n",ota_info.os.version);
 | ||
| 
 | ||
|     MQTT_PublishDataQs1("xiuosiot/ota/version",tempdatabuff,strlen(tempdatabuff));  //发送等级QS=1的PUBLISH报文
 | ||
| }
 | ||
| 
 | ||
| 
 | ||
| /*******************************************************************************
 | ||
| * 函 数 名: OTA_Download
 | ||
| * 功能描述: OTA下载数据
 | ||
| * 形    参: size:本次下载量,offset:本次下载偏移量
 | ||
| * 返 回 值: 无
 | ||
| *******************************************************************************/
 | ||
| static void OTA_Download(int size, int offset)
 | ||
| {
 | ||
|     uint8_t tempdatabuff[128];
 | ||
| 
 | ||
|     memset(tempdatabuff,0,128);
 | ||
|     sprintf(tempdatabuff,"{\"clientId\":\"%s\",\"fileId\":%d,\"fileOffset\":%d,\"size\":%d}",Platform_mqtt.ClientID,platform_ota.streamId,offset,size);
 | ||
| 
 | ||
|     MQTT_PublishDataQs0("xiuosiot/ota/files", tempdatabuff, strlen(tempdatabuff));
 | ||
| }
 | ||
| 
 | ||
| /*******************************************************************************
 | ||
| * 函 数 名: mqttCloudInteraction
 | ||
| * 功能描述: 设备通过MQTT协议与云平台交互
 | ||
| * 形    参: 无
 | ||
| * 返 回 值: 无
 | ||
| *******************************************************************************/
 | ||
| static void mqttCloudInteraction(void* parameter)
 | ||
| {
 | ||
|     int datalen;
 | ||
|     int ret = 0;
 | ||
|     int freecnt = 0;
 | ||
|     ota_info_t ota_info;
 | ||
|     uint32_t heart_time = 0;
 | ||
|     uint32_t flashdestination = DOWN_FLAH_ADDRESS;
 | ||
|     uint8_t topicdatabuff[2][64];
 | ||
|     char *ptr1, *ptr2;
 | ||
|     uint16_t payloadLen;
 | ||
| 
 | ||
|     mcuboot.flash_init();
 | ||
|     memset(&ota_info, 0, sizeof(ota_info_t));
 | ||
|     mcuboot.op_flash_read(FLAG_FLAH_ADDRESS, (void*)&ota_info, sizeof(ota_info_t));
 | ||
|     ota_info.status = OTA_STATUS_DOWNLOADING;
 | ||
|     UpdateOTAFlag(&ota_info);
 | ||
|     memset(topicdatabuff,0,sizeof(topicdatabuff)); 
 | ||
|     sprintf(topicdatabuff[0],"ota/%s/update",CLIENTID); 
 | ||
|     sprintf(topicdatabuff[1],"ota/%s/files",CLIENTID);   
 | ||
| 
 | ||
| reconnect:
 | ||
|     if((AdapterNetActive() == 0) && MQTT_Connect() && MQTT_SubscribeTopic(topicdatabuff[0]) && MQTT_SubscribeTopic(topicdatabuff[1]))
 | ||
|     {
 | ||
|         KPrintf("Log in to the cloud platform and subscribe to the topic successfully.\n"); 
 | ||
|         PropertyVersion();
 | ||
|     }
 | ||
|     else
 | ||
|     {
 | ||
|         KPrintf("Log in to the cloud platform failed, retry!\n");
 | ||
|         goto reconnect;  
 | ||
|     }
 | ||
| 
 | ||
| #ifdef USING_DOWNLOAD_JSON
 | ||
|     uint8_t jsontopicdatabuff[64];
 | ||
|     uint8_t jsonfilename[32];
 | ||
|     memset(jsontopicdatabuff,0,sizeof(jsontopicdatabuff));
 | ||
|     sprintf(jsontopicdatabuff,"protocol/%s/files",CLIENTID);
 | ||
|     if(!MQTT_SubscribeTopic(jsontopicdatabuff))
 | ||
|     {
 | ||
|         KPrintf("subscribe to the json download topic failed!\n"); 
 | ||
|     }
 | ||
| #endif
 | ||
| 
 | ||
|     while(1)
 | ||
|     {
 | ||
|         memset(MqttRxbuf,0,sizeof(MqttRxbuf));
 | ||
|         datalen = MQTT_Recv(MqttRxbuf, sizeof(MqttRxbuf));
 | ||
|         if(datalen <= 0)
 | ||
|         {
 | ||
|             freecnt++; 
 | ||
|             if((freecnt >= 20) && (CalculateTimeMsFromTick(CurrentTicksGain()) - heart_time >= HEART_TIME)) //连续20次未收到数据默认为为空闲状态,需每隔一段时间发送需要发送心跳包保活
 | ||
|             {
 | ||
|                 heart_time = CalculateTimeMsFromTick(CurrentTicksGain());
 | ||
|                 freecnt = 0;
 | ||
|                 if(!MQTT_SendHeart()) //发送心跳包失败可能连接断开,需要重连
 | ||
|                 {
 | ||
|                     KPrintf("The connection has been disconnected, reconnecting!\n");
 | ||
|                     goto reconnect;  
 | ||
|                 }
 | ||
|                 KPrintf("Send heartbeat packet successful!\n"); 
 | ||
|             }
 | ||
|         }
 | ||
|         else if(MqttRxbuf[0] == 0x30)
 | ||
|         {
 | ||
|             freecnt = 0;
 | ||
|             payloadLen = MQTT_DealPublishData(MqttRxbuf, datalen);
 | ||
| 
 | ||
|             // 1.获取新版本固件大小及版本信息
 | ||
|             ptr1 = strstr((char *)Platform_mqtt.cmdbuff,topicdatabuff[0]); 
 | ||
|             ptr2 = strstr((char *)Platform_mqtt.cmdbuff,"{\"fileSize\":");
 | ||
|             if((ptr1 != NULL) &&(ptr2 != NULL))
 | ||
|             {
 | ||
|                 if(sscanf(ptr2,"{\"fileSize\":%d,\"version\":\"%11s\",\"fileId\":%d,\"md5\"",&platform_ota.size,platform_ota.version,&platform_ota.streamId)==3)
 | ||
|                 {
 | ||
|                     KPrintf("OTA firmware information:file size is %d,file id is %d,file version is %s!\r\n",platform_ota.size,platform_ota.streamId,platform_ota.version);
 | ||
|                     KPrintf("------Start the firmware file transfer!------\r\n");
 | ||
|                     KPrintf("---------------------------------------------\r\n");
 | ||
|                     if(platform_ota.size > APP_FLASH_SIZE)
 | ||
|                     {
 | ||
|                         KPrintf("File size is larger than partition size,the partition size is %dk.\n",APP_FLASH_SIZE/1024);
 | ||
|                         ret = -1;
 | ||
|                         ota_info.status = OTA_STATUS_ERROR;
 | ||
|                         memset(ota_info.error_message,0,sizeof(ota_info.error_message));
 | ||
|                         strncpy(ota_info.error_message, "File size is larger than partition size!",sizeof(ota_info.error_message));
 | ||
|                         UpdateOTAFlag(&ota_info);
 | ||
|                         break;
 | ||
|                     }
 | ||
|                     if(mcuboot.op_flash_erase(DOWN_FLAH_ADDRESS,platform_ota.size) != kStatus_Success)
 | ||
|                     {
 | ||
|                         KPrintf("Failed to erase download partition!\n");
 | ||
|                         ret = -1;
 | ||
|                         ota_info.status = OTA_STATUS_ERROR;
 | ||
|                         memset(ota_info.error_message,0,sizeof(ota_info.error_message));
 | ||
|                         strncpy(ota_info.error_message, "Failed to erase download partition!",sizeof(ota_info.error_message));
 | ||
|                         UpdateOTAFlag(&ota_info);
 | ||
|                         break;
 | ||
|                     }
 | ||
|                     platform_ota.counter = (platform_ota.size%FRAME_LEN != 0)? (platform_ota.size/FRAME_LEN + 1):(platform_ota.size/FRAME_LEN);
 | ||
|                     platform_ota.num = 1;                                          //下载次数,初始值为1
 | ||
|                     platform_ota.downlen = FRAME_LEN;                              //记录本次下载量
 | ||
|                     OTA_Download(platform_ota.downlen,(platform_ota.num - 1)*FRAME_LEN); //发送要下载的数据信息给服务器
 | ||
|                 }
 | ||
|                 else
 | ||
|                 {
 | ||
|                     KPrintf("Failed to get ota information!\n");
 | ||
|                     ret = -1;
 | ||
|                     ota_info.status = OTA_STATUS_ERROR;
 | ||
|                     memset(ota_info.error_message,0,sizeof(ota_info.error_message));
 | ||
|                     strncpy(ota_info.error_message, "Failed to get ota information!",sizeof(ota_info.error_message));
 | ||
|                     UpdateOTAFlag(&ota_info);
 | ||
|                     break;
 | ||
|                 }
 | ||
|             }
 | ||
| 
 | ||
|             // 2.分片接收新版本固件
 | ||
|             if(strstr((char *)Platform_mqtt.cmdbuff,topicdatabuff[1]))
 | ||
|             {
 | ||
|                 memset(FrameBuf,0,sizeof(FrameBuf));
 | ||
|                 memcpy(FrameBuf, &MqttRxbuf[datalen-platform_ota.downlen], platform_ota.downlen);
 | ||
|                 if(mcuboot.op_flash_write(flashdestination,FrameBuf,platform_ota.downlen) != kStatus_Success)
 | ||
|                 {
 | ||
|                     KPrintf("current frame[%d] flash failed.\n",platform_ota.num-1);
 | ||
|                     ret = -1;
 | ||
|                     ota_info.status = OTA_STATUS_ERROR;
 | ||
|                     memset(ota_info.error_message,0,sizeof(ota_info.error_message));
 | ||
|                     sprintf(ota_info.error_message,"current frame[%d] flash failed.",platform_ota.num-1);  
 | ||
|                     UpdateOTAFlag(&ota_info);
 | ||
|                     break;
 | ||
|                 }
 | ||
|                 else
 | ||
|                 {
 | ||
|                     displayProgress((platform_ota.num*100)/platform_ota.counter);
 | ||
|                     flashdestination += platform_ota.downlen;
 | ||
|                     platform_ota.num++; 
 | ||
|                 }
 | ||
| 
 | ||
|                 if(platform_ota.num < platform_ota.counter) //如果小于总下载次数
 | ||
|                 {
 | ||
|                     platform_ota.downlen = FRAME_LEN;                                      //记录本次下载量
 | ||
|                     OTA_Download(platform_ota.downlen,(platform_ota.num - 1)*FRAME_LEN);   //发送要下载的数据信息给服务器
 | ||
|                 }
 | ||
|                 else if(platform_ota.num == platform_ota.counter) //如果等于总下载次数,说明是最后一次下载
 | ||
|                 {
 | ||
|                     if(platform_ota.size%FRAME_LEN == 0)    //判断固件大小是否是FRAME_LEN的整数倍
 | ||
|                     {
 | ||
|                         platform_ota.downlen = FRAME_LEN;                                    //记录本次下载量
 | ||
|                         OTA_Download(platform_ota.downlen,(platform_ota.num - 1)*FRAME_LEN); //发送要下载的数据信息给服务器
 | ||
|                     }
 | ||
|                     else
 | ||
|                     {
 | ||
|                         platform_ota.downlen = platform_ota.size%FRAME_LEN;                   //记录本次下载量
 | ||
|                         OTA_Download(platform_ota.downlen,(platform_ota.num - 1)*FRAME_LEN);  //发送要下载的数据信息给服务器
 | ||
|                     }
 | ||
|                 }
 | ||
| 
 | ||
|                 else //下载完毕
 | ||
|                 {
 | ||
|                     ret = 0;
 | ||
|                     break;
 | ||
|                 }
 | ||
|             }
 | ||
| #ifdef USING_DOWNLOAD_JSON
 | ||
|             // 3.下载json文件,SD卡要确保已经插入并mount成功
 | ||
|             extern bool GetSdMountStatus(void);
 | ||
|             if(strstr((char *)Platform_mqtt.cmdbuff,jsontopicdatabuff) && GetSdMountStatus())
 | ||
|             {
 | ||
|                 KPrintf("------Start download joson file !------\r\n");
 | ||
|                 memset(jsonfilename,0,sizeof(jsonfilename));
 | ||
|                 memset(FrameBuf,0,sizeof(FrameBuf));
 | ||
|                 memcpy(FrameBuf, &Platform_mqtt.cmdbuff[strlen(jsontopicdatabuff)],payloadLen-strlen(jsontopicdatabuff)); 
 | ||
|                 
 | ||
|                 cJSON *json_obj = cJSON_Parse(FrameBuf);
 | ||
|                 char* product_name = cJSON_GetObjectItem(json_obj, "productName")->valuestring;
 | ||
|                 sprintf(jsonfilename,"%s",product_name); 
 | ||
|                 strcat(jsonfilename,".json");
 | ||
| 
 | ||
|                 FILE *fp = fopen(jsonfilename, "w");
 | ||
|                 if(fp == NULL)
 | ||
|                 {
 | ||
|                    KPrintf("%s file create failed,please check!\r\n",jsonfilename);
 | ||
|                    cJSON_Delete(json_obj); 
 | ||
|                    continue;
 | ||
|                 }
 | ||
|                 else
 | ||
|                 {
 | ||
|                     KPrintf("%s file create success!\r\n",jsonfilename); 
 | ||
|                     char *json_print_str = cJSON_Print(json_obj);
 | ||
|                     fprintf(fp, "%s", json_print_str);
 | ||
|                     fclose(fp);
 | ||
|                     cJSON_free(json_print_str);
 | ||
|                     cJSON_Delete(json_obj);
 | ||
|                 }
 | ||
|                 KPrintf("------download %s file done!------\r\n",jsonfilename);
 | ||
|             }
 | ||
| #endif
 | ||
|         }
 | ||
|         else
 | ||
|         {
 | ||
|             freecnt = 0;
 | ||
|             continue;
 | ||
|         }
 | ||
|     }
 | ||
| 
 | ||
|     // 新版本固件接收完毕,写入描述信息
 | ||
|     if(0 == ret)
 | ||
|     {
 | ||
|         ota_info.down.size = platform_ota.size;
 | ||
|         ota_info.down.crc32 = calculate_crc32(DOWN_FLAH_ADDRESS, platform_ota.size);
 | ||
| 
 | ||
|         memset(ota_info.down.version,0,sizeof(ota_info.down.version)); 
 | ||
|         strncpy(ota_info.down.version, platform_ota.version, sizeof(ota_info.down.version));
 | ||
| 
 | ||
|         memset(ota_info.down.description,0,sizeof(ota_info.down.description)); 
 | ||
|         strncpy(ota_info.down.description, "MQTT OTA bin.",sizeof(ota_info.down.description));
 | ||
| 
 | ||
|         ota_info.status = OTA_STATUS_READY;
 | ||
|     
 | ||
|         memset(ota_info.error_message,0,sizeof(ota_info.error_message)); 
 | ||
|         strncpy(ota_info.error_message, "No error message!",sizeof(ota_info.error_message));
 | ||
| 
 | ||
|         UpdateOTAFlag(&ota_info);
 | ||
|         KPrintf("firmware file transfer successful,start reboot!\n");
 | ||
|     }
 | ||
|     else
 | ||
|     {
 | ||
|         KPrintf("firmware file transfer failed,start reboot!\n");
 | ||
|     }
 | ||
|     MQTT_Disconnect();
 | ||
|     mcuboot.flash_deinit();
 | ||
|     MdelayKTask(2000);
 | ||
|     mcuboot.op_reset();
 | ||
| }
 | ||
| #endif
 | ||
| 
 | ||
| #ifdef ALIBABA_PLATFORM
 | ||
| /*******************************************************************************
 | ||
| * 函 数 名: PropertyVersion
 | ||
| * 功能描述: 向服务器上传当前设备版本信息
 | ||
| * 形    参: 无
 | ||
| * 返 回 值: 无
 | ||
| *******************************************************************************/
 | ||
| static void PropertyVersion(void)
 | ||
| {
 | ||
|     uint8_t topicdatabuff[64];
 | ||
|     uint8_t tempdatabuff[128];
 | ||
|     ota_info_t ota_info;
 | ||
| 
 | ||
|     memset(topicdatabuff,0,64);
 | ||
|     sprintf(topicdatabuff,"/ota/device/inform/%s/%s", PLATFORM_PRODUCTKEY, CLIENT_DEVICENAME);
 | ||
| 
 | ||
|     memset(&ota_info, 0, sizeof(ota_info_t));
 | ||
|     mcuboot.op_flash_read(FLAG_FLAH_ADDRESS, (void*)&ota_info, sizeof(ota_info_t)); 
 | ||
|     
 | ||
|     memset(tempdatabuff,0,128);
 | ||
|     sprintf(tempdatabuff,"{\"id\": \"1\",\"params\": {\"version\": \"%s\"}}",ota_info.os.version);
 | ||
|     KPrintf("------current version is:%s------\r\n",ota_info.os.version);
 | ||
| 
 | ||
|     MQTT_PublishDataQs1(topicdatabuff,tempdatabuff,strlen(tempdatabuff));  //发送等级QS=1的PUBLISH报文
 | ||
| }
 | ||
| 
 | ||
| 
 | ||
| /*******************************************************************************
 | ||
| * 函 数 名: OTA_Download
 | ||
| * 功能描述: OTA下载数据
 | ||
| * 形    参: size:本次下载量,offset:本次下载偏移量
 | ||
| * 返 回 值: 无
 | ||
| *******************************************************************************/
 | ||
| static void OTA_Download(int size, int offset)
 | ||
| {
 | ||
|     uint8_t topicdatabuff[64];
 | ||
|     uint8_t tempdatabuff[128];
 | ||
| 
 | ||
|     memset(topicdatabuff,0,64);  
 | ||
|     sprintf(topicdatabuff,"/sys/%s/%s/thing/file/download",PLATFORM_PRODUCTKEY,CLIENT_DEVICENAME);
 | ||
| 
 | ||
|     memset(tempdatabuff,0,128);
 | ||
|     sprintf(tempdatabuff,"{\"id\": \"1\",\"params\": {\"fileInfo\":{\"streamId\":%d,\"fileId\":1},\"fileBlock\":{\"size\":%d,\"offset\":%d}}}",platform_ota.streamId,size,offset);
 | ||
| 
 | ||
|     MQTT_PublishDataQs0(topicdatabuff, tempdatabuff, strlen(tempdatabuff));
 | ||
| }
 | ||
| 
 | ||
| /*******************************************************************************
 | ||
| * 函 数 名: mqttCloudInteraction
 | ||
| * 功能描述: 设备通过MQTT协议与云平台交互
 | ||
| * 形    参: 无
 | ||
| * 返 回 值: 无
 | ||
| *******************************************************************************/
 | ||
| static void mqttCloudInteraction(void* parameter)
 | ||
| {
 | ||
|     int datalen;
 | ||
|     int ret = 0;
 | ||
|     int freecnt = 0;
 | ||
|     ota_info_t ota_info;
 | ||
|     uint32_t heart_time = 0;
 | ||
|     uint32_t flashdestination = DOWN_FLAH_ADDRESS;
 | ||
|     uint8_t topicdatabuff[2][64];
 | ||
|     char *ptr1, *ptr2;
 | ||
| 
 | ||
|     mcuboot.flash_init();
 | ||
|     memset(&ota_info, 0, sizeof(ota_info_t));
 | ||
|     mcuboot.op_flash_read(FLAG_FLAH_ADDRESS, (void*)&ota_info, sizeof(ota_info_t));
 | ||
|     ota_info.status = OTA_STATUS_DOWNLOADING;
 | ||
|     UpdateOTAFlag(&ota_info);
 | ||
|     memset(topicdatabuff,0,sizeof(topicdatabuff));
 | ||
|     sprintf(topicdatabuff[0],"/ota/device/upgrade/%s/%s",PLATFORM_PRODUCTKEY,CLIENT_DEVICENAME);  
 | ||
|     sprintf(topicdatabuff[1],"/sys/%s/%s/thing/file/download_reply",PLATFORM_PRODUCTKEY,CLIENT_DEVICENAME);  
 | ||
| 
 | ||
| reconnect:
 | ||
|     if((AdapterNetActive() == 0) && MQTT_Connect() && MQTT_SubscribeTopic(topicdatabuff[0]) && MQTT_SubscribeTopic(topicdatabuff[1]))
 | ||
|     {
 | ||
|         KPrintf("Log in to the cloud platform and subscribe to the topic successfully.\n"); 
 | ||
|         PropertyVersion();
 | ||
|     }
 | ||
|     else
 | ||
|     {
 | ||
|         KPrintf("Log in to the cloud platform failed, retry!\n");
 | ||
|         goto reconnect;  
 | ||
|     }
 | ||
| 
 | ||
|     while(1)
 | ||
|     {
 | ||
|         memset(MqttRxbuf,0,sizeof(MqttRxbuf));
 | ||
|         datalen = MQTT_Recv(MqttRxbuf, sizeof(MqttRxbuf));
 | ||
|         if(datalen <= 0)
 | ||
|         {
 | ||
|             freecnt++;
 | ||
|             if((freecnt >= 20) && (CalculateTimeMsFromTick(CurrentTicksGain()) - heart_time >= HEART_TIME)) //连续20次未收到数据默认为为空闲状态,需每隔一段时间发送需要发送心跳包保活
 | ||
|             {
 | ||
|                 heart_time = CalculateTimeMsFromTick(CurrentTicksGain());
 | ||
|                 freecnt = 0;
 | ||
|                 if(!MQTT_SendHeart()) //发送心跳包失败可能连接断开,需要重连
 | ||
|                 {
 | ||
|                     KPrintf("The connection has been disconnected, reconnecting!\n");
 | ||
|                     goto reconnect;  
 | ||
|                 }
 | ||
|                 KPrintf("Send heartbeat packet successful!\n"); 
 | ||
|             }
 | ||
|         }
 | ||
|         else if(MqttRxbuf[0] == 0x30)
 | ||
|         {
 | ||
|             freecnt = 0;
 | ||
|             MQTT_DealPublishData(MqttRxbuf, datalen);
 | ||
|             
 | ||
|             // 1.获取新版本固件大小及版本信息
 | ||
|             ptr1 = strstr((char *)Platform_mqtt.cmdbuff,topicdatabuff[0]); 
 | ||
|             ptr2 = strstr((char *)Platform_mqtt.cmdbuff,"{\"code\":\"1000\"");
 | ||
|             if((ptr1 != NULL) &&(ptr2 != NULL))
 | ||
|             {
 | ||
|                 if(sscanf(ptr2,"{\"code\":\"1000\",\"data\":{\"size\":%d,\"streamId\":%d,\"sign\":\"%*32s\",\"dProtocol\":\"mqtt\",\"version\":\"%11s\"",&platform_ota.size,&platform_ota.streamId,platform_ota.version)==3)
 | ||
|                 {
 | ||
|                     KPrintf("OTA firmware information:file size is %d,file id is %d,file version is %s!\r\n",platform_ota.size,platform_ota.streamId,platform_ota.version);
 | ||
|                     KPrintf("------Start the firmware file transfer!------\r\n");                    
 | ||
|                     KPrintf("---------------------------------------------\r\n");
 | ||
|                     if(platform_ota.size > APP_FLASH_SIZE)
 | ||
|                     {
 | ||
|                         KPrintf("File size is larger than partition size,the partition size is %dk.\n",APP_FLASH_SIZE/1024);
 | ||
|                         ret = -1;
 | ||
|                         ota_info.status = OTA_STATUS_ERROR;
 | ||
|                         memset(ota_info.error_message,0,sizeof(ota_info.error_message));
 | ||
|                         strncpy(ota_info.error_message, "File size is larger than partition size!",sizeof(ota_info.error_message));
 | ||
|                         UpdateOTAFlag(&ota_info);
 | ||
|                         break;
 | ||
|                     }
 | ||
|                     if(mcuboot.op_flash_erase(DOWN_FLAH_ADDRESS,platform_ota.size) != kStatus_Success)
 | ||
|                     {
 | ||
|                         KPrintf("Failed to erase download partition!\n");
 | ||
|                         ret = -1;
 | ||
|                         ota_info.status = OTA_STATUS_ERROR;
 | ||
|                         memset(ota_info.error_message,0,sizeof(ota_info.error_message));
 | ||
|                         strncpy(ota_info.error_message, "Failed to erase download partition!",sizeof(ota_info.error_message));
 | ||
|                         UpdateOTAFlag(&ota_info);
 | ||
|                         break;
 | ||
|                     }
 | ||
|                     platform_ota.counter = (platform_ota.size%FRAME_LEN != 0)? (platform_ota.size/FRAME_LEN + 1):(platform_ota.size/FRAME_LEN);
 | ||
|                     platform_ota.num = 1;                                          //下载次数,初始值为1
 | ||
|                     platform_ota.downlen = FRAME_LEN;                              //记录本次下载量
 | ||
|                     OTA_Download(platform_ota.downlen,(platform_ota.num - 1)*FRAME_LEN); //发送要下载的数据信息给服务器
 | ||
|                 }
 | ||
|                 else
 | ||
|                 {
 | ||
|                     KPrintf("Failed to get ota information!\n");
 | ||
|                     ret = -1;
 | ||
|                     ota_info.status = OTA_STATUS_ERROR;
 | ||
|                     memset(ota_info.error_message,0,sizeof(ota_info.error_message));
 | ||
|                     strncpy(ota_info.error_message, "Failed to get ota information!",sizeof(ota_info.error_message));
 | ||
|                     UpdateOTAFlag(&ota_info);
 | ||
|                     break;
 | ||
|                 }
 | ||
|             }
 | ||
| 
 | ||
|             // 2.分片接收新版本固件
 | ||
|             if(strstr((char *)Platform_mqtt.cmdbuff,topicdatabuff[1]))
 | ||
|             {
 | ||
|                 memset(FrameBuf,0,sizeof(FrameBuf));
 | ||
|                 memcpy(FrameBuf, &MqttRxbuf[datalen-platform_ota.downlen-2], platform_ota.downlen);
 | ||
|                 if(mcuboot.op_flash_write(flashdestination,FrameBuf,platform_ota.downlen) != kStatus_Success)
 | ||
|                 {
 | ||
|                     KPrintf("current frame[%d] flash failed.\n",platform_ota.num-1);
 | ||
|                     ret = -1;
 | ||
|                     ota_info.status = OTA_STATUS_ERROR;
 | ||
|                     memset(ota_info.error_message,0,sizeof(ota_info.error_message));
 | ||
|                     sprintf(ota_info.error_message,"current frame[%d] flash failed.",platform_ota.num-1);  
 | ||
|                     UpdateOTAFlag(&ota_info);
 | ||
|                     break;
 | ||
|                 }
 | ||
|                 else
 | ||
|                 {
 | ||
|                     displayProgress((platform_ota.num*100)/platform_ota.counter);
 | ||
|                     flashdestination += platform_ota.downlen;
 | ||
|                     platform_ota.num++; 
 | ||
|                 }
 | ||
| 
 | ||
|                 if(platform_ota.num < platform_ota.counter) //如果小于总下载次数
 | ||
|                 {
 | ||
|                     platform_ota.downlen = FRAME_LEN;                                      //记录本次下载量
 | ||
|                     OTA_Download(platform_ota.downlen,(platform_ota.num - 1)*FRAME_LEN);   //发送要下载的数据信息给服务器
 | ||
|                 }
 | ||
|                 else if(platform_ota.num == platform_ota.counter) //如果等于总下载次数,说明是最后一次下载
 | ||
|                 {
 | ||
|                     if(platform_ota.size%FRAME_LEN == 0)    //判断固件大小是否是FRAME_LEN的整数倍
 | ||
|                     {
 | ||
|                         platform_ota.downlen = FRAME_LEN;                                    //记录本次下载量
 | ||
|                         OTA_Download(platform_ota.downlen,(platform_ota.num - 1)*FRAME_LEN); //发送要下载的数据信息给服务器
 | ||
|                     }
 | ||
|                     else
 | ||
|                     {
 | ||
|                         platform_ota.downlen = platform_ota.size%FRAME_LEN;                   //记录本次下载量
 | ||
|                         OTA_Download(platform_ota.downlen,(platform_ota.num - 1)*FRAME_LEN);  //发送要下载的数据信息给服务器
 | ||
|                     }
 | ||
|                 }
 | ||
| 
 | ||
|                 else //下载完毕
 | ||
|                 {
 | ||
|                     ret = 0;
 | ||
|                     break;
 | ||
|                 }
 | ||
|             }
 | ||
| 
 | ||
|         }
 | ||
|         else
 | ||
|         {
 | ||
|             freecnt = 0;
 | ||
|             continue;
 | ||
|         }
 | ||
|     }
 | ||
| 
 | ||
|     // 新版本固件接收完毕,写入描述信息
 | ||
|     if(0 == ret)
 | ||
|     {
 | ||
|         ota_info.down.size = platform_ota.size;
 | ||
|         ota_info.down.crc32 = calculate_crc32(DOWN_FLAH_ADDRESS, platform_ota.size);
 | ||
| 
 | ||
|         memset(ota_info.down.version,0,sizeof(ota_info.down.version)); 
 | ||
|         strncpy(ota_info.down.version, platform_ota.version, sizeof(ota_info.down.version));
 | ||
| 
 | ||
|         memset(ota_info.down.description,0,sizeof(ota_info.down.description)); 
 | ||
|         strncpy(ota_info.down.description, "MQTT OTA bin.",sizeof(ota_info.down.description));
 | ||
| 
 | ||
|         ota_info.status = OTA_STATUS_READY;
 | ||
|     
 | ||
|         memset(ota_info.error_message,0,sizeof(ota_info.error_message)); 
 | ||
|         strncpy(ota_info.error_message, "No error message!",sizeof(ota_info.error_message));
 | ||
| 
 | ||
|         UpdateOTAFlag(&ota_info);
 | ||
|         KPrintf("firmware file transfer successful,start reboot!\n");
 | ||
|     }
 | ||
|     else
 | ||
|     {
 | ||
|         KPrintf("firmware file transfer failed,start reboot!\n");
 | ||
|     }
 | ||
|     MQTT_Disconnect();
 | ||
|     mcuboot.flash_deinit();
 | ||
|     MdelayKTask(2000);
 | ||
|     mcuboot.op_reset();
 | ||
| }
 | ||
| #endif
 | ||
| 
 | ||
| int OtaTask(void)
 | ||
| {
 | ||
|     int32 ota_task = 0;
 | ||
|     ota_task = KTaskCreate("mqtt_platform", mqttCloudInteraction, NULL,10240, 10);
 | ||
|     if(ota_task < 0) {
 | ||
|         KPrintf("matt platform task create failed ...%s %d.\n", __FUNCTION__,__LINE__);
 | ||
|         return ERROR;
 | ||
|     }
 | ||
| 
 | ||
|     StartupKTask(ota_task);
 | ||
|     return 0;
 | ||
| }
 | ||
| #endif
 | ||
| 
 | ||
| 
 | ||
| /*******************************************************************************
 | ||
| * 函 数 名: app_clear_jumpflag
 | ||
| * 功能描述: 跳转app成功后,在app中调用将lastjumpflag重置为0XCDCDCDCD
 | ||
| * 形    参: 无
 | ||
| * 返 回 值: 无
 | ||
| *******************************************************************************/
 | ||
| void app_clear_jumpflag(void)
 | ||
| {
 | ||
|     ota_info_t ota_info;
 | ||
|     mcuboot.flash_init();
 | ||
|     //跳转成功设置lastjumpflag为JUMP_SUCCESS_FLAG
 | ||
|     memset(&ota_info, 0, sizeof(ota_info_t));
 | ||
|     mcuboot.op_flash_read(FLAG_FLAH_ADDRESS, (void*)&ota_info, sizeof(ota_info_t));
 | ||
|     ota_info.lastjumpflag = JUMP_SUCCESS_FLAG;
 | ||
|     UpdateOTAFlag(&ota_info);
 | ||
|     mcuboot.flash_deinit();
 | ||
| }
 | ||
| 
 | ||
| 
 | ||
| /*******************************************************************************
 | ||
| * 函 数 名: ota_entry
 | ||
| * 功能描述: bootloader的入口函数
 | ||
| * 形    参: 无
 | ||
| * 返 回 值: 无
 | ||
| *******************************************************************************/
 | ||
| void ota_entry(void)
 | ||
| {
 | ||
|     uint8_t ch1, ch2;
 | ||
|     uint32_t ret;
 | ||
|     uint32_t timeout = 1000;
 | ||
| 
 | ||
|     mcuboot.board_init();
 | ||
| 
 | ||
|     mcuboot.print_string("Please press 'space' key into menu in 5s !!!\r\n");
 | ||
|     
 | ||
|     while(timeout)
 | ||
|     { 
 | ||
|         ret = (SerialKeyPressed((uint8_t*)&ch1));
 | ||
|         if(ret) break;
 | ||
|         timeout--;
 | ||
|         mcuboot.op_delay(5);
 | ||
|     }
 | ||
| 
 | ||
|     while(1)
 | ||
|     {
 | ||
| 
 | ||
|         if((ret)&&(ch1 == 0x20))
 | ||
|         {
 | ||
|             mcuboot.print_string("\r\nPlease slecet:");
 | ||
|             
 | ||
|             mcuboot.print_string("\r\n 1:run app");
 | ||
|             mcuboot.print_string("\r\n 2:update app");
 | ||
|             mcuboot.print_string("\r\n 3:reboot \r\n");
 | ||
| 
 | ||
| 
 | ||
|             ch2 = GetKey();
 | ||
|             switch(ch2)
 | ||
|             {
 | ||
|                 case 0x31:   
 | ||
|                     BootLoaderJumpApp();
 | ||
|                     break;
 | ||
| 
 | ||
|                 case 0x32:
 | ||
|                     Update();
 | ||
|                     BootLoaderJumpApp();
 | ||
|                     break;
 | ||
| 
 | ||
|                 case 0x33:
 | ||
|                     mcuboot.op_reset();
 | ||
|                 default:
 | ||
|                     break;
 | ||
|             }
 | ||
|         }
 | ||
|         //10s内不按下空格键默然进行升级,升级完成后跳转
 | ||
|         else
 | ||
|         {
 | ||
|             Update();
 | ||
|             BootLoaderJumpApp();
 | ||
|         } 
 | ||
|     }
 | ||
| } |