找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 3213|回复: 0
打印 上一主题 下一主题
收起左侧

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

[复制链接]
跳转到指定楼层
楼主
ID:161768 发表于 2017-1-14 23:24 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
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功能:清相关寄存器
  1. #include "my51.h"

  2. /******************定义命令字节******************/      
  3. #define read_cmd     0x01                   //字节读数据命令     
  4. #define wirte_cmd    0x02                   //字节编程数据命令     
  5. #define erase_cmd    0x03                   //扇区擦除数据命令               
  6.   
  7. /****************特殊功能寄存器声明****************/  
  8. sfr ISP_DATA = 0xe2;   
  9. sfr ISP_ADDRH = 0xe3;     
  10. sfr ISP_ADDRL = 0xe4;   
  11. sfr ISP_CMD = 0xe5;   
  12. sfr ISP_TRIG = 0xe6;      
  13. sfr ISP_CONTR = 0xe7;

  14. /*定义Flash 操作等待时间及允许IAP/ISP/EEPROM 操作的常数******************/
  15. //#define enable_waitTime 0x80 //系统工作时钟<30MHz 时,对IAP_CONTR 寄存器设置此值
  16. //#define enable_waitTime 0x81 //系统工作时钟<24MHz 时,对IAP_CONTR 寄存器设置此值
  17. //#define enable_waitTime 0x82  //系统工作时钟<20MHz 时,对IAP_CONTR 寄存器设置此值
  18. #define enable_waitTime 0x83 //系统工作时钟<12MHz 时,对IAP_CONTR 寄存器设置此值
  19. //#define enable_waitTime 0x84 //系统工作时钟<6MHz 时,对IAP_CONTR 寄存器设置此值
  20.   
  21.    
  22. void ISP_IAP_disable(void)                                     //关闭ISP_IAP
  23. {   
  24.         EA=1;                                                                          //恢复中断
  25.     ISP_CONTR = 0x00;     
  26.     ISP_CMD = 0x00;      
  27.     ISP_TRIG = 0x00;                                                                                      
  28. }

  29. void ISP_IAP_trigger()                       //触发
  30. {   
  31.         EA=0;                                    //下面的2条指令必须连续执行,故关中断                    
  32.     ISP_TRIG = 0x46;                         //送触发命令字0x46     
  33.     ISP_TRIG = 0xB9;                         //送触发命令字0xB9         
  34. }
  35.       

  36. void ISP_IAP_readData(u16 beginAddr, u8* pBuf, u16 dataSize) //读取数据
  37. {
  38.         ISP_DATA=0;                                                           //清零,不清也可以
  39.         ISP_CMD = read_cmd;                                                  //指令:读取
  40.         ISP_CONTR = enable_waitTime;                           //开启ISP_IAP,并送等待时间
  41.         while(dataSize--)                                                  //循环读取
  42.         {
  43.             ISP_ADDRH = (u8)(beginAddr >> 8);     //送地址高字节      
  44.             ISP_ADDRL = (u8)(beginAddr & 0x00ff); //送地址低字节
  45.                 ISP_IAP_trigger();                                        //触发
  46.                 beginAddr++;                                                //地址++
  47.                 *pBuf++ = ISP_DATA;                                        //将数据保存到接收缓冲区
  48.         }
  49.         ISP_IAP_disable();                                                //关闭ISP_IAP功能
  50. }      
  51.    
  52. void ISP_IAP_writeData(u16 beginAddr,u8* pDat,u16 dataSize) //写数据   
  53. {           
  54.     ISP_CONTR = enable_waitTime;                //开启ISP_IAP,并送等待时间      
  55.     ISP_CMD = wirte_cmd;                        //送字节编程命令字
  56.         while(dataSize--)
  57.         {
  58.             ISP_ADDRH = (u8)(beginAddr >> 8);       //送地址高字节      
  59.             ISP_ADDRL = (u8)(beginAddr & 0x00ff);   //送地址低字节           
  60.             ISP_DATA = *pDat++;                     //送数据      
  61.                 beginAddr++;
  62.                 ISP_IAP_trigger();                                                 //触发
  63.         }         
  64.     ISP_IAP_disable();   //关闭                                      
  65. }   
  66.         
  67. void ISP_IAP_sectorErase(u16 sectorAddr)             //扇区擦除
  68. {           
  69.     ISP_CONTR = enable_waitTime;                //开启ISP_IAP;并送等待时间      
  70.     ISP_CMD = erase_cmd;                        //送扇区擦除命令字         
  71.     ISP_ADDRH = (u8)(sectorAddr >> 8);                    //送地址高字节      
  72.     ISP_ADDRL = (u8)(sectorAddr & 0X00FF);          //送地址低字节           
  73.         ISP_IAP_trigger();                                                     //触发
  74.     ISP_IAP_disable();                          //关闭ISP_IAP功能         
  75. }
  76.    
  77. void main()                                                                     //测试
  78. {   
  79.         u8 buf[3]={0};                                                                  //接收数据缓冲区
  80.         u8 dat[5]={b(111010),b(1001),b(1),b(1011),b(1110)};//我写成二进制是为观察led灯   
  81.     ISP_IAP_sectorErase(0x2000);                //扇区擦除,一块512字节      
  82.     ISP_IAP_writeData(0x21f0,dat,sizeof(dat));  //写EEPROM               
  83.         ISP_IAP_readData(0x21f0,buf,sizeof(buf));         //读取
  84.         P1=buf[2];//在地址0x21f0处第1次写11010110,第2次写111010,读出结果是这2个值的相与10010
  85.     while(1); //所以如果一个地址处的值不是0xff时写入新的数据是不对的,要先擦除为0xff
  86. }
