什么是SPI SPI 是英语Serial Peripheral interface的缩写,顾名思义就是串行外围设备接口。是Motorola(摩托罗拉)首先在其MC68HCXX系列处理器上定义的。 SPI,是一种高速的,全双工,同步的通信总线,并且在芯片的管脚上只占用四根线,节约了芯片的管脚,同时为PCB的布局上节省空间,提供方便,主要应用在 EEPROM,FLASH,实时时钟,AD转换器,还有数字信号处理器和数字信号解码器之间。 7 G t& C# w& i. \+ \
% Z$ D# E6 X, b3 l SPI主从模式) g J9 y. M- F, N5 m
SPI分为主、从两种模式,一个SPI通讯系统需要包含一个(且只能是一个)主设备,一个或多个从设备。提供时钟的为主设备(Master),接收时钟的设备为从设备(Slave),SPI接口的读写操作,都是由主设备发起。当存在多个从设备时,通过各自的片选信号进行管理。 SPI是全双工且SPI没有定义速度限制,一般的实现通常能达到甚至超过10 Mbps
0 Q% K; e" s2 v0 M' {2 j! ^1 V" f3 Y
/ e5 Q: i: s* G SPI信号线
5 l1 m* Z! U, u0 [9 z1 l& H2 sSPI接口一般使用四条信号线通信:" N) C7 n) k% }! y
SDI(数据输入),SDO(数据输出),SCK(时钟),CS(片选) - MISO: 主设备输入/从设备输出引脚。该引脚在从模式下发送数据,在主模式下接收数据。
- MOSI: 主设备输出/从设备输入引脚。该引脚在主模式下发送数据,在从模式下接收数据。
- SCLK:串行时钟信号,由主设备产生。
- CS/SS:从设备片选信号,由主设备控制。它的功能是用来作为“片选引脚”,也就是选择指定的从设备,让主设备可以单独地与特定从设备通讯,避免数据线上的冲突。
" I n+ ]( |- G1 w4 @ D! ]' j4 F6 T. L
9 [6 c" ^. F) J2 j7 J% J0 `7 w
. |0 c4 Z7 o' p& n" S3 f9 G* o/ J
- [; h1 r( v ~2 C' e, D o硬件上为4根线。 SPI一对一
4 a! L# d9 q+ a7 O: W
4 v* t+ w, W0 v" H* C SPI一对多& } L9 L4 J: l) q
, A* I, \& O# i5 \! ~% X4 C! E& `; p
$ M3 t$ Z& |5 R3 H; L4 n) P4 Y SPI设备选择
/ R& v& Y; H% D* i$ g$ z
. z6 J$ |( i$ O7 m0 H: wSPI是[单主设备( single-master )]通信协议,这意味着总线中的只有一支中心设备能发起通信。当SPI主设备想读/写[从设备]时,它首先拉低[从设备]对应的SS线(SS是低电平有效),接着开始发送工作脉冲到时钟线上,在相应的脉冲时间上,[主设备]把信号发到MOSI实现“写”,同时可对MISO采样而实现“读”,如下图:
! Y$ t' F& f& w% }* @/ I' p+ s' {4 I" V
5 ]% R* |8 a1 f( j* e
; E/ M. k$ O) y; K9 ] SPI数据发送接收
8 k7 a" E; c9 T5 {$ k0 P m! {- z0 E# G% j; k5 {7 l. ]8 S8 t4 {9 A
SPI主机和从机都有一个串行移位寄存器,主机通过向它的SPI串行寄存器写入一个字节来发起一次传输。 - 首先拉低对应SS信号线,表示与该设备进行通信
- 主机通过发送SCLK时钟信号,来告诉从机写数据或者读数据+ }+ u% N% @ n
这里要注意,SCLK时钟信号可能是低电平有效,也可能是高电平有效,因为SPI有四种模式,这个我们在下面会介绍 - 主机(Master)将要发送的数据写到发送数据缓存区(Menory),缓存区经过移位寄存器(0~7),串行移位寄存器通过MOSI信号线将字节一位一位的移出去传送给从机,,同时MISO接口接收到的数据经过移位寄存器一位一位的移到接收缓存区。
- 从机(Slave)也将自己的串行移位寄存器(0~7)中的内容通过MISO信号线返回给主机。同时通过MOSI信号线接收主机发送的数据,这样,两个移位寄存器中的内容就被交换。
+ h' X8 O( ^2 l
4 K. [% [# k h- d+ q @' ?8 D5 U + a# T( d# v+ n. W. e- j7 P2 w4 {) U
4 P8 E& U3 B7 {+ Y
9 M1 d: ]5 y/ |8 |9 @" M- s/ V) k) v# B; P) v
1 A, O) r9 ~4 k- S* vSPI只有主模式和从模式之分,没有读和写的说法,外设的写操作和读操作是同步完成的。如果只进行写操作,主机只需忽略接收到的字节;反之,若主机要读取从机的一个字节,就必须发送一个空字节来引发从机的传输。也就是说,你发一个数据必然会收到一个数据;你要收一个数据必须也要先发一个数据。 G; G% a$ `' N1 Q6 S
5 X( S3 E3 V% G- d SPI通信的四种模式
: g1 d; l6 k( i( @7 } @6 d% U! Q& b1 \. p$ W
SPI的四种模式,简单地讲就是设置SCLK时钟信号线的那种信号为有效信号
SPI通信有4种不同的操作模式,不同的从设备可能在出厂是就是配置为某种模式,这是不能改变的;但我们的通信双方必须是工作在同一模式下,所以我们可以对我们的主设备的SPI模式进行配置,通过CPOL(时钟极性)和CPHA(时钟相位)来' e A2 g* m' M6 j; R
控制我们主设备的通信模式,具体如下:" b+ w/ a% z6 R
、 时钟极性(CPOL)定义了时钟空闲状态电平: - CPOL=0,表示当SCLK=0时处于空闲态,所以有效状态就是SCLK处于高电平时
- CPOL=1,表示当SCLK=1时处于空闲态,所以有效状态就是SCLK处于低电平时
7 A# i) z7 y R9 X* U% h
. j5 i/ _1 Z( I# C4 F- P0 Q
时钟相位(CPHA)定义数据的采集时间。 - CPHA=0,在时钟的第一个跳变沿(上升沿或下降沿)进行数据采样。,在第2个边沿发送数据
- CPHA=1,在时钟的第二个跳变沿(上升沿或下降沿)进行数据采样。,在第1个边沿发送数据
8 ~/ F7 {9 `. m* L! w
4 K6 i& _) d/ B
例如: Mode0:CPOL=0,CPHA=0:此时空闲态时,SCLK处于低电平,数据采样是在第1个边沿,也就是SCLK由低电平到高电平的跳变,所以数据采样是在上升沿(准备数据),(发送数据)数据发送是在下降沿。 Mode1:CPOL=0,CPHA=1:此时空闲态时,SCLK处于低电平,数据发送是在第1个边沿,也就是SCLK由低电平到高电平的跳变,所以数据采样是在下降沿,数据发送是在上升沿。 Mode2:CPOL=1,CPHA=0:此时空闲态时,SCLK处于高电平,数据采集是在第1个边沿,也就是SCLK由高电平到低电平的跳变,所以数据采集是在下降沿,数据发送是在上升沿。 Mode3:CPOL=1,CPHA=1:此时空闲态时,SCLK处于高电平,数据发送是在第1个边沿,也就是SCLK由高电平到低电平的跳变,所以数据采集是在上升沿,数据发送是在下降沿。 6 Y, Z% j M/ a. N. ?
6 X; J( M% W% a& O( `/ K8 A
* u0 s2 G: r# h/ }- h" m6 h# G
$ L3 E b- S" n, {0 S' V: y
) @( i! F: i. l' k. S
: u7 n1 N# \9 j3 l! z/ g+ V/ |
4 K/ h# y; O- Y5 ?5 s
% |. S" r9 }* Y! {; S" k$ q$ T" m# n: F/ ~8 V8 o: H
* r2 Y3 A/ p z' f: V( m0 c
0 H: i5 s; b" Q* y它们的区别是定义了在时钟脉冲的哪条边沿转换(toggles)输出信号,哪条边沿采样输入信号,还有时钟脉冲的稳定电平值(就是时钟信号无效时是高还是低)。每种模式由一对参数刻画,它们称为时钟极(clock polarity)CPOL与时钟期(clock phase)CPHA。
; P) v! s J. [/ ]& a, e. U+ J% M; r# U! h、 SPI的通信协议
/ Z8 X1 [. K' B; W+ @2 J
+ f% O1 ^8 ]' N I# Q
$ N2 Z6 g. b0 D5 ~, [
/ o$ w" h( m2 }7 Z6 k主从设备必须使用相同的工作模式——SCLK、CPOL 和 CPHA,才能正常工作。如果有多个从设备,并且它们使用了不同的工作模式,那么主设备必须在读写不同从设备时需要重新修改对应从设备的模式。以上SPI总线协议的主要内容。 是不是感觉,这就完了? SPI就是如此,他没有规定最大传输速率,没有地址方案,也没规定通信应答机制,没有规定流控制规则。 只要四根信号线连接正确,SPI模式相同,将CS/SS信号线拉低,即可以直接通信,一次一个字节的传输,读写数据同时操作,这就是SPI 些通信控制都得通过SPI设备自行实现,SPI并不关心物理接口的电气特性,例如信号的标准电压。 PS:/ N& G' Z! t9 j% Y" u; C+ P
这也是SPI接口的一个缺点:没有指定的流控制,没有应答机制确认是否接收到数据。
2 Z* |! o% e( {) P, H5 i+ E/ z
! I$ E0 A) z H SPI的三种模式
4 ?2 d5 r5 M' |1 I% g z! U7 \* Y8 ~4 g) t7 |3 g8 n
SPI工作在3中模式下,分别是运行、等待和停止。 运行模式(Run Mode)
) H+ v# v, e" {这是基本的操作模式 等待模式(Wait Mode)
" s' q+ j3 u; v' k( h" v& OSPI工作在等待模式是一种可配置的低功耗模式,可以通过SPICR2寄存器的SPISWAI位进行控制。在等待模式下,如果SPISWAI位清0,SPI操作类似于运行模式。如果SPISWAI位置1,SPI进入低功耗状态,并且SPI时钟将关闭。如果SPI配置为主机,所有的传输将停止,但是会在CPU进入运行模式后重新开始。如果SPI配置为从机,会继续接收和传输一个字节,这样就保证从机与主机同步。 停止模式(Stop Mode)
& f; A% y k) h4 e; v为了降低功耗,SPI在停止模式是不活跃的。如果SPI配置为主机,正在进行的传输会停止,但是在CPU进入运行模式后会重新开始。如果SPI配置为从机,会继续接受和发送一个字节,这样就保证了从机与主机同步。
; E9 T; J% `& l6 M+ O
0 f2 P$ s) K1 I* ^' L8 N7 c6 o
SPI原理图连接/ |2 x: ?* a3 o
- A/ t- Z- Q4 P& K
7 G( o& }& q* L1 o( A
, D, s# u6 j! Z% \1 v* F
* z! ]* H% F2 w5 m**STM32中SPI初始化配置; F' G/ p* }! R2 o% E% o h
& j* p& S# V9 a3 g3 R5 C
1.初始化GPIO口,配置相关引脚的复用功能,使能SPIx时钟。调用函数:void GPIO_Init(); 2.使能SPI时钟总线:RCC_APB2PeriphClockCmd(RCC_APB2Periph_SPI1,ENABLE) 3.配置SPI初始化的参数,设置SPI工作模式:SPI_Init(SPI1,&SPI_Initstructure) 4.使能SPI外设:SPI_Cmd(SPI1,ENABLE); 8 p; d+ N/ I* e [# t7 E/ t
SPI配置设置
8 L& E& ^; L9 j
7 v& F+ r0 x0 K/ \: B6 P
- typedef struct, P' Q5 I$ `. |5 \# _6 ^
- {
+ \) \/ S! F" u' a& d0 q- i - uint16_t SPI_Direction; /*!< 传输方向,两向全双工,单向接收等*/: m* f& @5 t) J8 v6 [1 A" [
- uint16_t SPI_Mode; /*!< 模式选择,确定主机还是从机 */3 G# o3 D3 W+ X# H" I
- uint16_t SPI_DataSize; /*!< 数据大小,8位还是16位 */& L) b5 \2 P+ k# F
- uint16_t SPI_CPOL; /*!< 时钟极性选择 */& M- Z" _# E' k. y3 W: f$ ^' K
- uint16_t SPI_CPHA; /*!< 时钟相位选择 */
6 l# y- z/ X: f, w - uint16_t SPI_NSS; /*!< 片选是硬件还是软件*/
8 |+ @ K9 y# S2 s, D$ A - uint16_t SPI_BaudRatePrescaler; /*!< 分频系数 */9 b T6 J: b: r* T
- uint16_t SPI_FirstBit; /*!< 指定数据传输是从MSB还是LSB位开始的。M
& c$ p% Z; L3 c: a* c - SB就是二进制第一位,LSB就是最后一位 */
* X$ n6 b. ^8 N2 k - uint16_t SPI_CRCPolynomial; /*!< CRC校验 ,设置 CRC 校验多项式,提高通' L6 h- A9 `7 L" E/ ^# O- K8 A
- 信可靠性,大于 1 即可*/
" p, m8 l, M! p, _3 y& H - }SPI_InitTypeDef;
2 n# N1 q1 K) [% ? |! x
复制代码 ! g% T7 F0 f9 n8 J& h
; U9 y6 e, h- Y+ J& d, j, p# H. U( V
# ~# Q! ]; h/ `5 b- d
- x5 F1 R( q" W/ ~6 K( j
- void SPI2_Init(void)
v" l& Z7 r1 \$ v - {
$ n$ p, N( g( g& q1 V1 ? - GPIO_InitTypeDef GPIO_InitStructure;
& W" a& Q* j9 Y+ y! q - SPI_InitTypeDef SPI_InitStructure;
- k; _ M I! K: j0 g - 9 _& }# h1 ^7 {5 t& f5 o* Y. V: c
- RCC_APB2PeriphClockCmd( RCC_APB2Periph_GPIOB, ENABLE );//PORTB时钟使能
/ S( V6 M# h u8 N* h, q# q! ` - RCC_APB1PeriphClockCmd( RCC_APB1Periph_SPI2, ENABLE );//SPI2时钟使能 6 ?( C1 b8 J+ \: a5 f
-
! b- F, F9 l, J" d: B7 W% H7 v( b/ g - GPIO_InitStructure.GPIO_Pin = GPIO_Pin_13 | GPIO_Pin_14 | GPIO_Pin_15;
0 Y. J5 n1 w: M6 _: ?+ d - GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP; //PB13/14/15复用推挽输出 4 J7 ~ R& ]' h! }4 G6 F
- GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;6 P k1 b2 T- c; S
- GPIO_Init(GPIOB, &GPIO_InitStructure);//初始化GPIOB7 y3 x# O X2 F0 m- A
-
$ f6 R0 c3 [) u$ _. Y/ O; }4 G4 Q - GPIO_SetBits(GPIOB,GPIO_Pin_13|GPIO_Pin_14|GPIO_Pin_15); //PB13/14/15上拉6 j# ]6 |: E' ?% K4 C
- 6 X2 m& p( E B _
- SPI_InitStructure.SPI_Direction = SPI_Direction_2Lines_FullDuplex; //设置SPI单向或者双向的数据模式:SPI设置为双线双向全双工$ ~9 |5 ]0 b0 p. a& D& z8 M; J+ I
- SPI_InitStructure.SPI_Mode = SPI_Mode_Master; //设置SPI工作模式:设置为主SPI0 V& B7 {4 R- ?2 r0 @. h/ j
- SPI_InitStructure.SPI_DataSize = SPI_DataSize_8b; //设置SPI的数据大小:SPI发送接收8位帧结构) z4 N) Q) o: g! d* ~+ m' ?
- SPI_InitStructure.SPI_CPOL = SPI_CPOL_High; //串行同步时钟的空闲状态为高电平! J5 F" z4 D2 _% S
- SPI_InitStructure.SPI_CPHA = SPI_CPHA_2Edge; //串行同步时钟的第二个跳变沿(上升或下降)数据被采样
$ D/ ]. Y2 w2 G8 F* E; P - SPI_InitStructure.SPI_NSS = SPI_NSS_Soft; //NSS信号由硬件(NSS管脚)还是软件(使用SSI位)管理:内部NSS信号有SSI位控制
k3 n& _# Y( ^ v, ? - SPI_InitStructure.SPI_BaudRatePrescaler = SPI_BaudRatePrescaler_256; //定义波特率预分频的值:波特率预分频值为256
4 @( c6 y: W8 F& X: P5 P/ | - SPI_InitStructure.SPI_FirstBit = SPI_FirstBit_MSB; //指定数据传输从MSB位还是LSB位开始:数据传输从MSB位开始
1 b8 J, I2 p% x1 ^; U$ ]- D - SPI_InitStructure.SPI_CRCPolynomial = 7; //CRC值计算的多项式" t* S2 k* \9 V# @
- SPI_Init(SPI2, &SPI_InitStructure); //根据SPI_InitStruct中指定的参数初始化外设SPIx寄存器
5 l$ L, d- V! ~( n( y; z1 ^ -
% X7 `& I* t2 U. I" |( l o/ b# P - SPI_Cmd(SPI2, ENABLE); //使能SPI外设
2 M( H' y r$ d2 j; T9 r# g! m -
3 V: P7 N5 m( O( T( z3 C. L. l7 { - SPI2_ReadWriteByte(0xff);//启动传输 1 H0 b; n) M0 G' U; ?
- ( w8 T' B9 @8 a% M5 h
- }
复制代码 2 Y% ?0 S* [# C
SPI发送函数(标准库/HAL库)
5 V) i1 d. x: a7 V3 a C1 G
- 标准库:2 {5 I4 k* d2 S5 R+ a
- u8 SPIx_ReadWriteByte(u8 TxData)/ i0 T/ ?' v0 i _0 `/ @8 D
- {5 {) f3 |0 k( d, m% \. r1 q
- u8 retry=0;. e1 Q$ Y$ o3 j& Y; a& r
- while (SPI_I2S_GetFlagStatus(SPI2, SPI_I2S_FLAG_TXE) == RESET)
( K. `$ Q( E. S, e) M- |% T" { - {: q. Y; B6 `- Z+ A8 v
- }//等待发送区空
D b L0 X+ o* [! | - SPI_I2S_SendData(SPI2, TxData); //通过外设SPIx发送一个byte 数据
8 e+ ~: v& x3 B1 E; n - while (SPI_I2S_GetFlagStatus(SPI2, SPI_I2S_FLAG_RXNE) == RESET)6 f+ n' m; J" z+ \0 ~
- {
. P8 ~( V9 k6 r/ G2 A - } //等待接收完一个byte' i2 u" E& i7 T) Q+ T# A/ B
- return SPI_I2S_ReceiveData(SPI2); //返回通过SPIx最近接收的数据& e7 ?: K( t# K0 q) O9 F# y
- }: t1 s U8 M, _" }5 D
- ( C' l# @$ X$ ^8 s* q
- HLA库:
* u( O. z- o& o) j6 d - uint8_t SPI_SendByte(uint8_t byte)
0 d) s2 I/ m8 {1 I9 r0 S5 Q - {
* I/ P) x) ^5 W+ N! j, K - uint8_t d_read,d_send=byte;1 }) L3 Y* \4 T
- if(HAL_SPI_TransmitReceive(&hspi1,&d_send,&d_read,1,0xFFFFFF)!=HAL_OK)" S! i( I' _; Q2 I
- d_read=0XFF;
% g% x' _5 ~9 Y V6 C3 s - return d_read;1 K' G5 z% i: \) z: X
复制代码
. O; z0 n0 ^/ U! Z. F2 e% b7 I$ L& k' y0 B
/ A) I# }+ _2 f$ ?$ ~' u0 J |