STM32-SPI通信
1 SPI的基本介绍
1.1 SPI的简介
SPI,是英语Serial Peripheral interface的缩写,顾名思义就是串行外围设备接口,是Motorola首先在其MC68HCXX系列处理器上定义的。
SPI主要用于MCU和一些外设进行通信的场合,例如:EEPROM、Flash、AD转换器等一些应用中,还有数字信号处理器和数字信号解码器之间。SPI是一种高速的,全双工,同步的通信总线,并且在芯片的管脚上只占用四根线,这里全双工指的是可以在同一时刻设备进行接收和发送同时进行,它有别于CAN总线或者RS585总线,因为这些总线在同一时刻只能进行数据的单向传输。换句话说,对于一个设备在同一时刻只能接收或者发送数据。同步通信指的是一种比特同步通信技术,要求发收双方具有同频同相的同步时钟信号,SPI是通过CLK和相位实现这一点的。SPI节约了芯片的管脚,同时为PCB的布局上节省空间,提供方便,正是出于这种简单易用的特性,现在越来越多的芯片集成了这种通信协议。
SPI分为主、从两种模式,一个SPI通讯系统需要包含一个(且只能是一个)主设备,一个或多个从设备。SPI接口的读写操作,都是由主设备发起。当存在多个从设备时,通过各自的片选信号进行管理。
优点:支持全双工通信、通信简单、数据传输速率快;
缺点:没有指定的流控制,没有应答机制确认是否接收到数据,所以跟IIC总线协议比较在数据的可靠性上有一定的缺陷。
STM32中SPI接口的特征:
1)3线全双工同步传输;
2)8或16位传输帧格式选择;
3)主或从操作,支持多主模式;
4)主模式和从模式下均可以由软件或硬件进行NSS管理:主/从操作模式的动态改变;
5)可编程的时钟极性和相位;
6)可编程的数据顺序,MSB在前或LSB在前;
7)可触发中断的专用发送和接收标志;
8)SPI总线忙状态标志;
9)支持可靠通信的硬件CRC;
10)可触发中断的主模式故障、过载以及CRC错误标志;
11)支持DMA功能的1字节发送和接收缓冲器:产生发送和接受请求。
2 SPI协议
2.1 SPI引脚说明
SPI的通信原理很简单,它以主从方式工作,这种模式通常有一个主设备和一个或多个从设备,需要至少4根线,事实上3根也可以(单向传输时)。这四根线分别是MISO、MOSI、SCLK、CS,具体的描述见下表:
名称 | 描述 |
MISO | 主机输入从机输出,主机通过该线接收数据,从机通过该线发送数据 |
MOSI | 主机输出从机输入,主机通过该线发送数据,从机通过该线接收数据 |
SCLK | 串行时钟同步输出,同步数据传输,使用主机输出 |
CS | 片选,主机输出,用来选中具体的从机 |
CS:控制芯片是否被选中的,也就是说只有片选信号为预先规定的使能信号时(一般默认为低电位),对此芯片的操作才有效,这就允许在同一总线上连接多个SPI设备成为可能。
也就是说:当有多个从设备的时候,因为每个从设备上都有一个片选引脚接入到主设备机中,当我们的主设备和某个从设备通信时将需要将从设备对应的片选引脚电平拉低。
MISO/MOSI/SCLK:通讯是通过数据交换完成的,这里先要知道SPI是串行通讯协议,也就是说数据是一位一位的传输的。这就是SCLK时钟线存在的原因,由SCLK提供时钟脉冲,MISO,MOSI则基于此脉冲完成数据传输。数据输出通过MOSI线,数据在时钟上升沿或下降沿时采样,同时也会有返回数据用于接受。完成一位数据传输,输入也使用同样原理。这样,在至少8次时钟信号的改变(上沿和下沿为一次),就可以完成8位数据的传输。
要注意的是:
①SCLK信号线只由主设备控制,从设备不能控制信号线。同样,在一个基于SPI的设备中,至少有一个主控设备;
②在点对点的通信中,SPI接口不需要进行寻址操作,且为全双工通信,显得简单高效。在多个从设备的系统中,每个从设备需要独立的使能信号,硬件上比I2C系统要稍微复杂一些。
4线SPI接线:SPI主从设备之间接线如下图所示。
3线SPI接线:将从设备的SS引脚接地,实现3线SPI这样就可以节省芯片资源,减少布线。
一主多从SPI接线(1):通过将每个从设备的SS引脚分别接到主模式设备,可以达到一个主设备控制多个从设备的目的。
一主多从SPI接线(2):此方法中加入译码器,可以有效减少片选信号的数量,节省主模式设备的芯片引脚。
2.2 SPI协议时序
SPI协议时序也被称为工作模式。 SPI一共有四种工作模式,通常情况下,从机的工作模式是固定的,主机需要根据从机的工作模式进行调整自身工作模式,来完成相互之间的通信。我们的通信双方必须是工作在同一模式下,所以我们可以对我们的主设备的SPI模式进行配置,通过CPOL(时钟极性)和CPHA(时钟相位)来控制我们主设备的通信模式,具体如下:
SPI模式 | CPOL | CPHA | 空闲状态时钟极性 | 采样跳变沿 |
---|---|---|---|---|
0 | 0 | 0 | 低电平 | 奇数沿采样,偶数沿输出 |
1 | 0 | 0 | 低电平 | 奇数沿输出,偶数沿采样 |
2 | 1 | 1 | 高电平 | 偶数沿采样,奇数沿输出 |
3 | 1 | 1 | 高电平 | 奇数沿输出,偶数沿采样 |
1)CPOL时钟极性选择,=0表示总线空闲为低电平(SCLK时钟空闲状态为低电平,此时MOSI和MISO上的数据可以变化);=1表示总线空闲位高电平(SCLK时钟空闲状态为低电平,此时MOSI和MISO上的数据可以变化)。
2)CPHA时钟相位选择,=0会在SCLK的第一个跳变沿采样,第二个跳变沿输出;=1会在SCLK的第二个跳变沿采样,第一个跳变沿输出。以上说法不好理解,可以这样描述,=0时,在奇数跳变沿采样,偶数跳变沿输出;=1时,则正好相反。
下面具体看一下各模式的时序图:
模式0:
从图中可以看出,空闲电平是低电平,采样边沿为奇数跳变沿,输出在偶数跳变沿。
模式1:
图中可以看出,空闲电平是低电平;奇数跳变沿输出,偶数跳变沿采样。
模式2:
图中看出,空闲电平是高电平;在奇数跳变沿进行采样,偶数跳变沿进行输出。
模式3:
从图中可以看出,空闲电平是高电平,;在奇数跳变沿进行输出,在偶数跳变沿进行采样。
2.3 SPI内部工作机制
下面对照一个SPI单主机与单从机连接图,理解其内部工作机制:
①硬件上为4根线;
②主机和从机都有一个串行移位寄存器,主机通过向它的SPI串行寄存器写入一个字节来发起一次传输;
③串行移位寄存器通过MOSI信号线将字节传送给从机,同时从机也将自己的串行移位寄存器中的内容通过MISO信号线返回给主机。这样,两个移位寄存器中的内容就被交换;
④外设的写操作和读操作是同步完成的。如果只进行写操作,主机只需忽略接收到的字节;反之,若主机要读取从机的一个字节,就必须发送一个空字节来引发从机的传输。
上面的过程转为动画:初始状态
主机读取一个bit过程:
交换后:
也就是说:SPI是一个环形总线结构,由CS、SCLK、MISO、MOSI构成,其时序其实很简单,主要是在SCLK的控制下,数据按照从高位到低位的方式依次移出主机寄存器和从机寄存器,并且依次移入从机寄存器和主机寄存器。当寄存器中的内容全部移出时,相当于完成了两个寄存器内容的交换。
假设主机的8位寄存器装的是待发送的数据10101010,上升沿发送、下降沿接收、高位先发送。那么第一个上升沿来的时候,主机将会通过MOSI信号线传输给从机最高位1,自身寄存器变成0101010x。同时,MISO信号线会从从机处返回一个数据给主机,那么这时寄存器为0101010MISO,这样在 8个时钟脉冲以后,两个寄存器的内容互相交换一次。这样就完成里一个SPI时序。
这个时候就会有一个疑问,或者说产生一个必然了:
为什么主机发送一个数据给从机,从机就同时通过MISO返回的一个数据给主机呢?
解释:主机和从机的发送数据是同时完成的,两者的接收数据也是同时完成的。也就是说,当上升沿主机发送数据的时候,从机也发送了数据。
所以为了保证主从机正确通信,应使得它们的SPI具有相同的时钟极性和时钟相位。
3 STM32的SPI接口
SPI可分为主、从两种模式,并且支持全双工模式,所以这也就导致STM32的SPI接口比较复杂。比如:配置SPI为主模式、配置SPI为从模式、配置SPI为单工通信、配置SPI为双工通信等等。这里的内容就非常庞大,涉及到的寄存器的位也比较多,所以就不介绍太多,想要了解更多可以去查看STM32F1xx官方资料的第23章节。
3.1 SPI接口的框图
3.2 SPI引脚
STM32的SPI接口通过4个引脚与外部器件相连,与标准的SPI协议是一致的:
①MISO:主设备输入/从设备输出引脚。该引脚在从模式下发送数据,在主模式下接收数据;
②MOSI:主设备输出/从设备输入引脚。该引脚在主模式下发送数据,在从模式下接收数据;
③SCK:串口时钟,作为主设备的输入,从设备的输入;
④NSS:从设备选择。这是一个可选的引脚,用来选择主/从设备。它的功能是用来作为“片选引脚”,让主设备可以单独地与特定从设备通讯,避免数据线上的冲突。
从选择(NSS)脚管理
有2种NSS模式:
1)软件NSS模式:可以通过设置SPI_CR1寄存器的SSM位来使能这种模式。在这种模式下NSS引脚可以用作它用,而内部NSS信号电平可以通过写SPI_CR1的SSI位来驱动;
2)硬件NSS模式,分两种情况:
①NSS输出被使能:当STM32F10xxx工作为主SPI,并且NSS输出已经通过SPI_CR2寄存器的SSOE位使能,这时NSS引脚被拉低,所有NSS引脚与这个主SPI的NSS引脚相连并配置为硬件NSS的SPI设备,将自动变成从SPI设备。 当一个SPI设备需要发送广播数据,它必须拉低NSS信号,以通知所有其它的设备它是主设备;如果它不能拉低NSS,这意味着总线上有另外一个主设备在通信,这时将产生一个硬件失败错误;
②NSS输出被关闭:允许操作于多主环境。
3.3 数据帧格式
1)根据SPI_CR1寄存器中的LSBFIRST位,输出数据位时可以左对齐(MSB对齐标准)也可以右对齐(LSB对齐标准)。
2)根据SPI_CR1寄存器的DFF位,每个数据帧可以是8位或是16位。所选择的数据帧格式对发送和/或接收都有效。
3.4 状态标志
应用程序通过3个状态标志可以完全监控SPI总线的状态:
1)发送缓冲器空闲标志(TXE)
此标志为1时表明发送缓冲器为空,可以写下一个待发送的数据进入缓冲器中。当写入SPI_DR时,TXE标志被清除。
2)接收缓冲器非空(RXNE)
此标志为1时表明在接收缓冲器中包含有效的接收数据。读SPI数据寄存器可以清除此标志。
2)忙(Busy)标志
BSY标志由硬件设置与清除(写入此位无效果),此标志表明SPI通信层的状态。
当它被设置为1时,表明SPI正忙于通信,但有一个例外:在主模式的双向接收模式下(MSTR=1、BDM=1并且BDOE=0),在接收期间BSY标志保持为低。
在软件要关闭SPI模块并进入停机模式(或关闭设备时钟)之前,可以使用BSY标志检测传输是否结束,这样可以避免破坏最后一次传输,因此需要严格按照下述过程执行。
3.5 SPI中断
4 STM32的SPI引脚
4.1 SPI引脚位置
4.2 外设的GPIO配置
5 SPI相关配置库函数
5.1 1个初始化函数
void SPI_Init(SPI_TypeDef* SPIx, SPI_InitTypeDef* SPI_InitStruct);
作用:初始化SPI的相关参数,比如方向(全双工)、主从模式、数据大小、CPOL、CPHA、片选软件模式、预分频系数等。
5.2 3个使能函数
void SPI_Cmd(SPI_TypeDef* SPIx, FunctionalState NewState);
void SPI_I2S_ITConfig(SPI_TypeDef* SPIx, uint8_t SPI_I2S_IT, FunctionalState NewState);
void SPI_I2S_DMACmd(SPI_TypeDef* SPIx, uint16_t SPI_I2S_DMAReq, FunctionalState NewState);
作用:使能SPI接口;使能SPI中断;使能SPI的DMA功能。
5.3 2个数据传输函数
void SPI_I2S_SendData(SPI_TypeDef* SPIx, uint16_t Data);
uint16_t SPI_I2S_ReceiveData(SPI_TypeDef* SPIx);
作用:分别用于SPI传输数据、接收数据。
5.4 4个状态位函数
FlagStatus SPI_I2S_GetFlagStatus(SPI_TypeDef* SPIx, uint16_t SPI_I2S_FLAG);
void SPI_I2S_ClearFlag(SPI_TypeDef* SPIx, uint16_t SPI_I2S_FLAG);
ITStatus SPI_I2S_GetITStatus(SPI_TypeDef* SPIx, uint8_t SPI_I2S_IT);
void SPI_I2S_ClearITPendingBit(SPI_TypeDef* SPIx, uint8_t SPI_I2S_IT);
作用:前两者用于获得和清除SPI的各种状态位;后两者则针对SPI的中断标志位。
6 SPI一般步骤
实验目标:利用SPI2进行初始化等操作。
1)配置相关引脚的复用功能,使能SPIx时钟。调用函数:void GPIO_Init();
2)初始化SPIx,设置SPIx工作模式。调用函数:void SPI_Init();
3)使能SPIx。调用函数:void SPI_Cmd();
4)SPI传输数据。调用函数:void SPI_I2S_SendData();uint16_t SPI_I2S_ReceiveData();
5)查看SPI传输状态。调用函数:SPI_I2S_GetFlagStatus(SPI2, SPI_I2S_FLAG_RXNE)。
下面按照这个一般步骤来进行一个简单的SPI程序:
void SPI2_Init(void)
{GPIO_InitTypeDef GPIO_InitStructure;SPI_InitTypeDef SPI_InitStructure;RCC_APB2PeriphClockCmd( RCC_APB2Periph_GPIOB, ENABLE );//PORTB时钟使能 RCC_APB1PeriphClockCmd( RCC_APB1Periph_SPI2, ENABLE );//SPI2时钟使能 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_13 | GPIO_Pin_14 | GPIO_Pin_15;GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP; //PB13/14/15复用推挽输出 GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;GPIO_Init(GPIOB, &GPIO_InitStructure);//初始化GPIOBGPIO_SetBits(GPIOB,GPIO_Pin_13|GPIO_Pin_14|GPIO_Pin_15); //PB13/14/15上拉SPI_InitStructure.SPI_Direction = SPI_Direction_2Lines_FullDuplex; //设置SPI单向或者双向的数据模式:SPI设置为双线双向全双工SPI_InitStructure.SPI_Mode = SPI_Mode_Master; //设置SPI工作模式:设置为主SPISPI_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; //定义波特率预分频的值:波特率预分频值为256SPI_InitStructure.SPI_FirstBit = SPI_FirstBit_MSB; //指定数据传输从MSB位还是LSB位开始:数据传输从MSB位开始SPI_InitStructure.SPI_CRCPolynomial = 7; //CRC值计算的多项式SPI_Init(SPI2, &SPI_InitStructure); //根据SPI_InitStruct中指定的参数初始化外设SPIx寄存器SPI_Cmd(SPI2, ENABLE); //使能SPI外设SPI2_ReadWriteByte(0xff);//启动传输 }
//SPI 速度设置函数
//SpeedSet:
//SPI_BaudRatePrescaler_2 2分频
//SPI_BaudRatePrescaler_8 8分频
//SPI_BaudRatePrescaler_16 16分频
//SPI_BaudRatePrescaler_256 256分频 void SPI2_SetSpeed(u8 SPI_BaudRatePrescaler)
{assert_param(IS_SPI_BAUDRATE_PRESCALER(SPI_BaudRatePrescaler));SPI2->CR1&=0XFFC7;SPI2->CR1|=SPI_BaudRatePrescaler; //设置SPI2速度 SPI_Cmd(SPI2,ENABLE); } //SPIx 读写一个字节
//TxData:要写入的字节
//返回值:读取到的字节
u8 SPI2_ReadWriteByte(u8 TxData)
{ u8 retry=0; while (SPI_I2S_GetFlagStatus(SPI2, SPI_I2S_FLAG_TXE) == RESET) //检查指定的SPI标志位设置与否:发送缓存空标志位{retry++;if(retry>200)return 0;} SPI_I2S_SendData(SPI2, TxData); //通过外设SPIx发送一个数据retry=0;while (SPI_I2S_GetFlagStatus(SPI2, SPI_I2S_FLAG_RXNE) == RESET) //检查指定的SPI标志位设置与否:接受缓存非空标志位{retry++;if(retry>200)return 0;} return SPI_I2S_ReceiveData(SPI2); //返回通过SPIx最近接收的数据
}
STM32-SPI通信相关推荐
- 通俗理解STM32 SPI通信(主从双机SPI通信)
STM32 SPI通信 高速全双工的通信总线 SPI 通讯使用 3 条总线及片选线,3 条总线分别为 SCK.MOSI.MISO,片选线为NSS(CS) NSS 信号线由高变低 ,是 SPI 通讯的起 ...
- Zigbee(cc2530)和STM32 SPI通信,温度采集,组网
实验板子: 1. zigbee底板3块(1终端.1路由.1协调器,均基于CC2530) 2. stm32核心板(采集温度) 说明:下面的程序都是在TI官方例程SampleApp工程下面进行的更改.Z- ...
- STM32 SPI通信(读写flash)
SPI通信 四线制 MISO:主设备数据输入,从设备数据输出 MOSI:主设备数据输出,从设备数据输入 SCLK:时钟信号 CS:片选引脚(低电平有效) 时序图 读写程序 flash读写函数: u8 ...
- STM32——SPI通信
文章目录 一.SPI通信 二.硬件电路 三.移位示意图 四.SPI时序基本单元 交换一个字节(模式0)[用的多] 交换一个字节(模式1) 交换一个字节(模式2) 交换一个字节(模式3) 五.SPI时序 ...
- STM32——SPI通信实验
程序配置过程: 1.使能SPIx和IO口时钟: RCC_AHBxPeriphClockCmd()/RCC_APBxPeriphClockCmd(); 2.初始化IO口为复用功能: void GPIO_ ...
- Spi通信,写操作时,发送缓冲区非空停留在while里面。arm a5与w5500芯片udp网络通信。网络调试助手界面及sscom无法点击
问题描述 w5500 与SAMA5D3x处理器通过硬件spi实现网络通信.采用w5500自带的函数进行udp通信,接口函数sendto,发送100Hz的数据一段时间后,程序停留在函数SPI_write ...
- STM32 OLED显示屏--SPI通信知识汇总
备注:在OLED显示屏部分,单片机是通过SPI和OLED进行通信的. 文章目录 目录 文章目录 一.SPI时序通信 二.SPI通信的分类 1.硬件SPI 2.软件SPI 三.硬件SPI SPI特性 硬 ...
- 基于STM32与NOR FLASH的SPI通信
SPI的通信很容易实现,相比之下,驱动FLASH反而耗费了我学习SPI整个过程的大部分时间.下面是我学习过程的一些记录. 硬件平台:秉火ISO_V2开发板 实现功能:STM32使用SPI协议读写板 ...
- pixy php,Pixy2与STM32进行SPI通信
简单来说,如果你通过检索从而看到这篇文章,那就假设你已经知道Pixy2是用来干什么的(一款功能强大的开源视觉传感器),以及你已拥有一些STM32基础.你可以通过访问官方手册来获得更多关于Pixy2的相 ...
- STM32 网口转SPI通信
提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档 文章目录 前言 一.网络通信与SPI通信是什么? 二.代码思路 1.流程图 三.测试代码流程与关键代码 总结 前言 `本文介绍利用ST ...
最新文章
- 自适应IFRAME的大小
- KafkaConsumer 长时间地在poll(long )方法中阻塞
- centos下安装python2.7.1 以及装完后 yum 不能用的问题
- 网页检测 AdBlock 的 6 种方法
- “哎哟,真的很快哦” 闪送宣布签约周杰伦为其品牌代言人
- 人工智能作业——搜索树博弈树一阶逻辑表达式CNF范式
- python和c 的区别-python和c语言的主要区别总结
- csv文件导入后台乱码_win7系统下excel打开csv文件出现乱码怎么修复
- 如何在 Web 前端做 3D 音效处理
- C语言科学计数法介绍和示例
- 读书寄语:所有的遇见都是有原因的,并不只是偶然
- android Compose Modifier介绍
- 下载并安装 Node
- WIN10:今天开机突然遇到在打拼音的时候,输入框不见了,已下是本人的解决办法
- 2020年深圳杯A、B、C、D四题详解+代码【无套路】
- 老杨说运维 | 企业数字化转型中,统一监控的必要性
- go 实现发送短信验证码登录
- 虚拟机VMware下载以及在VMware上安装Ubuntu18.04,保姆级教程,绝对成功!
- EtherCAT通讯DS402协议----控制模式
- Linux上传文件时文件名自动加引号问题
热门文章
- ifound Android wifi,方正新品记录仪iFound V1号称黑夜变白天,真的假的?
- 配置python程序debug/run,避免每次运行都会重复加载数据集或模型,节约大量等待时间
- 向量距离汇总(连续值与离散值),Latex与Python实现
- R语言快速学习第一部分(有其他语言基础)
- mysql使用cmd命令连接_通过cmd命令连接mysql
- java webservice报文过长_年薪百万IT大牛分享及(京东,阿里,58)Java初中高级765道面试题...
- h5移动端局部放大效果
- python接口自动化(九)--python中字典和json的区别(详解)
- 2018-2019-1 20165214 《信息安全系统设计基础》第八周学习总结
- JSON与Delphi Object的互换