stm32学习笔记(10)spi

前言

SPI(Serial Peripheral Interface)是嵌入式系统中应用最广泛的同步串行通信协议之一,其高效的全双工通信和灵活的硬件设计使其成为传感器、存储器和显示模块的理想选择。本文从基本通信原理和实现两个方面来介绍SPI通信。

工作原理

SPI是一种全双工、同步、主从式的串行通信协议,其核心在于通过时钟信号(SCLK)同步主从设备之间的数据交换。以下从硬件接口、时序逻辑两个维度详细解析其工作原理。

硬件接口

SPI通过4根信号线实现通信:
1. SCLK(Serial Clock)
   由主机生成,控制数据传输节奏。  
   时钟频率决定通信速率(如10 MHz、20 MHz)。  

2. MOSI(Master Out Slave In) 
   主机向从机发送数据的单向通道。  

3. MISO(Master In Slave Out)
   从机向主机返回数据的单向通道。  

4. SS/CS(Slave Select/Chip Select)
   主机通过拉低此信号选择目标从机(低电平有效)。  
   多从机系统中,每个从机需独立的SS信号。

时序逻辑

SPI的时序逻辑由时钟极性(CPOL)和时钟相位(CPHA)共同定义,决定了数据在时钟信号(SCLK)的哪个边沿被采样和更新。

CPOL(Clock Polarity,时钟极性)
定义SCLK在空闲状态(无数据传输时)的电平:  
     CPOL=0:SCLK空闲时为低电平。  
     CPOL=1:SCLK空闲时为高电平。  

CPHA(Clock Phase,时钟相位)
定义数据在时钟的哪个边沿被采样:  
    CPHA=0:数据在SCLK的第一个边沿(从空闲状态跳变的边沿)被采样。  
    CPHA=1:数据在SCLK的第二个边沿(与第一个边沿相反的边沿)被采样。  

例如:

当CPOL=0, CPHA=0时(Mode 0)
空闲状态:SCLK为低电平。  
数据传输:  
主机在SCLK的上升沿发送数据(MOSI),从机在上升沿采样数据。

主机在SCLK的下降沿接收数据(MISO),从机在下降沿更新输出。  

当CPOL=1, CPHA=1时(Mode 3)
空闲状态:SCLK为高电平。  
数据传输:  
主机在SCLK的**上升沿**发送数据(MOSI),从机在上升沿采样数据。  

主机在SCLK的**下降沿**接收数据(MISO),从机在下降沿更新输出。

注意
1、数据必须在采样边沿到来前保持稳定。  例如,Mode 0中,MOSI数据需在SCLK上升沿前稳定。  
2、数据在采样边沿后仍需保持稳定一段时间。
3、最大SCLK频率由主从设备中较低速的一方决定(如从机支持10MHz,主机不可超过此速率)。 
4、主从设备模式必须一致 若主机使用Mode 0,从机也必须配置为Mode 0,否则数据采样错位,会导致通信失败。

测试示例

spi.c

void SPI1_Init(void)
{	 
	RCC->APB2ENR|=1<<2;       //PORTA时钟使能 	 
	RCC->APB2ENR|=1<<12;      //SPI1时钟使能 
		   
	//这里只针对SPI口初始化
	GPIOA->CRL&=0X000FFFFF; 
	GPIOA->CRL|=0XBBB00000;//PA5.6.7复用 	    
	GPIOA->ODR|=0X7<<5;    //PA5.6.7上拉
		
	SPI1->CR1|=0<<10;//全双工模式	
	SPI1->CR1|=1<<9; //软件nss管理
	SPI1->CR1|=1<<8;  

	SPI1->CR1|=1<<2; //SPI主机
	SPI1->CR1|=0<<11;//8bit数据格式	
	SPI1->CR1|=1<<1; //空闲模式下SCK为1 CPOL=1
	SPI1->CR1|=1<<0; //数据采样从第二个时间边沿开始,CPHA=1  
	SPI1->CR1|=7<<3; //Fsck=Fcpu/256
	SPI1->CR1|=0<<7; //MSBfirst   
	SPI1->CR1|=1<<6; //SPI设备使能
	SPI1_ReadWriteByte(0xff);//启动传输(主要作用:维持MOSI为高)		 
}   
//SPI1 速度设置函数
//SpeedSet:0~7
//SPI速度=fAPB2/2^(SpeedSet+1)
//APB2时钟一般为72Mhz
void SPI1_SetSpeed(u8 SpeedSet)
{
	SpeedSet&=0X07;			//限制范围
	SPI1->CR1&=0XFFC7; 
	SPI1->CR1|=SpeedSet<<3;	//设置SPI1速度  
	SPI1->CR1|=1<<6; 		//SPI设备使能 
} 
//SPI1 读写一个字节
//TxData:要写入的字节
//返回值:读取到的字节
u8 SPI1_ReadWriteByte(u8 TxData)
{		
	u16 retry=0;				 
	while((SPI1->SR&1<<1)==0)//等待发送区空	
	{
		retry++;
		if(retry>0XFFFE)return 0;
	}			  
	SPI1->DR=TxData;	 	  //发送一个byte 
	retry=0;
	while((SPI1->SR&1<<0)==0) //等待接收完一个byte  
	{
		retry++;
		if(retry>0XFFFE)return 0;
	}	  						    
	return SPI1->DR;          //返回收到的数据				    
}

spi.h

#ifndef __SPI_H
#define __SPI_H
#include "sys.h"
#define SPI_SPEED_2   		0
#define SPI_SPEED_4   		1
#define SPI_SPEED_8   		2
#define SPI_SPEED_16  		3
#define SPI_SPEED_32 		4
#define SPI_SPEED_64 		5
#define SPI_SPEED_128 		6
#define SPI_SPEED_256 		7
						  	    													  
void SPI1_Init(void);			 //初始化SPI口
void SPI1_SetSpeed(u8 SpeedSet); //设置SPI速度   
u8 SPI1_ReadWriteByte(u8 TxData);//SPI总线读写一个字节
		 
#endif

上述代码使用的是stm32mini板上的SPI1进行通信。

如果有差别请根据手册上的总线结构来寻找相应的接口。

与其他通信方式的对比

总的来说这几种通信方式各有优劣,在实际运用中还是要多加考虑。如果在第一次进行spi通信时出现差错,可以尝试降低通信速率重新测试。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值