前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
社区首页 >专栏 >基于STM32的Flash擦除方式

基于STM32的Flash擦除方式

原创
作者头像
跋扈洋
发布于 2022-05-31 11:55:02
发布于 2022-05-31 11:55:02
3.4K30
代码可运行
举报
文章被收录于专栏:物联网知识物联网知识
运行总次数:0
代码可运行

前言

本文主要介绍STM32的内部Flash擦除方式和擦除长文件的功能函数怎样编写。并且介绍一些注意事项,如只想擦除当前地址,却发现上下地址都出现了擦除等问题。阅读完本文可以使你能够正常的完成Flash擦除。并对擦除时会影响的地址大小有一个深入的认识,并在对页擦除时,页的起始地址和大小有所了解。

介绍

STM32 FLASH

不同型号的 STM32,其 FLASH 容量也有所不同,最小的只有 16K 字节,最大的则达到了1024K 字节。本次实验选用的STM32 开发板是F103ZET6,其 FLASH 容量为 512K 字节,属于大容量产品(另外还有中容量和小容量产品),大容量产品的闪存模块组织如图 所示:

STM32 的闪存模块由:主存储器、信息块和闪存存储器接口寄存器等 3 部分组成。 主存储器,该部分用来存放代码和数据常数(如 const 类型的数据)。对于大容量产品,其被划分为 256 页,每页 2K 字节。注意,小容量和中容量产品则每页只有 1K 字节。从上图可以看出主存储器的起始地址就是0X08000000, B0、B1 都接 GND 的时候,就是从 0X08000000开始运行代码的。 信息块,该部分分为 2 个小部分,其中启动程序代码,是用来存储 ST 自带的启动程序,用于串口下载代码,当 B0 接 V3.3,B1 接 GND 的时候,运行的就是这部分代码。用户选择字节,则一般用于配置写保护、读保护等功能。 闪存存储器接口寄存器,该部分用于控制闪存读写等,是整个闪存模块的控制机构。对主存储器和信息块的写入由内嵌的闪存编程/擦除控制器(FPEC)管理;编程与擦除的高电压由内部产生。 在执行闪存写操作时,任何对闪存的读操作都会锁住总线,在写操作完成后读操作才能正确地进行;既在进行写或擦除操作时,不能进行代码或数据的读取操作。

闪存的编程和擦除

STM32 的闪存编程是由 FPEC(闪存编程和擦除控制器)模块处理的,这个模块包含 7 个 32 位寄存器,他们分别是:

  • FPEC 键寄存器(FLASH_KEYR)
  • 选择字节键寄存器(FLASH_OPTKEYR)
  • 闪存控制寄存器(FLASH_CR)
  • 闪存状态寄存器(FLASH_SR)
  • 闪存地址寄存器(FLASH_AR)
  • 选择字节寄存器(FLASH_OBR)
  • 写保护寄存器(FLASH_WRPR) STM32 复位后,FPEC 模块是被保护的,不能写入 FLASH_CR 寄存器;通过写入特定的序列到 FLASH_KEYR 寄存器可以打开 FPEC 模块,只有在写保护被解除后,我们才能操作相关寄存器。 STM32 闪存的编程每次必须写入 16 位(不能单纯的写入 8 位数据哦!),当 FLASH_CR 寄存器的 PG 位为’1’时,在一个闪存地址写入一个半字将启动一次编程;写入任何非半字的数据,FPEC 都会产生总线错误。在编程过程中(BSY 位为’1’),任何读写闪存的操作都会使 CPU暂停,直到此次闪存编程结束。 同样,STM32 的 FLASH 在编程的时候,也必须要求其写入地址的 FLASH 是被擦除了的(也就是其值必须是 0XFFFF),否则无法写入,在FLASH_SR 寄存器的 PGERR 位将得到一个警告。

