STM32學習筆記——SPI串行通訊(向原子哥學習)


 一、SPI  簡介

  SPI是 Serial Peripheral interface 的縮寫,就是串行外圍設備接口。SPI 接口主要應用在  EEPROM, FLASH,實時時鍾,AD 轉換器,還有數字信號處理器和數字信號解碼器之間。SPI,是一種高速的,全雙工,同步的通信總線,並且在芯片的管腳上只占用四根線,節約了芯片的管腳,同時為 PCB 的布局上節省空間,提供方便,正是出於這種簡單易用的特性,現在越來越多的芯片集成了這種通信協議,STM32 有 SPI 接口。下面是 SPI 的內部簡明圖:

 

圖1 SPI 內部結構簡明圖

  SPI 接口一般使用 4 條線通信:

    MISO  主設備數據輸入,從設備數據輸出。

    MOSI  主設備數據輸出,從設備數據輸入。

    SCLK 時鍾信號,由主設備產生。

    CS 從設備片選信號,由主設備控制。

  從圖中可以看出,主機和從機都有一個串行移位寄存器,主機通過向它的 SPI 串行寄存器寫入一個字節來發起一次傳輸。寄存器通過 MOSI 信號線將字節傳送給從機,從機也將自己的移位寄存器中的內容通過 MISO 信號線返回給主機。這樣,兩個移位寄存器中的內容就被交換。外設的寫操作和讀操作是同步完成的。如果只進行寫操作,主機只需忽略接收到的字節;反之,若主機要讀取從機的一個字節,就必須發送一個空字節來引發從機的傳輸。

二、

  SPI 主要特點有:可以同時發出和接收串行數據;可以當作主機或從機工作;提供頻率可編程時鍾;發送結束中斷標志;寫沖突保護;總線競爭保護等。

  SPI 總線四種工作方式  SPI  模塊為了和外設進行數據交換,根據外設工作要求,其輸出串行同步時鍾極性和相位可以進行配置,時鍾極性(CPOL)對傳輸協議沒有重大的影響。如果CPOL=0,串行同步時鍾的空閑狀態為低電平;如果 CPOL=1,串行同步時鍾的空閑狀態為高電平。時鍾相位(CPHA)能夠配置用於選擇兩種不同的傳輸協議之一進行數據傳輸。如果CPHA=0,在串行同步時鍾的第一個跳變沿(CPOL位為0時就是下降沿,CPOL位為’1’時就是上升沿)數據被采樣;如果 CPHA=1,在串行同步時鍾的第二個跳變沿(CPOL位為0時就是下降沿,CPOL位為’1’時就是上升沿)數據被采樣。SPI 主模塊和與之通信的外設備時鍾相位和極性應該一致。

  不同時鍾相位下的總線數據傳輸時序如圖 2 所示:

圖2  不同時鍾相位下的總線傳輸時序(CPHA=0/1)

  STM32 的 SPI 功能很強大,SPI 時鍾最多可以到 18Mhz,支持 DMA,可以配置為 SPI 協議或者 I2S 協議(僅大容量型號支持) 。

  使用 STM32 的 SPI2 的主模式,下面就來看看 SPI2 部分的設置步驟吧。SPI 相關的庫函數和定義分布在文件 stm32f10x_spi.c 以及頭文件 stm32f10x_spi.h 中。

三、STM32 的主模式配置步驟如下:

  1)配置相關引腳的復用功能,使能 SPI2 時鍾

  用 SPI2,第一步就要使能 SPI2 的時鍾。其次要設置 SPI2 的相關引腳為復用輸出,這樣才會連接到 SPI2 上否則這些 IO 口還是默認的狀態,也就是標准輸入輸出口。這里使用的是 PB13、14、15 這 3 個(SCK.、MISO、MOSI,CS 使用軟件管理方式),所以設置這三個為復用 IO。

GPIO_InitTypeDef GPIO_InitStructure;

RCC_APB2PeriphClockCmd(  RCC_APB2Periph_GPIOB, ENABLE );        //PORTB 時鍾使能

RCC_APB1PeriphClockCmd(  RCC_APB1Periph_SPI2, ENABLE );        //SPI2 時鍾使能  

GPIO_InitStructure.GPIO_Pin = GPIO_Pin_13 | GPIO_Pin_14 | GPIO_Pin_15;

GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;             //PB13/14/15 復用推挽輸出

GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_Init(GPIOB,
&GPIO_InitStructure);                  //初始化 GPIOB

  2)初始化 SPI2,設置 SPI2 工作模式

  接下來要初始化 SPI2,設置 SPI2 為主機模式,設置數據格式為 8 位,然設置 SCK 時鍾極性及采樣方式。並設置 SPI2 的時鍾頻率(最大 18Mhz),以及數據的格式(MSB 在前還是LSB 在前)。這在庫函數中是通過 SPI_Init 函數來實現的。

    void SPI_Init(SPI_TypeDef* SPIx, SPI_InitTypeDef* SPI_InitStruct);

  第一個參數是 SPI 標號,這里我們是使用的 SPI2。下面我們來看看第二個參數結構體類型 SPI_InitTypeDef 的定義:

