以下內容,將解釋以下幾個問題
1.IIC協議是什么?
2.IIC協議用來干什么?
3.IIC協議的通信過程?
1.IIC協議是什么?
IIC,即I2C,全稱 Inter-Integrated Circuit,字面上的意思是集成電路之間,它其實是I2C Bus簡稱,所以中文應該叫 集成電路總線 ,它是一種串行通信總線,使用多主從架構,由飛利浦公司在1980年代為了讓主板、嵌入式系統或手機用以連接低速周邊設備而發展。(百度百科)
2.IIC協議用來干什么?
簡單地說,IIC就是一種通信協議,是為了能讓主板,或嵌入式系統等與其他外設模塊進行通信而進行開發的。玩過stm32開發板的同學都知道,對于一塊stm32核心開發板而言,要想使用其他的外設模塊,就肯定要經過接線,寫代碼,燒錄運行的這個過程。
其實這個過程,就是一個stm32與外設模塊通信的過程。接線,就是搭建通信的線路。寫代碼,就是制定通信的傳輸協議。燒錄運行,就是正式的通信過程。只不過有的模塊通信過程很簡單,大家感覺不出來。
外設和芯片間的通信可以形象地比喻成兩個人講話:
你說的別人得能聽懂:雙方約定信號的協議
你的語速別人得能接受:雙方滿足時序要求
但是隨著科技的發展,模塊越來越多,總不可能,每個模塊都要制定一種通信協議,這樣不現實。所以,總要有一些代表性的協議能夠適應大部分的模塊的通信。IIC這是這樣一種協議,一個IIC總線上,可以掛載多個外接設備。
常用的串行通信協議有:
①UART串口通信
②IIC協議
③SPI協議
④USB協議(很難)
常用的并行通信協議有:
①8080
②6800
3 .IIC協議的通信過程( 此處重點 )
接線:要搭建IIC的通信線路,出除去電源之外,還需要兩條線,分別是SDA和SCLK
SDA:數據信號線,用于傳輸數據
SCLK:時鐘信號線,用于產生時鐘頻率,控制時序,實現協議過程
由此可以看出,由于是單總線進行數據傳輸,所以IIC協議是半雙工的。
搭建好線路之后,就要進行具體的通信了。
要通信,總得先發個開始信號吧。就像你要和別人說話,總要先喊他一聲一樣。如下圖所示,協議規定,當SCLK時鐘信號一直處于高電平狀態時,SDA線由高電平跳變到低電平這個動作,表示起始信號。注意此時就算SDA數據線的電平跳變完,SCLK線依然是高電平哦。當連接在IIC總線上的外設模塊檢測到這個信號時,就知道數據要開始傳輸了。對于結束信號同理,協議規定,當SCLK時鐘信號一直處于高電平狀態時,SDA線由低電平跳變到高電平這個動作,表示結束信號。
在明白如何開始之后,就要開始進行數據的傳輸了。
協議規定,在數據的傳輸過程中,SCLK為高電平時,外設模塊開始采集SDA數據線上的數據,此時要求SDA數據線上的電平狀態必須穩定(不然鬼知道這一位數據是0還是1),當SCLK為低電平時才允許SDA線上的數據跳變成另外一種狀態。
以下以傳輸1個bit的數據為例,如下圖所示:
現在,我想傳輸1bit數據,該位數據為1,從上文知道,我們在發完開始信號之后,此時SDA數據線的電平狀態為低電平,SCLK信號依然是高電平。難道這個時候外設就要開始讀取數據了嗎?
這顯然不是的,從發完開始信號到真正的數據傳輸之間,會有一段緩沖時間,讓我們去準備數據,在準備數據階段,先將SCLK信號拉低一段時間,在這期間將SDA數據線拉高一段時間(即數據1),然后再將SCLK信號拉高,此時這個時鐘信號的高電平被外設檢測到的話,外設就知道要讀取數據了,從而SDA上的數據就會被外設讀到了。依次類推,傳輸下一位數據。
一般,傳輸完1個字節(即8bit,高位先入)的數據,才算做一次完整的數據傳輸,因為對存儲單元而言,最小的單位便是字節。那如何確定,每次都完好地傳輸了一個字節呢?
這種情況就需要外設來做出回應了,就像打電話一樣,如果對方不在,或不想聽,說再多也沒用啊。那么外設如何做出回應呢?
協議規定,主機每傳完一個字節的數據即外設每收到一個字節的數據,外設就要在第9個時鐘脈沖到來的時候,將SDA數據線拉低進行應答(ACK),且必須是穩定的低電平,表示已經收到了一個字節的數據,拉高表示不進行應答(NACK;注意這里是外設將SDA數據線拉低,不是主機了哦。如下圖所示:
所以在主機傳完一個字節的數據之后,就應該釋放總線(協議規定,當SDA和SCLK同時為高時,表示空閑狀態)然后把SDA數據線連接的IO口從輸出模式轉換成輸入模式,這樣才能拿到SDA數據線上的應答信號。這樣,一個字節的數據就從主機到外設傳輸完畢了。
既然IIC是雙向通信的,那主機肯定也是需要從外設讀取數據的,那這個讀取的過程又是怎么實現的呢?畢竟外設對于我們而言是不能直接操作的,我們能操作的只有stm32。我們知道,一個IIC總線上,可以掛載多個設備,那么stm32如何確定是哪個外設正在跟我進行通信呢。對于此,那些生產外設模塊的廠商們就約定,要是這個設備使用IIC協議進行通信,那么就要給這個設備指定一個器件地址,以供芯片訪問。這個器件地址會在你購買其模塊的時候在使用手冊上注明。所以,要跟哪個模塊通信,就一定要通過查閱其使用手冊,找到它的器件地址。
所以,在上文所述的最開始的一個字節的數據傳輸過程中,這一個數據往往是器件地址。這樣,對應的外設才知道,是要跟我進行通信。讀取數據,也是同理,要想從外設中讀取到數據,主機要明確三點:從哪個外設中的哪個地方讀取數據,讀取到的數據要存到哪里。
所以主機,在開始讀數據之前,主機必須要先給外設發器件地址,數據所在的地址,外設才會知道你要從該地址讀取數據,從而把數據通過SDA線傳出來。至于具體的每個字節的傳輸過程,和上面所講的從主機到外設的過程差不多,只不過反了一個反向而已,并且主機的等待應答變成了主動應答。
/* 設置SDA總線為輸出模式 參數值:NULL 返回值:NULL*/ void IIC_setSDAMode_Out(){ GPIO_InitTypeDef GPIO_IIC; RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOE, ENABLE); GPIO_IIC.GPIO_Mode = GPIO_Mode_OUT; //輸出 GPIO_IIC.GPIO_OType = GPIO_OType_PP; //推挽 GPIO_IIC.GPIO_Pin = GPIO_Pin_15; //引腳 GPIO_IIC.GPIO_PuPd = GPIO_PuPd_UP; //上拉 GPIO_IIC.GPIO_Speed = GPIO_Speed_25MHz; //輸出 GPIO_Init(GPIOE, &GPIO_IIC);} /* 設置SDA總線為輸入模式 參數值:NULL 返回值:NULL*/ void IIC_setSDAMode_In(){ GPIO_InitTypeDef GPIO_IIC; RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOE, ENABLE); GPIO_IIC.GPIO_Mode = GPIO_Mode_IN; //輸出 GPIO_IIC.GPIO_Pin = GPIO_Pin_15; //引腳 GPIO_IIC.GPIO_PuPd = GPIO_PuPd_UP; //上拉 GPIO_Init(GPIOE, &GPIO_IIC); } /* IIC開始信號 參數值:NULL 返回值:NULL*/ void IIC_Start(){ IIC_setSDAMode_Out(); IIC_SDA_OUT(1); //總線釋放狀態 IIC_SCL_OUT(1); delay_us(5); IIC_SDA_OUT(0); //SDA跳變為低電平 delay_us(5); IIC_SCL_OUT(0); delay_us(5); } /* IIC停止信號 參數值:NULL 返回值:NULL*/ void IIC_Stop(){ IIC_setSDAMode_Out(); IIC_SDA_OUT(0); IIC_SCL_OUT(0); delay_us(5); IIC_SCL_OUT(1); //SDA跳變為高電平 delay_us(5); IIC_SDA_OUT(1); delay_us(5); } /* 主機寫入數據到外設中 參數值: data 要寫入的一個字節 返回值:NULL*/ void IIC_writeByte(u8 data){ IIC_setSDAMode_Out(); IIC_SCL_OUT(0); //只有時鐘線拉低,SDA上的數據才允許寫入 delay_us(5); //將數據一位一位的發出去 for(int i =0;i<8;i++) { if(data&(0x1<<(7-i))) //高位先入 { IIC_SDA_OUT(1); } else { IIC_SDA_OUT(0); } IIC_SCL_OUT(1); //讓外設讀取數據 delay_us(5); IIC_SCL_OUT(0); //重新拉低,準備寫入下一位數據 delay_us(5); }} /* 主機從外設中讀取一個字節的數據 參數值:NULL 返回值:NULL*/ u8 IIC_readByte(){ u8 data = 0; IIC_setSDAMode_In(); IIC_SCL_OUT(0); //先拉低,為讀取數據做準備 delay_us(5); for(int i=0;i<8;i++) { IIC_SCL_OUT(1); // SCL為高期間才可以讀取數據 delay_us(5); if(IIC_SDA_IN) { data|=(0x01<<(7-i)); }else{ data &= ~(0x1<<(7-i)); } IIC_SCL_OUT(0); delay_us(5); } return data; } /* 主機等待應答 參數值:NULL 返回值:ack 0 應答 1 不應答*/ u8 IIC_waitAck(){ u8 ack =0; IIC_setSDAMode_In(); IIC_SCL_OUT(0); //準備時序 delay_us(5); IIC_SCL_OUT(1); delay_us(5); if(IIC_SDA_IN) { ack =1; } else { ack =0; } IIC_SCL_OUT(0); //拉低,表示應答完成 delay_us(5); return ack; } /* 主機主動應答 參數值: ack 0 應答 1 不應答 返回值:NULL*/ void IIC_Ack(u8 ack){ IIC_setSDAMode_Out(); IIC_SCL_OUT(0); delay_us(5); if(ack) { IIC_SDA_OUT(1); } else { IIC_SCL_OUT(0); } IIC_SCL_OUT(1); delay_us(5); IIC_SCL_OUT(0); delay_us(5); }
審核編輯:湯梓紅
-
通信協議
+關注
關注
28文章
858瀏覽量
40268 -
總線
+關注
關注
10文章
2868瀏覽量
87991 -
IIC
+關注
關注
11文章
300瀏覽量
38277
原文標題:IIC協議超詳細解釋(適合小白入門)
文章出處:【微信號:傳感器與檢測技術,微信公眾號:傳感器與檢測技術】歡迎添加關注!文章轉載請注明出處。
發布評論請先 登錄
相關推薦
評論