STC微控制器的內部EEPROM是用DATAFLASH模擬出來的,不是真正的EEPROM儲存器,不能用普通的方法來操作
下面是一些注意點:
1.位元組寫之前要先將這個位元組所在扇區的其它有效資料讀取到RAM暫存(這步不是必須的)
2.暫存完之後再對整個扇區(512位元組)進行擦除操作,擦拭完後,整個扇區每個地址中資料都變成0xFF
3.將欲寫入的N個位元組資料,用位元組寫函式寫入EEPROM
4.將暫存到RAM的其它有用的EEPROM值再用位元組寫函式寫回EEPROM
5.STC用FLASH模擬出來的EEPROM的位元組寫功能只能將1變成0,而不能將0變成1,
只有扇區擦除後資料才是全1,
例如:在地址0x21f0處第1次寫11010110,第2次寫111010,讀出結果是這2個值的相與10010
所以如果一個地址處的值不是0xff時寫入新的資料是不對的,要先執行扇區擦除,變為0xff,
對於單個位元組的寫入,我們可以先檢查該地址處的資料是否為0xff,是的話就不用擦除扇區了
----------------------------------------------------------------------
STC89C52微控制器內部EEPROM 的讀寫過程
1 配置ISP_CONTR暫存器,使能第7位ISPEN,讓ISP_IAP功能生效,並配置低3位的等待時間
2 寫指令: 讀/寫/擦除扇區 這3個命令
3 賦值: ISP_ADDRH和ISP_ADDRL的地址值
4 關閉總中斷EA,因為下面要寫的2個觸發指令必須是連續操作的,不能被中斷
5 執行公用的 ISP_IAP 觸發指令,觸發後讀寫操作才能進行
6 開啟中斷 EA, 關閉ISP_IAP功能:清相關暫存器
#include "my51.h" /******************定義命令位元組******************/ #define read_cmd 0x01 //位元組讀資料命令 #define wirte_cmd 0x02 //位元組程式設計資料命令 #define erase_cmd 0x03 //扇區擦除資料命令 /****************特殊功能暫存器宣告****************/ sfr ISP_DATA = 0xe2; sfr ISP_ADDRH = 0xe3; sfr ISP_ADDRL = 0xe4; sfr ISP_CMD = 0xe5; sfr ISP_TRIG = 0xe6; sfr ISP_CONTR = 0xe7; /*定義Flash 操作等待時間及允許IAP/ISP/EEPROM 操作的常數******************/ //#define enable_waitTime 0x80 //系統工作時鐘<30MHz 時,對IAP_CONTR 暫存器設定此值 //#define enable_waitTime 0x81 //系統工作時鐘<24MHz 時,對IAP_CONTR 暫存器設定此值 //#define enable_waitTime 0x82 //系統工作時鐘<20MHz 時,對IAP_CONTR 暫存器設定此值 #define enable_waitTime 0x83 //系統工作時鐘<12MHz 時,對IAP_CONTR 暫存器設定此值 //#define enable_waitTime 0x84 //系統工作時鐘<6MHz 時,對IAP_CONTR 暫存器設定此值 void ISP_IAP_disable(void) //關閉ISP_IAP { EA=1; //恢復中斷 ISP_CONTR = 0x00; ISP_CMD = 0x00; ISP_TRIG = 0x00; } void ISP_IAP_trigger() //觸發 { EA=0; //下面的2條指令必須連續執行,故關中斷 ISP_TRIG = 0x46; //送觸發命令字0x46 ISP_TRIG = 0xB9; //送觸發命令字0xB9 } void ISP_IAP_readData(u16 beginAddr, u8* pBuf, u16 dataSize) //讀取資料 { ISP_DATA=0; //清零,不清也可以 ISP_CMD = read_cmd; //指令:讀取 ISP_CONTR = enable_waitTime; //開啟ISP_IAP,並送等待時間 while(dataSize--) //迴圈讀取 { ISP_ADDRH = (u8)(beginAddr >> 8); //送地址高位元組 ISP_ADDRL = (u8)(beginAddr & 0x00ff); //送地址低位元組 ISP_IAP_trigger(); //觸發 beginAddr++; //地址++ *pBuf++ = ISP_DATA; //將資料儲存到接收緩衝區 } ISP_IAP_disable(); //關閉ISP_IAP功能 } void ISP_IAP_writeData(u16 beginAddr,u8* pDat,u16 dataSize) //寫資料 { ISP_CONTR = enable_waitTime; //開啟ISP_IAP,並送等待時間 ISP_CMD = wirte_cmd; //送位元組程式設計命令字 while(dataSize--) { ISP_ADDRH = (u8)(beginAddr >> 8); //送地址高位元組 ISP_ADDRL = (u8)(beginAddr & 0x00ff); //送地址低位元組 ISP_DATA = *pDat++; //送資料 beginAddr++; ISP_IAP_trigger(); //觸發 } ISP_IAP_disable(); //關閉 } void ISP_IAP_sectorErase(u16 sectorAddr) //扇區擦除 { ISP_CONTR = enable_waitTime; //開啟ISP_IAP;並送等待時間 ISP_CMD = erase_cmd; //送扇區擦除命令字 ISP_ADDRH = (u8)(sectorAddr >> 8); //送地址高位元組 ISP_ADDRL = (u8)(sectorAddr & 0X00FF); //送地址低位元組 ISP_IAP_trigger(); //觸發 ISP_IAP_disable(); //關閉ISP_IAP功能 } void main() //測試 { u8 buf[3]={0}; //接收資料緩衝區 u8 dat[5]={b(111010),b(1001),b(1),b(1011),b(1110)};//我寫成二進位制是為觀察led燈 ISP_IAP_sectorErase(0x2000); //扇區擦除,一塊512位元組 ISP_IAP_writeData(0x21f0,dat,sizeof(dat)); //寫EEPROM ISP_IAP_readData(0x21f0,buf,sizeof(buf)); //讀取 P1=buf[2];//在地址0x21f0處第1次寫11010110,第2次寫111010,讀出結果是這2個值的相與10010 while(1); //所以如果一個地址處的值不是0xff時寫入新的資料是不對的,要先擦除為0xff }
#ifndef _MY51_H #define _MY51_H #include <reg52.h> //#include <math.h> #include <intrins.h> #include <stdio.h> #include "mytype.h" /*************二進位制輸入巨集****************************/ #ifndef _LongToBin_ #define _LongToBin_ #define LongToBin(n) \ ( \ ((n >> 21) & 0x80) | \ ((n >> 18) & 0x40) | \ ((n >> 15) & 0x20) | \ ((n >> 12) & 0x10) | \ ((n >> 9) & 0x08) | \ ((n >> 6) & 0x04) | \ ((n >> 3) & 0x02) | \ ((n ) & 0x01) \ ) #define bin(n) LongToBin(0x##n##l) #define BIN(n) bin(n) #define B(n) bin(n) #define b(n) bin(n) #endif /*************單個資料位的置位巨集*********************/ #ifndef _BIT_ #define _BIT_ #define BIT(n) (1<<n) #define bit(n) BIT(n) #endif #define high 1 //高電平 #define low 0 //低電平 #define led P1 //燈匯流排控制 sbit led0=P1^0; //8個led燈,陰極送低電平點亮 sbit led1=P1^1; sbit led2=P1^2; sbit led3=P1^3; sbit led4=P1^4; sbit led5=P1^5; sbit led6=P1^6; sbit led7=P1^7; sbit ledLock=P2^5; //led鎖存的狀態,0鎖定 ,1不鎖定 sbit beep=P2^3; //蜂鳴器 void delayms(u16 ms); //void delayXus(u8 us); //函式執行(8+6x)個機器週期, 即t=(8+6x)*1.085 ///////////////////////////////////////////////////////////////////////////// #endif