STM32 BootLoader 刷新项目 (十) Flash擦除-命令0x56

STM32 BootLoader 刷新项目 (十) Flash擦除-命令0x56

1. STM32F407 BootLoader 中的 Flash 擦除功能详解

在嵌入式系统中,BootLoader 的设计是非常关键的部分,它负责引导主程序的启动、升级以及安全管理。而在 STM32F407 等 MCU 上实现 BootLoader,Flash 操作则是其中的核心功能之一。本文将重点介绍 STM32F407 上的 Flash 擦除功能,并给出实现过程和注意事项。

image-20241114070905643

一、STM32F407 Flash 结构概述

STM32F407 芯片采用 Cortex-M4 内核,内置 Flash 存储用于程序和数据存储。STM32F407 的 Flash 大小为 512KB,分为若干个扇区(Sector),每个扇区的大小并不相同:

  • 扇区 0 到 3:16KB 每扇区
  • 扇区 4:64KB
  • 扇区 5 到 11:128KB 每扇区

image-20241114070535836

这种不同大小的扇区设计适合不同的应用需求,比如小容量扇区用于存储配置数据,大容量扇区用于存储固件。

二、Flash 擦除的原理

Flash 擦除是将 Flash 中的某个扇区的数据重置为 0xFF。由于 Flash 的物理特性,写操作只能将位设置为 0,而擦除操作将位恢复为 1。因此,在更新 Flash 数据时,通常需要先擦除再写入。

在 STM32F407 中,Flash 擦除只能以扇区为单位进行,这意味着无法擦除扇区中的部分数据。每次擦除扇区时,整个扇区的数据都会被清空。

三、Flash 擦除的操作流程

在 STM32F407 的 BootLoader 中实现 Flash 擦除,一般需要遵循以下步骤:

  1. 解锁 Flash 控制寄存器:在进行任何 Flash 操作之前,需要解锁 Flash 的写保护功能。
  2. 等待 Flash 空闲:检查 Flash 状态寄存器,确保没有其他操作正在进行。
  3. 启动扇区擦除:设置目标扇区并启动擦除命令。
  4. 等待擦除完成:监控状态寄存器中的 BSY 位,等待擦除操作完成。
  5. 锁定 Flash 控制寄存器:完成操作后,将 Flash 锁定,以防止误操作。

四、Flash 擦除实现代码

以下是 STM32F407 上实现 Flash 擦除的代码示例。本文假设使用了 STM32 标准外设库,便于调用硬件寄存器。

#include "stm32f4xx.h"/*** @brief  擦除 Flash 指定扇区* @param  sector 要擦除的扇区编号* @retval 0 表示成功,-1 表示失败*/
int Flash_EraseSector(uint8_t sector)
{// 解锁 Flash 控制寄存器FLASH_Unlock();// 等待 Flash 空闲while (FLASH_GetStatus() == FLASH_BUSY);// 擦除操作FLASH_ClearFlag(FLASH_FLAG_EOP | FLASH_FLAG_OPERR | FLASH_FLAG_WRPERR | FLASH_FLAG_PGAERR | FLASH_FLAG_PGPERR | FLASH_FLAG_PGSERR);FLASH_Status status = FLASH_EraseSector(sector, VoltageRange_3);// 等待擦除完成if (status != FLASH_COMPLETE){FLASH_Lock();return -1; // 擦除失败}// 锁定 Flash 控制寄存器FLASH_Lock();return 0; // 擦除成功
}

五、代码详解

  • FLASH_Unlock():此函数用于解锁 Flash 控制寄存器,以允许擦除和写入操作。
  • FLASH_GetStatus():用于检查 Flash 是否处于忙碌状态。擦除操作会占用一定的时间,必须等待 Flash 空闲后才能继续。
  • FLASH_ClearFlag():清除可能出现的错误标志位,确保擦除操作不会受到之前错误的影响。
  • FLASH_EraseSector():这是 STM32 库中的擦除函数,输入参数为扇区编号和电压范围(STM32F4 系列通常使用 VoltageRange_3)。
  • FLASH_Lock():操作完成后锁定 Flash 控制寄存器,以防止意外操作。

