当前位置:   article > 正文

SPI 通信协议_spi datasize

spi datasize

在这里插入图片描述

【 1. 概述 】

  • SPI ( Serial Peripheral interface),串行外围设备接口。由摩托罗拉公司设计并实现。
  • 特点
    SPI 是一种 高速、全双工、同步、串行 的通信总线,采用 主机(Master)-从机(Slave)的通信方式

1.1 SPI 引脚接口

  • MISO (Master Input Slave Output):主设备数据输入,从设备数据输出。
  • MOSI (Master Output Slave Input):主设备数据输出,从设备数据输入。
  • SCLK (Serial Clock),也叫做 SCK :串行时钟信号。SPI规定只能由主设备产生,并根据此时钟信号一位一位地传输。
  • SS (Slave Select),也叫做 NSS、CS、SSEL :从设备片选信号,由主设备控制,为一个主设备连接多个从设备提供了可能。

【 2. 基本原理 】

2.1 全双工特征下的SPI传输

  • 没有只进行读操作或者只进行写操作的说法,因为是全双工的传输方式:写操作和读操作是同步完成的。每次SPI传输数据时,主从设备都是在互相交换信息,即主/从机发一个数据必然会收到一个数据。
    • 如果只进行写操作,主机只需忽略接收到的字节;反之,若主机要读取从机的一个字节,就必须发送一个空字节来引发从机的发送。
      在这里插入图片描述
  • 数据交换与移位寄存器
    SPI是一个环形总线结构,主要是在 sck 的控制下,两个双向移位寄存器进行数据交换。
    SPI总线的主机和从机都有一个移位寄存器。当主机向自己的移位寄存器写入数据时,数据会通过MOSI信号线进入到从机的移位寄存器;类似地,从机移位寄存器里的数据,通过MISO信号线进入到主机的移位寄存器。这样,主机和从机就完成了一次数据交换。下面这张图,是SPI通信的简明原理图:
    在这里插入图片描述

2.2 二线式、三线式的SPI

  • 二线式SPI
    • 如果主机只给从机发送命令,从机不需要回复数据的时候,那MISO就可以不要。
    • 而在主机只读取从机的数据,不需要给从机发送指令的时候,那MOSI可以不要。
  • 三线式SPI
    • 当一个主机一个从机的时候,从机的片选有时可以固定为有效电平而一直处于使能状态,那么CS可以不要;此时如果再加上主机只给从机发送数据,那么CS和MISO都可以不要;如果主机只读取从机送来的数据,CS和MOSI都可以不要。

2.3 SPI 模式

  • 问题背景
    主机和从机要交换数据,就牵涉到一个问题,即主机在什么时刻输出数据到MOSI上而从机在什么时刻采样这个数据,或者从机在什么时刻输出数据到MISO上而主机什么时刻采样这个数据。同步通信的特点就是所有数据的变化和采样都是伴随着时钟沿进行的,也就是说数据总是在时钟的边沿附近变化或被采样。那么,如果一方在上升沿输出数据到MOSI上,另一方就只能在下降沿去采样这个数据;反之如果一方在下降沿输出数据,那么另一方就必须在上升沿采样这个数据。

  • SPI共有4种通信模式,主机和从机需要工作在相同的模式下才能正常通信。
    这4种模式是基于CPOL和CPHA进行划分:
    在这里插入图片描述在这里插入图片描述

  • CPHA (Clock Phase),时钟相位 ,用于决定何时进行信号采样。

    • CPHA=1,就表示数据的输出是在一个时钟周期的第一个沿上,至于这个沿是上升沿还是下降沿,这要是CPOL的值而定,CPOL=1那就是下降沿,CPOL=0就是上升沿。那么数据的采样自然就是在第二个沿上了。
    • CPHA=0,就表示数据的采样是在一个时钟周期的第一个沿上,如果CPHA=1就取下降沿,CPHA=0就取上升沿。那么数据的输出自然就在第二个沿上了。

仔细想一下,当CPHA=0时,这里会有一个问题:就是当一帧数据开始传输第一bit时,在第一个时钟沿上就采样该数据了,那么它是在什么时候输出来的呢?有两种情况:一是SS片选引脚使能的边沿,二是上一帧数据的最后一个时钟沿,有时两种情况还会同时生效。

  • CPOL(Clock Polarity),时钟极性 ,用于决定空闲时刻的电平状态。通信的整个过程分为空闲状态和通信状态,SCLK在数据发送之前和数据发送之后的状态为 空闲状态
    • 如果CPOL=0,那么空闲状态SCLK是低电平。
    • 如果CPOL=1,那么空闲状态SCLK是高电平。

2.4 SPI 时序图

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

根据时序图推导SPI模式

  • 我们对CPOL=1,CPHA=1即 SPI 模式3进行讲解:
    在这里插入图片描述
  • 当数据发送前以及发送后,SCK都是高电平,因此也可以直接通过时序图推出 CPOL=1
  • 可以看出,在SCK第一个沿的时候,MOSI和MISO的数据会发生变化,而在SCK第二个沿的时候,MOSI和MISO的数据是稳定的。又因为 SPI 只能在数据稳定时进行采样,数据不稳定时输出,所以图中是在第一个边沿输出第二个边沿采样,即 CPHA=1
  • 注意最后最隐蔽的SSEL片选,一般情况下,这个引脚通常用来决定是哪个从机和主机进行通信。

