單片機(jī)EEPROM多字節(jié)讀寫(xiě)操作時(shí)序
掃描二維碼
隨時(shí)隨地手機(jī)看文章
我們讀取 EEPROM 的時(shí)候很簡(jiǎn)單,EEPROM 根據(jù)我們所送的時(shí)序,直接就把數(shù)據(jù)送出來(lái)了,但是寫(xiě) EEPROM 卻沒(méi)有這么簡(jiǎn)單了。給 EEPROM 發(fā)送數(shù)據(jù)后,先保存在了 EEPROM的緩存,EEPROM 必須要把緩存中的數(shù)據(jù)搬移到“非易失”的區(qū)域,才能達(dá)到掉電不丟失的效果。而往非易失區(qū)域?qū)懶枰欢ǖ臅r(shí)間,每種器件不完全一樣,ATMEL 公司的 24C02 的這個(gè)寫(xiě)入時(shí)間最高不超過(guò) 5ms。在往非易失區(qū)域?qū)懙倪^(guò)程,EEPROM 是不會(huì)再響應(yīng)我們的訪問(wèn)的,不僅接收不到我們的數(shù)據(jù),我們即使用 I2C 標(biāo)準(zhǔn)的尋址模式去尋址,EEPROM 都不會(huì)應(yīng)答,就如同這個(gè)總線上沒(méi)有這個(gè)器件一樣。數(shù)據(jù)寫(xiě)入非易失區(qū)域完畢后,EEPROM 再次恢復(fù)正常,可以正常讀寫(xiě)了。
細(xì)心的同學(xué),在看上一節(jié)程序的時(shí)候會(huì)發(fā)現(xiàn),我們寫(xiě)數(shù)據(jù)的那段代碼,實(shí)際上我們有去讀應(yīng)答位 ACK,但是讀到了應(yīng)答位我們也沒(méi)有做任何處理。這是因?yàn)槲覀円淮沃粚?xiě)一個(gè)字節(jié)的數(shù)據(jù)進(jìn)去,等到下次重新上電再寫(xiě)的時(shí)候,時(shí)間肯定遠(yuǎn)遠(yuǎn)超過(guò)了 5ms,但是如果我們是連續(xù)寫(xiě)入幾個(gè)字節(jié)的時(shí)候,就必須得考慮到應(yīng)答位的問(wèn)題了。寫(xiě)入一個(gè)字節(jié)后,再寫(xiě)入下一個(gè)字節(jié)之前,我們必須要等待 EEPROM 再次響應(yīng)才可以,大家注意我們程序的寫(xiě)法,可以學(xué)習(xí)一下。
之前我們知道編寫(xiě)多.c 文件移植的方便性了,本節(jié)程序和上一節(jié)的 Lcd1602.c 文件和I2C.c 文件完全是一樣的,因此這次我們只把 main.c 文件給大家發(fā)出來(lái),幫大家分析明白。
而同學(xué)們卻不能這樣,同學(xué)們是初學(xué),很多知識(shí)和技巧需要多練才能鞏固下來(lái),因此每個(gè)程序還是建議大家在你的 Keil 軟件上一個(gè)代碼一個(gè)代碼的敲出來(lái)。
/*****************************I2C.c 文件程序源代碼*******************************/
(此處省略,可參考之前章節(jié)的代碼)
/***************************Lcd1602.c 文件程序源代碼*****************************/
(此處省略,可參考之前章節(jié)的代碼)
/*****************************main.c 文件程序源代碼******************************/
#include
extern void InitLcd1602();
extern void LcdShowStr(unsigned char x, unsigned char y, unsigned char *str);
extern void I2CStart();
extern void I2CStop();
extern unsigned char I2CReadACK();
extern unsigned char I2CReadNAK();
extern bit I2CWrite(unsigned char dat);
void E2Read(unsigned char *buf, unsigned char addr, unsigned char len);
void E2Write(unsigned char *buf, unsigned char addr, unsigned char len);
void MemToStr(unsigned char *str, unsigned char *src, unsigned char len);
void main(){
unsigned char i;
unsigned char buf[5];
unsigned char str[20];
InitLcd1602(); //初始化液晶
E2Read(buf, 0x90, sizeof(buf)); //從 E2 中讀取一段數(shù)據(jù)
MemToStr(str, buf, sizeof(buf)); //轉(zhuǎn)換為十六進(jìn)制字符串
LcdShowStr(0, 0, str); //顯示到液晶上
for (i=0; i buf[i] = buf[i] + 1 + i; } E2Write(buf, 0x90, sizeof(buf)); //再寫(xiě)回到 E2 中 while(1); } /* 將一段內(nèi)存數(shù)據(jù)轉(zhuǎn)換為十六進(jìn)制格式的字符串, str-字符串指針,src-源數(shù)據(jù)地址,len-數(shù)據(jù)長(zhǎng)度 */ void MemToStr(unsigned char *str, unsigned char *src, unsigned char len){ unsigned char tmp; while (len--){ tmp = *src >> 4; //先取高 4 位 if (tmp <= 9){ //轉(zhuǎn)換為 0-9 或 A-F *str++ = tmp + '0'; }else{ *str++ = tmp - 10 + 'A'; } tmp = *src & 0x0F; //再取低 4 位 if (tmp <= 9){ //轉(zhuǎn)換為 0-9 或 A-F *str++ = tmp + '0'; }else{ *str++ = tmp - 10 + 'A'; } *str++ = ' '; //轉(zhuǎn)換完一個(gè)字節(jié)添加一個(gè)空格 src++; } } /* E2 讀取函數(shù),buf-數(shù)據(jù)接收指針,addr-E2 中的起始地址,len-讀取長(zhǎng)度 */ void E2Read(unsigned char *buf, unsigned char addr, unsigned char len){ do { //用尋址操作查詢當(dāng)前是否可進(jìn)行讀寫(xiě)操作 I2CStart(); if (I2CWrite(0x50<<1)){ //應(yīng)答則跳出循環(huán),非應(yīng)答則進(jìn)行下一次查詢 break; } I2CStop(); } while(1); I2CWrite(addr); //寫(xiě)入起始地址 I2CStart(); //發(fā)送重復(fù)啟動(dòng)信號(hào) I2CWrite((0x50<<1)|0x01); //尋址器件,后續(xù)為讀操作 while (len > 1){ //連續(xù)讀取 len-1 個(gè)字節(jié) *buf++ = I2CReadACK(); //最后字節(jié)之前為讀取操作+應(yīng)答 len--; } *buf = I2CReadNAK(); //最后一個(gè)字節(jié)為讀取操作+非應(yīng)答 I2CStop(); } /* E2 寫(xiě)入函數(shù),buf-源數(shù)據(jù)指針,addr-E2 中的起始地址,len-寫(xiě)入長(zhǎng)度 */ void E2Write(unsigned char *buf, unsigned char addr, unsigned char len){ while (len--){ do { //用尋址操作查詢當(dāng)前是否可進(jìn)行讀寫(xiě)操作 I2CStart(); if (I2CWrite(0x50<<1)){ //應(yīng)答則跳出循環(huán),非應(yīng)答則進(jìn)行下一次查詢 break; } I2CStop(); } while(1); I2CWrite(addr++); //寫(xiě)入起始地址 I2CWrite(*buf++); //寫(xiě)入一個(gè)字節(jié)數(shù)據(jù) I2CStop(); //結(jié)束寫(xiě)操作,以等待寫(xiě)入完成 } } 函數(shù) MemToStr:可以把一段內(nèi)存數(shù)據(jù)轉(zhuǎn)換成十六進(jìn)制字符串的形式。由于我們從EEPROM 讀出來(lái)的是正常的數(shù)據(jù),而 1602 液晶接收的是 ASCII 碼字符,因此我們要通過(guò)液晶把數(shù)據(jù)顯示出來(lái)必須先通過(guò)一步轉(zhuǎn)換。算法倒是很簡(jiǎn)單,就是把每一個(gè)字節(jié)的數(shù)據(jù)高 4 位和低 4 位分開(kāi),和 9 進(jìn)行比較,如果小于等于 9,則直接加?0?轉(zhuǎn)為 0~9 的 ASCII 碼;如果大于 9,則先減掉 10 再加?A?即可轉(zhuǎn)為 A~F 的 ASCII 碼。 函數(shù) E2Read:我們?cè)谧x之前,要查詢一下當(dāng)前是否可以進(jìn)行讀寫(xiě)操作,EEPROM 正常響應(yīng)才可以進(jìn)行。進(jìn)行后,讀最后一個(gè)字節(jié)之前的,全部給出 ACK,而讀完了最后一個(gè)字節(jié),我們要給出一個(gè) NAK。 函數(shù) E2Write:每次寫(xiě)操作之前,我們都要進(jìn)行查詢判斷當(dāng)前 EEPROM 是否響應(yīng),正常響應(yīng)后才可以寫(xiě)數(shù)據(jù)。