在我们日常的开发中STM32的Flash擦除最常用的就是页擦除,所以我们在这里着重介绍一下页擦除。 STM32 的页擦除顺序为:

  • 检查 FLASH_CR 的 LOCK 是否解锁,如果没有则先解锁
  • 检查 FLASH_SR 寄存器的 BSY 位,以确认没有其他正在进行的闪存操作
  • 设置 FLASH_CR 寄存器的 PER 位为’1’
  • 用 FLASH_AR 寄存器选择要擦除的页 -设置 FLASH_CR 寄存器的 STRT 位为’1’
  • 等待 BSY 位变为’0’
  • 读出被擦除的页并做验证 该寄存器我们本章只用到了它的 LOCK、STRT、PER 和 PG 等 4 个位。 LOCK 位,该位用于指示 FLASH_CR 寄存器是否被锁住,该位在检测到正确的解锁序列后,硬件将其清零。在一次不成功的解锁操作后,在下次系统复位之前,该位将不再改变。 STRT 位,该位用于开始一次擦除操作。在该位写入 1 ,将执行一次擦除操作。 PER 位,该位用于选择页擦除操作,在页擦除的时候,需要将该位置1。 PG 位,该位用于选择编程操作,在往 FLASH 写数据的时候,该位需要置1。

Flash擦除的标准库函数

  1. 解锁函数:void FLASH_Unlock(void); 对 FLASH 进行写操作前必须先解锁,解锁操作也就是必须在 FLASH_KEYR 寄存器写入特定的序列,固件库函数实现很简单:只需要直接调用 FLASH_Unlock();即可。
  2. 锁定函数:void FLASH_Lock(void); 有解锁当然就有上锁,为了保护Flash,读写和擦除全部需要的Flash后需要上锁,只需要调用: FLASH_Lock();
  3. 擦除函数 固件库我们主要使用两个 FLASH 擦除函数: FLASH_Status FLASH_ErasePage(uint32_t Page_Address); FLASH_Status FLASH_EraseAllPages(void); 顾名思义,第一个函数是页擦除函数,根据页地址擦除特定的页数据。 第二个函数是擦除所有的页数据。
  4. 获取 FLASH 状态 主要是用的函数是:FLASH_Status FLASH_GetStatus(void); 返回值是通过枚举类型定义的,分别为: FLASH_BUSY = 1,//忙 FLASH_ERROR_PG,//编程错误 FLASH_ERROR_WRP,//写保护错误 FLASH_COMPLETE,//操作完成 FLASH_TIMEOUT//操作超时
  5. 等待操作完成函数 在执行闪存写操作时,任何对闪存的读操作都会锁住总线,在写操作完成后读操作才能正确地进行;既在进行写或擦除操作时,不能进行代码或数据的读取操作。 所以在每次操作之前,我们都要等待上一次操作完成这次操作才能开始。使用的函数是:FLASH_Status FLASH_WaitForLastOperation(uint32_t Timeout)入口参数为等待时间,返回值是 FLASH 的状态,这个很容易理解,这个函数本身我们在固件库中使用得不多,但是在固件库函数体中间可以多次看到。

软件设计

直接使用固件库函数擦除当前地址所在的内容

直接使用固件库擦除选定的地址的内容,每次会擦除选定地址的当前页。 注意:这里有一个很容易混淆的点,擦除当前页,并不是擦除从这个地址之后的一页,而是STM32规定的该地址所在的页。不知道的可以看Flash的图,也可以自己计算,其实就是0x0800 0000 每次加2k(中小容量加1K),比如0x0800 0810,此时就会擦除0x0800 0800-0x0800 0FFF的所有内容,而不是0x0800 0810到0x0800 100F的内容。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
FLASH_Unlock();	//解锁
FLASH_ErasePage(0X08003A98);
FLASH_Lock();//上锁

此语句就可擦除0x0800 3A98所在页的全部内容。 可能有人疑惑,难道不能擦除单个字节的内容嘛,很遗憾,确实不能,并非是程序的问题,而是硬件设计,就是按页擦除。

