news 2026/6/15 19:17:42

STM32 SPI通信协议实战指南——从模式配置到FLASH读写

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
STM32 SPI通信协议实战指南——从模式配置到FLASH读写

1. SPI协议基础与STM32硬件连接

SPI(Serial Peripheral Interface)是一种高速全双工同步串行通信协议,在嵌入式系统中广泛应用。我第一次接触SPI是在做一个温湿度传感器项目时,当时被它简单的四线制连接方式惊艳到了——相比I2C的两根线,SPI虽然多用了两根线,但传输速度能轻松达到MHz级别。

STM32的SPI接口通常包含以下四个信号线:

  • SCK:时钟线,由主机产生
  • MOSI:主机输出从机输入
  • MISO:主机输入从机输出
  • NSS:片选信号(低电平有效)

以STM32F103为例,SPI1的默认引脚映射是:

PA4 - NSS PA5 - SCK PA6 - MISO PA7 - MOSI

硬件连接时有个容易踩的坑:如果使用硬件NSS模式,需要特别注意上拉电阻的配置。我在一个项目中因为没加10K上拉电阻,导致通信一直不稳定,后来用逻辑分析仪抓包才发现NSS信号有抖动。

2. SPI模式配置关键细节

SPI有四种工作模式,由CPOL(时钟极性)和CPHA(时钟相位)组合决定。刚开始学的时候我总记混这两个参数,后来发现可以用"时钟空闲状态"和"采样边沿"来理解:

  • CPOL=0:时钟空闲时为低电平
  • CPOL=1:时钟空闲时为高电平
  • CPHA=0:在第一个时钟边沿采样
  • CPHA=1:在第二个时钟边沿采样

最常用的模式是Mode0和Mode3。FLASH芯片通常支持这两种模式,比如W25Q64芯片就同时兼容两种模式。配置时一定要确保主从设备模式一致,我有次调试时主机设成Mode0而从机是Mode3,结果数据完全对不上。

STM32的SPI初始化代码示例:

SPI_InitTypeDef SPI_InitStruct; 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; // Mode3 SPI_InitStruct.SPI_CPHA = SPI_CPHA_2Edge; // Mode3 SPI_InitStruct.SPI_NSS = SPI_NSS_Soft; // 软件控制片选 SPI_InitStruct.SPI_BaudRatePrescaler = SPI_BaudRatePrescaler_4; // 18MHz/4=4.5MHz SPI_InitStruct.SPI_FirstBit = SPI_FirstBit_MSB; SPI_Init(SPI1, &SPI_InitStruct); SPI_Cmd(SPI1, ENABLE);

3. FLASH芯片读写实战

以W25Q128FV FLASH芯片为例,其典型操作流程包括:写使能、页编程、扇区擦除、读取数据等。在实际项目中,我发现有几点需要特别注意:

  1. 写操作前必须先发WREN指令
