51微控制器學習筆記:連續讀寫STC89C52RC內部EEPROM儲存器

xouou_53320發表於2013-02-26

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

 

相關文章