STM32F4X 内部FLASH使用
STM32F4X 内部FLASH使用
- STM32F4X 内部FLASH
- STM32F4X内部FLASH结构
- STM32F40X和STM32F41X内部FLASH结构
- STM32F42X和STM32F43X内部FLASH结构
- STM32F4X内部FLASH操作
- 例程
- internal_flash.h
- internal_flash.c
- main.c
在嵌入式开发中,经常需要实时保存一些数据。如果工程的代码量比较大的话可以使用一些外部的存储器件进行数据的保存。如果工程的代码量比较小则可以利用MCU的内部FLASH进行数据的存储。
STM32F4X 内部FLASH
当我们把程序编译好之后下载到MCU之后,下载器会把程序文件下载到MCU的内部FLASH中,内部FLASH除非人为擦除,否则会永久保存。内部FLASH除了可以保存代码之外,还可以保存数据。STM32F4X的内部FLASH有以下特点。
- STM32F40X和STM32F41X的内部FLASH容量是1MB,而STM32F42X和STM32F43X的内部FLASH容量是2MB。
- STM32F4X的内部FLASH支持字节、半字、字和双字写入。
- STM32F4X的内部FLASH支持扇区擦除和全擦除。
STM32F4X内部FLASH结构
STM32F40X和STM32F41X内部FLASH结构

可以看到STM32F40X和STM32F41X将1MB的内部FLASH分成了12个扇区,每个扇区的大小都不一样,分别是4个16KB的扇区,1个64KB的扇区和7个128KB的扇区。
STM32F42X和STM32F43X内部FLASH结构