void Flash_WriteEnable(void) { CS_LOW(); SPI_SendByte(W25X_WriteEnable); CS_HIGH(); Delay_us(5); // 等待写使能完成 }
  1. 页编程要注意地址对齐:W25Q128的页大小是256字节,跨页写入会导致数据回卷。我有次没注意这点,导致关键配置数据被覆盖。

  2. 读取状态寄存器判断忙状态

uint8_t Flash_WaitBusy(void) { uint8_t status; do { CS_LOW(); SPI_SendByte(W25X_ReadStatusReg); status = SPI_SendByte(Dummy_Byte); CS_HIGH(); } while(status & 0x01); // 检查BUSY位 return status; }

完整的读数据函数示例:

void Flash_ReadData(uint32_t addr, uint8_t *pData, uint16_t len) { CS_LOW(); SPI_SendByte(W25X_ReadData); SPI_SendByte((addr >> 16) & 0xFF); // 24位地址 SPI_SendByte((addr >> 8) & 0xFF); SPI_SendByte(addr & 0xFF); while(len--) { *pData++ = SPI_SendByte(Dummy_Byte); } CS_HIGH(); }

4. 常见问题排查技巧

调试SPI通信时我总结了几条实用经验:

  1. 用示波器或逻辑分析仪检查信号质量:曾经遇到SCK信号振铃导致通信失败,后来在SCK线上加了个33Ω电阻解决了问题。

  2. 检查供电电压稳定性:FLASH芯片在电压不稳时可能出现奇怪的行为,建议在VCC对地加个0.1μF去耦电容。

  3. 注意时钟相位关系:Mode0和Mode3的主要区别在于第一个时钟边沿是上升还是下降,用错模式会导致采样点不对。

  4. DMA传输优化:对于大数据量传输,可以使用DMA提高效率。但要注意设置正确的数据宽度和FIFO阈值:

DMA_InitStruct.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte; DMA_InitStruct.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte; DMA_InitStruct.DMA_BufferSize = BufferSize; SPI_I2S_DMACmd(SPI1, SPI_I2S_DMAReq_Tx, ENABLE);
  1. 跨字节传输问题:当连续发送多字节时,建议在字节间加入微小延时(1-2个时钟周期),特别是对低速从设备。
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/6/15 12:17:42

闲鱼数据采集自动化工具:一站式解决方案

闲鱼数据采集自动化工具:一站式解决方案 【免费下载链接】xianyu_spider 闲鱼APP数据爬虫 项目地址: https://gitcode.com/gh_mirrors/xia/xianyu_spider 在当今电商数据分析领域,手动采集闲鱼商品信息面临效率低下、数据不完整、重复劳动等痛点&…

作者头像 李华
网站建设 2026/6/15 12:23:22

CarPropertyService深度优化指南:如何设计高可用的车辆属性订阅系统

CarPropertyService深度优化指南:构建工业级车辆属性订阅系统 在智能座舱系统开发中,车辆属性管理一直是核心挑战之一。随着车载功能复杂度呈指数级增长,传统的事件处理机制已难以满足高并发、低延迟的需求。本文将揭示如何通过CarPropertySe…

作者头像 李华
网站建设 2026/6/15 17:17:31

Xinference-v1.17.1安装与验证:从零开始搭建AI推理环境

Xinference-v1.17.1安装与验证:从零开始搭建AI推理环境 1. 为什么选Xinference?一个统一接口跑遍所有开源模型 你是不是也遇到过这些情况:想试一个新发布的LLM,得单独装HuggingFace Transformers、改一堆配置、调CUDA版本&#…

作者头像 李华
网站建设 2026/6/15 14:42:27

【2024升级版】闲鱼数据猎人:3步掌握AI驱动的电商情报采集技术

【2024升级版】闲鱼数据猎人:3步掌握AI驱动的电商情报采集技术 【免费下载链接】xianyu_spider 闲鱼APP数据爬虫 项目地址: https://gitcode.com/gh_mirrors/xia/xianyu_spider 价值定位:为什么选择闲鱼数据猎人 在电商情报竞争白热化的今天&…

作者头像 李华
网站建设 2026/6/15 13:25:21

ChatGLM-6B市场调研支持:用户反馈自动归类与总结

ChatGLM-6B市场调研支持:用户反馈自动归类与总结 1. 为什么市场团队需要一个“会读反馈”的AI助手? 你有没有遇到过这样的场景: 刚结束一场大型产品上线,客服后台涌进2000条用户留言; 电商大促后,商品评论…

作者头像 李华
网站建设 2026/6/15 13:20:56

从零实现基于xTaskCreate的简单调度系统

以下是对您提供的博文内容进行 深度润色与工程化重构后的版本 。我以一位深耕嵌入式系统多年、既写过百万行裸机代码也调过千级任务 FreeRTOS 系统的工程师视角,彻底重写了全文—— 去除所有AI腔调、模板化表达和教科书式结构,代之以真实开发现场的语言节奏、踩坑经验、设…

作者头像 李华