news 2026/5/8 10:33:52

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/5/6 14:45:11

开源串流技术突破:自建游戏服务器实现毫秒级延迟优化的探索之旅

开源串流技术突破:自建游戏服务器实现毫秒级延迟优化的探索之旅 【免费下载链接】Sunshine Sunshine: Sunshine是一个自托管的游戏流媒体服务器,支持通过Moonlight在各种设备上进行低延迟的游戏串流。 项目地址: https://gitcode.com/GitHub_Trending/…

作者头像 李华
网站建设 2026/5/5 12:30:37

4步掌握ncmdump高效转换技术:专业格式处理指南

4步掌握ncmdump高效转换技术:专业格式处理指南 【免费下载链接】ncmdump 项目地址: https://gitcode.com/gh_mirrors/ncmd/ncmdump 在数字化内容管理领域,文件转换效率提升已成为优化工作流的关键环节。无论是音乐爱好者处理加密音频文件&#x…

作者头像 李华
网站建设 2026/5/5 8:39:23

MedGemma Medical Vision Lab详细步骤:从零部署多模态医学AI研究平台

MedGemma Medical Vision Lab详细步骤:从零部署多模态医学AI研究平台 1. 这不是诊断工具,而是你的医学AI研究搭档 你有没有试过——刚下载好一张胸部X光片,想快速验证某个视觉-语言对齐实验的效果,却卡在环境配置上?…

作者头像 李华
网站建设 2026/5/5 8:39:22

一键部署MedGemma X-Ray:医疗影像智能分析如此简单

一键部署MedGemma X-Ray:医疗影像智能分析如此简单 你是否曾为一张胸部X光片反复比对标准图谱?是否在带教学生时,苦于找不到足够多、质量高、带结构化解读的典型片例?又或者,在科研中需要快速验证某种影像特征与AI识别…

作者头像 李华
网站建设 2026/5/5 8:39:23

SenseVoice Small语音识别实测:多语言支持+GPU加速体验

SenseVoice Small语音识别实测:多语言支持GPU加速体验 你有没有试过把一段会议录音拖进语音识别工具,结果等了半分钟,只出来几行断断续续的字?或者刚切到粤语模式,系统就报错“模型未加载”?又或者上传一个…

作者头像 李华