Ⅰ
開始和停止條件
SCL時鐘電平為高:
SDA數(shù)據(jù)線由高 -> 低 為總線開始條件;
SDA數(shù)據(jù)線由低 -> 高 為總線結束條件;
(注意:開始之后將SCL變?yōu)榈碗娖?,防止誤操作SDA使其通信停止,見源代碼)
時序圖:
源代碼程序:
/************************************************
函數(shù)名稱 : I2C_Start
功 能 : I2C開始
參 數(shù) : 無
返 回 值 : 無
作 者 : strongerHuang
*************************************************/
voidI2C_Start(void)
{
I2C_SCL_HIGH; //SCL高
I2C_Delay();
I2C_SDA_HIGH; //SDA高 -> 低
I2C_Delay();
I2C_SDA_LOW; //SDA低
I2C_Delay();
I2C_SCL_LOW; //SCL低(待寫地址/數(shù)據(jù))
I2C_Delay();
}
/************************************************
函數(shù)名稱 : I2C_Stop
功 能 : I2C停止
參 數(shù) : 無
返 回 值 : 無
作 者 : strongerHuang
*************************************************/
void I2C_Stop(void)
{
I2C_SDA_LOW; //SDA低 -> 高
I2C_Delay();
I2C_SCL_HIGH; //SCL高
I2C_Delay();
I2C_SDA_HIGH; //SDA高
I2C_Delay();
}
Ⅱ
數(shù)據(jù)位傳輸
SCL時鐘電平為低, 可以改換SDA數(shù)據(jù)線的電平,在SCL上升沿的過程將SDA數(shù)據(jù)發(fā)送出去。
(切記:請先將SCL變?yōu)榈碗娖剑俑淖僑DA電平狀態(tài)。 主要用于I2C讀寫B(tài)yte函數(shù),這兩個函數(shù)網(wǎng)上很多人寫的不規(guī)范,引用需注意,在下面我會舉例說明)
時序圖:
發(fā)送一位“高”數(shù)據(jù)流程:
SCL_LOW時鐘低 -> SDA_HIGH數(shù)據(jù) -> SCL_HIGH時鐘高
Ⅲ
應答位信息
I2C是以字節(jié)(8位)的方式進行傳輸,總線上每傳輸完1字節(jié)之后會有一個應答信號,主器件(主機)需要產(chǎn)生對應的一個額外時鐘。
應答位產(chǎn)生及接收:
1.在(主機)寫數(shù)據(jù)的時候是從機應答(給主機),主機檢測;
2.在(主機)讀數(shù)據(jù)的時候是主機應答(給從機),從機檢測;
(我們借助I2C讀寫函數(shù)一起理解)
1.主機寫,從機應答,主機讀取應答
時序圖:
源代碼:
/************************************************
函數(shù)名稱 : I2C_GetAck
功 能 : I2C主機讀取應答(或非應答)位
參 數(shù) : 無
返 回 值 : I2C_ACK ----- 應答
I2C_NOACK --- 非應答
作 者 : strongerHuang
*************************************************/
uint8_t I2C_GetAck(void)
{
uint8_t ack;
I2C_SCL_LOW; //SCL低 -> 高
I2C_Delay();
I2C_SDA_HIGH; //釋放SDA(開漏模式有效)
I2C_Delay();
I2C_SCL_HIGH; //SCL高(讀取應答位)
I2C_Delay();
if(I2C_SDA_READ)
ack = I2C_NOACK;//非應答
else
ack = I2C_ACK; //應答
I2C_SCL_LOW; //SCL低
I2C_Delay();
returnack;
}
2.主機讀,主機產(chǎn)生應答
時序圖:
源代碼:
/************************************************
函數(shù)名稱 : I2C_PutAck
功 能 : I2C主機產(chǎn)生應答(或非應答)位
參 數(shù) : I2C_ACK ----- 應答
I2C_NOACK --- 非應答
返 回 值 : 無
作 者 : strongerHuang
*************************************************/
voidI2C_PutAck(uint8_t Ack)
{
I2C_SCL_LOW; //SCL低
I2C_Delay();
if(I2C_ACK == Ack)
I2C_SDA_LOW; //應答
else
I2C_SDA_HIGH; //非應答
I2C_Delay();
I2C_SCL_HIGH; //SCL高 -> 低
I2C_Delay();
I2C_SCL_LOW; //SCL低
I2C_Delay();
}
Ⅳ
I2C寫一字節(jié)
這里說的I2C寫,是主機往從機接入1Byte的數(shù)據(jù);
“寫”要求按照上面的“數(shù)據(jù)為傳輸”來操作:在SCL時鐘為低電平時準備好,待SCL為高電平時發(fā)送出去。
寫完一字節(jié)(8位)之后,讀取從機的應答位:
若為0,表示從機應答,可以繼續(xù)下一步操作;
若為1,表示從機非應答,不能進行下一步操作。
注意:
I2C寫一字節(jié),不是EEPROM寫一字節(jié)(需要區(qū)分開來)
寫一字節(jié)時序(前面8位數(shù)據(jù) + 最后1為應答):
源代碼:
/************************************************
函數(shù)名稱 : I2C_WriteByte
功 能 : I2C寫一字節(jié)
參 數(shù) : Data --- 數(shù)據(jù)
返 回 值 : 無
作 者 : strongerHuang
*************************************************/
void I2C_WriteByte(uint8_t Data)
{
uint8_t cnt;
for(cnt=0; cnt<8; cnt++)
{
I2C_SCL_LOW; //SCL低(SCL為低電平時變化SDA有效)
I2C_Delay();
if(Data & 0x80)
I2C_SDA_HIGH;//SDA高
else
I2C_SDA_LOW; //SDA低
Data <<= 1;
I2C_Delay();
I2C_SCL_HIGH; //SCL高(發(fā)送數(shù)據(jù))
I2C_Delay();
}
I2C_SCL_LOW; //SCL低(等待應答信號)
I2C_Delay();
I2C_GetAck(); //讀取應答位
}
提示:
網(wǎng)上常見幾種關于“I2C寫數(shù)據(jù)函數(shù)”的不規(guī)范寫法, 或許整個I2C驅動能通信成功,但各個函數(shù)之間依賴關系很強,不便理解,也不是標準的函數(shù)。
1.首先將SCL置高:
voidI2C_WriteByte(uint8_t Data)
{
uint8_t cnt;
for(cnt=0; cnt<8; cnt++)
{
I2C_SCL_HIGH;
if(Data & 0x80)
I2C_SDA_HIGH;
else
I2C_SDA_LOW;
Data <<= 1;
I2C_SCL_LOW;
}
I2C_GetAck();
}
這種程序的寫法有一個致命的地方(有可能停止,或重新開始I2C通信):
首先將SCL置高:
A.若之前SDA是低電平,第一位寫入高電平,將停止I2C通信。
B.若之前SDA是高電平,第一位寫入低電平,將重新開始I2C通信。
2.寫完8位數(shù)據(jù)之后,未將SCL置低(也就是SCL保持高電平狀態(tài))
由于寫完8位數(shù)據(jù)之后,將要讀取應答信號,也就是要SDA將從輸出狀態(tài)變?yōu)檩斎霠顟B(tài)。
這個時候SCL為高,如果SDA最后一位是低且SDA是開漏模式,需要將SDA釋放,也就是要將SDA置位高,那么,這個時候就進行了一個停止操作。
3.時序混亂
void I2C_WriteByte(uint8_t Data)
{
uint8_t cnt;
I2C_SCL_HIGH;
for(cnt=0; cnt<8; cnt++)
{
if(Data & 0x80)
I2C_SDA_HIGH;
else
I2C_SDA_LOW;
Data <<= 1;
I2C_SCL_LOW;
I2C_SCL_HIGH;
}
I2C_GetAck();
}
多種問題的例子,有可能產(chǎn)生以下問題:
A.有可能多寫1位數(shù)據(jù);
B.有可能停止I2C通信;
C.有可能重新開始I2C通信。
Ⅴ
I2C讀一字節(jié)
I2C的讀一字節(jié)函數(shù),其實和“寫一字節(jié)”類似,只是數(shù)據(jù)傳輸方向相反,應答的方向也是相反。
讀完一字節(jié)(8位)之后,由主機產(chǎn)生應答(或非應答)位:
若產(chǎn)生應答,表示可以繼續(xù)讀下一字節(jié)操作(從設備地址指向下一字節(jié));
若產(chǎn)生非應答,表示不可以繼續(xù)讀下一字節(jié)操作;
網(wǎng)上I2C讀數(shù)據(jù)程序和“寫數(shù)據(jù)”類似,存在很多不標準的版本,參考時請注意。
讀一字節(jié)時序(主機讀取前面8位數(shù)據(jù) + 主機產(chǎn)生1為非應答<連續(xù)讀,主機產(chǎn)生應答位>):
源代碼:
/************************************************
函數(shù)名稱 : I2C_ReadByte
功 能 : I2C讀一字節(jié)
參 數(shù) : ack --------- 產(chǎn)生應答(或者非應答)位
返 回 值 : data -------- 讀取的一字節(jié)數(shù)據(jù)
作 者 : strongerHuang
*************************************************/
uint8_t I2C_ReadByte(uint8_t ack)
{
uint8_t cnt;
uint8_t data;
I2C_SCL_LOW; //SCL低
I2C_Delay();
I2C_SDA_HIGH; //釋放SDA(開漏模式有效)
for(cnt=0; cnt<8; cnt++)
{
I2C_SCL_HIGH; //SCL高(讀取數(shù)據(jù))
I2C_Delay();
data <<= 1;
if(I2C_SDA_READ)
data |= 0x01; //SDA為高(數(shù)據(jù)有效)
I2C_SCL_LOW; //SCL低
I2C_Delay();
}
I2C_PutAck(ack); //產(chǎn)生應答(或者非應答)位
return data; //返回數(shù)據(jù)
}
-
協(xié)議
+關注
關注
2文章
606瀏覽量
39352 -
I2C
+關注
關注
28文章
1495瀏覽量
124699 -
SCL
+關注
關注
1文章
240瀏覽量
17185
發(fā)布評論請先 登錄
相關推薦
I2C總線通信原理 如何設計I2C總線電路
基于I2C總線的智能家居應用
I2C總線的優(yōu)缺點分析
I2C總線的工作模式介紹
I2C總線協(xié)議詳細解析
I2C總線故障排除技巧
I2C總線與單片機的連接
I2C總線應用實例分析
I2C總線與SPI總線的比較
I2C總線上拉電阻的必要性
簡單認識I2C通信協(xié)議
I2C總線協(xié)議的工作原理和尋址格式
什么是I2C協(xié)議 I2C總線的控制邏輯
![什么是<b class='flag-5'>I2C</b><b class='flag-5'>協(xié)議</b> <b class='flag-5'>I2C</b><b class='flag-5'>總線</b>的控制邏輯](https://file1.elecfans.com/web2/M00/C3/E6/wKgZomXvre-AWsW5AABL2e5FJAM091.png)
詳解I2C接口協(xié)議
![詳解<b class='flag-5'>I2C</b>接口<b class='flag-5'>協(xié)議</b>](https://file1.elecfans.com/web2/M00/C3/DA/wKgaomXpJVWAAqdsAAAPYf0u4CY429.png)
評論