SPI协议
SPI协议网上资料比较多,但是也比较乱,当初在网上搜集的错误资料导致现在比较混乱。
SPI协议资料比较正规的是:
1.SPI的规约协议英文文档,例如《摩托罗拉spi协议规范》
2.许多IC的datasheet中关于SPI协议的介绍
下图摘自《摩托罗拉spi协议规范》:
- CPOL和CPHA的描述和定义
注意画线部分的描述:第一个边沿锁存(或者叫采样),第二个边沿发送,注意,发送的是之前锁存好的。 - 时序图
- 工作模式的定义
上图《Application Note MLX90363 Getting Started Guide》中关于SPI协议的相关介绍,可以得出SPI 模式0-3的定义
总结:
- CPHA=0,第一个跳变沿采样,第二个跳变沿传送。CPHA=1,则反之。
- CPOL=0,表示空闲为低电平,CPOL=1,则反之
-
模式 CPOL CPHA 0 0 0 1 0 1 2 1 0 3 1 1 IO口模拟
模拟的关键点主要是对采样和发送的理解,例如以模式1为例,根据前面介绍可以得知上升沿发送,下降沿采样。
关键点:- 上升沿发送。需理解电平从低到高也是时间的。上升沿发送标准的是,在低电平上升至1/2高电平时进行发送,所以如果是软件模拟,必须是SCL=1后,不要有任何时延,立即拉低或拉高Data口。例图:1/2处发送
- 下降沿采样。同理也是高电平下降至1/2除,不要有任何时延,立即读取Data口
- 开始和结束时的设置,以Mode1为例,应该在CS有效之前拉低SCLK半个时钟,结束之后在CS无效之前拉低SCLK半个时钟。CS有效之后应延时半个时钟,CS无效之后也应延时半个时钟
代码如下
//-最小为10-
#define N_Delay 20
unsigned char SPI_Send(unsigned char Data)
{
int i = 0;
unsigned char RecvByte = 0; COMM_CLK_L();
SPI_Delay(N_Delay);
COMM_CS_L();
SPI_Delay(N_Delay); for (i = 0; i < 8; i++)
{
//-发送-
COMM_CLK_H();
if ((Data & 0x80) != 0)
{
COMM_MOSI_H();
}
else
{
COMM_MOSI_L();
}
Data <<= 1;
SPI_Delay(N_Delay); //-接收-
COMM_CLK_L();
if (COMM_MISO_Get() != 0)
{
RecvByte |= 0x01 << (7 - i);
}
SPI_Delay(N_Delay); }
COMM_CLK_L();
SPI_Delay(N_Delay);
COMM_CS_H();
SPI_Delay(N_Delay);
return RecvByte;
}产生的波形如图: