【软件设计】通过软件设计提高 Flash 的擦写次数

article/2025/7/17 4:36:14

目录

  • 0. 个人简介 && 授权须知
  • 1. Flash 和 EEROM 基本情况
  • 2. 场景要求
  • 3. 软件设计思路
  • 4. 代码展示
    • 4.1 flash.h
    • 4.2 flash.c

0. 个人简介 && 授权须知

image-20230911133730620

📋 个人简介

  • 💖 作者简介:大家好,我是喜欢记录零碎知识点的菜鸟打工人。😎
  • 📝 个人主页:欢迎访问我的博客主页🔥…
    • https://blog.csdn.net/qq_39217004?spm=1010.2135.3001.5343
  • 🎉 支持我:点赞👍+收藏⭐️+留言📝
  • 📣 系列专栏:嵌入式Linux开发 🍁 🍁
  • 💬格言:写文档啊不是写文章,重要的还是直白!🔥

转载文章,禁止声明原创;不允许直接二次转载,转载请根据原文链接联系作者

若无需改版,在文首清楚标注作者及来源/原文链接,并删除【原创声明】,即可直接转载。
但对于未注明转载来源/原文链接的文章,我将保留追述的权利。

https://blog.csdn.net/qq_39217004?spm=1010.2135.3001.5343

作者:积跬步、至千里

image-20230911133724204

场景分析:
在嵌入式开发中,EEPROM 因其非易失性存储特性,常用于保存配置参数等数据。EEPROM 的擦写次数一般在10万次左右。

单片机一般通过 IIC 接口外加一个 EEROM 存储芯片

但是,有些小项目为了节省成本或者以前维护的一些项目,只能把 flash 划分出一段空间,当数据存储空间来用。

这样的缺点是

  1. 升级固件时如果固件变大了,一旦覆盖了原来划分的数据存储区域,那数据就没了。
  2. Flash 的擦写次数一般都是在 1万次 左右,这个次数可是远小于 EEROM

1万次,在很多场景下并不够。
因此本文参考
老板说:单片机,Flash模拟EEPROM,16字节,算法轮询存储给我做到100万次的存储次数 ,
的思路,通过软件设计,提高flash的擦写次数。

1. Flash 和 EEROM 基本情况

  1. 独立的EEPROM芯片是可以直接写字节的,即使覆盖写也无须擦除,
  2. 单片机的FLASH 写入数据之前,必须按页来擦除,先擦除再写入

2. 场景要求

程序一共有15个字节的内容需要断电保存,每改变其中一个字节就需要保存一次,做到100万次的一个存储次数。

单片机的 Flash 一共是 128KB ,从0x08000000----0x0801FFFF,一共64页,每页2KB。

3. 软件设计思路

以每个 page 2KB 的大小为一个节点,假设数据要存储到【最后一个page】,设计思路如下:

  1. 从第一个 page 开始写入,第一次写前16字节,然后更新写入地址索引到第16个字节
  2. 第二次从 17-31 字节写入,然后更新写入地址索引到第32个字节,依次循环

这样来算,每个 page 可以存储 2048/16 = 128 次,写满一页擦除一次,理论上存储次数能达到128 × 1万次/页 = 128万次。

4. 代码展示

4.1 flash.h

#ifndef FLASH__H
#define FLASH__H#include "stm32g0xx_hal.h"
#include <string.h>// FLASH配置
#define FLASH_BASE_ADDR 0x0801F800 // FLASH最后一页起始地址 (128KB - 2KB)
#define PAGE_SIZE 2048             // STM32G071页面大小为2KB
#define STATE_SIZE 16             // 结构体大小(填充到24字节)typedef struct {unsigned int  color;                  // 颜色unsigned int  seconds;                // 秒数unsigned char mode;                   // 模式unsigned char number;                 // 序号unsigned char padinng[5];             // 预留5unsigned char checksum;               // 1字节,校验和
}dataState;extern dataState old_state;
extern dataState current_state;
void printState(dataState *state);
HAL_StatusTypeDef flash_program(unsigned int addr, unsigned char* data, unsigned int len);
void read_flash(unsigned int addr, unsigned char* data, unsigned int len);
void init_flash_addr(void);
void save_state(dataState* state);
void get_state(dataState* state);
void update_state(dataState* state);#endif

