#stm32整理(一)flash读写

以这篇未开始我将进行stm32学习整理为期一个月左右完成stm32知识学习整理内容顺序没有一定之规写到哪想到哪想到哪写到哪,主要是扫除自己知识上的盲区完成一些基本外设操作。

以stm32f07为例子进行flash读写操作

stm32flash简介

参考资料正点原子和野火开发手册 stm32f4中文参考手册和datasheet

Flash 接口可管理 CPU 通过 **AHB I-Code(指令指令总线) 和 D-Code (数据总线)**对 Flash 进行的访问。该接口可针对 Flash 执行擦除和编程操作,并实施读写保护机制。Flash 接口通过指令预取和缓存机制加速
代码执行。
关于这两条总线先不细说参考链接 AHB I-Code(指令指令总线) 和 D-Code (数据总线)请参考 Cortex-M3 I-Code,D-Code,系统总线及其他总线接口

主要特性

● Flash 读操作
● Flash 编程/擦除操作
● 读/写保护
● I-Code 上的预取操作
● I-Code 上的 64 个缓存(128 位宽)
● D-Code 上的 8 个缓存(128 位宽)

结构框图

在这里插入图片描述

Flash 具有以下主要特性:
● 对于 STM32F40x 和 STM32F41x,容量高达 1 MB;对于 STM32F42x 和 STM32F43x,
容量高达 2 MB
● 128 位宽数据读取
字节、半字、字和双字数据写入
扇区擦除与全部擦除
● 存储器组织结构
Flash 结构如下:
— 主存储器块,分为 4 个 16 KB 扇区、1 个 64 KB 扇区和 7 个 128 KB 扇区
— 系统存储器,器件在系统存储器自举模式下从该存储器启动
— 512 字节 OTP(一次性可编程),用于存储用户数据
OTP 区域还有 16 个额外字节,用于锁定对应的 OTP 数据块。
— 选项字节,用于配置读写保护、BOR 级别、软件/硬件看门狗以及器件处于待机或
停止模式下的复位。
● 低功耗模式(有关详细信息,请参见参考手册的“电源控制 (PWR)”部分)

在这里插入图片描述

主存储器
一般我们说 STM32 内部 FLASH 的时候,都是指这个主存储器区域,它是存储用户应
用程序的空间,芯片型号说明中的 256K FLASH、512K FLASH 都是指这个区域的大
小。

系统存储区
系统存储区是用户不能访问的区域,它在芯片出厂时已经固化了启动代码,它负责实现串口、USB 以及 CAN 等 ISP 烧录功能 ISP烧录就是,芯片通过某些方式进入芯片内部预置的ISP升级程序,开启升级功能,然后与外部通信,然后通过相关的协议,完成程序区的擦除,写入,校验和相关的配置等一些列操作的过程.

选项字节
选项字节用于配置 FLASH 的读写保护、待机/停机复位、软件/硬件看门狗等功能,这
部分共 16 字节。可以通过修改 FLASH 的选项控制寄存器修改。

OTP介绍
OTP:one-time programmable,只允许一次编程,也就是只能从1写0,不能从0写1。这里可能有人要问,这不是flash的特性么?需要注意的是,flash是允许擦除的,是允许从0写1的。而OTP不允许擦除,就算在ICP烧录代码时,也不会丢。

3接口

在这里插入图片描述

4读写操作

执行任何 Flash 编程操作(擦除或编程)时,CPU 时钟频率 (HCLK) 不能低于 1 MHz。如果
在 Flash 操作期间发生器件复位,无法保证 Flash 中的内容。
在对 STM32F4xx 的 Flash 执行写入或擦除操作期间,任何读取 Flash 的尝试都会导致总线
阻塞。只有在完成编程操作后,才能正确处理读操作。这意味着,写/擦除操作进行期间不能
从 Flash 中执行代码或数据获取操作。也就是说读写之间不能操作

4.1解锁

