一、SPI协议简介
一般主从方式工作,这种模式通常有一个主设备和一个或多个从设备,通常采用的是4根线,它们是MISO(主机输入从机输出)、MOSI(主机输出,针对主机来说)、SCLK(时钟,主机产生)、CS(片选,一般由主机发送或者直接使能,通常为低电平有效)
●SPI接口介绍
SCK:时钟信号,由主设备产生,所以主设备SCK信号为推挽输出模式,从设备的SCK信号为浮空输入模式。
CS:使能信号,由主设备控制从设备,,所以主设备CS信号为推挽输出模式,从设备的CS信号为浮空输入模式。
MOSI:主设备数据输出,从设备数据输入,所以主设备MOSI信号为推挽输出模式,从设备的MOSI信号为浮空输入模式。
MISO:主设备数据输入,从设备数据输出,所以主设备MISO信号为浮空输入模式,从设备的MISO信号为推挽输出模式。
二、四种模式(本次模拟采用的模式0)
模式0:CPOL=0,CPHA =0 SCK空闲为低电平,数据在SCK的上升沿被采样(提取数据)
模式1:CPOL=0,CPHA =1 SCK空闲为低电平,数据在SCK的下降沿被采样(提取数据)
模式2:CPOL=1,CPHA =0 SCK空闲为高电平,数据在SCK的下降沿被采样(提取数据)
模式3:CPOL=1,CPHA =1 SCK空闲为高电平,数据在SCK的上升沿被采样(提取数据)
三、双机通信实现
1、GPIO引脚
PC8为CS引脚 PC10为CLK引脚 PC11为MISO引脚 PC12为MOSI引脚
2、时序
首先主机将片选拉低(使能片选),在时钟为低电平时向从机发送数据,从机通过检测MOSI线上的高低电平实现数据的接收。在时钟为高电平时主机检测MISO线上的高低电平来实现数据的接收。
主机在时钟信号为低电平时发送单字节的最高位,然后将该字节左移一位,然后将时钟信号拉高,此时从机检测到时钟信号为高电平(时钟上升沿),从而检测MOSI线上的高低电平并将得到的高低电平放到变量中。同时从机向主机发送数据,即改变MISO线上的高低电平。此过程重复8次,即可完成发送和接收一个字节的数据。
四、代码如下
1、主设备GPIO的配置
void SPI_GPIO_Init(void){
GPIO_InitTypeDef GPIO_InitStructure; //定义结构体类型的变量
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC,ENABLE); //使能GPIOC的端口时种
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_8|GPIO_Pin_10|GPIO_Pin_12;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; //推挽输出
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; //IO口速度为50Mhz
GPIO_Init(GPIOC,&GPIO_InitStructure);
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_11;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING; //浮空输入
GPIO_Init(GPIOC,&GPIO_InitStructure);
GPIO_SetBits(GPIOC,GPIO_Pin_8); //CS拉高
GPIO_ResetBits(GPIOC,GPIO_Pin_10);//CLK拉低
GPIO_ResetBits(GPIOC,GPIO_Pin_11);
}
2、主设备向从设备写以及读
/*
*********************************************************************************************************
* 函 数 名: SPI_WRByte
* 形 参: spi; wdat:写入的数据
* 返 回 值: spi读一个字节
* 功能说明: 主机spi同时读写一个字节的时序 MSB
*********************************************************************************************************
*/
u8 Master_SPI_WRByte(u8 wdat)
{
u8 i=0,rdat=0;
SPI_CS_LOW;
for(i=0;i<8;i++)
{
if(wdat&0x80)
SPI_MOSI_HIGH;
else
SPI_MOSI_LOW;
wdat<<=1;
delay_us(3);
SPI_CLK_HIGH;
delay_us(2);
rdat<<=1;
if(SPI_MISO_READ)
rdat |= 0x01;
delay_us(1);
SPI_CLK_LOW;
}
SPI_CS_HIGH;
return rdat;
}
3、从设备GPIO设置
void SPI_GPIO_Init(void){
GPIO_InitTypeDef GPIO_InitStructure; //定义结构体类型的变量
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC,ENABLE); //使能GPIOC的端口时种
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_11;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; //推挽输出
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; //IO口速度为50Mhz
GPIO_Init(GPIOC,&GPIO_InitStructure);
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_8|GPIO_Pin_10|GPIO_Pin_12;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING; //浮空输入
GPIO_Init(GPIOC,&GPIO_InitStructure);
GPIO_ResetBits(GPIOC,GPIO_Pin_11);
}
4、从设备读写函数
u8 Slave_SPI_RWByte(u8 wdat){
u8 i=0,dat=0;
if(!SPI_CS_READ){ //检测到片选拉低
for(i=0;i<8;i++)
{
while(!SPI_CLK_READ); //检测到时钟上升沿的到来
if(wdat&0x80)SPI_MISO_HIGH; //MSB
else SPI_MISO_LOW;
wdat<<=1;
dat<<=1;
if(SPI_MOSI_READ)dat |= 0x01;
while(SPI_CLK_READ);
}
}
return dat;
}
5、头文件
#define SELECT 1 //0表示主机 其它表示从机
#if SELECT==0
//主机
//CS引脚
#define SPI_CS_HIGH PCout(8)=1
#define SPI_CS_LOW PCout(8)=0
//CLK引脚
#define SPI_CLK_HIGH PCout(10)=1
#define SPI_CLK_LOW PCout(10)=0
//MISO引脚
#define SPI_MISO_READ PCin(11)
//MOSI引脚
#define SPI_MOSI_HIGH PCout(12)=1
#define SPI_MOSI_LOW PCout(12)=0
void SPI_GPIO_Init(void);
u8 Master_SPI_WRByte(u8 wdat);
#else
//从机
//CS引脚
#define SPI_CS_READ PCin(8)
//CLK引脚
#define SPI_CLK_READ PCin(10)
//MISO引脚
#define SPI_MISO_HIGH PCout(11)=1
#define SPI_MISO_LOW PCout(11)=0
//MOSI引脚
#define SPI_MOSI_READ PCin(12)
void SPI_GPIO_Init(void);
u8 Slave_SPI_RWByte(u8 wdat);
#endif
6、主函数
int main(void)
{
u8 key;
delay_init(); //延时函数初始化
led_init(); //初始化与LED连接的硬件接口
key_init();
SPI_GPIO_Init();
while(1)
{
#if SELECT==0
key=key_scan(0);
switch(key){
case key_up_value:{
if(Master_SPI_WRByte(0x50)==0x60)LED1=!LED1;
break;
};
case key_down_value:{
break;
};
case key_left_value:{
break;
};
case key_right_value:{
break;
};
}
#else
Slave_SPI_RWByte(0x60);
#endif
五、实验说明
本实验通过主机向从机发送0x50命令来控制从机LED1亮灭。其他应用层的使用读者可自行完成。