STM32F42X和STM32F43X的内部FLASH结构跟STM32F40X和STM32F41X的差不多,只是有点细微的差别。STM32F42X和STM32F43X将内部2MB的FLASH分成了2个块,每个块是1MB,每个块分成了12个扇区,每个扇区的大小都不一样,分别是4个16KB的扇区,1个64KB的扇区和7个128KB的扇区。
STM32F4X内部FLASH操作
- 先将内部FLASH解锁
- 擦除扇区
- 往FLASH中写入数据
- 读出写入的数据进行校验
- FLASH上锁
例程
internal_flash.h
#ifndef __INTERNAL_FLASH__H
#define __INTERNAL_FLASH__H#include "stm32f4xx.h"
#include "stdio.h"#define ADDR_FLASH_SERTOR_0 (0x8000000)
#define ADDR_FLASH_SERTOR_1 (0x8004000)
#define ADDR_FLASH_SERTOR_2 (0x8008000)
#define ADDR_FLASH_SERTOR_3 (0x800C000)
#define ADDR_FLASH_SERTOR_4 (0x8010000)
#define ADDR_FLASH_SERTOR_5 (0x8020000)
#define ADDR_FLASH_SERTOR_6 (0x8040000)
#define ADDR_FLASH_SERTOR_7 (0x8060000)
#define ADDR_FLASH_SERTOR_8 (0x8080000)
#define ADDR_FLASH_SERTOR_9 (0x80A0000)
#define ADDR_FLASH_SERTOR_10 (0x80C0000)
#define ADDR_FLASH_SERTOR_11 (0x80E0000) int flash_test(void);
unsigned int get_sector_index(unsigned int start_address);
int compare_u8_data(const unsigned char *src,unsigned int start_address,int len);
int compare_u16_data(const unsigned short *src,unsigned int start_address,int len);
int compare_u32_data(const unsigned int *src,unsigned int start_address,int len);
#endif
internal_flash.c
#include "internal_flash.h"static unsigned char test_data_u8[] =
{
//随机数个数:100;每行显示个数:10;取值范围:0到255157, 115, 145, 130, 66, 251, 215, 97, 214, 233,
69, 161, 177, 204, 140, 72, 143, 247, 248, 120,
219, 190, 224, 68, 45, 233, 65, 194, 17, 98,
166, 97, 10, 49, 75, 125, 214, 170, 129, 49,
244, 229, 84, 109, 100, 111, 108, 72, 174, 63,
49, 151, 37, 236, 51, 190, 109, 29, 200, 198,
59, 64, 247, 176, 92, 127, 148, 9, 249, 239,
173, 191, 25, 11, 37, 51, 103, 98, 159, 143,
54, 140, 157, 90, 194, 98, 233, 74, 184, 111,
53, 89, 249, 161, 254, 26, 155, 68, 182, 78,
};static unsigned short test_data_u16[] =
{
//随机数个数:100;每行显示个数:10;取值范围:256到6553510301, 24195, 26063, 23491, 30667, 1467, 25287, 28733, 28285, 649,
20974, 22870, 22983, 28195, 8943, 22043, 4696, 15632, 15502, 9559,
17054, 31636, 5956, 17834, 30344, 11887, 27433, 31707, 22671, 11207,
29976, 4975, 16375, 32866, 5901, 3451, 29778, 13592, 22481, 12726,
19015, 2184, 1498, 13778, 32145, 28939, 364, 28429, 27722, 31462,
29189, 16899, 5994, 22212, 27384, 17606, 3211, 11363, 17240, 18224,
7407, 1421, 9495, 13688, 1475, 25930, 5999, 2308, 28069, 24404,
2177, 15230, 23397, 17355, 6717, 28714, 2568, 13098, 13953, 18295,
15891, 11898, 11909, 4308, 32774, 30661, 26998, 20164, 24614, 22222,
26209, 16693, 10111, 5631, 5459, 27150, 6698, 19363, 19391, 16391,
};static unsigned int test_data_u32[] =
{
//随机数个数:100;每行显示个数:10;取值范围:65536到10000077260, 82517, 87839, 66237, 89695, 72376, 96222, 92678, 95797, 93033,
80246, 71041, 65849, 67378, 98262, 66674, 82041, 77441, 81670, 96973,
83937, 83480, 75533, 69009, 76105, 91653, 83731, 84311, 86195, 87253,
93123, 75563, 95693, 82693, 81139, 82842, 83646, 88761, 89735, 68620,
77854, 96176, 71826, 87637, 91718, 72568, 84671, 71302, 70603, 94686,
77552, 75529, 86341, 96681, 77604, 87789, 84565, 75144, 86686, 78462,
78822, 67912, 67140, 80409, 96178, 91978, 83013, 97086, 80136, 78389,
89854, 81895, 71373, 87784, 95007, 83992, 85698, 90803, 83209, 82360,
75268, 88518, 68987, 94879, 80601, 86505, 96622, 84208, 74665, 75118,
90429, 83369, 98123, 86628, 97033, 95608, 88707, 88590, 68700, 93137,
};unsigned int get_sector_index(unsigned int start_address)
{if(start_address >= ADDR_FLASH_SERTOR_0 && start_address < ADDR_FLASH_SERTOR_1)return FLASH_Sector_0;else if(start_address >= ADDR_FLASH_SERTOR_1 && start_address < ADDR_FLASH_SERTOR_2)return FLASH_Sector_1;else if(start_address >= ADDR_FLASH_SERTOR_2 && start_address < ADDR_FLASH_SERTOR_3)return FLASH_Sector_2;else if(start_address >= ADDR_FLASH_SERTOR_3 && start_address < ADDR_FLASH_SERTOR_4)return FLASH_Sector_3;else if(start_address >= ADDR_FLASH_SERTOR_4 && start_address < ADDR_FLASH_SERTOR_5)return FLASH_Sector_4;else if(start_address >= ADDR_FLASH_SERTOR_5 && start_address < ADDR_FLASH_SERTOR_6)return FLASH_Sector_5;else if(start_address >= ADDR_FLASH_SERTOR_6 && start_address < ADDR_FLASH_SERTOR_7)return FLASH_Sector_6;else if(start_address >= ADDR_FLASH_SERTOR_7 && start_address < ADDR_FLASH_SERTOR_8)return FLASH_Sector_7;else if(start_address >= ADDR_FLASH_SERTOR_8 && start_address < ADDR_FLASH_SERTOR_9)return FLASH_Sector_8;else if(start_address >= ADDR_FLASH_SERTOR_9 && start_address < ADDR_FLASH_SERTOR_10)return FLASH_Sector_9;else if(start_address >= ADDR_FLASH_SERTOR_10 && start_address < ADDR_FLASH_SERTOR_11)return FLASH_Sector_10;elsereturn FLASH_Sector_11;}int flash_test(void)
{unsigned int i,write_setor,ret = 0;FLASH_Unlock();write_setor = get_sector_index(ADDR_FLASH_SERTOR_6); // 判断FLASH地址的扇区索引/* 字节读写测试 */if(FLASH_EraseSector(write_setor,VoltageRange_3) != FLASH_COMPLETE) // 扇区擦除printf("flash erase error\r\n");elseprintf("flash erase success\r\n");for(i = 0;i < sizeof(test_data_u8);i++)FLASH_ProgramByte(ADDR_FLASH_SERTOR_6 + i,test_data_u8[i]); // 写数据if(compare_u8_data(test_data_u8,ADDR_FLASH_SERTOR_6,sizeof(test_data_u8)) != 0) // 比较{printf("flash write u8 error\r\n");ret = 1;goto end;}else printf("flash write u8 success\r\n");/* 半字读写测试 */if(FLASH_EraseSector(write_setor,VoltageRange_3) != FLASH_COMPLETE) // 扇区擦除 printf("flash erase error\r\n");elseprintf("flash erase success\r\n");for(i = 0;i < sizeof(test_data_u16);i++)FLASH_ProgramHalfWord(ADDR_FLASH_SERTOR_6 + i * 2,test_data_u16[i]); // 写数据if(compare_u16_data(test_data_u16,ADDR_FLASH_SERTOR_6,sizeof(test_data_u16)) != 0) // 比较{printf("flash write u16 error\r\n");ret = 1;goto end;}else printf("flash write u16 success\r\n");/* 字读写测试 */if(FLASH_EraseSector(write_setor,VoltageRange_3) != FLASH_COMPLETE) // 扇区擦除 printf("flash erase error\r\n");elseprintf("flash erase success\r\n");for(i = 0;i < sizeof(test_data_u32);i++)FLASH_ProgramWord(ADDR_FLASH_SERTOR_6 + i * 4,test_data_u32[i]); // 写数据if(compare_u32_data(test_data_u32,ADDR_FLASH_SERTOR_6,sizeof(test_data_u32)) != 0) // 比较 {printf("flash write u32 error\r\n");ret = 1;goto end;}else printf("flash write u32 success\r\n");
end: FLASH_Lock();return ret;
}int compare_u8_data(const unsigned char *src,unsigned int start_address,int len)
{int i;for(i = 0;i < len;i++){if(*((unsigned char *)(start_address + i)) != src[i])return 1;}return 0;
}
int compare_u16_data(const unsigned short *src,unsigned int start_address,int len)
{int i;for(i = 0;i < len;i++){if(*((unsigned short *)(start_address + i * 2)) != src[i])return 1;}return 0;
}
int compare_u32_data(const unsigned int *src,unsigned int start_address,int len)
{int i;for(i = 0;i < len;i++){if(*((unsigned int *)(start_address + i * 4)) != src[i])return 1;}return 0;
}
main.c
#include "stm32f4xx.h"
#include "delay.h"
#include "usart.h"
#include "internal_flash.h"int main(void)
{int i;NVIC_PriorityGroupConfig(2);system_tick_init();bsp_usart_init(115200);if(flash_test() == 0)printf("STM32F4X Internal Test Scuess\r\n");elseprintf("STM32F4X Internal Test Error\r\n");while(1){}}