复位后你先解锁才能操作
复位后,Flash 控制寄存器 (FLASH_CR) 不允许执行写操作,以防因电气干扰等原因出现对
Flash 的意外操作。此寄存器的解锁顺序如下:

  1. 在 Flash 密钥寄存器 (FLASH_KEYR) 中写入 KEY1 = 0x45670123
  2. 在 Flash 密钥寄存器 (FLASH_KEYR) 中写入 KEY2 = 0xCDEF89AB
  3. 如果顺序出现错误,将返回总线错误并锁定 FLASH_CR 寄存器,直到下一次复位。
    也可通过软件将 FLASH_CR 寄存器中的 LOCK 位置为 1 来锁定 FLASH_CR 寄存器。

当 FLASH_SR 寄存器中的 BSY 位为 1 时,将不能在写模式下访问 FLASH_CR 寄存器。
BSY 位为 1 时,对该寄存器的任何写操作尝试都会导致 AHB 总线阻塞,直到 BSY 位清零。

4.2擦除/编程位数

通过 FLASH_CR 寄存器中的 PSIZE 字段配置并行位数。并行位数表示每次对 Flash 进行写
操作时将编程的字节数。PSIZE 受限于电源电压以及是否使用外部 VPP 电源。因此,在进行
任何编程/擦除操作前,必须在 FLASH_CR 寄存器中对其进行正确配置。
编程就是读写 擦除受外部电压影响
Flash 擦除操作只能针对扇区或整个 Flash(批量擦除)执行。擦除时间取决于 PSIZE 编程
值。有关擦除时间的详细信息,请参见器件数据手册的电气特性部分。
在这里插入图片描述
DW:64 W:32 HW:16 Byte:8位

写到这我想先写一个关于stm32程序存在了哪这涉及到了程序编译过程我先写这个但是排在第二篇吧!
#stm32整理(二)关于MDK的编译过程及文件类型全解

4.3擦除

Flash 擦除操作可针对扇区或整个 Flash(批量擦除)执行。执行批量擦除时,不会影响 OTP 扇区或配置扇区。
扇区擦除
扇区擦除的具体步骤如下:

  1. 检查 FLASH_SR 寄存器中的 BSY 位,以确认当前未执行任何 Flash 操作
  2. 在 FLASH_CR 寄存器中,将 SER 位置 1,并从主存储块的 12 个 (STM32F405xx/07xx
    和 STM32F415xx/17xx) 或 24 个 (STM32F42xxx 和 STM32F43xxx) 扇区中选择要擦除
    的扇区 (SNB)
  3. 将 FLASH_CR 寄存器中的 STRT 位置 1
  4. 等待 BSY 位清零
    批量擦除
    要执行批量擦除,建议采用以下步骤:
  5. 检查 FLASH_SR 寄存器中的 BSY 位,以确认当前未执行任何 Flash 操作
  6. 将 FLASH_CR 寄存器中的 MER 位置 1(STM32F405xx/07xx 和 STM32F415xx/17xx
    器件)
  7. 将 FLASH_CR 寄存器中的 MER 和 MER1 位置 1(STM32F42xxx 和 STM32F43xxx
    器件)
    . 将 FLASH_CR 寄存器中的 STRT 位置 1
  8. 等待 BSY 位清零
    注意: 如果 FLASH_CR 寄存器中的 MERx 位和 SER 位均置为 1,则无法执行扇区擦除和批量擦除。

这里写一段flash 扇区删除的代码

Flash 状态寄存器 (FLASH_SR)
Flash status register
Flash 状态寄存器提供正在执行的编程和擦除操作的相关信息。
偏移地址:0x0C
复位值:0x0000 0000
访问:无等待周期,按字、半字和字节访问
在这里插入图片描述

在这里插入图片描述
在这里插入图片描述
用于 STM32F405xx/07xx 和 STM32F415xx/17xx 的 Flash 控制寄存器
(FLASH_CR)

Flash control register
Flash 控制寄存器用于配置和启动 Flash 操作。
偏移地址:0x10
复位值:0x8000 0000
访问:当前未执行任何 Flash 操作时无等待周期,按字、半字和字节访问。

