日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

SPI协议简介—学习笔记

發布時間:2023/12/20 编程问答 53 豆豆
生活随笔 收集整理的這篇文章主要介紹了 SPI协议简介—学习笔记 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

SPI協議簡介—學習筆記

1、SPI百度百科

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

2、SPI模式

SPI分為主、從兩種模式,一個SPI通訊系統需要包含一個(且只能是一個)主設備,一個或多個從設備。

提供時鐘的為主設備(Master),接收時鐘的設備為從設備(Slave),SPI接口的讀寫操作,都是由主設備發起。當存在多個從設備時,通過各自的片選信號進行管理。

SPI是全雙工且SPI沒有定義速度限制,一般的實現通常能達到甚至超過10 Mbps

3、SPI信號線

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

SDI(數據輸入),SDO(數據輸出),SCK(時鐘),CS(片選)

  • MISO: 主設備輸入/從設備輸出引腳。該引腳在從模式下發送數據,在主模式下接收數據。

  • MOSI: 主設備輸出/從設備輸入引腳。該引腳在主模式下發送數據,在從模式下接收數據。

  • SCLK:串行時鐘信號,由主設備產生。

  • CS/SS:從設備片選信號,由主設備控制。它的功能是用來作為“片選引腳”,也就是選擇指定的從設備,讓主設備可以單獨地與特定從設備通訊,避免數據線上的沖突。

1.SPI一對一:

2.SPI一對多:

4、SPI設備選擇

SPI是單主設備( single-master )通信協議,這意味著總線中的只有一支中心設備能發起通信。

