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