STM32F429 SPI读写W25Q128 Flash实战:从引脚配置到数据存储的完整流程

news2026/5/8 20:40:03
STM32F429 SPI读写W25Q128 Flash实战从引脚配置到数据存储的完整流程在嵌入式系统开发中外部Flash存储器扩展是常见需求。W25Q128作为一款16MB容量的SPI Flash芯片以其高性价比和易用性成为许多项目的首选。本文将手把手带你完成STM32F429与W25Q128的完整通信实现从硬件连接到数据存取涵盖工程实践中那些容易被忽视的关键细节。1. 硬件设计与初始化配置1.1 引脚连接与SPI模式选择W25Q128与STM32F429的连接需要特别注意信号完整性和电气特性。推荐使用以下连接方式W25Q128引脚STM32F429引脚功能说明CSPF6片选信号低电平有效DOPF8数据输出MISOWP3.3V写保护高电平禁用保护DIPF9数据输入MOSICLKPF7时钟信号HOLD3.3V保持信号高电平禁用VCC3.3V电源GNDGND地线硬件设计时需注意在CLK信号线串联22Ω电阻可减少信号反射电源引脚建议并联0.1μF和4.7μF电容组合长距离布线时建议在MOSI/MISO线上串联33Ω电阻1.2 SPI外设初始化代码实现针对W25Q128的SPI初始化需要特别注意时钟极性和相位配置。以下是经过优化的初始化代码void SPI5_Init(void) { GPIO_InitTypeDef GPIO_InitStruct {0}; SPI_InitTypeDef SPI_InitStruct {0}; // 启用GPIOF和SPI5时钟 RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOF, ENABLE); RCC_APB2PeriphClockCmd(RCC_APB2Periph_SPI5, ENABLE); // 配置PF6(CS)为推挽输出 GPIO_InitStruct.GPIO_Pin GPIO_Pin_6; GPIO_InitStruct.GPIO_Mode GPIO_Mode_OUT; GPIO_InitStruct.GPIO_OType GPIO_OType_PP; GPIO_InitStruct.GPIO_Speed GPIO_Speed_50MHz; GPIO_InitStruct.GPIO_PuPd GPIO_PuPd_UP; GPIO_Init(GPIOF, GPIO_InitStruct); // 配置PF7-9为复用功能 GPIO_InitStruct.GPIO_Pin GPIO_Pin_7 | GPIO_Pin_8 | GPIO_Pin_9; GPIO_InitStruct.GPIO_Mode GPIO_Mode_AF; GPIO_InitStruct.GPIO_OType GPIO_OType_PP; GPIO_InitStruct.GPIO_Speed GPIO_Speed_50MHz; GPIO_InitStruct.GPIO_PuPd GPIO_PuPd_NOPULL; GPIO_Init(GPIOF, GPIO_InitStruct); // 引脚复用映射 GPIO_PinAFConfig(GPIOF, GPIO_PinSource7, GPIO_AF_SPI5); GPIO_PinAFConfig(GPIOF, GPIO_PinSource8, GPIO_AF_SPI5); GPIO_PinAFConfig(GPIOF, GPIO_PinSource9, GPIO_PinAF_SPI5); // SPI参数配置 SPI_InitStruct.SPI_Direction SPI_Direction_2Lines_FullDuplex; SPI_InitStruct.SPI_Mode SPI_Mode_Master; SPI_InitStruct.SPI_DataSize SPI_DataSize_8b; SPI_InitStruct.SPI_CPOL SPI_CPOL_High; // 空闲时高电平 SPI_InitStruct.SPI_CPHA SPI_CPHA_2Edge; // 第二个边沿采样 SPI_InitStruct.SPI_NSS SPI_NSS_Soft; SPI_InitStruct.SPI_BaudRatePrescaler SPI_BaudRatePrescaler_4; // 22.5MHz SPI_InitStruct.SPI_FirstBit SPI_FirstBit_MSB; SPI_InitStruct.SPI_CRCPolynomial 7; SPI_Init(SPI5, SPI_InitStruct); SPI_Cmd(SPI5, ENABLE); FLASH_CS_HIGH(); // 初始时取消片选 }关键配置说明CPOL/CPHAW25Q128支持模式0(0,0)和模式3(1,1)模式3在高速下更稳定时钟分频STM32F429的APB2时钟为90MHz分频4得到22.5MHz通信速率软件NSS硬件NSS信号在DMA传输时可能有问题推荐使用GPIO模拟2. Flash基础操作与状态管理2.1 基本读写函数实现SPI通信的基础是字节传输函数需要正确处理超时情况uint8_t SPI5_ReadWriteByte(uint8_t TxData) { uint32_t timeout SPI_TIMEOUT; // 等待发送缓冲区空 while (SPI_I2S_GetFlagStatus(SPI5, SPI_I2S_FLAG_TXE) RESET) { if ((timeout--) 0) return SPI_TIMEOUT; } SPI_I2S_SendData(SPI5, TxData); timeout SPI_TIMEOUT; // 等待接收缓冲区非空 while (SPI_I2S_GetFlagStatus(SPI5, SPI_I2S_FLAG_RXNE) RESET) { if ((timeout--) 0) return SPI_TIMEOUT; } return SPI_I2S_ReceiveData(SPI5); }2.2 Flash状态机管理W25Q128内部操作需要时间完成必须正确检测状态寄存器#define W25X_WriteEnable 0x06 #define W25X_ReadStatusReg1 0x05 #define W25X_BUSY_MASK 0x01 void W25Q128_WaitForReady(void) { uint8_t status; do { FLASH_CS_LOW(); SPI5_ReadWriteByte(W25X_ReadStatusReg1); status SPI5_ReadWriteByte(0xFF); FLASH_CS_HIGH(); } while (status W25X_BUSY_MASK); } void W25Q128_WriteEnable(void) { FLASH_CS_LOW(); SPI5_ReadWriteByte(W25X_WriteEnable); FLASH_CS_HIGH(); }注意每次写操作前必须发送写使能指令且该指令会在上电复位或写禁用指令后失效3. 存储操作实战从扇区擦除到数据写入3.1 扇区擦除实现Flash存储器的特性决定了必须先擦除后写入擦除最小单位为4KB扇区#define W25X_SectorErase 0x20 void W25Q128_SectorErase(uint32_t addr) { W25Q128_WriteEnable(); FLASH_CS_LOW(); SPI5_ReadWriteByte(W25X_SectorErase); SPI5_ReadWriteByte((addr 16) 0xFF); SPI5_ReadWriteByte((addr 8) 0xFF); SPI5_ReadWriteByte(addr 0xFF); FLASH_CS_HIGH(); W25Q128_WaitForReady(); }擦除时间参数操作类型典型时间最大时间扇区擦除60ms300ms块擦除0.7s2s整片擦除25s60s3.2 页编程与数据写入W25Q128支持页编程256字节操作跨页写入需要特殊处理#define W25X_PageProgram 0x02 #define PAGE_SIZE 256 void W25Q128_PageWrite(uint8_t *pBuffer, uint32_t addr, uint16_t len) { if (len PAGE_SIZE) len PAGE_SIZE; W25Q128_WriteEnable(); FLASH_CS_LOW(); SPI5_ReadWriteByte(W25X_PageProgram); SPI5_ReadWriteByte((addr 16) 0xFF); SPI5_ReadWriteByte((addr 8) 0xFF); SPI5_ReadWriteByte(addr 0xFF); while (len--) { SPI5_ReadWriteByte(*pBuffer); } FLASH_CS_HIGH(); W25Q128_WaitForReady(); } void W25Q128_BufferWrite(uint8_t *pBuffer, uint32_t addr, uint32_t len) { uint32_t pageOffset addr % PAGE_SIZE; uint32_t remain len; // 处理起始不完整页 if (pageOffset 0) { uint32_t bytesToWrite MIN(PAGE_SIZE - pageOffset, len); W25Q128_PageWrite(pBuffer, addr, bytesToWrite); pBuffer bytesToWrite; addr bytesToWrite; remain - bytesToWrite; } // 写入完整页 while (remain PAGE_SIZE) { W25Q128_PageWrite(pBuffer, addr, PAGE_SIZE); pBuffer PAGE_SIZE; addr PAGE_SIZE; remain - PAGE_SIZE; } // 写入剩余数据 if (remain 0) { W25Q128_PageWrite(pBuffer, addr, remain); } }4. 高级应用数据类型存储与文件系统集成4.1 结构化数据存储方案实际项目中常需要存储结构化数据推荐采用以下格式#pragma pack(push, 1) typedef struct { uint32_t magic; // 标识符 0x55AA55AA uint16_t version; // 数据结构版本 uint32_t crc; // 数据区CRC校验 uint32_t timestamp; // 最后更新时间 uint8_t data[]; // 实际数据区 } FlashDataHeader; #pragma pack(pop) void SaveStructuredData(void *data, uint16_t size, uint32_t sectorAddr) { uint8_t buffer[4096]; FlashDataHeader *header (FlashDataHeader *)buffer; // 准备数据头 header-magic 0x55AA55AA; header-version 1; header-timestamp HAL_GetTick(); memcpy(header-data, data, size); header-crc Calculate_CRC32(header-data, size); // 擦除后写入 W25Q128_SectorErase(sectorAddr); W25Q128_BufferWrite(buffer, sectorAddr, sizeof(FlashDataHeader) size); }4.2 与FatFs文件系统集成在Flash上实现文件系统可极大简化数据管理FatFs是轻量级解决方案首先实现底层磁盘接口DSTATUS disk_initialize(BYTE pdrv) { // 初始化SPI Flash return RES_OK; } DRESULT disk_read(BYTE pdrv, BYTE *buff, LBA_t sector, UINT count) { uint32_t addr sector * FLASH_SECTOR_SIZE; W25Q128_BufferRead(buff, addr, count * FLASH_SECTOR_SIZE); return RES_OK; } DRESULT disk_write(BYTE pdrv, const BYTE *buff, LBA_t sector, UINT count) { uint32_t addr sector * FLASH_SECTOR_SIZE; for (uint32_t i 0; i count; i) { W25Q128_SectorErase(addr i * FLASH_SECTOR_SIZE); } W25Q128_BufferWrite(buff, addr, count * FLASH_SECTOR_SIZE); return RES_OK; }然后进行文件系统格式化void FormatFlashFilesystem(void) { FATFS fs; uint8_t work[FF_MAX_SS]; // 擦除前4MB空间用于文件系统 for (uint32_t i 0; i 1024; i) { W25Q128_SectorErase(i * 4096); } // 创建FAT文件系统 f_mkfs(0:, FM_FAT32, 0, work, sizeof(work)); f_mount(fs, 0:, 1); }5. 性能优化与错误处理5.1 DMA加速SPI传输对于大数据量传输使用DMA可显著提升性能void SPI5_DMA_Init(void) { DMA_InitTypeDef DMA_InitStruct; // 启用DMA2时钟 RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_DMA2, ENABLE); // 配置DMA发送通道 DMA_InitStruct.DMA_Channel DMA_Channel_2; DMA_InitStruct.DMA_PeripheralBaseAddr (uint32_t)SPI5-DR; DMA_InitStruct.DMA_Memory0BaseAddr (uint32_t)0; // 动态设置 DMA_InitStruct.DMA_DIR DMA_DIR_MemoryToPeripheral; DMA_InitStruct.DMA_BufferSize 0; // 动态设置 DMA_InitStruct.DMA_PeripheralInc DMA_PeripheralInc_Disable; DMA_InitStruct.DMA_MemoryInc DMA_MemoryInc_Enable; DMA_InitStruct.DMA_PeripheralDataSize DMA_PeripheralDataSize_Byte; DMA_InitStruct.DMA_MemoryDataSize DMA_MemoryDataSize_Byte; DMA_InitStruct.DMA_Mode DMA_Mode_Normal; DMA_InitStruct.DMA_Priority DMA_Priority_High; DMA_InitStruct.DMA_FIFOMode DMA_FIFOMode_Disable; DMA_InitStruct.DMA_FIFOThreshold DMA_FIFOThreshold_HalfFull; DMA_InitStruct.DMA_MemoryBurst DMA_MemoryBurst_Single; DMA_InitStruct.DMA_PeripheralBurst DMA_PeripheralBurst_Single; DMA_DeInit(DMA2_Stream3); DMA_Init(DMA2_Stream3, DMA_InitStruct); // 启用DMA中断 NVIC_InitTypeDef NVIC_InitStruct; NVIC_InitStruct.NVIC_IRQChannel DMA2_Stream3_IRQn; NVIC_InitStruct.NVIC_IRQChannelPreemptionPriority 0; NVIC_InitStruct.NVIC_IRQChannelSubPriority 0; NVIC_InitStruct.NVIC_IRQChannelCmd ENABLE; NVIC_Init(NVIC_InitStruct); DMA_ITConfig(DMA2_Stream3, DMA_IT_TC, ENABLE); // 关联DMA到SPI SPI_I2S_DMACmd(SPI5, SPI_I2S_DMAReq_Tx, ENABLE); }5.2 错误检测与恢复机制可靠的Flash操作需要完善的错误处理#define FLASH_OP_TIMEOUT 1000 // 1秒超时 typedef enum { FLASH_OK 0, FLASH_TIMEOUT, FLASH_VERIFY_FAIL, FLASH_WRITE_PROTECTED, FLASH_INVALID_SECTOR } FlashStatus; FlashStatus W25Q128_VerifyWrite(uint8_t *expected, uint32_t addr, uint32_t len) { uint8_t *readback malloc(len); uint32_t startTime HAL_GetTick(); // 等待写入完成 while ((HAL_GetTick() - startTime) FLASH_OP_TIMEOUT) { uint8_t status; FLASH_CS_LOW(); SPI5_ReadWriteByte(W25X_ReadStatusReg1); status SPI5_ReadWriteByte(0xFF); FLASH_CS_HIGH(); if (!(status W25X_BUSY_MASK)) break; } if ((HAL_GetTick() - startTime) FLASH_OP_TIMEOUT) { free(readback); return FLASH_TIMEOUT; } // 读取验证 W25Q128_BufferRead(readback, addr, len); if (memcmp(expected, readback, len) ! 0) { free(readback); return FLASH_VERIFY_FAIL; } free(readback); return FLASH_OK; } void FlashErrorHandler(FlashStatus status) { switch(status) { case FLASH_TIMEOUT: printf([ERROR] Flash operation timeout\r\n); break; case FLASH_VERIFY_FAIL: printf([ERROR] Data verification failed\r\n); break; case FLASH_WRITE_PROTECTED: printf([ERROR] Flash is write protected\r\n); break; case FLASH_INVALID_SECTOR: printf([ERROR] Invalid sector address\r\n); break; default: printf([ERROR] Unknown flash error\r\n); } // 尝试恢复操作 W25Q128_WriteDisable(); HAL_Delay(100); SPI5_Init(); // 重新初始化SPI }

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

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

相关文章

SpringBoot-17-MyBatis动态SQL标签之常用标签

文章目录 1 代码1.1 实体User.java1.2 接口UserMapper.java1.3 映射UserMapper.xml1.3.1 标签if1.3.2 标签if和where1.3.3 标签choose和when和otherwise1.4 UserController.java2 常用动态SQL标签2.1 标签set2.1.1 UserMapper.java2.1.2 UserMapper.xml2.1.3 UserController.ja…

wordpress后台更新后 前端没变化的解决方法

使用siteground主机的wordpress网站,会出现更新了网站内容和修改了php模板文件、js文件、css文件、图片文件后,网站没有变化的情况。 不熟悉siteground主机的新手,遇到这个问题,就很抓狂,明明是哪都没操作错误&#x…

网络编程(Modbus进阶)

思维导图 Modbus RTU(先学一点理论) 概念 Modbus RTU 是工业自动化领域 最广泛应用的串行通信协议,由 Modicon 公司(现施耐德电气)于 1979 年推出。它以 高效率、强健性、易实现的特点成为工业控制系统的通信标准。 包…

UE5 学习系列(二)用户操作界面及介绍

这篇博客是 UE5 学习系列博客的第二篇,在第一篇的基础上展开这篇内容。博客参考的 B 站视频资料和第一篇的链接如下: 【Note】:如果你已经完成安装等操作,可以只执行第一篇博客中 2. 新建一个空白游戏项目 章节操作,重…

IDEA运行Tomcat出现乱码问题解决汇总

最近正值期末周,有很多同学在写期末Java web作业时,运行tomcat出现乱码问题,经过多次解决与研究,我做了如下整理: 原因: IDEA本身编码与tomcat的编码与Windows编码不同导致,Windows 系统控制台…

利用最小二乘法找圆心和半径

#include <iostream> #include <vector> #include <cmath> #include <Eigen/Dense> // 需安装Eigen库用于矩阵运算 // 定义点结构 struct Point { double x, y; Point(double x_, double y_) : x(x_), y(y_) {} }; // 最小二乘法求圆心和半径 …

使用docker在3台服务器上搭建基于redis 6.x的一主两从三台均是哨兵模式

一、环境及版本说明 如果服务器已经安装了docker,则忽略此步骤,如果没有安装,则可以按照一下方式安装: 1. 在线安装(有互联网环境): 请看我这篇文章 传送阵>> 点我查看 2. 离线安装(内网环境):请看我这篇文章 传送阵>> 点我查看 说明&#xff1a;假设每台服务器已…

XML Group端口详解

在XML数据映射过程中&#xff0c;经常需要对数据进行分组聚合操作。例如&#xff0c;当处理包含多个物料明细的XML文件时&#xff0c;可能需要将相同物料号的明细归为一组&#xff0c;或对相同物料号的数量进行求和计算。传统实现方式通常需要编写脚本代码&#xff0c;增加了开…

LBE-LEX系列工业语音播放器|预警播报器|喇叭蜂鸣器的上位机配置操作说明

LBE-LEX系列工业语音播放器|预警播报器|喇叭蜂鸣器专为工业环境精心打造&#xff0c;完美适配AGV和无人叉车。同时&#xff0c;集成以太网与语音合成技术&#xff0c;为各类高级系统&#xff08;如MES、调度系统、库位管理、立库等&#xff09;提供高效便捷的语音交互体验。 L…

(LeetCode 每日一题) 3442. 奇偶频次间的最大差值 I (哈希、字符串)

题目&#xff1a;3442. 奇偶频次间的最大差值 I 思路 &#xff1a;哈希&#xff0c;时间复杂度0(n)。 用哈希表来记录每个字符串中字符的分布情况&#xff0c;哈希表这里用数组即可实现。 C版本&#xff1a; class Solution { public:int maxDifference(string s) {int a[26]…

【大模型RAG】拍照搜题技术架构速览:三层管道、两级检索、兜底大模型

摘要 拍照搜题系统采用“三层管道&#xff08;多模态 OCR → 语义检索 → 答案渲染&#xff09;、两级检索&#xff08;倒排 BM25 向量 HNSW&#xff09;并以大语言模型兜底”的整体框架&#xff1a; 多模态 OCR 层 将题目图片经过超分、去噪、倾斜校正后&#xff0c;分别用…

【Axure高保真原型】引导弹窗

今天和大家中分享引导弹窗的原型模板&#xff0c;载入页面后&#xff0c;会显示引导弹窗&#xff0c;适用于引导用户使用页面&#xff0c;点击完成后&#xff0c;会显示下一个引导弹窗&#xff0c;直至最后一个引导弹窗完成后进入首页。具体效果可以点击下方视频观看或打开下方…

接口测试中缓存处理策略

在接口测试中&#xff0c;缓存处理策略是一个关键环节&#xff0c;直接影响测试结果的准确性和可靠性。合理的缓存处理策略能够确保测试环境的一致性&#xff0c;避免因缓存数据导致的测试偏差。以下是接口测试中常见的缓存处理策略及其详细说明&#xff1a; 一、缓存处理的核…

龙虎榜——20250610

上证指数放量收阴线&#xff0c;个股多数下跌&#xff0c;盘中受消息影响大幅波动。 深证指数放量收阴线形成顶分型&#xff0c;指数短线有调整的需求&#xff0c;大概需要一两天。 2025年6月10日龙虎榜行业方向分析 1. 金融科技 代表标的&#xff1a;御银股份、雄帝科技 驱动…

观成科技:隐蔽隧道工具Ligolo-ng加密流量分析

1.工具介绍 Ligolo-ng是一款由go编写的高效隧道工具&#xff0c;该工具基于TUN接口实现其功能&#xff0c;利用反向TCP/TLS连接建立一条隐蔽的通信信道&#xff0c;支持使用Let’s Encrypt自动生成证书。Ligolo-ng的通信隐蔽性体现在其支持多种连接方式&#xff0c;适应复杂网…

铭豹扩展坞 USB转网口 突然无法识别解决方法

当 USB 转网口扩展坞在一台笔记本上无法识别,但在其他电脑上正常工作时,问题通常出在笔记本自身或其与扩展坞的兼容性上。以下是系统化的定位思路和排查步骤,帮助你快速找到故障原因: 背景: 一个M-pard(铭豹)扩展坞的网卡突然无法识别了,扩展出来的三个USB接口正常。…

未来机器人的大脑:如何用神经网络模拟器实现更智能的决策?

编辑&#xff1a;陈萍萍的公主一点人工一点智能 未来机器人的大脑&#xff1a;如何用神经网络模拟器实现更智能的决策&#xff1f;RWM通过双自回归机制有效解决了复合误差、部分可观测性和随机动力学等关键挑战&#xff0c;在不依赖领域特定归纳偏见的条件下实现了卓越的预测准…

Linux应用开发之网络套接字编程(实例篇)

服务端与客户端单连接 服务端代码 #include <sys/socket.h> #include <sys/types.h> #include <netinet/in.h> #include <stdio.h> #include <stdlib.h> #include <string.h> #include <arpa/inet.h> #include <pthread.h> …

华为云AI开发平台ModelArts

华为云ModelArts&#xff1a;重塑AI开发流程的“智能引擎”与“创新加速器”&#xff01; 在人工智能浪潮席卷全球的2025年&#xff0c;企业拥抱AI的意愿空前高涨&#xff0c;但技术门槛高、流程复杂、资源投入巨大的现实&#xff0c;却让许多创新构想止步于实验室。数据科学家…

深度学习在微纳光子学中的应用

深度学习在微纳光子学中的主要应用方向 深度学习与微纳光子学的结合主要集中在以下几个方向&#xff1a; 逆向设计 通过神经网络快速预测微纳结构的光学响应&#xff0c;替代传统耗时的数值模拟方法。例如设计超表面、光子晶体等结构。 特征提取与优化 从复杂的光学数据中自…