新闻  |   论坛  |   博客  |   在线研讨会
分享一个OTA升级相关的应用实践!(1)
鱼鹰谈单片机 | 2023-04-23 19:25:26    阅读:522   发布文章

大家好,我是杂烩君。

本次与大家分享一个ota升级相关的应用实践。

应用场景

某项目中,有三块控制板协同工作,WiFi模块挂在其中一块板上:

图片


其中,三块板子都有升级的需求。即board1需要从云端下载各板子的升级文件之后通过串口分发给另外两块板子。

思路及一些缩减代码

作为board1的开发者,除了处理好给board2、board3分包分发升级文件之外,还需要处理好整个升级过程的可视化反馈,即需要在手机APP上显示出当前的升级进度。

为了显示这个升级进度,可能需要考虑如下情况:

1、下载升级包的过程与传输升级包(以下我称为安装)的进度需不需要分开?

这里我们选择分开,即手机APP触发升级,下载过程,手机APP显示下载进度,下载进度走完100之后,显示安装进度。

图片图片

一方面比较方便地能看出当前处于升级的那个过程,对于研发而言可以方便定位分析问题,对于用户体验上也不会给用户增加更多的使用成本。

此处,下载升级包的过程其实不受我们控制的,这一块阿里的SDK已经给我们做好,但有些云平台的设备SDK可能没有没有做这个固件下载的过程,可能只提供了固件所在文件服务器的http链接,需要我们自行进行下载,固件下载过程可参照这篇文章:C语言实现http下载器(附代码)

我们要处理的其实就是下载完升级包之后的事情,如解压升级包、安装。

2、每次升级可能不是升级所有的板子,如何处理?

启动安装的时候,通过比对各板子升级文件的md5值与本地保存的md5值确认是否是新的固件,并且通过这个比对我们就知道了当前升级包中的需要升级哪些板子。

三块板子组合的升级情况全部列举如下:

左右滑动查看全部代码>>>

// OTA升级情况
#define DEV_NON_SELECTED      0   // 设备没有被选中
#define BOARD1_DEV_SELECTED   1   // board1设备被选中
#define BOARD2_DEV_SELECTED   2   // board2设备被选中
#define BOARD3_DEV_SELECTED   4   // board3设备被选中
typedef enum _ota_update_case
{
    UPDATE_SELECTED_NULL         = -1,                                                              // 初始值
    UPDATE_NON_DEV               = 0,                                                               // 没有设备要升级
    UPDATE_BOARD1_DEV            = BOARD1_DEV_SELECTED,                                             // 只升级board1设备
    UPDATE_BOARD2_DEV            = BOARD2_DEV_SELECTED,                                             // 只升级board2设备
    UPDATE_BOARD3_DEV            = BOARD3_DEV_SELECTED,                                             // 只升级board3设备
    UPDATE_BOARD1_AND_BOARD2_DEV = BOARD1_DEV_SELECTED + BOARD2_DEV_SELECTED,                       // 升级board1与board2设备
    UPDATE_BOARD1_AND_BOARD3_DEV = BOARD1_DEV_SELECTED + BOARD3_DEV_SELECTED,                       // 升级board1与board3设备
    UPDATE_BOARD2_AND_BOARD3_DEV = BOARD2_DEV_SELECTED + BOARD3_DEV_SELECTED,                       // 升级board2与board3设备
    UPDATE_ALL_DEV               = BOARD1_DEV_SELECTED + BOARD2_DEV_SELECTED + BOARD3_DEV_SELECTED, // 升级所有设备
}ota_update_case_e;

获取当前升级包属于哪一种升级情况:

左右滑动查看全部代码>>>