注意FLASH的值擦完是FF写的时候是把1变0
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

//1. 检查 FLASH_SR 寄存器中的 BSY 位,以确认当前未执行任何 Flash 操作 这相当于第一步
/*** @brief       得到FLASH的错误状态* @param       无* @retval      执行结果*   @arg       0   : 已完成*   @arg       其他 : 错误编号*/
static uint8_t stmflash_get_error_status(void)
{uint32_t res = 0;res = FLASH->SR;if (res & (1 << 16)) return 1;  /* BSY=1, 繁忙 */if (res & (1 << 7))  return 2;  /* PGSERR=1,编程序列错误 */if (res & (1 << 6))  return 3;  /* PGPERR=1,编程并行位数错误 */if (res & (1 << 5))  return 4;  /* PGAERR=1,编程对齐错误 */if (res & (1 << 4))  return 5;  /* WRPERR=1,写保护错误 */return 0;   /* 没有任何状态/操作完成. */
}
/*** @brief       等待操作完成* @param       time : 要延时的长短* @retval      执行结果*   @arg       0   : 已完成*   @arg       0XFF: 超时*   @arg       其他 : 错误编号*/
static uint8_t stmflash_wait_done(uint32_t time)
{uint8_t res;do{res = stmflash_get_error_status();  if (res != 1){break;      /* 非忙, 无需等待了, 直接退出 */}time--;} while (time);if (time == 0)res = 0XFF;   /* 超时 */return res;
}/*** @brief       获取某个地址所在的flash扇区* @param       faddr   : flash地址* @retval      0~11, 即addr所在的扇区*/
static uint8_t stmflash_get_flash_sector(uint32_t addr)
{if (addr < ADDR_FLASH_SECTOR_1)return 0;else if (addr < ADDR_FLASH_SECTOR_2)return 1;else if (addr < ADDR_FLASH_SECTOR_3)return 2;else if (addr < ADDR_FLASH_SECTOR_4)return 3;else if (addr < ADDR_FLASH_SECTOR_5)return 4;else if (addr < ADDR_FLASH_SECTOR_6)return 5;else if (addr < ADDR_FLASH_SECTOR_7)return 6;else if (addr < ADDR_FLASH_SECTOR_8)return 7;else if (addr < ADDR_FLASH_SECTOR_9)return 8;else if (addr < ADDR_FLASH_SECTOR_10)return 9;else if (addr < ADDR_FLASH_SECTOR_11)return 10;return 11;
}static uint8_t stmflash_erase_sector(uint32_t saddr)
{uint8_t res = 0;res = stmflash_wait_done(0XFFFFFFFF);   /* 等待上次操作结束 */ //1. 检查 FLASH_SR 寄存器中的 BSY 位,以确认当前未执行任何 Flash 操作if (res == 0){FLASH->CR &= ~(3 << 8);             /* 清除PSIZE原来的设置 */ //默认8位编程FLASH->CR |= 2 << 8;                /* 设置为32bit宽,确保VCC=2.7~3.6V之间!! */FLASH->CR &= ~(0X1F << 3);          /* 清除原来的设置 */FLASH->CR |= saddr << 3;            /* 设置要擦除的扇区 */FLASH->CR |= 1 << 1;                /* 扇区擦除 */ // SER置1 激活扇区擦除FLASH->CR |= 1 << 16;               /* 开始擦除 */res = stmflash_wait_done(0XFFFFFFFF);   /* 等待操作结束 */ //位 16 STRT:启动 (Start) 该位置 1 后可触发擦除操作。//  **注意这里没有设置MER,MER是按块擦除这里是按扇区擦除**if (res != 1)                       /* 非忙 */{FLASH->CR &= ~(1 << 1);         /* 清除扇区擦除标志 */// SER:扇区擦除 (Sector Erase)}}return res;
}

4.4编程(写)

