木桥 发表于 2014-8-7 10:17:31

请教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
}




lcw_swust 发表于 2014-8-7 10:42:57

作为主机,为了获取从机的数据,主机需要发送SCK时钟,从机根据时钟返回数据。
看起来是主机发送了数据,其实是为了发送时钟。

木桥 发表于 2014-8-7 11:15:42

lcw_swust 发表于 2014-8-7 10:42
作为主机,为了获取从机的数据,主机需要发送SCK时钟,从机根据时钟返回数据。
看起来是主机发送了数据,其 ...

请问可不可以这样理解:在主机向从机发送一字节的数据时,从机的8位数据也位移到了主机的数据寄存器中,因此主机寄存器中读到不是自己赋值给SPDAT的数据而是从机给SPDAT的数据啊?

qufuta 发表于 2014-8-7 11:27:14

其实你没有真正理解SPI,SPI总线协议,是以主机发出的始终为基础进行数据传递的,也就是说,无论数据方向,主机都要无条件地提供始终,这也是SPI通信的一个好处,数据传送永远是同步的

木桥 发表于 2014-8-7 11:36:05

qufuta 发表于 2014-8-7 11:27
其实你没有真正理解SPI,SPI总线协议,是以主机发出的始终为基础进行数据传递的,也就是说,无论数据方向, ...

谢谢指点,其实我上面的分析也就是说明主机这个过程中一直在提供时钟吧!

lcw_swust 发表于 2014-8-7 11:38:20

本帖最后由 lcw_swust 于 2014-8-7 11:42 编辑

木桥 发表于 2014-8-7 11:15
请问可不可以这样理解:在主机向从机发送一字节的数据时,从机的8位数据也位移到了主机的数据寄存器中, ...

对啊,可以这样想:SPDAT在发送时往左边移出去一位,同时右边移入一位读入的数据。
还有一个概念,貌似叫“影子寄存器”,对同一个寄存器的读与写操作对应的是不同的地址。
也就是说,往SPDAT里写入数据,和读取SPDAT的数据,这两个操作有可能是对应硬件中不同的地址。

木桥 发表于 2014-8-7 11:48:34

lcw_swust 发表于 2014-8-7 11:38
对啊,可以这样想:SPDAT在发送时往左边移出去一位,同时右边移入一位读入的数据。
还有一个概念,貌似叫 ...

非常感谢!
页: [1]
查看完整版本: 请教STC12的硬件SPI问题