news 2026/4/23 12:15:29

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/4/23 10:48:11

显卡驱动深度清理解决方案:驱动残留解决与系统优化指南

显卡驱动深度清理解决方案:驱动残留解决与系统优化指南 【免费下载链接】display-drivers-uninstaller Display Driver Uninstaller (DDU) a driver removal utility / cleaner utility 项目地址: https://gitcode.com/gh_mirrors/di/display-drivers-uninstaller…

作者头像 李华
网站建设 2026/4/23 12:14:28

FSMN-VAD能否用于直播切片?场景可行性分析

FSMN-VAD能否用于直播切片?场景可行性分析 你有没有遇到过这样的情况:刚结束一场两小时的技术直播,回看时发现干货只集中在三个15分钟片段里,其余全是寒暄、调试报错、网络卡顿重连——手动拖进度条找重点,光剪辑就花…

作者头像 李华
网站建设 2026/4/18 7:55:28

Qwen3-32B智能代理开发:Python爬虫数据采集与处理实战

Qwen3-32B智能代理开发:Python爬虫数据采集与处理实战 1. 智能爬虫代理的价值与应用场景 在数据驱动的商业环境中,网页数据采集已成为企业获取市场情报、竞品分析和用户洞察的重要手段。传统爬虫面临三大核心挑战:动态网页解析困难、反爬机…

作者头像 李华
网站建设 2026/4/14 8:45:16

Git-RSCLIP零样本分类效果实测:小样本场景下优于Fine-tuning基线

Git-RSCLIP零样本分类效果实测:小样本场景下优于Fine-tuning基线 1. 为什么遥感图像分类需要新思路? 你有没有遇到过这样的问题:手头只有几十张农田、几幅机场航拍图,想快速建一个分类器识别地物类型,但标注成本太高…

作者头像 李华
网站建设 2026/4/8 17:02:10

深度解析DDPG与TD3:连续动作空间中的强化学习优化策略

1. 连续动作空间中的强化学习挑战 在强化学习领域,连续动作空间问题一直是个棘手的问题。想象一下你在教机器人走路,它的每一步动作都需要精确控制关节角度和力度,这些动作值不是简单的"左转"或"右转"这样的离散指令&…

作者头像 李华
网站建设 2026/4/23 12:14:51

IndexTTS-2-LLM部署问题全解:scipy依赖冲突处理步骤详解

IndexTTS-2-LLM部署问题全解:scipy依赖冲突处理步骤详解 1. 为什么你启动IndexTTS-2-LLM时总卡在“ImportError: cannot import name cython_special”? 你不是一个人。 刚拉下kusururi/IndexTTS-2-LLM镜像,执行python app.py或uvicorn启动…

作者头像 李华