4. 通信流程

  1. 主机先将NSS片选信号拉低,这样保证开始接收数据。
  2. 当从机接收端检测到时钟的边沿信号时,它将立即读取数据线上的信号,这样就得到了一位数据(1bit);
    由于时钟是随数据一起发送的,因此指定数据的传输速度并不重要,尽管设备将具有可以运行的最高速度。
  3. 主机发送到从机时:主机产生相应的时钟信号,然后数据一位一位地将从MOSI信号线上进行发送到从机;
  4. 主机接收从机数据:如果从机需要将数据发送回主机,则主机将继续生成预定数量的时钟信号,并且从机会将数据通过MISO信号线发送;

在这里插入图片描述

  • 实例
    假设下面主机的8位寄存器装的是待发送的数据10101010,上升沿发送、下降沿接收、高位先发送。那么第一个上升沿来的时候 MOSI 数据将=1,主机寄存器=0101010x。下降沿到来的时候,MISO上的电平将锁存到从机寄存器中去,那么这时从机寄存器=0101010+MISO,这样在 8个时钟脉冲以后,两个寄存器的内容互相交换一次。这样就完成里一个 SPI 时序。
    在这里插入图片描述

【 3. SPI底层驱动 】

//以下是SPI模块的初始化代码,配置成主机模式 						  
//SPI口初始化
//这里是对SPI1的初始化
void SPI1_Init(void)
{	 
	GPIO_InitTypeDef  GPIO_InitStructure;
	SPI_InitTypeDef  SPI_InitStructure;
	RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOB, ENABLE);//使能GPIOB时钟
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_SPI1, ENABLE);//使能SPI1时钟
 
  	//GPIOFB3,4,5初始化设置
  	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_3|GPIO_Pin_4|GPIO_Pin_5;//PB3~5复用功能输出	
  	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF;//复用功能
  	GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;//推挽输出
  	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_100MHz;//100MHz
  	GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_UP;//上拉
 	GPIO_Init(GPIOB, &GPIO_InitStructure);//初始化
	
	GPIO_PinAFConfig(GPIOB,GPIO_PinSource3,GPIO_AF_SPI1); //PB3复用为 SPI1
	GPIO_PinAFConfig(GPIOB,GPIO_PinSource4,GPIO_AF_SPI1); //PB4复用为 SPI1
	GPIO_PinAFConfig(GPIOB,GPIO_PinSource5,GPIO_AF_SPI1); //PB5复用为 SPI1
 
	//这里只针对SPI口初始化
	RCC_APB2PeriphResetCmd(RCC_APB2Periph_SPI1,ENABLE);//复位SPI1
	RCC_APB2PeriphResetCmd(RCC_APB2Periph_SPI1,DISABLE);//停止复位SPI1

	SPI_InitStructure.SPI_Direction = SPI_Direction_2Lines_FullDuplex;  //设置SPI单向或者双向的数据模式:SPI设置为双线双向全双工
	SPI_InitStructure.SPI_Mode = SPI_Mode_Master;		//设置SPI工作模式:设置为主SPI
	SPI_InitStructure.SPI_DataSize = SPI_DataSize_8b;		//设置SPI的数据大小:SPI发送接收8位帧结构
	SPI_InitStructure.SPI_CPOL = SPI_CPOL_High;		//串行同步时钟的空闲状态为高电平
	SPI_InitStructure.SPI_CPHA = SPI_CPHA_2Edge;	//串行同步时钟的第二个跳变沿(上升或下降)数据被采样
	SPI_InitStructure.SPI_NSS = SPI_NSS_Soft;		//NSS信号由硬件(NSS管脚)还是软件(使用SSI位)管理:内部NSS信号有SSI位控制
	SPI_InitStructure.SPI_BaudRatePrescaler = SPI_BaudRatePrescaler_256;		//定义波特率预分频的值:波特率预分频值为256
	SPI_InitStructure.SPI_FirstBit = SPI_FirstBit_MSB;	//指定数据传输从MSB位还是LSB位开始:数据传输从MSB位开始
	SPI_InitStructure.SPI_CRCPolynomial = 7;	//CRC值计算的多项式
	SPI_Init(SPI1, &SPI_InitStructure);  //根据SPI_InitStruct中指定的参数初始化外设SPIx寄存器
 
	SPI_Cmd(SPI1, ENABLE); //使能SPI外设

	SPI1_ReadWriteByte(0xff);//启动传输		 
}   
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39
  • 40
  • 41
//SPI1速度设置函数
//SPI速度=fAPB2/分频系数
//@ref SPI_BaudRate_Prescaler:SPI_BaudRatePrescaler_2~SPI_BaudRatePrescaler_256  
//fAPB2时钟一般为84Mhz:
void SPI1_SetSpeed(u8 SPI_BaudRatePrescaler)
{
  	assert_param(IS_SPI_BAUDRATE_PRESCALER(SPI_BaudRatePrescaler));//判断有效性
	SPI1->CR1&=0XFFC7;//位3-5清零,用来设置波特率
	SPI1->CR1|=SPI_BaudRatePrescaler;	//设置SPI1速度 
	SPI_Cmd(SPI1,ENABLE); //使能SPI1
} 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
//SPI1 读写一个字节
//TxData:要写入的字节
//返回值:读取到的字节
u8 SPI1_ReadWriteByte(u8 TxData)
{		 			 
 
  	while (SPI_I2S_GetFlagStatus(SPI1, SPI_I2S_FLAG_TXE) == RESET){}//等待发送区空  
	
	SPI_I2S_SendData(SPI1, TxData); //通过外设SPIx发送一个byte  数据
		
   	while (SPI_I2S_GetFlagStatus(SPI1, SPI_I2S_FLAG_RXNE) == RESET){} //等待接收完一个byte  
 
	return SPI_I2S_ReceiveData(SPI1); //返回通过SPIx最近接收的数据	
 		    
}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/天景科技苑/article/detail/944391
推荐阅读
相关标签
  

闽ICP备14008679号