相关文章:
STM32F4X 内部FLASH使用
STM32F4X 内部FLASH使用 STM32F4X 内部FLASHSTM32F4X内部FLASH结构STM32F40X和STM32F41X内部FLASH结构STM32F42X和STM32F43X内部FLASH结构 STM32F4X内部FLASH操作例程internal_flash.hinternal_flash.cmain.c 在嵌入式开发中,经常需要实时保存一些数据。如果工程的代…...
减小windows或linux虚拟机导出ova体积大小
减小windows或linux虚拟机导出ova体积大小 删除无用的文件,比如日志或者命令,程序等;去除磁盘碎片将不用的内存空间填充为0,便于vmdk压缩。 例子: 日志文件置空: 批量置空 /sf/data/log/ 目录下的日志文…...
WPF livecharts 折线图遮挡数字问题
在WPF里使用livecharts,如果折线图或者柱状图有多个的时候,可能会出现两个数字遮挡问题,这时候要设置DataLabelsTemplate 属性。 如LineSeries设置代码如下: 第一个折线图的DataLabelsTemplate var stackPanelFactory new Fra…...
电力系统数字化升级改造之配电室无人值守
随着科技的不断进步,电力系统的数字化升级改造已成为必然趋势。其中,配电室的无人值守是其中重要的一环。 配电室是电力系统的重要组成部分,其运行状态直接影响到电力系统的稳定性和可靠性。然而,传统的配电室存在很多问题&am…...
集合Set
目录 一、去重问题 一、去重问题 题目描述: 小明想在学校中请一些同学一起做一项问卷调查,为了实验的客观性他先用计算机生成了N个1~1000之间的随机整数(N<1000),N是用户输入的,对于其中重复的数字,只保留一个&am…...
TCP/IP(二)导论
一 知识铺垫 以下内容参照 <<电子科技大学TCPIP协议原理>>全 ① 协议和标准 一组规则: 交通规则、学生上学的学生守则等;数据通信的规则,有一个专门的名称叫作协议 protocol语义:具体描述在通信当中,每一个信息的具体含义. 二进制bit流…...
Java之UDP,TCP的详细解析
练习四:文件名重复 public class UUIDTest { public static void main(String[] args) { String str UUID.randomUUID().toString().replace("-", ""); System.out.println(str);//9f15b8c356c54f55bfcb0ee3023fce8a } } public class Client…...
【总结】kubernates crd client-java 关于自定义资源的增删改查
Java model 准备 首先使用 crd.yml 和 kubernetes CRD 自动生成 Java model 类,这是一切的前提,之前在这个地方也卡了很久。如何生成在另外一个文章中已经有所记录。 使用 crd.yml 和 kubernetes CRD 自动生成 Java model 类 CustomObjectsApi 文档学习…...
蓝牙主要知识,一文概览
蓝牙知识相关 文章目录 蓝牙知识相关1.蓝牙版本的发展简史2.低功耗BLE PHY2.1 频段**2.2 BLE调制方案—GFSK**2.3 **蓝牙 LE 传输速度、功率和接收器灵敏度**2.4 **BLE 时分双工 (TDD)**3.BT主从连接过程3.1 主设备工作模式3.1.1 积木编程控制台3.2 从设备工作模式3.2.1 蓝牙遥…...
Linux 守护进程
一 何为守护进程 守护进程( Daemon )也称为精灵进程,是运行在后台的一种特殊进程,它独立于控制终端并且周期性 地执行某种任务或等待处理某些事情的发生,主要表现为以下两个特点: 长期运行。守护进程是一…...
自动驾驶技术的基础知识
自动驾驶技术是现代汽车工业中的一项革命性发展,它正在改变着我们对交通和出行的理解。本文将介绍自动驾驶技术的基础知识,包括其概念、历史发展、分类以及关键技术要素。 1. 自动驾驶概念 自动驾驶是一种先进的交通技术,它允许汽车在没有人…...
解决:yarn 无法加载文件 “C:\Users\XXXXX\AppData\Roaming\npm\yarn.ps1,因为在此系统上禁止运行脚本“ 的问题
1、问题描述: 报错的整体代码为: yarn : 无法加载文件 C:\Users\admin\AppData\Roaming\npm\yarn.ps1,因为在此系统上禁止运行脚本 // 整体的报错代码为 : yarn : 无法加载文件 C:\Users\admin\AppData\Roaming\npm\yarn.ps1&…...
【JVM--StringTable字符串常量池】
文章目录 1. String 的基本特性2. 字符串拼接操作3. intern()的使用4. StringTable 的垃圾回收 1. String 的基本特性 String 声明为 final 的,不可被继承String 实现了 Serializable 接口:表示字符串是支持序列化的。String 实现了 Comparable 接口&am…...
Large Language Models Meet Knowledge Graphs to Answer Factoid Questions
本文是LLM系列文章,针对《Large Language Models Meet Knowledge Graphs to Answer Factoid Questions》的翻译。 大型语言模型与知识图谱相遇,回答虚假问题 摘要1 引言2 相关工作3 提出的方法4 实验设计5 结果与讨论6 结论 摘要 最近,有研…...
blender 之视频渲染(以三维重建path为例)
blender 之视频渲染(以三维重建path为例) 1.新建轨迹路径2.设置相机,使其按照path运动3.将相机视角对准物体4.修改帧率5.设置输出路径6.设置输出格式7.渲染 1.新建轨迹路径 新建轨迹 选中新建的BezierCycle,按住S,拖…...
leetcode做题笔记166. 分数到小数
给定两个整数,分别表示分数的分子 numerator 和分母 denominator,以 字符串形式返回小数 。 如果小数部分为循环小数,则将循环的部分括在括号内。 如果存在多个答案,只需返回 任意一个 。 对于所有给定的输入,保证 …...
Android Studio新建项目缓慢解决方案
关于Android Studio2022新建项目时下载依赖慢的解决方案 起因解决方案gradle下载慢解决方案kotlin依赖下载慢解决方案 结尾 起因 新建Android Studio项目时,常会因为网络问题导致部分依赖下载缓慢,其中gradle和kotlin最拖慢进度。 解决方案 gradle下载…...
AmdU (5-azidomethyl-2‘-deoxyuridine)的反应原理|59090-48-1
产品简介:叠氮甲基dU(AmdU)是一种核苷类化合物,它含有叠氮基团,这种结构特点使其在细胞学和生物学领域得到了广泛应用。与胸腺嘧啶核苷相似的结构,使得叠氮甲基dU(AmdU)能够被细胞聚…...
使用nvm安装多个node版本
github下载地址: Releases coreybutler/nvm-windows (github.com) 安装了 nvm(Node Version Manager)后,可以使用以下步骤安装第二个 Node.js 版本: 打开终端或命令提示符。 使用以下命令列出可用的 Node.js 版本: …...
基于springboot实现自习室预订系统的设计与实现项目【项目源码+论文说明】分享
基于springboot实现自习室预订系统的设计与实现演示 摘要 在网络高速发展的时代,众多的软件被开发出来,给学生带来了很大的选择余地,而且人们越来越追求更个性的需求。在这种时代背景下,学院只能以学生为导向,所以自习…...
应用升级/灾备测试时使用guarantee 闪回点迅速回退
1.场景 应用要升级,当升级失败时,数据库回退到升级前. 要测试系统,测试完成后,数据库要回退到测试前。 相对于RMAN恢复需要很长时间, 数据库闪回只需要几分钟。 2.技术实现 数据库设置 2个db_recovery参数 创建guarantee闪回点,不需要开启数据库闪回。…...
云计算——弹性云计算器(ECS)
弹性云服务器:ECS 概述 云计算重构了ICT系统,云计算平台厂商推出使得厂家能够主要关注应用管理而非平台管理的云平台,包含如下主要概念。 ECS(Elastic Cloud Server):即弹性云服务器,是云计算…...
Redis相关知识总结(缓存雪崩,缓存穿透,缓存击穿,Redis实现分布式锁,如何保持数据库和缓存一致)
文章目录 1.什么是Redis?2.为什么要使用redis作为mysql的缓存?3.什么是缓存雪崩、缓存穿透、缓存击穿?3.1缓存雪崩3.1.1 大量缓存同时过期3.1.2 Redis宕机 3.2 缓存击穿3.3 缓存穿透3.4 总结 4. 数据库和缓存如何保持一致性5. Redis实现分布式…...
MFC内存泄露
1、泄露代码示例 void X::SetApplicationBtn() {CMFCRibbonApplicationButton* pBtn GetApplicationButton();// 获取 Ribbon Bar 指针// 创建自定义按钮CCustomRibbonAppButton* pCustomButton new CCustomRibbonAppButton();pCustomButton->SetImage(IDB_BITMAP_Jdp26)…...
Mybatis逆向工程,动态创建实体类、条件扩展类、Mapper接口、Mapper.xml映射文件
今天呢,博主的学习进度也是步入了Java Mybatis 框架,目前正在逐步杨帆旗航。 那么接下来就给大家出一期有关 Mybatis 逆向工程的教学,希望能对大家有所帮助,也特别欢迎大家指点不足之处,小生很乐意接受正确的建议&…...
现代密码学 | 椭圆曲线密码学—附py代码
Elliptic Curve Cryptography 椭圆曲线密码学(ECC)是一种基于有限域上椭圆曲线数学特性的公钥加密技术。其核心原理涉及椭圆曲线的代数性质、离散对数问题以及有限域上的运算。 椭圆曲线密码学是多种数字签名算法的基础,例如椭圆曲线数字签…...
Pinocchio 库详解及其在足式机器人上的应用
Pinocchio 库详解及其在足式机器人上的应用 Pinocchio (Pinocchio is not only a nose) 是一个开源的 C 库,专门用于快速计算机器人模型的正向运动学、逆向运动学、雅可比矩阵、动力学和动力学导数。它主要关注效率和准确性,并提供了一个通用的框架&…...
《C++ 模板》
目录 函数模板 类模板 非类型模板参数 模板特化 函数模板特化 类模板的特化 模板,就像一个模具,里面可以将不同类型的材料做成一个形状,其分为函数模板和类模板。 函数模板 函数模板可以简化函数重载的代码。格式:templa…...
JS设计模式(4):观察者模式
JS设计模式(4):观察者模式 一、引入 在开发中,我们经常会遇到这样的场景:一个对象的状态变化需要自动通知其他对象,比如: 电商平台中,商品库存变化时需要通知所有订阅该商品的用户;新闻网站中࿰…...
接口自动化测试:HttpRunner基础
相关文档 HttpRunner V3.x中文文档 HttpRunner 用户指南 使用HttpRunner 3.x实现接口自动化测试 HttpRunner介绍 HttpRunner 是一个开源的 API 测试工具,支持 HTTP(S)/HTTP2/WebSocket/RPC 等网络协议,涵盖接口测试、性能测试、数字体验监测等测试类型…...