當SPI主設備想讀/寫[從設備]時:

  • 它首先拉低[從設備]對應的SS線(SS是低電平有效
  • 接著開始發送工作脈沖到時鐘線
  • 在相應的脈沖時間上,[主設備]把信號發到MOSI實現“寫”,同時可對MISO采樣而實現“讀”
  • 如下圖:

    5、SPI數據發送接收

    SPI主機和從機都有一個串行移位寄存器,主機通過向它的SPI串行寄存器寫入一個字節來發起一次傳輸。

  • 首先拉低對應SS信號線,表示與該設備進行通信
  • 主機通過發送SCLK時鐘信號,來告訴從機寫數據或者讀數據
  • 這里要注意,SCLK時鐘信號可能是低電平有效,也可能是高電平有效,因為SPI有四種模式,下面會介紹

  • 主機(Master)將要發送的數據寫到發送數據緩存區(Menory),緩存區經過移位寄存器(0~7),串行移位寄存器通過MOSI信號線將字節一位一位的移出去傳送給從機,同時MISO接口接收到的數據經過移位寄存器一位一位的移到接收緩存區

  • **從機(Slave)**也將自己的串行移位寄存器(0~7)中的內容通過MISO信號線返回給主機。同時通過MOSI信號線接收主機發送的數據

  • 這樣,兩個移位寄存器中的內容就被交換。

    SPI只有主模式和從模式之分,沒有讀和寫的說法,外設的寫操作和讀操作是同步完成的。

    如果只進行寫操作,主機只需忽略接收到的字節;反之,若主機要讀取從機的一個字節,就必須發送一個空字節來引發從機的傳輸。

    也就是說,發一個數據必然會收到一個數據;要收一個數據必須也要先發一個數據。

    6、SPI通信的四種模式

    SPI的四種模式,簡單地講就是設置SCLK時鐘信號線的那種信號為有效信號

    它們的區別是定義了在時鐘脈沖的哪條邊沿轉換(toggles)輸出信號,哪條邊沿采樣輸入信號,還有時鐘脈沖的穩定電平值(就是時鐘信號無效時是高還是低)。每種模式由一對參數刻畫,它們稱為時鐘極(clock polarity)CPOL與時鐘期(clock phase)CPHA。

    • SPI通信有4種不同的操作模式,不同的從設備可能在出廠是就是配置為某種模式,這是不能改變的
    • 但我們的通信雙方必須是工作在同一模式下
    • 所以我們可以對我們的主設備的SPI模式進行配置,通過CPOL(時鐘極性)和CPHA(時鐘相位)來控制我們主設備的通信模式

    具體如下:

    1. 時鐘極性(CPOL)定義了時鐘空閑狀態電平:

    CPOL=0,表示當SCLK=0時處于空閑態,所以有效狀態就是SCLK處于電平時**(高有效)**
    CPOL=1,表示當SCLK=1時處于空閑態,所以有效狀態就是SCLK處于電平時**(低有效)**

    2. 時鐘相位(CPHA)定義數據的采集時間。

    CPHA=0,在時鐘的第1個跳變沿(上升沿或下降沿)進行數據采樣。在第2個邊沿發送數據**(1采2發)**
    CPHA=1,在時鐘的第2個跳變沿(上升沿或下降沿)進行數據采樣。在第1個邊沿發送數據**(1發2采)**

    例如:

  • Mode0:CPOL=0,CPHA=0:此時空閑態時,SCLK處于低電平;數據采樣是在第1個邊沿,也就是SCLK由低電平到高電平的跳變;所以數據采樣是在上升沿,數據發送是在下降沿。

  • Mode1:CPOL=0,CPHA=1:此時空閑態時,SCLK處于低電平;數據發送是在第1個邊沿,也就是SCLK由低電平到高電平的跳變;所以數據采樣是在下降沿,數據發送是在上升沿。

  • Mode2:CPOL=1,CPHA=0:此時空閑態時,SCLK處于高電平;數據采樣是在第1個邊沿,也就是SCLK由高電平到低電平的跳變;所以數據采集是在下降沿,數據發送是在上升沿。

  • Mode3:CPOL=1,CPHA=1:此時空閑態時,SCLK處于高電平;數據發送是在第1個邊沿,也就是SCLK由高電平到低電平的跳變;所以數據采集是在上升沿,數據發送是在下降沿。

  • 注:數據采樣=準備數據

    CPOL=0:高有效,0到1,即上升沿是第一個邊沿

    CPOL=1:低有效,1到0,即下降沿是第一個邊沿

    列個表就是:

    四個模式的時序圖:

    注:Toggling edge=切換(發送)邊緣、Sampling edge=取樣邊緣

    **自己碎碎念:**拿SPI MODE 1舉例,CPOL=0,CPHA=1,即高有效、1發2采,也就是低電平之后第一個邊沿(也就是上升沿)發送數據

    對照下邊這個圖就更明白了:

    7、SPI的通信協議

    主從設備必須使用相同的工作模式——SCLK、CPOL 和 CPHA,才能正常工作。

    如果有多個從設備,并且它們使用了不同的工作模式,那么主設備必須在讀寫不同從設備時需要重新修改對應從設備的模式。

    SPI就是如此,他沒有規定最大傳輸速率,沒有地址方案,也沒規定通信應答機制,沒有規定流控制規則。

    只要四根信號線連接正確,SPI模式相同,將CS/SS信號線拉低,即可以直接通信,一次一個字節的傳輸,讀寫數據同時操作,這就是SPI

    SPI并不關心物理接口的電氣特性,例如信號的標準電壓。

    這也是SPI接口的一個缺點:沒有指定的流控制,沒有應答機制確認是否接收到數據。

    8、SPI的三種模式

    SPI工作在三種模式下,分別是:運行、等待停止。

    1.運行模式(Run Mode)
    這是基本的操作模式

    2.等待模式(Wait Mode)
    SPI工作在等待模式是一種可配置的低功耗模式,可以通過SPICR2寄存器的SPISWAI位進行控制。在等待模式下,如果SPISWAI位清0,SPI操作類似于運行模式。如果SPISWAI位置1,SPI進入低功耗狀態,并且SPI時鐘將關閉。如果SPI配置為主機,所有的傳輸將停止,但是會在CPU進入運行模式后重新開始。如果SPI配置為從機,會繼續接收和傳輸一個字節,這樣就保證從機與主機同步。

    3.停止模式(Stop Mode)
    為了降低功耗,SPI在停止模式是不活躍的。如果SPI配置為主機,正在進行的傳輸會停止,但是在CPU進入運行模式后會重新開始。如果SPI配置為從機,會繼續接受和發送一個字節,這樣就保證了從機與主機同步。

    9、原理圖

    10、代碼實現過程

    STM32中SPI初始化配置

  • 初始化GPIO口,配置相關引腳的復用功能,使能SPIx時鐘。調用函數:void GPIO_Init();
  • 使能SPI時鐘總線:RCC_APB2PeriphClockCmd(RCC_APB2Periph_SPI1,ENABLE)
  • 配置SPI初始化的參數,設置SPI工作模式:SPI_Init(SPI1,&SPI_Initstructure)
  • 使能SPI外設:SPI_Cmd(SPI1,ENABLE);
  • SPI配置設置

    typedef struct { uint16_t SPI_Direction; /*!< 傳輸方向,兩向全雙工,單向接收等*/ uint16_t SPI_Mode; /*!< 模式選擇,確定主機還是從機 */ uint16_t SPI_DataSize; /*!< 數據大小,8位還是16位 */ uint16_t SPI_CPOL; /*!< 時鐘極性選擇 */ uint16_t SPI_CPHA; /*!< 時鐘相位選擇 */ uint16_t SPI_NSS; /*!< 片選是硬件還是軟件*/ uint16_t SPI_BaudRatePrescaler; /*!< 分頻系數 */ uint16_t SPI_FirstBit; /*!< 指定數據傳輸是從MSB還是LSB位開始的。M SB就是二進制第一位,LSB就是最后一位 */ uint16_t SPI_CRCPolynomial; /*!< CRC校驗 ,設置 CRC 校驗多項式,提高通 信可靠性,大于 1 即可*/ }SPI_InitTypeDef;

    void SPI2_Init(void) {GPIO_InitTypeDef GPIO_InitStructure;SPI_InitTypeDef SPI_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);//初始化GPIOBGPIO_SetBits(GPIOB,GPIO_Pin_13|GPIO_Pin_14|GPIO_Pin_15); //PB13/14/15上拉SPI_InitStructure.SPI_Direction = SPI_Direction_2Lines_FullDuplex; //設置SPI單向或者雙向的數據模式:SPI設置為雙線雙向全雙工SPI_InitStructure.SPI_Mode = SPI_Mode_Master; //設置SPI工作模式:設置為主SPISPI_InitStructure.SPI_DataSize = SPI_DataSize_8b; //設置SPI的數據大小:SPI發送接收8位幀結構SPI_InitStructure.SPI_CPOL = SPI_CPOL_High; //串行同步時鐘的空閑狀態為高電平SPI_InitStructure.SPI_CPHA = SPI_CPHA_2Edge; //串行同步時鐘的第二個跳變沿(上升或下降)數據被采樣SPI_InitStructure.SPI_NSS = SPI_NSS_Soft; //NSS信號由硬件(NSS管腳)還是軟件(使用SSI位)管理:內部NSS信號有SSI位控制SPI_InitStructure.SPI_BaudRatePrescaler = SPI_BaudRatePrescaler_256; //定義波特率預分頻的值:波特率預分頻值為256SPI_InitStructure.SPI_FirstBit = SPI_FirstBit_MSB; //指定數據傳輸從MSB位還是LSB位開始:數據傳輸從MSB位開始SPI_InitStructure.SPI_CRCPolynomial = 7; //CRC值計算的多項式SPI_Init(SPI2, &SPI_InitStructure); //根據SPI_InitStruct中指定的參數初始化外設SPIx寄存器SPI_Cmd(SPI2, ENABLE); //使能SPI外設SPI2_ReadWriteByte(0xff);//啟動傳輸 }

    SPI發送函數(標準庫/HAL庫)

    u8 SPIx_ReadWriteByte(u8 TxData) {u8 retry=0;while (SPI_I2S_GetFlagStatus(SPI2, SPI_I2S_FLAG_TXE) == RESET){}//等待發送區空SPI_I2S_SendData(SPI2, TxData); //通過外設SPIx發送一個byte 數據while (SPI_I2S_GetFlagStatus(SPI2, SPI_I2S_FLAG_RXNE) == RESET){} //等待接收完一個bytereturn SPI_I2S_ReceiveData(SPI2); //返回通過SPIx最近接收的數據 }HLA庫: uint8_t SPI_SendByte(uint8_t byte) {uint8_t d_read,d_send=byte;if(HAL_SPI_TransmitReceive(&hspi1,&d_send,&d_read,1,0xFFFFFF)!=HAL_OK)d_read=0XFF;return d_read;

    本學習筆記整理參考來源:

    https://blog.csdn.net/as480133937/article/details/105764119

    總結

    以上是生活随笔為你收集整理的SPI协议简介—学习笔记的全部內容,希望文章能夠幫你解決所遇到的問題。

    如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。