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