擦除对应地址和大小的Flash

我们在开发中,不可能每次都计算用擦除多少页的地址,或者要擦除的范围是多少,这里我们就可以编写一个函数来帮我们实现。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
void STMFLASH_Erase(u32 EraseAddr,u16 NumToErase)	
{
	u32 secpos;	   //扇区地址
	u16 secoff;	   //扇区内偏移地址(16位字计算)
	u16 secremain; //扇区内剩余地址(16位字计算)	   
 	u16 i;    
	u32 offaddr;   //去掉0X08000000后的地址
	if(EraseAddr<STM32_FLASH_BASE||(EraseAddr>=(STM32_FLASH_BASE+1024*STM32_FLASH_SIZE)))return;//非法地址
	FLASH_Unlock();						//解锁
	offaddr=EraseAddr-STM32_FLASH_BASE;		//实际偏移地址.
	secpos=offaddr/STM_SECTOR_SIZE;			//扇区地址  0~127 for STM32F103RBT6
	secoff=(offaddr%STM_SECTOR_SIZE)/2;		//在扇区内的偏移(2个字节为基本单位.)
	secremain=STM_SECTOR_SIZE/2-secoff;		//扇区剩余空间大小   
	if(NumToErase<=secremain)
		secremain=NumToErase;//不大于该扇区范围
	while(1) 
	{	FLASH_ErasePage(secpos*STM_SECTOR_SIZE+STM32_FLASH_BASE);//擦除这个扇区		
		if(NumToErase==secremain)
			break; //擦除结束了
		else     //擦除未结束
		{
			secpos++;				//扇区地址增1
			secoff=0;				//偏移位置为0 	 
			EraseAddr+=secremain;	//地址偏移	   
		  NumToErase-=secremain;	//字节(16位)数递减
			if(NumToErase>(STM_SECTOR_SIZE/2))
				secremain=STM_SECTOR_SIZE/2;//下一个扇区还是擦除不完
			else secremain=NumToErase;//下一个扇区可以擦除完了
		}	 
	};	
	FLASH_Lock();//上锁
}

调用本函数就可以擦除EraseAddr地址开始,NumToErase大小的Flash了。注意NumToErase是16位,也就是半字(两个字节),如果你打算擦除1980字节的程序,NumToErase就应该是990。 因为函数中已经编写了解锁和上锁,所以就不用在使用时再加了,直接调用STMFLASH_Erase(0X0x0800 0810,2049); 这一句程序可以实现擦除0x0800 0810开始4098字节所在页的内容。 注意,这里的擦除也是所在页,并不是正好擦除该地址后面4098字节的内容。如果我们填入的起始地址不是STM32设定的某页的起始地址,那么擦除的时候,就会也把前面的一部分Flash内容进行擦除,比如STMFLASH_Erase(0X0x0800 0810,2049); 就是擦除了0x0800 0800-0x0800 0FFF(第二页)和0x0800 1000-0x0801 17FF(第三页)的所有内容。 可以看出来,虽然我们只想擦除0X0x0800 0810后4098个字节的全部内容,但我们不可避免的擦除了4K的全部内容,编写程序时,需要注意。

如何查看Flash的内容

Keil的软件调试中,有专门可以查看所连接的板子的Flash的内容。 首先我们点击调试按钮,如下图。

之后我们在View中找到memory Windows,选择memory1。之后我们就可以看到右下角出现Flash的内容,第一次打开是空白,因为还没有选定地址,会出现下图内容。

我们在地址处填写想查看的地址,比如查看0x0800 3A98,输入完成后点击回车,就会出现Flash地址对应的内容。

原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。

如有侵权,请联系 cloudcommunity@tencent.com 删除。

原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。

如有侵权,请联系 cloudcommunity@tencent.com 删除。

