请教STC12的硬件SPI问题
请问在“BYTE SPISwap(BYTE dat)”中,为什么作为接受函数的时候还是会执行SPDAT = dat; //trigger SPI send
while (!(SPSTAT & SPIF)); //wait send complete
SPSTAT = SPIF | WCOL; //clear SPI status
这些语句呢?不是只是在发送时候才会用到吗?
代码如下/*------------------------------------------------------------------*/
/* --- STC MCU Limited ---------------------------------------------*/
/* --- STC12C5Axx Series MCU SPI Demo(1 master and 1 slave) --------*/
/* --- Mobile: (86)13922805190 -------------------------------------*/
/* --- Fax: 86-755-82905966 ----------------------------------------*/
/* --- Tel: 86-755-82948412 ----------------------------------------*/
/* --- Web: www.STCMCU.com -----------------------------------------*/
/* If you want to use the program or the program referenced in the*/
/* article, please specify in which data and procedures from STC */
/*------------------------------------------------------------------*/
#include "reg51.h"
//#define MASTER //define:master undefine:slave
#define FOSC 18432000L
#define BAUD (256 - FOSC / 32 / 115200)
typedef unsigned char BYTE;
typedef unsigned int WORD;
typedef unsigned long DWORD;
sfrAUXR = 0x8e; //Auxiliary register
sfr SPSTAT = 0xcd; //SPI status register
#define SPIF 0x80 //SPSTAT.7
#define WCOL 0x40 //SPSTAT.6
sfr SPCTL = 0xce; //SPI control register
#define SSIG 0x80 //SPCTL.7
#define SPEN 0x40 //SPCTL.6
#define DORD 0x20 //SPCTL.5
#define MSTR 0x10 //SPCTL.4
#define CPOL 0x08 //SPCTL.3
#define CPHA 0x04 //SPCTL.2
#define SPDHH 0x00 //CPU_CLK/4
#define SPDH 0x01 //CPU_CLK/16
#define SPDL 0x02 //CPU_CLK/64
#define SPDLL 0x03 //CPU_CLK/128
sfr SPDAT = 0xcf; //SPI data register
sbit SPISS = P1^3; //SPI slave select, connect to slave' SS(P1.4) pin
void InitUart();
void InitSPI();
void SendUart(BYTE dat); //send data to PC
BYTE RecvUart(); //receive data from PC
BYTE SPISwap(BYTE dat); //swap SPI data between master and slave
///////////////////////////////////////////////////////////
void main()
{
InitUart(); //initial UART
InitSPI(); //initial SPI
while (1)
{
#ifdef MASTER //for master (receive UART data from PC and send it to slave, in the meantime
// receive SPI data from slave and send it to PC)
SendUart(SPISwap(RecvUart()));
#else //for salve (receive SPI data from master and
ACC = SPISwap(ACC); // send previous SPI data to master)
#endif
}
}
///////////////////////////////////////////////////////////
void InitUart()
{
SCON = 0x5a; //set UART mode as 8-bit variable baudrate
TMOD = 0x20; //timer1 as 8-bit auto reload mode
AUXR = 0x40; //timer1 work at 1T mode
TH1 = TL1 = BAUD; //115200 bps
TR1 = 1;
}
///////////////////////////////////////////////////////////
void InitSPI()
{
SPDAT = 0; //initial SPI data
SPSTAT = SPIF | WCOL; //clear SPI status
#ifdef MASTER
SPCTL = SPEN | MSTR; //master mode
#else
SPCTL = SPEN; //slave mode
#endif
}
///////////////////////////////////////////////////////////
void SendUart(BYTE dat)
{
while (!TI); //wait pre-data sent
TI = 0; //clear TI flag
SBUF = dat; //send current data
}
///////////////////////////////////////////////////////////
BYTE RecvUart()
{
while (!RI); //wait receive complete
RI = 0; //clear RI flag
return SBUF; //return receive data
}
///////////////////////////////////////////////////////////
BYTE SPISwap(BYTE dat)
{
#ifdef MASTER
SPISS = 0; //pull low slave SS
#endif
SPDAT = dat; //trigger SPI send
while (!(SPSTAT & SPIF)); //wait send complete
SPSTAT = SPIF | WCOL; //clear SPI status
#ifdef MASTER
SPISS = 1; //push high slave SS
#endif
return SPDAT; //return received SPI data
}
作为主机,为了获取从机的数据,主机需要发送SCK时钟,从机根据时钟返回数据。
看起来是主机发送了数据,其实是为了发送时钟。 lcw_swust 发表于 2014-8-7 10:42
作为主机,为了获取从机的数据,主机需要发送SCK时钟,从机根据时钟返回数据。
看起来是主机发送了数据,其 ...
请问可不可以这样理解:在主机向从机发送一字节的数据时,从机的8位数据也位移到了主机的数据寄存器中,因此主机寄存器中读到不是自己赋值给SPDAT的数据而是从机给SPDAT的数据啊? 其实你没有真正理解SPI,SPI总线协议,是以主机发出的始终为基础进行数据传递的,也就是说,无论数据方向,主机都要无条件地提供始终,这也是SPI通信的一个好处,数据传送永远是同步的 qufuta 发表于 2014-8-7 11:27
其实你没有真正理解SPI,SPI总线协议,是以主机发出的始终为基础进行数据传递的,也就是说,无论数据方向, ...
谢谢指点,其实我上面的分析也就是说明主机这个过程中一直在提供时钟吧! 本帖最后由 lcw_swust 于 2014-8-7 11:42 编辑
木桥 发表于 2014-8-7 11:15
请问可不可以这样理解:在主机向从机发送一字节的数据时,从机的8位数据也位移到了主机的数据寄存器中, ...
对啊,可以这样想:SPDAT在发送时往左边移出去一位,同时右边移入一位读入的数据。
还有一个概念,貌似叫“影子寄存器”,对同一个寄存器的读与写操作对应的是不同的地址。
也就是说,往SPDAT里写入数据,和读取SPDAT的数据,这两个操作有可能是对应硬件中不同的地址。 lcw_swust 发表于 2014-8-7 11:38
对啊,可以这样想:SPDAT在发送时往左边移出去一位,同时右边移入一位读入的数据。
还有一个概念,貌似叫 ...
非常感谢!
页:
[1]