六、Flash 擦除的注意事项

  1. 擦除单位:在 STM32F407 中,Flash 擦除只能以扇区为单位进行,无法进行字节或页的擦除。因此,设计 BootLoader 时要合理分配数据的存储位置,避免不必要的擦除操作。
  2. 电压范围:擦除 Flash 时,电源电压必须在一定范围内,低电压下可能导致擦除失败甚至损坏 Flash。STM32F407 提供了多种电压模式,通常选择 VoltageRange_3 即可。
  3. 错误处理:Flash 擦除失败可能会出现各种错误,例如写保护、编程错误等。建议在擦除操作前后清除并检查错误标志。
  4. 数据备份:在擦除前备份必要的数据。在 BootLoader 中进行固件更新时,最好将旧固件暂时保存至其他存储介质中,以应对擦除或写入失败的情况。
  5. 上电时间:Flash 擦除时间较长,尤其是 128KB 扇区,因此在设计 BootLoader 时需要考虑电源的持续供电能力,避免中途掉电。

2. BootLoader Flash划分

由下图可以看出本BootLoader的Flash的划分,其中划给BootLoader为32KB,从0x0800 0000-0x0800 7FFF,占用Sector 0-1两个段。App应用程序占用0x0800 8000-0x080F FFFF,总共10个Sector。

image-20241114070819942

下面是BootLoader的跳转过程,关于具体的跳转过程,可以参考上一篇文章: STM32 BootLoader 刷新项目 (九) 跳转指定地址-命令0x55

image-20241114230528904

image-20241114230647450

3. 0x56命令介绍–Flash擦除

在本篇文章,我们的主要是介绍0x56的命令,这个命令主要是在BootLoader中擦除指定Flash Sector的命令。

通过上位机发送8 Byte的数据,其中第1 Byte为整个数据的长度,第2Byte为指令码,第3 Byte为要擦除Flash的起始Sector序号,第4 Byte是擦除Sector的长度,第5-8 Byte为前6个Byte的CRC校验值。上位机通过串口UART发送给下位机,下位机回复地址是否擦除成功的标志。

image-20241114230143679

下面是发送命令过程中上位机与BootLoader之间的交互。

image-20241114230607647

4. Flash擦除命令程序设计

下面我们来进行程序设计,下面是读取上位机指令,并解析指令的过程,通过switch case判断执行哪种命令。目前通过上位机执行BL_FLASH_ERASE指令,然后执行bootloader_handle_flash_erase_cmd(bl_rx_buffer)函数。

下面是代码的详细注释和分析:

bootloader_uart_read_data 函数

该函数用于从主机接收命令数据并调用相应的处理函数来执行命令。它是 BootLoader 的核心输入接口,通过 UART 接收主机的命令,并根据命令类型分发到对应的处理函数。

void bootloader_uart_read_data(void)
{uint8_t rcv_len = 0;// 打印调试信息,提示接收到命令printmsg_Host("BL_DEBUG_MSG: Receive CMD\n\r");// 无限循环,持续接收命令并处理while (1){// 将 LED2 熄灭,表示进入命令接收状态HAL_GPIO_WritePin(LED2_GPIO_Port, LED2_Pin, GPIO_PIN_RESET);// 清空接收缓冲区,防止残留数据干扰memset(bl_rx_buffer, 0, 200);// 读取命令包的第一个字节,即命令的长度字段HAL_UART_Receive(C_UART, bl_rx_buffer, 1, HAL_MAX_DELAY);// 获取命令包长度,长度字节位于第一个字节rcv_len = bl_rx_buffer[0];// 根据命令包长度读取剩余的字节HAL_UART_Receive(C_UART, &bl_rx_buffer[1], rcv_len, HAL_MAX_DELAY);// 根据命令码执行相应的命令处理函数switch (bl_rx_buffer[1]){case BL_GET_VER:// 处理获取版本命令bootloader_handle_getver_cmd(bl_rx_buffer);break;case BL_GET_HELP:// 处理获取帮助命令bootloader_handle_gethelp_cmd(bl_rx_buffer);break;case BL_GET_CID:// 处理获取芯片 ID 命令bootloader_handle_getcid_cmd(bl_rx_buffer);break;case BL_GET_RDP_STATUS:// 处理获取读保护状态命令bootloader_handle_getrdp_cmd(bl_rx_buffer);break;case BL_GO_TO_ADDR:// 处理跳转到地址命令bootloader_handle_go_cmd(bl_rx_buffer);break;case BL_FLASH_ERASE:// 处理 Flash 擦除命令bootloader_handle_flash_erase_cmd(bl_rx_buffer);break;default:// 未知命令,打印调试信息printmsg("BL_DEBUG_MSG: Invalid command code received from host \n");break;}}
}
  • 初始化和循环:函数会无限循环等待主机发送的命令并进行解析。
  • 数据接收:首先读取一个字节的长度字段,然后根据这个长度读取完整的命令数据。
  • 命令解析:通过 switch 语句解析命令码,并调用相应的处理函数。
  • 错误处理:如果命令码不在预期范围内,打印调试信息。