评论
登录后参与评论
3 条评论
热度
最新
666666666666
666666666666
回复回复点赞举报
777777777777777777
777777777777777777
回复回复点赞举报
6666666666666666666666
6666666666666666666666
回复回复点赞举报
推荐阅读
编辑精选文章
换一批
【STM32】硬件资源及芯片介绍
以精英板STM32F103为例。STM32是Cortex M3架构,拥有更强劲的性能、更高的代码密度、位带操作、可嵌套中断、低成 本、低功耗等众多优势。
DevFrank
2024/07/24
2140
分享STM32 FLASH 擦除(以及防止误擦除程序代码)、写入
编译环境:我用的是(Keil)MDK4.7.2 stm32库版本:我用的是3.5.0 一、本文不对FLASH的基础知识做详细的介绍,不懂得地方请查阅有关资料。   对STM32 内部FLASH进行编程操作,需要遵循以下流程:   FLASH解锁   清除相关标志位   擦除FLASH(先擦除后写入的原因是为了工业上制作方便,即物理实现方便)   写入FLASH   锁定FLASH 实例: #define FLASH_PAGE_SIZE ((uint16_t)0x400) //如果一页为1K大小 #define WRITE_START_ADDR ((uint32_t)0x08008000)//写入的起始地址 #define WRITE_END_ADDR ((uint32_t)0x0800C000)//结束地址 uint32_t EraseCounter = 0x00, Address = 0x00;//擦除计数,写入地址 uint32_t Data = 0x3210ABCD;//要写入的数据 uint32_t NbrOfPage = 0x00;//记录要擦除的页数 volatile FLASH_Status FLASHStatus = FLASH_COMPLETE;/*FLASH擦除完成标志*/ void main() { /*解锁FLASH*/  FLASH_Unlock(); /*计算需要擦除FLASH页的个数 */  NbrOfPage = (WRITE_END_ADDR - WRITE_START_ADDR) / FLASH_PAGE_SIZE; /* 清除所有挂起标志位 */   FLASH_ClearFlag(FLASH_FLAG_EOP | FLASH_FLAG_PGERR | FLASH_FLAG_WRPRTERR); /* 擦除FLASH 页*/  for(EraseCounter = 0; (EraseCounter < NbrOfPage) && (FLASHStatus == FLASH_COMPLETE); EraseCounter++)    {    FLASHStatus = FLASH_ErasePage(WRITE_START_ADDR + (FLASH_PAGE_SIZE * EraseCounter));    } /* 写入FLASH */  Address = WRITE_START_ADDR;  while((Address < WRITE_END_ADDR) && (FLASHStatus == FLASH_COMPLETE))    {    FLASHStatus = FLASH_ProgramWord(Address, Data);    Address = Address + 4;    } /* 锁定FLASH */  FLASH_Lock(); } 二、FLASH 擦除(以及防止误擦除程序代码) 1、擦除函数 FLASH_Status FLASH_ErasePage(u32 Page_Address)只要()里面的数是flash第xx页中对应的任何一个地址!就是擦除xx页全部内容! 防止误擦除有用程序代码的方法 方法一:首先要计算程序代码有多少,把FLASH存取地址设置在程序代码以外的地方,这样就不会破坏用户程序。原则上从0x0800 0000 + 0x1000 以后的FLASH空间都可以作为存储使用。如果代码量占了 0x3000, 那么存储在 0x0800 0000+ 0x4000 以后的空间就不会破坏程序了。 方法二:先在程序中定义一个const 类型的常量数组,并指定其存储位置(方便找到写入、读取位置),这样编译器就会分配你指定的空间将常量数组存入FLASH中。当你做擦除。读写操作时,只要在这个常量数组所在的地址范围就好。   const uint8_t table[10] __at(0x08010000) = {0x55} ;   MDK3.03A开始就支持关键字 __at() 。   需要加#include <absacc.h> 方法三:在程序中定义一个const 类型的常量数组,无需指定其存储位置。只要定义一个32位的变量存储这个数组的FLASH区地址就行。   uint32_t address;//STM32的地址是32位的   const uint8_t imageBuffer[1024] = {0,1,2,3,4,5,6,7};   address = (uint32_t) imageBuffer;/*用强制类型转换的方式,可以把FLASH中存储的imageBuffer[1024]的地址读到RAM中的变量address 里,方便找到写入、读取位
用户4645519
2020/09/07
3.5K0
【STM32F429开发板用户手册】第47章 STM32F429的SPI 总线应用之SPI Flash的MDK下载算法制作
最新教程下载:http://www.armbbs.cn/forum.php?mod=viewthread&tid=93255 第47章 STM32F429的SPI 总线应用之SPI Fla
Simon223
2020/11/26
8270
【STM32F429开发板用户手册】第47章       STM32F429的SPI 总线应用之SPI Flash的MDK下载算法制作
【STM32笔记】使用STM32内部Flash额外的空间来存储数据
STM32 芯片内部的 FLASH 存储器,主要用于存储我们代码。如果内部FLASH存储完我们的代码还有剩余的空间,那么这些剩余的空间我们就可以利用起来,存储一些需要掉电保存的数据。
正念君
2019/11/27
7.1K0
【STM32H7教程】第71章 STM32H7的内部Flash应用之模拟EEPROM
完整教程下载地址:http://www.armbbs.cn/forum.php?mod=viewthread&tid=86980 第71章       STM32H7的内部Flash应用之模拟EEPR
Simon223
2020/03/11
2.1K0
(39)STM32——FLASH闪存
        通过这两个步骤,即可解锁 FLASH_CR,如果写入错误,那么 FLASH_CR 将被锁定,直到下次复位后才可以再次解锁。