// 获取升级情况
static ota_update_case_e get_ota_update_case(void)
{
    md5sum_t calc_board1_img_md5 = {0};
    md5sum_t calc_board2_img_md5 = {0};
    md5sum_t calc_board3_img_md5 = {0};
    ota_update_case_e ota_update_case = UPDATE_SELECTED_NULL;
    ota_update_case_e board1_update_case = UPDATE_SELECTED_NULL;
    ota_update_case_e board2_update_case = UPDATE_SELECTED_NULL;
    ota_update_case_e board3_update_case = UPDATE_SELECTED_NULL;

    // 读取升级文件md5参数表
    bzero(&s_flash_ota_file_md5_list, sizeof(s_flash_ota_file_md5_list));
    read_ota_md5_list_from_file(&s_flash_ota_file_md5_list, sizeof(s_flash_ota_file_md5_list)); 

    // 计算各升级文件的md5值
    calc_file_md5(OTA_BOARD1_DEV_IMG_FILE, &calc_board1_img_md5);
    calc_file_md5(OTA_BOARD2_DEV_IMG_FILE, &calc_board2_img_md5);
    calc_file_md5(OTA_BOARD3_DEV_IMG_FILE, &calc_board3_img_md5);

    // 检测是否选中board1设备
    if (0 == access(OTA_BOARD1_DEV_IMG_FILE, F_OK) && 
        FALSE == check_md5_equal(&calc_board1_img_md5, &s_flash_ota_file_md5_list[OTA_IMG_INDEX_BOARD1_DEV]))
    {
        LOG_D("BOARD1_DEV_SELECTED");
        board1_update_case = BOARD1_DEV_SELECTED;
    }
    else
    {
        LOG_D("board1 install file not exist or unchanged, not install");
        board1_update_case = DEV_NON_SELECTED;
    }

    // 检测是否选中board2设备
    if (0 == access(OTA_BOARD2_DEV_IMG_FILE, F_OK) && 
        FALSE == check_md5_equal(&calc_board2_img_md5, &s_flash_ota_file_md5_list[OTA_IMG_INDEX_BOARD2_DEV]))
    {
        LOG_D("BOARD2_DEV_SELECTED");
        board2_update_case = BOARD2_DEV_SELECTED;
    }
    else
    {
        LOG_D("board2 install file not exist or unchanged, not install");
        board2_update_case = DEV_NON_SELECTED;
    }

    // 检测是否选中board3设备
    if (0 == access(OTA_BOARD3_DEV_IMG_FILE, F_OK) && 
        FALSE == check_md5_equal(&calc_board3_img_md5, &s_flash_ota_file_md5_list[OTA_IMG_INDEX_BOARD3_DEV]))
    {
        LOG_D("BOARD3_DEV_SELECTED");
        board3_update_case = BOARD3_DEV_SELECTED;
    }
    else
    {
        LOG_D("board3 install file not exist or unchanged, not install");
        board3_update_case = DEV_NON_SELECTED;
    }

    // 升级的情况
    ota_update_case = board1_update_case + board2_update_case + board3_update_case;
    LOG_D("ota_update_case = %d", ota_update_case);

    return ota_update_case;
}
3、各种不同升级包情况,如何处理?

因为WiFi挂在board1上,所以整个升级过程,如果board1需要升级的话,board1需要放在最后升级,因为board1升级的过程中会与手机APP断连。

三块板的固件安装:

static void start_board1_ota_install(void)
static void start_board2_ota_install(void)
static void start_board3_ota_install(void)
  • start_board1_ota_install为安装board1固件的过程。安装board1固件的过程很简单,可以看做一次板子重启,升级安装命令如:
#define OTA_BOARD1_DEV_IMG_FILE   OTA_FIREWARE_EXTRACT_FILE_PATH "board1_update.img"         // board1固件
#define SHELL_CMD_START_BOARD1_DEV_OTA_INSTALL    "updateEngine --image_url=" OTA_BOARD1_DEV_IMG_FILE " --misc=update --savepath=" OTA_BOARD1_DEV_IMG_FILE " --reboot"
  • start_board2_ota_install为安装board2固件的过程。安装board2固件的过程即board1往board2发送board2固件的过程,整个过程通过数据传输量占总数据量的大小可表明安装进度。
  • start_board3_ota_install安装board3固件的过程,安装board3固件的过程即board1往board3发送board3固件的过程,整个过程通过数据传输量占总数据量的大小可表明安装进度。


*博客内容为网友个人发布,仅代表博主个人观点,如有侵权请联系工作人员删除。

参与讨论
登录后参与讨论
推荐文章
最近访客