bootloader_handle_flash_erase_cmd 函数

该函数处理 BL_FLASH_ERASE 命令,通过接收到的参数执行 Flash 擦除操作。

void bootloader_handle_flash_erase_cmd(uint8_t *pBuffer)
{uint8_t erase_status = 0x00;printmsg("BL_DEBUG_MSG: bootloader_handle_flash_erase_cmd\n");// 获取命令包的总长度uint32_t command_packet_len = bl_rx_buffer[0] + 1;// 从命令包中提取主机发送的 CRC32 值uint32_t host_crc = *((uint32_t *)(bl_rx_buffer + command_packet_len - 4));// 验证接收到的数据包的 CRC 校验if (!bootloader_verify_crc(&bl_rx_buffer[0], command_packet_len - 4, host_crc)){// 如果校验成功printmsg("BL_DEBUG_MSG: checksum success !!\n");// 发送 ACK 响应,确认命令长度为 1 字节bootloader_send_ack(pBuffer[0], 1);// 打印擦除信息:初始扇区和扇区数量printmsg("BL_DEBUG_MSG: initial_sector: %d, no_of_sectors: %d\n", pBuffer[2], pBuffer[3]);// 熄灭 LED 表示擦除操作开始HAL_GPIO_WritePin(LED2_GPIO_Port, LED2_Pin, GPIO_PIN_RESET);// 调用 `execute_flash_erase` 执行擦除操作,传入初始扇区和扇区数量erase_status = execute_flash_erase(pBuffer[2], pBuffer[3]);// 重新点亮 LED 表示擦除操作完成HAL_GPIO_WritePin(LED2_GPIO_Port, LED2_Pin, GPIO_PIN_SET);// 打印擦除操作的状态printmsg("BL_DEBUG_MSG: flash erase status: %#x\n", erase_status);// 发送擦除状态回主机bootloader_uart_write_data(&erase_status, 1);}else{// 如果 CRC 校验失败,发送 NACK 响应printmsg("BL_DEBUG_MSG: checksum fail !!\n");bootloader_send_nack();}
}
  • 参数解析和 CRC 校验:从命令包中提取 CRC 值并进行校验,确保数据完整性。
  • 发送 ACK 或 NACK:根据 CRC 校验结果发送确认或拒绝响应。
  • 擦除操作:如果校验通过,提取擦除操作的初始扇区和扇区数量,然后调用 execute_flash_erase 函数执行擦除操作。
  • 状态回传:擦除完成后,将操作状态通过 UART 返回给主机。

execute_flash_erase 函数

execute_flash_erase 函数负责执行 Flash 的具体擦除操作,支持部分扇区擦除和全片擦除。

