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芯片为例,其典型操作流程包括:写使能、页编程、扇区擦除、读取数据等。在实际项目中,我发现有几点需要特别注意:
- 写操作前必须先发WREN指令:
void Flash_WriteEnable(void) { CS_LOW(); SPI_SendByte(W25X_WriteEnable); CS_HIGH(); Delay_us(5); // 等待写使能完成 }页编程要注意地址对齐:W25Q128的页大小是256字节,跨页写入会导致数据回卷。我有次没注意这点,导致关键配置数据被覆盖。
读取状态寄存器判断忙状态:
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通信时我总结了几条实用经验:
用示波器或逻辑分析仪检查信号质量:曾经遇到SCK信号振铃导致通信失败,后来在SCK线上加了个33Ω电阻解决了问题。
检查供电电压稳定性:FLASH芯片在电压不稳时可能出现奇怪的行为,建议在VCC对地加个0.1μF去耦电容。
注意时钟相位关系:Mode0和Mode3的主要区别在于第一个时钟边沿是上升还是下降,用错模式会导致采样点不对。
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-2个时钟周期),特别是对低速从设备。