STM32.SPI

SPI(Serial Peripheral Interface)串行外设接口,是一种高速,全双工,同步的只占用四根线的通信总线。
四根线:
MISO(Master Input Slave Output)主设备数据输入,从设备数据输出
MOSI(Master Output Slave Input) 主设备数据输出,从设备数据输入
SCLK(Serial Clock)时钟信号,由主设备产生;
CS (Chip Select)从设备使能信号(低电平使能),由主设备控制

CS是主设备选择从设备的控制信号,增加CS片选就可以增加从机设备,因是主从方式工作,只有一个主设备。
SPI是串行通讯协议,所以说数据是一位一位的传输的,这就引出了SCLK(提供时钟脉冲),在时钟脉冲的基础上通过MISO或MOSI进行数据传。
在这里插入图片描述

主从之间的数据传输流程图,在传输时用户可选择MSB或LSB先行,通过SPI_CR1寄存器LSBFIRST位设置,SPI的初始化就是对SPI_CR1寄存器进行配置
在这里插入图片描述

SPI_CR寄存器的CPOL和CPHA位,有四种时序关系,通过配置去选择在时钟的哪个边沿进行捕获
例如
在这里插入图片描述
SPI配置(这里是硬件SPI,所以也可以进行软件SPI,但那个需要你对SPI通信的时序了解)

void SPI1_Init(void)
{

	GPIO_InitTypeDef GPIO_InitStruct;
	SPI_InitTypeDef SPI_InitStruct;

	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA | RCC_APB2Periph_AFIO, ENABLE);
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_SPI1, ENABLE);

	GPIO_InitStruct.GPIO_Pin = GPIO_Pin_4;
	GPIO_InitStruct.GPIO_Mode =	GPIO_Mode_Out_PP;
	GPIO_InitStruct.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_Init(GPIOA,&GPIO_InitStruct);
	 //SCK & MOSI
	GPIO_InitStruct.GPIO_Pin = GPIO_Pin_5 | GPIO_Pin_7;
  	GPIO_InitStruct.GPIO_Speed = GPIO_Speed_50MHz;
 	GPIO_InitStruct.GPIO_Mode = GPIO_Mode_AF_PP;
 	GPIO_Init(GPIOA, &GPIO_InitStruct);
 	//MISO
 	GPIO_InitStruct.GPIO_Pin = GPIO_Pin_6;
 	GPIO_InitStruct.GPIO_Mode = GPIO_Mode_IN_FLOATING;
 	GPIO_Init(GPIOA, &GPIO_InitStruct);
	
	SPI_InitStruct.SPI_Direction = SPI_Direction_2Lines_FullDuplex;
	SPI_InitStruct.SPI_Mode = SPI_Mode_Master;
	SPI_InitStruct.SPI_DataSize = SPI_DataSize_16b;
	SPI_InitStruct.SPI_CPOL = SPI_CPOL_High;							
	SPI_InitStruct.SPI_CPHA = SPI_CPHA_2Edge;	
	SPI_InitStruct.SPI_NSS = SPI_NSS_Soft;
	SPI_InitStruct.SPI_BaudRatePrescaler = SPI_BaudRatePrescaler_8;
	SPI_InitStruct.SPI_FirstBit = SPI_FirstBit_MSB;
	SPI_InitStruct.SPI_CRCPolynomial = 7;
	SPI_Init(SPI1,&SPI_InitStruct);
	SPI_Cmd(SPI1,ENABLE);
	//默认上拉
	GPIOA->BSRR = GPIO_Pin_4;
}

SPI读写

u8 SPI1_RW(u8 data)
{

	u8 retry = 200;									//类似于等待时间,超时退出
	while((SPIx->SR & SPI_I2S_FLAG_BSY)==SET)		//判断是否发送完成
		if((retry--)==0) return printf("发送等待失败!\n");
	SPI1->DR = data1;														
	retry = 200;
	while((SPIx->SR & SPI_I2S_FLAG_RXNE)==RESET)	//判断接收缓冲器是否接收完成
		if((retry--)==0) return printf("发送等待失败!\n");
	return SPI1->DR;
}

这里一次发送16位数据,当需要发送32位数据时,你可能会想到发两次来达到这个要求,通过示波器你会发现有个问题,两个16位数据之间的时间问题
在这里插入图片描述

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Mapoplus

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值