复制代码
  1. #ifndef _MY51_H
  2. #define _MY51_H
  3. #include <reg52.h>
  4. //#include <math.h>
  5. #include <intrins.h>
  6. #include <stdio.h>
  7. #include "mytype.h"

  8. /*************二进制输入宏****************************/
  9. #ifndef _LongToBin_
  10. #define _LongToBin_
  11. #define LongToBin(n) \                           
  12. (                                         \
  13. ((n >> 21) & 0x80) | \
  14. ((n >> 18) & 0x40) | \
  15. ((n >> 15) & 0x20) | \
  16. ((n >> 12) & 0x10) | \
  17. ((n >> 9)  & 0x08) | \
  18. ((n >> 6)  & 0x04) | \
  19. ((n >> 3)  & 0x02) | \
  20. ((n ) & 0x01)                  \
  21. )
  22. #define  bin(n)  LongToBin(0x##n##l)
  23. #define  BIN(n)  bin(n)
  24. #define  B(n)    bin(n)
  25. #define  b(n)    bin(n)                          
  26. #endif

  27. /*************单个数据位的置位宏*********************/
  28. #ifndef   _BIT_
  29. #define   _BIT_
  30. #define   BIT(n)   (1<<n)
  31. #define   bit(n)   BIT(n)
  32. #endif

  33. #define high        1   //高电平
  34. #define low                0   //低电平

  35. #define led P1      //灯总线控制

  36. sbit led0=P1^0;     //8个led灯,阴极送低电平点亮
  37. sbit led1=P1^1;
  38. sbit led2=P1^2;
  39. sbit led3=P1^3;
  40. sbit led4=P1^4;
  41. sbit led5=P1^5;
  42. sbit led6=P1^6;
  43. sbit led7=P1^7;        
  44. sbit ledLock=P2^5;        //led锁存的状态,0锁定 ,1不锁定

  45. sbit beep=P2^3;     //蜂鸣器

  46. void delayms(u16 ms);
  47. //void delayXus(u8 us); //函数执行(8+6x)个机器周期, 即t=(8+6x)*1.085
  48. /////////////////////////////////////////////////////////////////////////////


  49. #endif
复制代码
分享到:  QQ好友和群QQ好友和群 QQ空间QQ空间 腾讯微博腾讯微博 腾讯朋友腾讯朋友
收藏收藏1 分享淘帖 顶 踩
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

手机版|小黑屋|51黑电子论坛 |51黑电子论坛6群 QQ 管理员QQ:125739409;技术交流QQ群281945664

Powered by 单片机教程网

快速回复 返回顶部 返回列表