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