STM32 SPI 收发数据 ---规则 + 问题解析

规则:

1) 
高速同步串行口。3~4线接口(CS ,CLK ,MOSI,MISO),收发独立、可同步进行。
2)SPI分为主从模式,主模式提供时钟和片选选择信号.
3) 模式控制:CPOL用来控制时钟信号(clk)在空闲时候的状态;CPHA用来控制采样时刻时CLK的边缘动作。
CPOL CPHA 模式:
0 0 CLK空闲时为低电平,CLK上升沿采样数据。
0 1 CLK空闲为低电平,CLK下降沿采样数据。
1 0 CLK空闲时为高电平,CLK上升沿采样数据。
1 1 CLK空闲时为高电平,CLK下降沿采样数据。


1)SPI配置(3.01库):
SPI_InitStructure.SPI_Direction = SPI_Direction_2Lines_FullDuplex; //双工模式
SPI_InitStructure.SPI_Mode = SPI_Mode_Master; //SPI主模式
SPI_InitStructure.SPI_DataSize = SPI_DataSize_8b; //8bit数据
SPI_InitStructure.SPI_CPOL = SPI_CPOL_High; //CLK空闲时为高电平 
SPI_InitStructure.SPI_CPHA = SPI_CPHA_2Edge; //CLK上升沿采样,因为上升沿是第二个边沿动作,所以也可以理解为第二个边沿采样
SPI_InitStructure.SPI_NSS = SPI_NSS_Soft; //片选用软件控制
SPI_InitStructure.SPI_BaudRatePrescaler = SPI_BaudRatePrescaler_4; //SPI频率
SPI_InitStructure.SPI_FirstBit = SPI_FirstBit_MSB; //高位在前
SPI_InitStructure.SPI_CRCPolynomial = 7; //crc7,stm32spi带硬件ecc
SPI_Init(SPI1, &SPI_InitStructure);


2)CS信号:
主模式下要为从设备提供片选信号,值得注意的是STM32的主频相当较高,要提防数据没有完全发送前拉高CS信号。 


3)SPI读写:(非中断模式)
a)写一个字节:
while (SPI_I2S_GetFlagStatus(SPI1, SPI_I2S_FLAG_TXE) == RESET);
//确保发生前Buffer为空,也就是说上一次已经发生完成
SPI_I2S_SendData(SPI1, Data); //往寄存器中写入一个字节
while (SPI_I2S_GetFlagStatus(SPI1, SPI_I2S_FLAG_RXNE) == RESET);
//等待接受到一个字节数据,为什么要这么做?加这一句的原因是为了确保这个字节已经发送出去,因为发生和接受是并行同步进行,那就是说你发生出去一个字节意味着你收到一个字节。所以这样判断完全没有问题,再说必要性,如果你不加这句你就会容易犯过早拉高CS信号的错误,你想想如果在SPI_I2S_SendData(SPI1, Data)后面立即拉高CS是什么后果。
SPI_I2S_ReceiveData(SPI1); //都会接收到的数据,看起来没什么必要,但以用stm32的经验推荐这样做,也许会有意想不到的收获。


SPI_Writebyte(u8 data)
{
while (SPI_I2S_GetFlagStatus(SPI1, SPI_I2S_FLAG_TXE) == RESET);
SPI_I2S_SendData(SPI1, Data); 
while (SPI_I2S_GetFlagStatus(SPI1, SPI_I2S_FLAG_RXNE) == RESET); 
SPI_I2S_ReceiveData(SPI1); 
}


b)读一个字节:


读的时候要注意一个问题,因为从模式是没法提供时钟的,所以主模式下必须要在接收的同时提供时钟。办法就是发送一个字节来实现,因为还是上面说的,发送一个字节就意味着收到一个字节,代码和写完全一样,只要把读出来的字节保存即可。


u8 SPI_Readbyte(u8 data)
{ 
while (SPI_I2S_GetFlagStatus(SPI1, SPI_I2S_FLAG_TXE) == RESET);
SPI_I2S_SendData(SPI1, Data);    //发送与应用不相关的数据,为后续的读数据提供时钟信号
while (SPI_I2S_GetFlagStatus(SPI1, SPI_I2S_FLAG_RXNE) == RESET); 
return SPI_I2S_ReceiveData(SPI1); 
}


总结:上面的程序是最求稳定而设定的,如果你对速度有要求,你可以做相应的精简,比如读写直接对寄存器进行操作,另外配置SPI前要对从模式的模式了解清楚,包括从设备支持的时钟范围和模式(CPOL,CPHA状态)。

以上出自:STM32 SPI使用知识


问题及解析:

1、全双工模式下,发送一个8位的数据,接收到的数据是什么? 

 经实验观察,没看出什么联系。当然,这里所说的是发送无用的数据,而非 某些特定环境 说:发送一个数据(命令),应该能够接收到什么.....


2、接收数据时,示波器观察到MISO引脚 不是方波,下降沿特别圆滑,看着像是电容放电的节奏。

那阵子貌似在弄 ADS1256这款A/D片子,遇到这么个情况。  最后定位为 :

SPI配置:SPI_InitStructure.SPI_BaudRatePrescaler = **;---合适就好,过犹不及................



STM32SPI外设是一种串行外设接口,用于在微控制器和外部设备之间进行通信。SPI外设可以配置为主机模式或从机模式。在主机模式下,STM32产生SCK信号,控制通信的时序;在从机模式下,STM32接收外部设备提供的SCK信号。\[3\] SPI外设的配置包括以下几个方面: 1. 单双向模式:设置SPI数据传输方向,可以是单向或双向。 2. 主/从机模式:设置SPI的工作模式,可以是主机模式或从机模式。 3. 数据长度:设置SPI数据长度,可以是8位或16位。 4. 时钟极性和相位:设置SPI的时钟极性和相位,用于控制数据采样的时机。 5. NSS引脚控制:设置NSS引脚由硬件控制还是软件控制。 6. 时钟分频因子:设置SPI的时钟分频因子,用于控制通信速率。 7. MSB/LSB先行:设置数据传输的位序,是从高位到低位还是从低位到高位。 8. CRC校验表达式:设置CRC校验的表达式,用于数据完整性校验。\[1\] 在配置完成后,可以调用库函数SPI_Init将配置写入寄存器,并调用SPI_Cmd使能SPI外设。在数据传输过程中,可以通过检查寄存器位来了解发送接收的状态,并根据需要进行处理。也可以使用DMA方式来收发数据。\[2\] 总结起来,STM32SPI外设可以通过配置来实现不同的工作模式和通信参数,以满足不同应用的需求。 #### 引用[.reference_title] - *1* *3* [STM32SPI详解](https://blog.csdn.net/wangjiaweiwei/article/details/125646382)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^koosearch_v1,239^v3^insert_chatgpt"}} ] [.reference_item] - *2* [STM32SPI详细解析](https://blog.csdn.net/qq_43940175/article/details/123460281)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^koosearch_v1,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]
评论 7
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值