小点点
2022/12/12
1.4K0
(39)STM32——FLASH闪存
如何使用STM32单片机记录硬件复位次数?
STM32 单片机的备份寄存器是一种特殊的寄存器,即使在主电源 VDD 掉电的情况下,其内容也可以由备份电池供电来保持数据。
不脱发的程序猿
2025/01/19
1320
如何使用STM32单片机记录硬件复位次数?
【STM32H7教程】第70章 STM32H7的内部Flash基础知识和HAL库API
完整教程下载地址:http://www.armbbs.cn/forum.php?mod=viewthread&tid=86980 第70章       STM32H7的内部Flash基础知识和HAL库
Simon223
2020/03/11
1.9K0
M-Arch(番外7)GD32L233评测-FLASH读写
闪存控制器(FMC),提供了片上闪存需要的所有功能。一般而言,MCU的Flash包括4个部分:
滚神大人
2022/03/22
6020
M-Arch(番外7)GD32L233评测-FLASH读写
M-Arch(4)第三个示例:Flash读写操作
本文我们将回顾下FMC的知识,并给出Flash读写的接口设计和示例,这在设计升级程序时十分重要。
滚神大人
2021/09/10
8320
stm32入门教程_单片机STM32
首先你得知道学习stm32,实际就是在学ARM内核,stm32内核就是ARM的; ARM使用RISC精简指令集模式开发; ARM公司全称Acorn Risc Machine; ARM处理器本身是32位设计,但也具备16位指令集,与等价32位处理器相比代码量节省35%,还能具备32位处理器的所有优势; ARM公司是英国的; ARM公司是全球知识产权提供商,他不做生产制造; 全世界超过95%的智能手机和平板电脑都采用ARM架构; 同时日本软银收购了ARM公司,成为物联网的领军者; ARM11系列就是应用到手机上的芯片,包括ARMv6、ARM6T2、ARMv6KZ、ARMv6K; ARM12系列时候,名字就不叫ARM12了,叫成Cortex; 杨桃首页:
全栈程序员站长
2022/10/29
1.6K0
stm32入门教程_单片机STM32
单片机STM32学习笔记之寄存器映射详解
  我们知道,存储器本身没有地址,给存储器分配地址的过程叫存储器映射,那什么叫寄存器映射?寄存器到底是什么?   在存储器Block2 这块区域,设计的是片上外设,它们以四个字节为一个单元,共32bi