4.2 flash.c

// 初始化:查找最新有效数据
void init_flash_addr(void) {dataState temp_state;uint32_t addr = FLASH_BASE_ADDR;uint32_t last_valid_addr = FLASH_BASE_ADDR;int found_valid_data = 0;while (addr < FLASH_BASE_ADDR + PAGE_SIZE) {read_flash(addr, (uint8_t*)&temp_state, STATE_SIZE);// 检查是否全0xFFuint8_t all_ff[STATE_SIZE];memset(all_ff, 0xFF, STATE_SIZE);int is_all_ff = (memcmp(&temp_state, all_ff, STATE_SIZE) == 0);if (!is_all_ff && temp_state.checksum == calculate_checksum(&temp_state)) {last_valid_addr = addr;found_valid_data = 1;memcpy(&current_state, &temp_state, STATE_SIZE);        } else {     break;}addr += STATE_SIZE;}flash_addr = last_valid_addr + (found_valid_data ? STATE_SIZE : 0);if(found_valid_data)printf("init first,found valid data,last_valid_addr:%X\r\n",last_valid_addr);elseprintf("init first,it is all ff,it is the first data\r\n");if (flash_addr > FLASH_BASE_ADDR + PAGE_SIZE)     {printf("init erase page 2KB\r\n");erase_page(FLASH_BASE_ADDR);flash_addr = FLASH_BASE_ADDR;}if (!found_valid_data) {printf("not found valid data\r\n");current_state.color = 100;current_state.seconds = 200;current_state.mode = 1;current_state.number = 1;current_state.checksum = calculate_checksum(&current_state);__disable_irq(); flash_program(FLASH_BASE_ADDR, (uint8_t*)&current_state, STATE_SIZE);__enable_irq(); flash_addr = FLASH_BASE_ADDR + STATE_SIZE;       }printState(&current_state);
}
  1. 第一步,先从第一个地址读取第一个16字节,然后判断是不是全部等于0xFF,如果第一次是就证明是第一次,下一步flash_addr就不需要增加STATE_SIZE,写入地址索引就是FLASH_BASE_ADDR
  2. 第二步,如果不全是0xFF并且校验字节通过,证明这是一组有效数据,我们先将此数据更新到current_state,但是这里还不能证明是最后一组有效数据,因为最后一组有效数据才是我们要找到的数据。
  3. 继续检查下一组数据,直到检查到一组数据是全0xFF,证明上一组数据就是最后一组有效数据,就跳出,此时我们也就找到了最后一组有效数据的起始地址。
  4. 此地址增加STATE_SIZE就是最新可以存储数据的地址索引了。
  5. 如果没找到有效数据,证明是第一次,就写入默认数值并保存,更新索引。

保存数据save_state函数:

oid save_state(dataState* state) {dataState last_state;if (flash_addr > FLASH_BASE_ADDR) {read_flash(flash_addr - STATE_SIZE, (uint8_t*)&last_state, STATE_SIZE);if (memcmp(&last_state, state, STATE_SIZE) == 0) {printf("数据没有变化,直接返回");return; }}__disable_irq(); if (flash_addr + STATE_SIZE > FLASH_BASE_ADDR + PAGE_SIZE) {printf("erase page 2KB\r\n");erase_page(FLASH_BASE_ADDR);flash_addr = FLASH_BASE_ADDR;   }state->checksum = calculate_checksum(state);flash_program(flash_addr, (uint8_t*)state, STATE_SIZE);flash_addr += STATE_SIZE; __enable_irq(); 
}

函数就比较简单了,首先将要保存的数据与最新存储的数据做对比,如果没变化,就不操作;如果地址超出范围了,就先擦除整个页,更新写索引到FLASH_BASE_ADDR,接着保存数据到当下最新写地址索引即可。


http://www.hkcw.cn/article/iUllAjxgNa.shtml

相关文章

C 语言练习--初级

#学习C 代码&#xff0c; 做小练习时&#xff0c;自己运行代码竟然发现很多错误&#xff0c;记录一下。 1、计算器 根据输入的数值和符合&#xff0c;输出相应结果。 结果&#xff1a; #include "stdio.h"int Primary_math(int a, int b, char sign){int num0;swit…

pikachu通关教程-CSRF XSS

XSS XSS漏洞原理 XSS被称为跨站脚本攻击&#xff08;Cross Site Scripting&#xff09;&#xff0c;由于和层叠样式表&#xff08;Cascading Style Sheets&#xff0c;CSS&#xff09;重名&#xff0c;改为XSS。主要基于JavaScript语言进行恶意攻击&#xff0c;因为js非常灵活…

E. Melody 【CF1026 (Div. 2)】 (求欧拉路径之Hierholzer算法)

E. Melody 思路 将所有出现过的音量和音高看作一个点&#xff0c;一个声音看作一条边&#xff0c;连接起来。那么很容易知道要找的就是图上的一条欧拉路径&#xff08;类似一笔画问题&#xff09; 又已知存在欧拉路径的充要条件为&#xff1a;度数为奇数的点的个数为0或者2个…

历年中国科学技术大学计算机保研上机真题

2025中国科学技术大学计算机保研上机真题 2024中国科学技术大学计算机保研上机真题 2023中国科学技术大学计算机保研上机真题 在线测评链接&#xff1a;https://pgcode.cn/school?classification1 拆分数字 题目描述 给定一个数字&#xff0c;拆分成若干个数字之和&#xff…

2025陕西省赛补题

A 贪心 题意&#xff1a;给一个长度为n的序列&#xff0c;每次操作可以花费 w [ c [ i ] ] ( r − l 1 ) w[c[i]](r-l1) w[c[i]](r−l1)的代价&#xff0c;把区间 [ l , r ] [l,r] [l,r]染成染色 。 思路&#xff1a;对任意颜色&#xff0c;[l,r]中如果有cnt个连续的该颜色段…

Linux详谈进程地址空间

目录 第一谈&#xff1a;简单了解 第二谈&#xff1a;与操作系统的联系 内核空间与用户空间 步骤1&#xff1a;用户态代码执行 步骤2&#xff1a;跳转到内核代码 步骤3&#xff1a;内核代码访问用户数据 步骤4&#xff1a;返回到用户态 对于操作系统的本质&#xff1a;…

RabbitMQ vs MQTT:深入比较与最新发展

RabbitMQ vs MQTT&#xff1a;深入比较与最新发展 引言 在消息队列和物联网&#xff08;IoT&#xff09;通信领域&#xff0c;RabbitMQ 和 MQTT 是两种备受瞩目的技术&#xff0c;各自针对不同的需求和场景提供了强大的解决方案。随着 2025 年的到来&#xff0c;这两项技术都…

【Dify学习笔记】:Dify离线安装插件教程

Dify离线安装插件教程 1.本地下载插件 插件点击详情页面&#xff0c;安装右边的下载按钮&#xff0c;下载到本地 2.dify插件打包工具 dify-plugin-repackaging 下载后&#xff0c;进入到工具所在目录dify-plugin-repackaging/ git clone https://github.com/junjiem/dif…

2025年全国青少年信息素养大赛 scratch图形化编程挑战赛 小高组初赛 内部模拟试卷解析

2025年信息素养大赛初赛scratch模拟题 博主推荐 所有考级比赛学习相关资料合集【推荐收藏】 scratch资料 Scratch3.0系列视频课程资料零基础学习scratch3.0【入门教学 免费】零基础学习scratch3.0【视频教程 114节 免费】 历届蓝桥杯scratch国赛真题解析历届蓝桥杯scratch…

eBest智能价格引擎系统 助力屈臣氏饮料落地「价格大脑」+「智慧通路」数字基建​

从价格策略到终端执行&#xff0c;数字化正在重构饮料行业竞争壁垒&#xff01; 近日&#xff0c;eBest为屈臣氏饮料提供的智能价格引擎系统已正式上线并投入运营。同时&#xff0c;基于eBest SFA方案且与屈臣氏饮料业务场景深度耦合的Smart Field Operation智慧通路项目正式启…

开发效率提升小技巧:快速提取图标资源的解决方案

在日常使用电脑的过程中&#xff0c;我们经常会遇到想要提取某个软件图标的情况&#xff0c;比如用于美化桌面、制作快捷方式&#xff0c;或者个人收藏等。这一款高效又实用的图标提取工具&#xff0c;帮助你轻松获取软件中的图标资源&#xff01; ResHacker 是一个绿色免安装…

keepalived定制日志bug

keepalived定制日志bug 源码安装apt安装endl 源码安装 在/etc/rsyslog.d/目录下创建 keepalived的日志配置文件keepalived.conf [rootubuntu24-13:~]# vim /etc/rsyslog.d/keepalived.conf [rootubuntu24-13:~]# cat /etc/rsyslog.d/keepalived.conf local6.* /var/log/keepa…

SpringCloud——Docker

1.命令解读 docker run -d 解释&#xff1a;创建并运行一个容器&#xff0c;-d则是让容器以后台进程运行 --name mysql 解释&#xff1a; 给容器起个名字叫mysql -p 3306:3306 解释&#xff1a;-p 宿主机端口:容器内端口&#xff0c;设置端口映射 注意&#xff1a; 1、…

2.测试项目启动和研读需求文档

软件质量需求 定义: 用于确定测试目标&#xff0c;反映用户对软件的要求分类依据: 分为功能和非功能两大类&#xff0c;其中非功能包含性能、界面等8个子类 软件质量需求的分类 功能需求: 软件能做什么的核心能力非功能需求: 性能&#xff1a;运行效率和资源占用界面&#xf…

在 Android 上备份短信:保护您的对话

尽管我们的Android手机有足够的存储空间来存储无数的短信&#xff0c;但由于设备故障、意外删除或其他意外原因&#xff0c;您可能会丢失重要的对话。幸运的是&#xff0c;我们找到了 5 种有效的 Android SMS 备份解决方案&#xff0c;确保您的数字聊天和信息保持安全且可访问。…

02业务流程的定义

1.要想用好业务流程&#xff0c;首先必须得了解流程与认识流程&#xff0c;什么是业务流程。在认识流程之前&#xff0c;首先要理清两个基本概念&#xff0c;业务和流程。 业务指的是&#xff1a;个人的或者摸个机构的专业工作。流程&#xff0c;原本指的是水的路程&#xff0…

PHP7+MySQL5.6 查立得源码授权系统DNS验证版

# PHP7MySQL5.6 查立得源码授权系统DNS验证版 ## 一、系统概述 本系统是一个基于PHP7和MySQL5.6的源码授权系统&#xff0c;使用DNS TXT记录验证域名所有权&#xff0c;实现对软件源码的授权保护。 系统支持多版本管理&#xff0c;可以灵活配置不同版本的价格和下载路径&#…

vue+threeJs 绘制3D圆形

嗨&#xff0c;我是小路。今天主要和大家分享的主题是“vuethreeJs 绘制圆形”。 今天找到一个用three.js绘制图形的项目&#xff0c;主要是用来绘制各种形状。 项目案例示意图 1.THREE.ShapeGeometry 定义&#xff1a;是 Three.js 中用于从 2D 路径形状&#xff08…

vue+threeJs 生成一个圆柱体

嗨&#xff0c;我是小路。今天主要和大家分享的主题是“vuethreeJs 生成一个圆柱体”。 案例示例图 1.CylinderGeometry 定义&#xff1a;创造一个圆柱体。 属性列表列表说明 radiusTop 顶部半径 radiusBottom 底部半径 height 高 radialSegments 横向分段&#xff…

VUE中created() 和 mounted()俩种生命周期钩子函数的区别

在 Vue.js 中&#xff0c;created() 和 mounted() 是两个关键的生命周期钩子函数&#xff0c;它们的主要区别在于​​调用时机​​和​​可访问的实例属性​​&#xff1a; 调用时机 ​​created()​​ 在 Vue 实例创建完成后立即调用&#xff08;​​数据初始化完成&#xff…