typedef struct

{

uint16_t SPI_Direction;

uint16_t SPI_Mode;

uint16_t SPI_DataSize;

uint16_t SPI_CPOL;

uint16_t SPI_CPHA;

uint16_t SPI_NSS;

uint16_t SPI_BaudRatePrescaler;

uint16_t SPI_FirstBit;

uint16_t SPI_CRCPolynomial;

}SPI_InitTypeDef;

結構體成員變量比較多,這里我們挑取幾個重要的成員變量講解一下:

  第一個參數 SPI_Direction 是用來設置 SPI 的通信方式,可以選擇為半雙工,全雙工,以及串行發和串行收方式,這里我們選擇全雙工模式 SPI_Direction_2Lines_FullDuplex。

  第二個參數 SPI_Mode 用來設置 SPI 的主從模式,這里我們設置為主機模式 SPI_Mode_Master,當然有需要你也可以選擇為從機模式 SPI_Mode_Slave。

  第三個參數 SPI_DataSiz 為 8 位還是 16 位幀格式選擇項,這里我們是 8 位傳輸,選擇 SPI_DataSize_8b。

  第四個參數 SPI_CPOL 用來設置時鍾極性,我們設置串行同步時鍾的空閑狀態為高電平所以我們選擇 SPI_CPOL_High。

  第五個參數 SPI_CPHA 用來設置時鍾相位,也就是選擇在串行同步時鍾的第幾個跳變沿(上升或下降)數據被采樣,可以為第一個或者第二個條邊沿采集,這里我們選擇第二個跳變沿,所以選擇 SPI_CPHA_2Edge

  第六個參數 SPI_NSS 設置 NSS 信號由硬件(NSS 管腳)還是軟件控制,這里我們通過軟件控制 NSS 關鍵,而不是硬件自動控制,所以選擇 SPI_NSS_Soft。

  第七個參數 SPI_BaudRatePrescaler 很關鍵,就是設置 SPI 波特率預分頻值也就是決定 SPI 的時鍾的參數 , 從不分頻道 256 分頻 8 個可選值,初始化的時候我們選擇 256 分頻值 SPI_BaudRatePrescaler_256,  傳輸速度為 36M/256=140.625KHz。

  第八個參數 SPI_FirstBit 設置數據傳輸順序是 MSB 位在前還是 LSB 位在前, ,這里我們選擇SPI_FirstBit_MSB 高位在前。

  第九個參數 SPI_CRCPolynomial 是用來設置 CRC 校驗多項式,提高通信可靠性,大於 1 即可。

設置好上面 9 個參數,就可以初始化 SPI 外設了。初始化的范例格式為:

SPI_InitTypeDef SPI_InitStructure;

SPI_InitStructure.SPI_Direction = SPI_Direction_2Lines_FullDuplex;     //雙線雙向全雙工

SPI_InitStructure.SPI_Mode = SPI_Mode_Master;                  //主 SPI

SPI_InitStructure.SPI_DataSize = SPI_DataSize_8b;               // SPI 發送接收 8 位幀結構

SPI_InitStructure.SPI_CPOL = SPI_CPOL_High;                 //串行同步時鍾的空閑狀態為高電平

SPI_InitStructure.SPI_CPHA = SPI_CPHA_2Edge;                //第二個跳變沿數據被采樣

SPI_InitStructure.SPI_NSS = SPI_NSS_Soft;                   //NSS 信號由軟件控制
  
SPI_InitStructure.SPI_BaudRatePrescaler = SPI_BaudRatePrescaler_256;   //預分頻 256

SPI_InitStructure.SPI_FirstBit = SPI_FirstBit_MSB;              //數據傳輸從 MSB 位開始

SPI_InitStructure.SPI_CRCPolynomial = 7;                    //CRC 值計算的多項式

SPI_Init(SPI2, &SPI_InitStructure);                       //根據指定的參數初始化外設 SPIx 寄存器

  3)使能 SPI2
  初始化完成之后接下來是要使能 SPI2 通信了,在使能 SPI2 之后,我們就可以開始 SPI 通訊了。使能 SPI2 的方法是:
     SPI_Cmd(SPI2, ENABLE); //使能 SPI 外設

  4)SPI 傳輸數據

  通信接口當然需要有發送數據和接受數據的函數,固件庫提供的發送數據函數原型為:

    void SPI_I2S_SendData(SPI_TypeDef* SPIx, uint16_t Data);//往 SPIx 數據寄存器寫入數據  Data,從而實現發送。

  固件庫提供的接受數據函數原型為:

    uint16_t SPI_I2S_ReceiveData(SPI_TypeDef* SPIx)  ;//從 SPIx 數據寄存器讀出接受到的數據。

  5)查看 SPI 傳輸狀態

  在 SPI 傳輸過程中,我們經常要判斷數據是否傳輸完成,發送區是否為空等等狀態,這是通過函數 SPI_I2S_GetFlagStatus 實現的,這個函數很簡單就不詳細講解,判斷發送是否完成的方法是:

    SPI_I2S_GetFlagStatus(SPI2, SPI_I2S_FLAG_RXNE);


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM