首页 | 新闻 | 新品 | 文库 | 方案 | 视频 | 下载 | 商城 | 开发板 | 数据中心 | 座谈新版 | 培训 | 工具 | 博客 | 论坛 | 百科 | GEC | 活动 | 主题月 | 电子展
返回列表 回复 发帖

51单片机学习笔记:连续读写STC89C52RC内部EEPROM存储器

51单片机学习笔记:连续读写STC89C52RC内部EEPROM存储器

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功能:清相关寄存器
C代码  
  • #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++; //送数据
返回列表