Flash 编程顺序如下:

  1. 检查 FLASH_SR 中的 BSY 位,以确认当前未执行任何主要 Flash 操作。
  2. 将 FLASH_CR 寄存器中的 PG 位置 1。
  3. 针对所需存储器地址(主存储器块或 OTP 区域内)执行数据写入操作:
    — 并行位数为 x8 时按字节写入
    — 并行位数为 x16 时按半字写入
    — 并行位数为 x32 时按字写入
    — 并行位数为 x64 时按双字写入
  4. 等待 BSY 位清零

注意: 把 Flash 的单元从“1”写为“0”时,无需执行擦除操作即可进行连续写操作。把 Flash 的
单元从“0”写为“1”时,则需要执行 Flash 擦除操作。
如果同时发出擦除和编程操作请求,首先执行擦除操作。

这里放一段写代码

//从指定地址开始写入指定长度的数据
//WriteAddr:起始地址(此地址必须为2的倍数!!)
//pBuffer:数据指针
//NumToWrite:半字(16位)数(就是要写入的16位数据的个数.)
void Flash_Write(uint32_t WriteAddr, uint16_t *pBuffer, uint16_t NumToWrite)
{uint8_t status = 0;uint32_t addrx =0;uint32_t endaddr=0;uint8_t sector;if(WriteAddr<STM32_FLASH_BASE||WriteAddr%2||WriteAddr>(STM32_FLASH_BASE+STM32_FLASH_SIZE))//判断地址是否符合这里我们需注意基地址和flash大小可以根据手册查{return;//return 语句是提前结束函数的唯一办法。return 后面可以跟一份数据,表示将这份数据返回到函数外面;return 后面也可以不跟任何数据,表示什么也不返回,仅仅用来结束函数。}FLASH_Unlock();//解锁之前介绍过 直接用库函数中的函数FLASH_DataCacheCmd(DISABLE);//关闭数据缓存,这里不关闭数据缓存擦除时可能会发生缓存不一致现象 这里我没验证相关可以参考 https://shequ.stmicroelectronics.cn/thread-621109-1-1.htmladdrx=WriteAddr;//开始地址endaddr=WriteAddr+NumToWrite*2;//结束地址sector=stmflash_get_flash_sector(addrx);//获取扇区if(addrx<0X1FFF0000)//地址0x1FFF 0000是系统存储器的地址{while(addrx<endaddr){if(Flash_ReadHalfWord(addrx)!=0XFFFF)//读到非零擦除{status = stmflash_erase_sector(sector);if(status)break;}else{addrx+=2;}}}if(status==0){status = stmflash_wait_done(0XFFFFF);//这一句其实没用while(WriteAddr<endaddr){if(stmflash_wait_done(0XFFFFF)==0)status=FLASH_ProgramHalfWord(WriteAddr,*pBuffer);//半字写入elsebreak;WriteAddr+=2;pBuffer++;}}   									//这里缺一点如果写入错误应该如何判断写入问题FLASH_DataCacheCmd(ENABLE);//使能数据缓冲FLASH_Lock();//上锁}

读函数

//读取指定地址的半字(16位数据)
//faddr:读地址
//返回值:对应数据.
static uint16_t Flash_ReadHalfWord(uint32_t faddr)
{return *(vu16 *)faddr;//(vu16 *)将32位地址强制转换为16为__IO uint16_t 16位地址 第二个*才是返回该地址所存储的值。
}

关于f1的和这个不一样先不写了先写到这里但是流程是差不多的。

1、这里注意我们在进行擦除和写入时操作的基本单元是sector 所以在进行操作时哪怕对某一个地址中的值进行修改需要先将扇区中的值读出来然后更改这个值再写进去,所以当数据量特别大的时候我们也需要一个特别大的全局buffer现将值读出来有点不实用,这里我想了一个思路还是要计算当前要写的字节在sector的大小然后把他给到一个临时buffer中然后修改临时buffer的值将buffer再写回sector中,但是同样存在临时buffer太大栈溢出导致硬件中断错误
2、注意这里我们写的数据没有超过该secotr的大小如果超过了那么就坏了,还要根据扇区大小不同判断剩余字节数

先写到这吧

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

如何选择传感器输出模式——电流输出还是电压输出?

一 背景及挑战 传感器在汽车测试系统中发挥着采集和传输信息的作用&#xff0c;可称之为汽车的“神经元”。 按照功能可将传感器分为压力传感器、流量传感器、温湿度传感器和电流传感器等。传感器的主要指标是精度、测量范围和响应时间等。在满足指标的情况下&#xff0c;通常…

笔记软件Notability mac中文版软件功能

Notability mac是一款帮助用户备注文件的得力工具&#xff0c;Notability Mac版可用于注释文稿、草拟想法、录制演讲、记录备注等。它将键入、手写、录音和照片结合在一起&#xff0c;便于您根据需要创建相应的备注。 Mac Notability mac中文版软件功能 将手写&#xff0c;照片…

springboot和flask整合nacos,使用openfeign实现服务调用,使用gateway实现网关的搭建(附带jwt续约的实现)

环境准备&#xff1a; 插件版本jdk21springboot 3.0.11 springcloud 2022.0.4 springcloudalibaba 2022.0.0.0 nacos2.2.3&#xff08;稳定版&#xff09;python3.8 nacos部署&#xff08;docker&#xff09; 先创建目录&#xff0c;分别创建config&#xff0c;logs&#xf…

APISpace 全国快递物流地图轨迹查询API接口案例代码

1.全国快递物流地图轨迹查询接口详解 1.1 接口请求 请求方式&#xff1a;POST请求地址&#xff1a;https://eolink.o.apispace.com/wldtgj1/paidtobuy_api/trace_map请求头&#xff1a; 标签必填说明X-APISpace-Token是鉴权私钥&#xff0c;登陆 APISpace 后在管理后台的[访…

证照之星XE专业版下载专业证件照制作工具

值得肯定的是智能背景替换功能&#xff0c;轻松解决背景处理这一世界难题。不得不提及的是新增打印字体设置&#xff0c;包含字体选择、字号大小、字体颜色等。不同领域的应用证明了万能制作&#xff0c;系统支持自定义证照规格&#xff0c;并预设了17种常用的证件照规格。人所…

紧急:发现NGINX Ingress Controller for Kubernetes中的新安全漏洞

导语 大家好&#xff0c;今天我要向大家紧急报告一则消息&#xff1a;我们在NGINX Ingress Controller for Kubernetes中发现了三个新的安全漏洞&#xff01;这些漏洞可能被黑客利用&#xff0c;从集群中窃取机密凭据。在本文中&#xff0c;我们将详细介绍这些漏洞的细节&#…

ROCKCHIP ~ Camera 闪光灯

一、闪光灯基本原理 工作模式 Camera flash led分flash和torch两种模式。 flash&#xff1a; 拍照时上光灯瞬间亮一下&#xff0c;电流比较大&#xff0c;目前是1000mA&#xff0c;最大电流不能超过led最大承受能力 torch&#xff1a; 只用于录video或者拿led当手电筒的情况&…

python多环境并存

1. 现况简介 1.1 本人windows所存Python版本 Python 2.7 Python 3.6 Python 3.7 1.2 Python 各版本路径如下 Python 2.7Python 3.6Python 3.7C:\Server\Python27C:\Server\Python36C:\Server\Python37 1.3 系统环境变量配置如下 2. 解决方案 2.1 进入目录 cd C:\Server…

qt5工程打包成可执行exe程序

一、编译生成.exe 1.1、在release模式下编译生成.exe 1.2、建一个空白文件夹package&#xff0c;再将在release模式下生成的.exe文件复制到新建的文件夹中package。 1.3、打开QT5的命令行 1.4、用命令行进入新建文件夹package&#xff0c;使用windeployqt对生成的exe文件进行动…

84.在排序数组中查找元素的第一个和最后一个位置(力扣)

目录 问题描述 代码解决以及思想 知识点 问题描述 代码解决以及思想 class Solution { public:vector<int> searchRange(vector<int>& nums, int target) {int left 0; // 定义左边界int right nums.size() - 1; // 定义右…

2023年10月13日,美国材料与试验协会(ASTM)发布了新版玩具安全标准ASTM F963-23

新标准发布 2023年10月13日&#xff0c;美国材料与试验协会&#xff08;ASTM&#xff09;发布了新版玩具安全标准ASTM F963-23。 主要更新内容 与ASTM F963-17相比&#xff0c;此次更新包括&#xff1a;单独描述了基材重金属元素的豁免情况&#xff0c;更新了邻苯二甲酸酯的管…

【ROS入门】雷达、摄像头及kinect信息仿真以及显示

文章结构 雷达信息仿真以及显示Gazebo仿真雷达配置雷达传感器信息xacro文件集成启动仿真环境 Rviz显示雷达数据 摄像头信息仿真以及显示Gazebo仿真摄像头新建xacro文件&#xff0c;配置摄像头传感器信息xacro文件集成启动仿真环境 Rviz显示摄像头数据 kinect信息仿真以及显示Ga…

基于深度学习网络的美食检测系统matlab仿真

目录 1.算法运行效果图预览 2.算法运行软件版本 3.部分核心程序 4.算法理论概述 5.算法完整程序工程 1.算法运行效果图预览 2.算法运行软件版本 matlab2022a 3.部分核心程序 % 图像大小 image_size [224 224 3]; num_classes size(VD,2)-1;% 目标类别数量…

可视化 | 数据可视化降维算法梳理

文章目录 &#x1f4da;数据描述&#x1f407;iris&#x1f407;MNIST &#x1f4da;PCA&#x1f407;算法流程&#x1f407;图像描述 &#x1f4da;Kernel-PCA&#x1f407;算法流程&#x1f407;图像描述 &#x1f4da;MDS&#x1f407;算法流程&#x1f407;图像描述 &#…

Linux两条服务器实现相互免密登录

1.准备两台虚拟机&#xff0c;一台充当服务器端&#xff08;server&#xff09;&#xff0c;一台充当客户端&#xff08;client&#xff09; 服务器端&#xff08;server&#xff09;&#xff1a;192.168.75.139 客户端&#xff08;client&#xff09;&#xff1a;192.168.75…

Fegin ----微服务 SpringCloud

FeignClient 是一个注解&#xff0c;用于创建一个声明式的 REST 客户端&#xff0c;用于访问其他服务的 REST API。通过 Feign Client&#xff0c;可以将远程服务的 API 当做本地服务进行调用&#xff0c;Feign Client 能够自动生成 API 客户端的实现类&#xff0c;在使用 API …

node使用fs模块(一)—— 写入文件的基本使用

文章目录 前言一、写入文件的使用&#xff08;fs.writeFile&#xff09;1.参数说明2.基本使用(1)新建app.js 文件(2)代码如下(3)执行命令(4&#xff09;效果 3.写入文件的同步和异步&#xff08;1&#xff09;默认异步&#xff08;2&#xff09; 同步方法&#xff08;writeFile…

jmeter界面压测过程卡死解决思路

1、排查压测机的资源是否充足&#xff1b; 2、检查jmeter压测脚本&#xff0c;除聚合报告的所有组件关闭&#xff1b; 我在压测过程中出现频繁卡死&#xff0c;就是查看结果数和断言结果信息量过多导致&#xff1a; 3、直接用非gui界面形式&#xff0c;也就是脚本形式压测。

3D虚拟样板间场景制作软件的应用优势及价值

高端家装样板间不仅代表着高品质的家居生活&#xff0c;还是一种生活态度和品味的体现。为了让客户能够更好地体验到高端家装样板间的魅力&#xff0c;许多家装公司和房地产开发商开始尝试使用VR技术来进行样板间的拍摄和展示。通过VR全景拍摄&#xff0c;客户可以在线上进行参…

css小程序踩坑记录

写标签设置距离 一直设置不动 写个双层 设置动了 神奇 好玩