uint8_t execute_flash_erase(uint8_t sector_number, uint8_t number_of_sector)
{// STM32F407 芯片共有 12 个扇区(0 到 11)// 如果 sector_number = 0xff,表示全片擦除FLASH_EraseInitTypeDef flashErase_handle;uint32_t sectorError;HAL_StatusTypeDef status;// 如果指定的扇区数量超过有效范围,返回无效扇区if (number_of_sector > 11)return INVALID_SECTOR;// 检查扇区号是否有效if ((sector_number == 0xFF) || (sector_number <= 11)){// 判断是否全片擦除if (sector_number == (uint8_t)0xFF){flashErase_handle.TypeErase = FLASH_TYPEERASE_MASSERASE;}else{// 部分扇区擦除// 计算剩余可擦除的扇区数uint8_t remaining_sector = 12 - sector_number;if (number_of_sector > remaining_sector){number_of_sector = remaining_sector;}flashErase_handle.TypeErase = FLASH_TYPEERASE_SECTORS;flashErase_handle.Sector = sector_number;flashErase_handle.NbSectors = number_of_sector;}// 设置擦除银行(STM32F4 系列有两个 Flash Bank,选取 Bank 1)flashErase_handle.Banks = FLASH_BANK_1;// 解锁 Flash 寄存器,允许擦除操作HAL_FLASH_Unlock();// 设置电压范围,适应 STM32F4 的工作电压范围flashErase_handle.VoltageRange = FLASH_VOLTAGE_RANGE_3;// 执行擦除操作,传入擦除配置和错误记录变量status = (uint8_t)HAL_FLASHEx_Erase(&flashErase_handle, &sectorError);// 锁定 Flash 控制寄存器,防止误操作HAL_FLASH_Lock();return status;  // 返回擦除操作的状态}return INVALID_SECTOR;  // 如果无效扇区,返回无效状态
}
  • 全片和部分擦除:函数支持全片擦除(sector_number == 0xFF)和从指定扇区开始的部分擦除。
  • 擦除范围检查:确保扇区数量不超过剩余的可擦除范围。
  • 解锁和锁定:在擦除操作前解锁 Flash 控制寄存器,擦除完成后重新锁定。
  • 电压范围设置:设置为 FLASH_VOLTAGE_RANGE_3,适配 STM32F4 的工作电压。
  • 状态返回:返回擦除状态,以便调用方确认擦除是否成功。

总结

  • bootloader_uart_read_data:从 UART 接收命令,解析命令码并调用对应的命令处理函数。
  • bootloader_handle_flash_erase_cmd:处理 Flash 擦除命令,进行 CRC 校验

5. 实战演练

下面是上位机的命令菜单,通过在终端调用Python脚本,然后在终端输入下位机连接的串口号,即可进入命令界面,目前可支持如下命令:

image-20240713104433991

下面我们执行命令–7,执行BL_FLASH_ERASE,即为在BootLoader中执行Flash擦除命令0x56,第一步输入要开始擦除的Sector起始段,第二步输入擦除Sector的长度,最后BootLoader向上位机返回是否擦除成功的Status。

image-20241115071431811

下面我们来看Flash擦除命令是否擦除成功,用STM32CubeProgrammer工具,用ST-LINK连接Debug口,上一步中我们擦除的Sector 3的Flash,这里面我们输入Sector 2的起始地址0x0800 C000,可以看出全部擦除为FF。至此Flash擦除命令已经讲完。

image-20241115072819345

6. 结论

Flash 擦除是 STM32F407 BootLoader 设计中的重要环节,它为固件更新和数据存储提供了基础。本文详细介绍了 Flash 擦除的原理和实现方法,并提供了相关代码和注意事项。在实际应用中,设计人员可以根据需要将擦除功能整合至 BootLoader 的主流程中,确保系统的可靠性和可升级性。

希望本篇文章能为读者在 STM32F407 的 BootLoader 开发中提供一些帮助,若有疑问,欢迎交流讨论。

7. 系列文章

STM32 BootLoader 刷新项目 (一) STM32CubeMX UART串口通信工程搭建

STM32 BootLoader 刷新项目 (二) 方案介绍

STM32 BootLoader 刷新项目 (三) 程序框架搭建及刷新演示

STM32 BootLoader 刷新项目 (四) 通信协议

STM32 BootLoader 刷新项目 (五) 获取软件版本号-命令0x51

STM32 BootLoader 刷新项目 (六) 获取帮助-命令0x52

STM32 BootLoader 刷新项目 (七) 获取芯片ID-0x53

STM32 BootLoader 刷新项目 (八) 读取Flash保护ROP-0x54

STM32 BootLoader 刷新项目 (九) 跳转指定地址-命令0x55

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.xdnf.cn/news/16054.html

