stm32通過i2c存儲(chǔ)數(shù)據(jù)在eeprom
掃描二維碼
隨時(shí)隨地手機(jī)看文章
首先我們來認(rèn)識(shí)一下i2c通訊協(xié)議
i2c總線只需要串行數(shù)據(jù)SDA線以及串行時(shí)鐘SCL線,兩條線都是雙向的。每個(gè)從器件都有一個(gè)唯一的地址以便識(shí)別。
i2c傳輸過程:start-從機(jī)地址-應(yīng)答/非應(yīng)答-R/W(1為讀/0為寫)-數(shù)據(jù)傳輸-應(yīng)答/非應(yīng)答-stop
數(shù)據(jù)傳輸每個(gè)字節(jié)都需要應(yīng)答/非應(yīng)答信號(hào)
模擬i2c傳輸協(xié)議:
根據(jù)時(shí)序圖可以知道:start就是拉高SCL,給SDA下降沿;stop是拉高SCL,給SDA上升沿。
當(dāng)傳輸數(shù)據(jù)時(shí)即SCL為高電平期間,SDA上的數(shù)據(jù)必須保持穩(wěn)定,只有在SCL上的信號(hào)為低電平期間,SDA上的高電平或低電平狀態(tài)才允許變化。
數(shù)據(jù)傳輸一般選擇8位,在8位數(shù)據(jù)傳輸之后,必須進(jìn)行應(yīng)答/非應(yīng)答信號(hào)的接受。
應(yīng)答/非應(yīng)答:應(yīng)答的意思是在數(shù)據(jù)傳輸之后,接收數(shù)據(jù)的器件必須傳輸一個(gè)應(yīng)答信號(hào)給主機(jī),如果沒有傳輸數(shù)據(jù),那么判斷數(shù)據(jù)傳輸不成功,需要stop或者reset。軟件模擬應(yīng)答信號(hào)就是在數(shù)據(jù)傳輸結(jié)束之后,需要將SDA拉高釋放總線,然后從機(jī)會(huì)將SDA置低,主機(jī)據(jù)此判斷SDA電平,確定是否收到應(yīng)答信號(hào)(SDA=0為應(yīng)答,反之則然)。
EEPROM:24系列EEPROM是一種普遍使用的非易失性存儲(chǔ)器,24Cxx中xx的單位是kb,例如24C08的存儲(chǔ)容量是8kb。
器件地址:一般datasheet中會(huì)有給出器件地址。
字節(jié)地址:對(duì)EEPROM進(jìn)行讀寫時(shí),需要選定字節(jié)地址,以24C08為例,讀寫地址在 0x00-0x3ff。
讀/寫時(shí)序:對(duì)于24C08來說,內(nèi)部分為頁(yè),每頁(yè)共可存儲(chǔ)16字節(jié)數(shù)據(jù),即高四位一致的地址?;诖丝梢灾?道,在寫數(shù)據(jù)或者讀數(shù)據(jù)時(shí),要確定是否讀/寫數(shù)據(jù)是連續(xù)的,而且是不是在同一頁(yè)中,如果所讀/寫 數(shù)據(jù)超過一頁(yè),那么則需要進(jìn)行換頁(yè)操作。
寫一整個(gè)字節(jié)的數(shù)據(jù)到EEPROM:
void I2C_EE_ByteWrite(u8* pBuffer,u8 WriteAddr)
{
I2C_GenerateSTART(I2C1,ENABLE); //start while(!I2C_CheckEvent(I2C1,I2C_EVENT_MASTER_MODE_SELECT)); //設(shè)置主機(jī)模式(R/W)I2C_Send7bitAddress(I2C1,EEPROM_ADDRESS,I2C_Direction_Transmitter); //向指定的從IIC設(shè)備傳送地址字,選擇為發(fā)送方向
while(!I2C_CheckEvent(I2C1,I2C_EVENT_MASTER_TRANSMITTER_MODE_SELECTED)); //等待這次選擇過程成功
I2C_SendData(I2C1,WriteAddr); //通過外設(shè)I2C1發(fā)送地址
while(!I2C_CheckEvent(I2C1,I2C_EVENT_MASTER_BYTE_TRANSMITTED)); //等待字節(jié)發(fā)送完成I2C_SendData(I2C1,*pBuffer); //寫入數(shù)據(jù)到EEPROM內(nèi)部 while(!I2C_CheckEvent(I2C1,I2C_EVENT_MASTER_BYTE_TRANSMITTED)); //等待字節(jié)發(fā)送完成
I2C_GenerateSTOP(I2C1,ENABLE); //stop
}
整個(gè)函數(shù)理解起來很簡(jiǎn)單,就是按照步驟走的:start-設(shè)置主從模式-傳送地址到i2c-然后通過i2c總線把地址發(fā)送到從機(jī)-寫數(shù)據(jù)-stop (這其中的I2C_CheckEvent()函數(shù)就是應(yīng)答/非應(yīng)答信號(hào))。
假如寫多頁(yè)數(shù)據(jù)原理上很簡(jiǎn)單,就是地址遞增就好了,這里就不寫了。
讀取從機(jī)某個(gè)地址存儲(chǔ)的數(shù)據(jù):
void I2C_EE_BufferRead(u8* pBuffer,u8 ReadAddr,u16 NumByteToRead)
{
I2C_EE_WaitEepromStandbyState(); //等待EEPROM
while(I2C_GetFlagStatus(I2C1,I2C_FLAG_BUSY)); //檢測(cè)總線是否忙碌
I2C_GenerateSTART(I2C1,ENABLE);
while(!I2C_CheckEvent(I2C1,I2C_EVENT_MASTER_MODE_SELECT));
I2C_Send7bitAddress(I2C1,EEPROM_ADDRESS,I2C_Direction_Transmitter); //向指定的從iic設(shè)備傳輸?shù)刂?,選擇發(fā)送方向
while(!I2C_CheckEvent(I2C1,I2C_EVENT_MASTER_TRANSMITTER_MODE_SELECTED));
I2C_Cmd(I2C1,ENABLE);
I2C_SendData(I2C1,ReadAddr); //發(fā)送要讀取的EEPROM數(shù)據(jù)的起始地址
while(!I2C_CheckEvent(I2C1,I2C_EVENT_MASTER_BYTE_TRANSMITTED));
I2C_GenerateSTART(I2C1,ENABLE); //再次發(fā)送起始條件
while(!I2C_CheckEvent(I2C1,I2C_EVENT_MASTER_MODE_SELECT));
I2C_Send7bitAddress(I2C1,EEPROM_ADDRESS,I2C_Direction_Receiver); //向指定的從iic設(shè)備傳送地址,選擇接受方向
while(!I2C_CheckEvent(I2C1,I2C_EVENT_MASTER_RECEIVER_MODE_SELECTED));
while(NumByteToRead) //直到讀取完成
{
if(NumByteToRead==1)
{
I2C_AcknowledgeConfig(I2C1,DISABLE); //禁止IIC的應(yīng)答功能
I2C_GenerateSTOP(I2C1,ENABLE);
}
if(I2C_CheckEvent(I2C1,I2C_EVENT_MASTER_BYTE_RECEIVED)) //檢測(cè)是否接收到數(shù)據(jù)
{
*pBuffer=I2C_ReceiveData(I2C1); //讀取通過I2C1最近接受的數(shù)據(jù)
pBuffer++;
NumByteToRead--;
}
}
I2C_AcknowledgeConfig(I2C1,ENABLE); //使能I2C的應(yīng)答功能
}
讀數(shù)據(jù)步驟:檢測(cè)總線是否空閑-start-發(fā)送器件地址和寫模式(稱之為偽寫)-發(fā)送讀取數(shù)據(jù)的地址-start-發(fā)送7位器件地址和讀模式-接受數(shù)據(jù)
注意:1,在使用i2c時(shí),SCL和SDA需要上拉
2,很多人說硬件i2c存在bug,一般都是用模擬i2c,或者把i2c放在優(yōu)先級(jí)比較高的DMA中使用,暫時(shí)還沒有定論。