RT-Thread—STM32—在线升级(Ymodem_OTA)


概述

本教程主要根据官方推荐的教程进行改编,详细信息请参考
OTA Downloader软件包
STM32 通用 Bootloader

本例程的模板使用通用模板环境搭建里面的模板RT-Thread—STM32—FAL库
文末也有我移植好的例程,不过建议大家从头开始移植,加深印象^_^


bootloader的制作

文末有我已经做好的Bootloader文件,可供参考

图片

  • 点击生成Bootloader
  • 可以选择邮箱下载,或者手动下载

请添加图片描述

烧录Bootloader

  • 选择合适的工具烧录BootLoader
  • 这里我选择的是STM32CubeProgrammer(使用的是ST-Link/V2)
  • 连接之后下载刚刚生成的Bootloader文件(xxxx.bin)

请添加图片描述

  • 连接串口,测试打印信息
  • 能看到我们之前制作Bootloader时,相关的参数以及logo,说明Bootloader烧录成功,如下图所示
  • 博主使用的是Xshell软件(建议使用Xshell软件)
  • Xhell官网

请添加图片描述

制作APP程序

ENV配置

  • 选择ota功能

请添加图片描述

  • 使能Ymodem OTA
  • 然后更新配置保存退出

请添加图片描述

代码修改

  • 打开fal_cfg.h文件(此过程一定要和Bootloader制作是保持地址对应,否者没法升级)
  • 更改app的开始地址
#define RT_APP_PART_ADDR 0x08020000        // app区的开始地址
  • 更改分区表
#define FAL_PART_TABLE                                                               \
{                                                                                    \
    {FAL_PART_MAGIC_WORD,"bl",            "onchip_flash",    0,            128*1024,    0}, \
    {FAL_PART_MAGIC_WORD,"app",            "onchip_flash",    128*1024,    128*1024,    0}, \
    {FAL_PART_MAGIC_WORD,"download",    "onchip_flash",    256*1024,    128*1024,    0}, \
    {FAL_PART_MAGIC_WORD,"easyflash",    "onchip_flash",    384*1024,    128*1024,    0}, \
}
  • 修改后如下图所示

请添加图片描述

  • 打开Option for Target…
  • 在Linker中点击Edit…修改link.sct文件,使开始地址为Bootloader设置的地址,如下图所示

请添加图片描述

  • 打开fal_cfg.h文件
  • 添加打印版本信息的代码
    rt_kprintf("/****************************************************/\n");
    rt_kprintf("/*************Version : 2.0.0\n");
    rt_kprintf("/****************************************************/\n");
  • 添加更改中断向量表相关的参数
/**
 * Function    ota_app_vtor_reconfig
 * Description Set Vector Table base location to the start addr of app(RT_APP_PART_ADDR).
*/
static int ota_app_vtor_reconfig(void)
{
    #define NVIC_VTOR_MASK   0x3FFFFF80
    /* Set the Vector Table base location by user application firmware definition */
    SCB->VTOR = RT_APP_PART_ADDR & NVIC_VTOR_MASK;

    return 0;
}
INIT_BOARD_EXPORT(ota_app_vtor_reconfig);

请添加图片描述

  • 查看烧录设置
  • Option for Target...–>Debug–>Settings–>FlashDownload
  • 确保Download Function选择Erase Sectors

请添加图片描述

  • 编译并且烧录程序

请添加图片描述

  • 打开串口会看到Bootloader已经成功跳转到我们的程序并且正常运行了

请添加图片描述

  • 修改版本号,只编译编译
  • 千万不要烧录!!!
  • 千万不要烧录!!!
  • 千万不要烧录!!!

请添加图片描述

  • 打开目录packages \ ota_downloader-latest \ tools \ ota_packager
  • 找到如下所示的生成软件包生成工具,并且打开

请添加图片描述

  • 点击选择固件找到主目录下的rtthread.bin文件
  • 添加固件区名和`固件版本```然后打包
  • 成功后会在rtthread.bin文件的同一目录下生成rtthread.rbl文件