如若内容造成侵权/违法违规/事实不符,请联系一条长河网进行投诉反馈,一经查实,立即删除!

相关文章

J.U.C - 深入解读重入锁和读写锁

文章目录 概述synchronized的缺陷1&#xff09;synchronized不能控制阻塞&#xff0c;不能灵活控制锁的释放。2&#xff09;在读多写少的场景中&#xff0c;效率低下。 独占锁ReentrantLock原理ReentrantLock概述AQS同步队列1. AQS实现原理2. 线程被唤醒时&#xff0c;AQS队列的…

异地双活容灾技术研究

摘要 随着技术快速发展&#xff0c;尤其是人工智能、大数据等新兴技术的应用&#xff0c;对数据安全提出了新的挑战&#xff0c;平台部署在机房云资源池&#xff0c;当云平台因人为错误原因出现基础设施故障&#xff0c;或自然灾害使得云平台的机房出现停电、断网等故障&#x…

从Facebook到Meta:公司转型背后的战略与意义

2021年&#xff0c;Facebook宣布更名为Meta&#xff0c;转型聚焦于“元宇宙”——这一虚拟世界的构建标志着公司从传统社交平台向更前沿的科技领域迈进。本文将探讨这一转型的背景、战略布局及其深远意义。 一、转型背景&#xff1a;应对市场和技术的挑战 自2004年成立以来&am…

前端在PC端实现支付思路流程

一.去支付 1.前端点击“去支付”按钮&#xff0c;请求订单详情接口&#xff0c;传递订单的id、订单号给后端和请求支付方式接口 2.后端返回支付信息和支付方式数据 二.弹出支付窗口 接收支付信息和支付方式数据后&#xff0c;前端弹出支付弹窗 三.确认支付 前端无论选择任何…

STM32 51单片机设计半导体制冷片温控设计

目录 前言 一、本设计主要实现哪些很“开门”功能&#xff1f; 二、电路设计原理图 1.电路图采用Altium Designer进行设计&#xff1a; 三、实物设计图 四、程序源代码设计 五、获取资料内容 前言 基于STM32与51单片机的半导体制冷片温控设计 前言 随着现代工业、医疗…

开源大模型技术路线及趋势

1. 三个维度 大模型研发力量 学术机构 大模型创业公司 科技大厂 旗舰开源模型的形态/模态 基础大模型 -> instruct大模型 -> 多模态&#xff08;VL大模型 -> 类gpt-4o大模型&#xff09; 时间线 23年上半年 23年下半年 24年 25年 2. 核心观察 学术机构及…

Vue学习笔记

Vue学习笔记 以下内容为黑马SpringbootVue3中的Vue3教程部分 Vue概述 什么是Vue&#xff1f; Vue 是一款用于构建用户界面的渐进式的JavaScript框架。&#xff08;官方:https://cn.vuejs.org/&#xff09; 快速入门 准备 准备html页面&#xff0c;并引入Vue模块&#xff08;官…

Java 使用MyBatis-Plus数据操作关键字冲突报错You have an error in your SQL syntax问题

问题 这个报错是出现在Java Spring boot项目中&#xff0c;使用MyBatis-Plus通过创建的实体类对数据库的操作过程中&#xff0c;通过实体创建数据库表是没有问题的&#xff0c;而在接口调用服务类操作数据库的时候&#xff0c;会出现报错。报错详情如下&#xff1a; 服务请求异…

Chrome DevTools Protocol 进阶: Page域

前言 本章开始我们将进一步学习 Chrome DevTools Protocol&#xff08;CDP&#xff09;&#xff0c;首先切入的内容是 CDP 中的域。 在 Chrome DevTools Protocol&#xff08;CDP&#xff09; 中&#xff0c;Page 域 是一个至关重要的部分&#xff0c;它负责控制浏览器页面的…

电动采光排烟大窗施工常见问题

电动采光排烟天窗施工常见问题可能涉及多个方面&#xff0c;包括施工前的准备、施工过程中的操作以及施工后的维护等。以下是对这些常见问题的归纳和分析。 一、施工前的准备问题 1、现场勘查不足&#xff0c;可能导致天窗尺寸、形状和材质不符合建筑设计要求&#xff1b; 2…