用户6754675
2019/12/25
2.1K0
【STM32H7教程】第67章 STM32H7的系统bootloader基础知识
完整教程下载地址:http://www.armbbs.cn/forum.php?mod=viewthread&tid=86980 第67章       STM32H7的系统bootloader基础知识
Simon223
2020/03/06
2.4K0
[硬件]关于SPI Flash那些你不知道的事儿
以华邦W25Q128为例,详解SPI Flash的特点,读写注意事项,和地址范围等。
单片机点灯小能手
2020/07/17
6K0
[硬件]关于SPI Flash那些你不知道的事儿
单片机内部FLASH的字节操作
一般32位单片机的内部FALSH是不支持字节操作的,有的可以按字节读取,但是不能按字节写入。
知否知否应是绿肥红瘦
2025/02/19
900
单片机内部FLASH的字节操作
【不是问题的问题】为什么STM32的Flash地址要设置到0x08000000
我们言简意赅的普及下这个知识点,争取让大家不伤脑细胞 一、背景知识: M3,M4内核芯片上电复位后,要固定从0x0000 0000地址读取中断向量表,获取复位中断服务程序的入口地址后,进入复位中断服务程序,其中0x0000 0000是栈顶地址,0x0000 0004存的是复位中断服务程序地址。
Simon223
2021/11/01
2.2K0
【不是问题的问题】为什么STM32的Flash地址要设置到0x08000000
stm32flash的读写特性
4、一般都是在最后几页进行数据保存的,确保数据量不超过flash的的大小,比如f103大容量是2k字节,其实一个扇区只能写入1k数量的2字节的数据。
用户4645519
2020/09/07
5550
STM32寄存器讲解
原理讲解 芯片讲解 STM32F103芯片 我们看到的 STM32 芯片是已经封装好的成品,主要由内核和片上外设组成。若与电脑类比,内核与外设就如同电脑上的 CPU与主板、内存、显卡、硬盘的关系。 STM32F103采用的是 Cortex-M3内核,内核即 CPU,由 ARM公司设计。ARM公司并不生产芯片,而是出售其芯片技术授权。芯片生产厂商(SOC)如 ST、TI、Freescale,负责在内核之外设计部件并生产整个芯片,这些内核之外的部件被称为核外外设或片上外设。如 GPIO、USART(串口)、I2C、SPI等都叫做片上外设。(采用野火官方的介绍)。
跋扈洋
2021/02/02
1.7K0
STM32寄存器讲解
M-Arch(雅特力M4)【AT-START-F425测评】No.05 FLASH
今儿这个是flash的读写测试,AT32F425R8T7-7拥有64K的flash空间。
滚神大人
2022/06/09
6050
M-Arch(雅特力M4)【AT-START-F425测评】No.05 FLASH
【stm32f407】SPI实验 驱动W25Q128「建议收藏」
SPI 是英语SerialPeripheral interface的缩写,顾名思义就是串行外围设备接口。是Motorola首先在其MC68HCXX系列处理器上定义的。SPI接口主要应用在EEPROM,FLASH,实时时钟,AD转换器,还有数字信号处理器和数字信号解码器之间。SPI,是一种高速的,全双工,同步的通信总线,并且在芯片的管脚上只占用四根线,节约了芯片的管脚,同时为PCB的布局上节省空间,提供方便,正是出于这种简单易用的特性,现在越来越多的芯片集成了这种通信协议,STM32F4也有SPI接口。下面我们看看SPI的内部简明图
Java架构师必看
2022/04/11
1.8K0
【stm32f407】SPI实验 驱动W25Q128「建议收藏」
推荐阅读
相关推荐
【STM32】硬件资源及芯片介绍
更多 >
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档