请添加图片描述

  • 打开串口输入help会打印帮助信息
  • 输入ymodem_ota执行升级命令

请添加图片描述

  • 在黑窗口点击鼠标右键–>传输–>YMODEM(Y)
  • 选择刚刚生成的rtthread.rbl文件,打开进行升级,如下图所示

请添加图片描述

请添加图片描述

  • 成功之后,会看到版本变化了,说明升级成功,如下图所示

请添加图片描述


源代码获取

源码已放到码云 ! ! ! ( 请点击文首链接进入仓库 )


  • 4
    点赞
  • 61
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 8
    评论
以下是STM32F103的Ymodem升级程序示例代码: ``` #include "stm32f10x.h" #include "stdio.h" #define FLASH_APP1_ADDR 0x08010000 // APP1地址 #define FLASH_APP2_ADDR 0x08020000 // APP2地址 #define FLASH_APP_SIZE 0x00010000 // 每个APP的大小为64KB #define FLASH_PAGE_SIZE 0x400 // Flash每页大小为1KB #define YMODEM_SOH 0x01 #define YMODEM_STX 0x02 #define YMODEM_EOT 0x04 #define YMODEM_ACK 0x06 #define YMODEM_NAK 0x15 #define YMODEM_CAN 0x18 #define YMODEM_C 0x43 #define YMODEM_TIMEOUT 1000 // 超时时间 #define YMODEM_MAX_SIZE 64 // Ymodem每个数据包大小为128字节,实际使用时使用64字节即可 static uint8_t ymodem_get_char(uint8_t *data); static uint8_t ymodem_wait_ack(void); static uint8_t ymodem_calc_checksum(uint8_t *data, uint32_t size); static uint8_t ymodem_write_packet(uint8_t *data, uint32_t size); static void ymodem_write_flash(uint8_t *data, uint32_t size); static uint8_t ymodem_file_receive(uint32_t addr); int main(void) { if (ymodem_file_receive(FLASH_APP1_ADDR) == YMODEM_ACK) { // APP1接收成功,重启执行APP1 NVIC_SystemReset(); } else if (ymodem_file_receive(FLASH_APP2_ADDR) == YMODEM_ACK) { // APP2接收成功,重启执行APP2 NVIC_SystemReset(); } while (1); } static uint8_t ymodem_get_char(uint8_t *data) { uint32_t timer = YMODEM_TIMEOUT; while (timer--) { if (USART_GetFlagStatus(USART1, USART_FLAG_RXNE) != RESET) { *data = USART_ReceiveData(USART1); return YMODEM_ACK; } } return YMODEM_NAK; } static uint8_t ymodem_wait_ack(void) { uint8_t ret = 0; uint8_t data = YMODEM_NAK; uint32_t timer = YMODEM_TIMEOUT; while (timer--) { ret = ymodem_get_char(&data); if (ret == YMODEM_ACK) { return YMODEM_ACK; } } return YMODEM_NAK; } static uint8_t ymodem_calc_checksum(uint8_t *data, uint32_t size) { uint8_t checksum = 0; uint32_t i; for (i = 0; i < size; i++) { checksum += data[i]; } return checksum; } static uint8_t ymodem_write_packet(uint8_t *data, uint32_t size) { uint8_t ret = 0; uint8_t packet[YMODEM_MAX_SIZE + 6]; // 数据包大小为64字节,加上头部3字节和尾部3字节 uint32_t i; packet[0] = size == 128 ? YMODEM_SOH : YMODEM_STX; // 根据数据包大小确定头类型 packet[1] = 0xFF - size; // 数据包大小的补码 packet[2] = size == 128 ? 0x00 : 0x01; // 序号,128字节为0,64字节为1 for (i = 0; i < size; i++) { packet[i + 3] = data[i]; } for (i = size + 3; i < YMODEM_MAX_SIZE + 3; i++) { packet[i] = 0x00; // 填充0 } uint8_t checksum = ymodem_calc_checksum(&packet[3], size); packet[YMODEM_MAX_SIZE + 3] = checksum; packet[YMODEM_MAX_SIZE + 4] = 0x00; // 结束标志 packet[YMODEM_MAX_SIZE + 5] = 0x00; // 结束标志 for (i = 0; i < 3; i++) { ret = USART_SendData(USART1, packet[i]); while (USART_GetFlagStatus(USART1, USART_FLAG_TXE) == RESET); } for (i = 3; i < YMODEM_MAX_SIZE + 6; i++) { ret = USART_SendData(USART1, packet[i]); while (USART_GetFlagStatus(USART1, USART_FLAG_TXE) == RESET); } return ymodem_wait_ack(); } static void ymodem_write_flash(uint8_t *data, uint32_t size) { uint32_t i; uint32_t flash_addr = FLASH_APP1_ADDR; // 默认写入APP1 uint8_t flash_erase = 0; // 是否需要擦除 // 判断写入目标地址 if (*((uint32_t *)data) == 0x08020000) { flash_addr = FLASH_APP2_ADDR; } // 判断是否需要擦除 if (flash_addr == FLASH_APP1_ADDR) { if (*((uint32_t *)FLASH_APP1_ADDR) != 0xFFFFFFFF) { flash_erase = 1; } } else { if (*((uint32_t *)FLASH_APP2_ADDR) != 0xFFFFFFFF) { flash_erase = 1; } } // 擦除Flash if (flash_erase) { FLASH_Unlock(); FLASH_ErasePage(flash_addr); FLASH_Lock(); } // 写入数据到Flash FLASH_Unlock(); for (i = 0; i < size; i += 4) { FLASH_ProgramWord(flash_addr + i, *((uint32_t *)(data + i))); } FLASH_Lock(); } static uint8_t ymodem_file_receive(uint32_t addr) { uint8_t ret = 0; uint8_t data[YMODEM_MAX_SIZE]; uint32_t size = 0; uint32_t i; // 发送开始传输命令 uint8_t cmd = YMODEM_C; for (i = 0; i < 3; i++) { ret = USART_SendData(USART1, cmd); while (USART_GetFlagStatus(USART1, USART_FLAG_TXE) == RESET); } while (1) { // 等待数据包 ret = ymodem_wait_ack(); if (ret != YMODEM_SOH && ret != YMODEM_STX) { return YMODEM_NAK; } // 接收数据包 for (i = 0; i < YMODEM_MAX_SIZE; i++) { ret = ymodem_get_char(&data[i]); if (ret != YMODEM_ACK) { return YMODEM_NAK; } } // 校验数据包 if (data[YMODEM_MAX_SIZE - 2] != ymodem_calc_checksum(data, YMODEM_MAX_SIZE - 2)) { return YMODEM_NAK; } // 处理数据包 if (data[0] == YMODEM_SOH) { size = 128; } else { size = 64; } // 数据包序号为0时表示结束 if (data[1] == 0x00) { // 发送确认结束命令 for (i = 0; i < 3; i++) { ret = USART_SendData(USART1, YMODEM_ACK); while (USART_GetFlagStatus(USART1, USART_FLAG_TXE) == RESET); } break; } // 写入Flash ymodem_write_flash(&data[3], size); // 发送确认命令 for (i = 0; i < 3; i++) { ret = USART_SendData(USART1, YMODEM_ACK); while (USART_GetFlagStatus(USART1, USART_FLAG_TXE) == RESET); } } return YMODEM_ACK; } ``` 该示例代码实现了一个Ymodem协议的文件接收程序,可以将接收到的数据包写入Flash中。在主函数中,先尝试接收APP1,如果接收成功则重启执行APP1;如果接收失败则尝试接收APP2,如果接收成功则重启执行APP2。如果两个APP都接收失败,则程序将被卡在while(1)循环中,等待重新启动。 在使用该示例代码时,需要在USART1的RX和TX引脚上接上一个串口转USB模块,并将该模块连接到PC上。然后,在PC上使用Ymodem协议的传输工具发送.bin文件即可。 需要注意的是,该示例代码只是一个简单的示例,实际使用时还需要根据具体的应用场景进行修改和完善。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 8
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

iot 小胡

从未指望过会有人打赏...

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值