关于VUE NPM安装失败的问题

最近使用 npm install --registryhttps://registry.npmmirror.com 安装一个新项目的依赖&#xff0c;各种失败。 最后发现是package-lock里面有老的淘宝的域名&#xff0c;整体替换掉就行了

51单片机应用开发---LCD1602显示应用

实现目标 1、了解LCD1602液晶屏&#xff1b; 2、掌握驱动程序的编写&#xff1b; 3. 具体目标&#xff1a;在屏幕上显示字符。 一、LCD1206概述 1.1 定义 LCD1602(Liquid Crystal Display)液晶显示屏是一种字符型液晶显示模块,可以显示ASCII码的标准字符和其它的一些内置…

解决Clion相对路径失效的问题

Clion相对路径失效的可能原因 工作目录不对相对路径错误 解决方法 设置工作目录 点击 工作文件 将工作路径置空 (或设置正确的工作路径) 之后便可完美解决 设置全局路径 这样即可解决 希望能帮助到你

香港电讯连续蝉联香港01「大湾区企业同创大奖」

香港电讯非常荣幸地连续第四年荣获「香港01大湾区企业同创大奖」殊荣&#xff0c;以表扬我们在大湾区业务所取得的卓越成绩&#xff0c;以及我们积极促进区内跨境业务方面的贡献。 面对企业拓展业务的迫切需求&#xff0c;香港电讯提供全面的跨境数码方案&#xff0c;包括SD-WA…

Vue3.js - 一文看懂Vuex

1. 前言 Vuex 是 Vue.js 的官方状态管理库&#xff0c;用于在 Vue 应用中管理组件之间共享的状态。Vuex 适用于中大型应用&#xff0c;它将组件的共享状态集中管理&#xff0c;可以避免组件间传递 props 或事件的复杂性。 2. 核心概念 我们可以将Vuex想象为一个大型的Vue&…

PdServer:调用MidjourneyAPI完成静夜思图文生成

欢迎沟通讨论&#xff0c;WX: cdszsz。公号&#xff1a;AIGC中文站。 今天我们将使用PdServer&#xff0c;通过Qwen大模型完成古诗的解析与prompt的生成&#xff0c;然后调用MidjourneyAPI完成图片的生成。有了文案和图片&#xff0c;我们就可以将其生成为一个古诗讲读视频。从…

融合创新:CNN+LSTM在深度学习中的高效应用,助力科研发表高影响因子文章!

2024深度学习发论文&模型涨点之——CNNLSTM 结合CNN&#xff08;卷积神经网络&#xff09;和LSTM&#xff08;长短期记忆网络&#xff09;是一种常见的深度学习模型组合&#xff0c;广泛应用于各种任务&#xff0c;特别是时间序列预测和序列数据处理。 CNNLSTM是一种深度学…

20241114在飞凌的OK3588-C的核心板上跑Linux R4时通过iperf3测试以太网卡的实际网速

20241114在飞凌的OK3588-C的核心板上跑Linux R4时通过iperf3测试以太网卡的实际网速 2024/11/14 21:09 《OK3588-C_Linux5.10.66Qt5.15.2_用户使用手册_V1.2_20240321.pdf》 飞凌文档上的配置有点不一样&#xff0c;需要修正&#xff1a; 3.2.15 以太网配置 OK3588-C板载两个千…

鸿蒙开发,Arkts 如何调用接口

面向万物互联时代&#xff0c;华为提出了“一次开发多端部署、可分可合自由流转、统一生态原生智能”三大应用与服务开发理念。针对多设备、多入口、服务可分可合等特性&#xff0c;华为提供了多种能力协助开发者降低开发门槛。在此背景下&#xff0c;HarmonyOS基于JS/TS语言体…

JavaScript方法修改 input type=file 样式

html中的<input type "file">的样式很难修改&#xff0c;又跟页面风格很不匹配。我就尝试了几种方法&#xff0c;但是不管是用label还是用opacity:0都很麻烦&#xff0c;还老是出问题&#xff0c;所以最后还是用JavaScript来解决。 下面附上代码&#xff1a;…