找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 2738|回复: 14
收起左侧

STC89C54的EEPROM连续存储多个数据如何实现?

[复制链接]
ID:636538 发表于 2019-11-14 10:37 | 显示全部楼层 |阅读模式
求助大神有人使用EEPROM存储连续存储多个数据例如2688字节,可以做到这个吗,在连续读取出来
回复

使用道具 举报

ID:213173 发表于 2019-11-14 14:20 | 显示全部楼层
STC89C54的EEPROM有45K, 读写是以扇区为单位,1个扇区一次最多保存512个字节,共80多个扇区。2688字节需要分6个扇区保存。起始地址0x4000,0x4200、0x4400、0x4800........。可以任意保存在任何扇区。
因为每次写之前,哪怕只写一个字节也需要把1整个扇区全部删除再写。如果数据量大,只能分解后删1个扇区写一个扇区,再删下1个扇区写下1个扇区,如此重复,直到写完。读的时候可以按扇区中任意地址读出一个字节或一串字节。最多一次读512个字节。
回复

使用道具 举报

ID:636538 发表于 2019-11-14 17:46 | 显示全部楼层
我是一次性删除7个扇区想依次向里面存入数据,在扇区的0x5200开始依次存入,读取的时候2个字节开始读出,但是就读出000,就没有东西啦
回复

使用道具 举报

ID:213173 发表于 2019-11-14 19:47 | 显示全部楼层
SKYLAR123 发表于 2019-11-14 17:46
我是一次性删除7个扇区想依次向里面存入数据,在扇区的0x5200开始依次存入,读取的时候2个字节开始读出,但 ...

不是这样玩的!写操作只能分扇区一个一个的操作,读无所谓。EEPROM中如果没有存储信息,读出来是0xff,因为这是物理特性决定的。
回复

使用道具 举报

ID:245004 发表于 2019-11-14 21:31 | 显示全部楼层
官方数据手册写的很详细,先定义起始地址,然后每个地址递增0x200(512字节),如果用7个扇区,擦除需要分7次分别进行。
void IapEraseSector(uint addr) 函数传递的地址为你使用的7个扇区地址。
官方的例程都挺好用的,稍作修改就可以,关键你要查清地址和选好时钟定义。
你可以先用一个扇区测试,读、写、擦除函数确定没问题了再增加多个扇区测试,问题都是分解解决的。


回复

使用道具 举报

ID:636538 发表于 2019-11-15 09:47 | 显示全部楼层
那向一个扇区中依次写入数据共512个,只是在开始擦除一次可以吗
回复

使用道具 举报

ID:636538 发表于 2019-11-15 11:20 | 显示全部楼层
谢谢大神的回答,我还有疑问我在同一扇区写入一个字节擦除一次,怎么能做到连续存储呢,写一次就要擦出一次,在起始地址上加两个字节再写入又要擦除,前面存入的数据,不就被擦出了吗?
回复

使用道具 举报

ID:636538 发表于 2019-11-15 11:48 | 显示全部楼层
wulin 发表于 2019-11-14 14:20
STC89C54的EEPROM有45K, 读写是以扇区为单位,1个扇区一次最多保存512个字节,共80多个扇区。2688字节需要 ...

那向一个扇区中依次写入数据共512个,只是在开始擦除一次可以吗
回复

使用道具 举报

ID:213173 发表于 2019-11-15 16:52 | 显示全部楼层
SKYLAR123 发表于 2019-11-15 11:48
那向一个扇区中依次写入数据共512个,只是在开始擦除一次可以吗

擦除只要代入扇区首地址,整个扇区就擦除。如果要改写其中某一个字节需要先读出全部有效数据保存,改写其中需要变更的数据后再全部写入。
回复

使用道具 举报

ID:213173 发表于 2019-11-16 17:01 | 显示全部楼层
SKYLAR123 发表于 2019-11-15 11:48
那向一个扇区中依次写入数据共512个,只是在开始擦除一次可以吗

给你写了一个EEPROM连续写3072个字节(6个扇区)的示例程序,经实际电路测试无误,使用芯片STC89C52RC,改用STC89C54只要修改EEPROM扇区首地址就可以。有详细注释便于理解。
  1. #include <reg52.h>      //52系列单片机头文件
  2. #include <intrins.h>
  3. #include <stdlib.h>        //包含产生随机数用到的rand()函数。
  4. #define uchar unsigned char
  5. #define uint unsigned int
  6. //定义ISP的操作命令
  7. #define RdCommand 0x01                //读命令
  8. #define PrgCommand 0x02                //写命令
  9. #define EraseCommand 0x03        //擦除命令
  10. #define WaitTime 0x01                 //定义CPU的等待时间,写入硬件延时
  11. /*STC89系列EEPROM寄存器声明*/
  12. sfr ISP_DATA=0xe2;                //0000,0000 EEPROM数据寄存器
  13. sfr ISP_ADDRH=0xe3;                //0000,0000 EEPROM地址高字节
  14. sfr ISP_ADDRL=0xe4;                //0000,0000 EEPROM地址第字节
  15. sfr ISP_CMD=0xe5;                //xxxx,xx00 EEPROM命令寄存器
  16. sfr ISP_TRIG=0xe6;                //0000,0000 EEPRPM命令触发寄存器
  17. sfr ISP_CONTR=0xe7;                //0000,x000 EEPROM控制寄存器

  18. sbit LED1=P1^0;
  19. sbit LED2=P1^7;

  20. /* 测试常量数组 */
  21. uchar code table[512] =
  22. {
  23.     0x00,0x01,0x02,0x03,0x04,0x05,0x06,0x07,0x08,0x09,0x0a,0x0b,0x0c,0x0d,0x0e,0x0f,
  24.     0x10,0x11,0x12,0x13,0x14,0x15,0x16,0x17,0x18,0x19,0x1a,0x1b,0x1c,0x1d,0x1e,0x1f,
  25.     0x20,0x21,0x22,0x23,0x24,0x25,0x26,0x27,0x28,0x29,0x2a,0x2b,0x2c,0x2d,0x2e,0x2f,
  26.     0x30,0x31,0x32,0x33,0x34,0x35,0x36,0x37,0x38,0x39,0x3a,0x3b,0x3c,0x3d,0x3e,0x3f,
  27.     0x40,0x41,0x42,0x43,0x44,0x45,0x46,0x47,0x48,0x49,0x4a,0x4b,0x4c,0x4d,0x4e,0x4f,
  28.     0x50,0x51,0x52,0x53,0x54,0x55,0x56,0x57,0x58,0x59,0x5a,0x5b,0x5c,0x5d,0x5e,0x5f,
  29.     0x60,0x61,0x62,0x63,0x64,0x65,0x66,0x67,0x68,0x69,0x6a,0x6b,0x6c,0x6d,0x6e,0x6f,
  30.     0x70,0x71,0x72,0x73,0x74,0x75,0x76,0x77,0x78,0x79,0x7a,0x7b,0x7c,0x7d,0x7e,0x7f,
  31.     0x80,0x81,0x82,0x83,0x84,0x85,0x86,0x87,0x88,0x89,0x8a,0x8b,0x8c,0x8d,0x8e,0x8f,
  32.     0x90,0x91,0x92,0x93,0x94,0x95,0x96,0x97,0x98,0x99,0x9a,0x9b,0x9c,0x9d,0x9e,0x9f,
  33.     0xa0,0xa1,0xa2,0xa3,0xa4,0xa5,0xa6,0xa7,0xa8,0xa9,0xaa,0xab,0xac,0xad,0xae,0xaf,
  34.     0xb0,0xb1,0xb2,0xb3,0xb4,0xb5,0xb6,0xb7,0xb8,0xb9,0xba,0xbb,0xbc,0xbd,0xbe,0xbf,
  35.     0xc0,0xc1,0xc2,0xc3,0xc4,0xc5,0xc6,0xc7,0xc8,0xc9,0xca,0xcb,0xcc,0xcd,0xce,0xcf,
  36.     0xd0,0xd1,0xd2,0xd3,0xd4,0xd5,0xd6,0xd7,0xd8,0xd9,0xda,0xdb,0xdc,0xdd,0xde,0xdf,
  37.     0xe0,0xe1,0xe2,0xe3,0xe4,0xe5,0xe6,0xe7,0xe8,0xe9,0xea,0xeb,0xec,0xed,0xee,0xef,
  38.     0xf0,0xf1,0xf2,0xf3,0xf4,0xf5,0xf6,0xf7,0xf8,0xf9,0xfa,0xfb,0xfc,0xfd,0xfe,0xff,
  39.     0xff,0xfe,0xfd,0xfc,0xfb,0xfa,0xf9,0xf8,0xf7,0xf6,0xf5,0xf4,0xf3,0xf2,0xf1,0xf0,
  40.     0xef,0xee,0xed,0xec,0xeb,0xea,0xe9,0xe8,0xe7,0xe6,0xe5,0xe4,0xe3,0xe2,0xe1,0xe0,
  41.     0xdf,0xde,0xdd,0xdc,0xdb,0xda,0xd9,0xd8,0xd7,0xd6,0xd5,0xd4,0xd3,0xd2,0xd1,0xd0,
  42.     0xcf,0xce,0xcd,0xcc,0xcb,0xca,0xc9,0xc8,0xc7,0xc6,0xc5,0xc4,0xc3,0xc2,0xc1,0xc0,
  43.     0xbf,0xbe,0xbd,0xbc,0xbb,0xba,0xb9,0xb8,0xb7,0xb6,0xb5,0xb4,0xb3,0xb2,0xb1,0xb0,
  44.     0xaf,0xae,0xad,0xac,0xab,0xaa,0xa9,0xa8,0xa7,0xa6,0xa5,0xa4,0xa3,0xa2,0xa1,0xa0,
  45.     0x9f,0x9e,0x9d,0x9c,0x9b,0x9a,0x99,0x98,0x97,0x96,0x95,0x94,0x93,0x92,0x91,0x90,
  46.     0x8f,0x8e,0x8d,0x8c,0x8b,0x8a,0x89,0x88,0x87,0x86,0x85,0x84,0x83,0x82,0x81,0x80,
  47.     0x7f,0x7e,0x7d,0x7c,0x7b,0x7a,0x79,0x78,0x77,0x76,0x75,0x74,0x73,0x72,0x71,0x70,
  48.     0x6f,0x6e,0x6d,0x6c,0x6b,0x6a,0x69,0x68,0x67,0x66,0x65,0x64,0x63,0x62,0x61,0x60,
  49.     0x5f,0x5e,0x5d,0x5c,0x5b,0x5a,0x59,0x58,0x57,0x56,0x55,0x54,0x53,0x52,0x51,0x50,
  50.     0x4f,0x4e,0x4d,0x4c,0x4b,0x4a,0x49,0x48,0x47,0x46,0x45,0x44,0x43,0x42,0x41,0x40,
  51.     0x3f,0x3e,0x3d,0x3c,0x3b,0x3a,0x39,0x38,0x33,0x36,0x35,0x34,0x33,0x32,0x31,0x30,
  52.     0x2f,0x2e,0x2d,0x2c,0x2b,0x2a,0x29,0x28,0x27,0x26,0x25,0x24,0x23,0x22,0x21,0x20,
  53.     0x1f,0x1e,0x1d,0x1c,0x1b,0x1a,0x19,0x18,0x17,0x16,0x15,0x14,0x13,0x12,0x11,0x10,
  54.     0x0f,0x0e,0x0d,0x0c,0x0b,0x0a,0x09,0x08,0x07,0x06,0x05,0x04,0x03,0x02,0x01,0x00
  55. };

  56. uint code table1[]={0x2000,0x2200,0x2400,0x2600,0x2800,0x2a00};//STC89C52RC EEPROM 扇区地址
  57. //uint code table1[]={0x4000,0x4200,0x4400,0x4600,0x4800,0x4a00};//STC89C54 EEPROM 扇区地址
  58. uint Random;//随机数变量
  59. /* ================ 打开 ISP,IAP 功能 ================= */
  60. void ISP_IAP_enable(void)
  61. {
  62.         EA = 0;       /* 关中断   */
  63.         ISP_CONTR = ISP_CONTR & 0x18;       /* 0001,1000 */
  64.         ISP_CONTR = ISP_CONTR | WaitTime; /* 写入硬件延时 */
  65.         ISP_CONTR = ISP_CONTR | 0x80;       /* ISPEN=1  */
  66. }
  67. /* =============== 关闭 ISP,IAP 功能 ================== */
  68. void ISP_IAP_disable(void)
  69. {
  70.         ISP_CONTR = ISP_CONTR & 0x7f; /* ISPEN = 0 */
  71.         ISP_TRIG = 0x00;
  72.         EA   =   1;   /* 开中断 */
  73. }
  74. /* ================ 公用的触发代码 ==================== */
  75. void ISPgoon(void)
  76. {
  77.         ISP_IAP_enable();   /* 打开 ISP,IAP 功能 */
  78.         ISP_TRIG = 0x46;  /* 触发ISP_IAP命令字节1 */
  79.         ISP_TRIG = 0xb9;  /* 触发ISP_IAP命令字节2 */
  80.         _nop_();
  81. }
  82. /* ==================== 字节读 ======================== */
  83. uchar byte_read(uint byte_addr)
  84. {
  85.         ISP_ADDRH = (uchar)(byte_addr >> 8);/* 地址赋值 */
  86.         ISP_ADDRL = (uchar)(byte_addr & 0x00ff);
  87.         ISP_CMD   = ISP_CMD & 0xf8;   /* 清除低3位  */
  88.         ISP_CMD   = ISP_CMD | RdCommand; /* 写入读命令 */
  89.         ISPgoon();       /* 触发执行  */
  90.         ISP_IAP_disable();    /* 关闭ISP,IAP功能 */
  91.         return (ISP_DATA);    /* 返回读到的数据 */
  92. }
  93. /* ================== 扇区擦除 ======================== */
  94. void SectorErase(uint sector_addr)
  95. {
  96.         uint iSectorAddr;
  97.         iSectorAddr = (sector_addr & 0xfe00); /* 取扇区地址 */
  98.         ISP_ADDRH = (uchar)(iSectorAddr >> 8);
  99.         ISP_ADDRL = 0x00;
  100.         ISP_CMD = ISP_CMD & 0xf8;   /* 清空低3位  */
  101.         ISP_CMD = ISP_CMD | EraseCommand; /* 擦除命令3  */
  102.         ISPgoon();       /* 触发执行  */
  103.         ISP_IAP_disable();    /* 关闭ISP,IAP功能 */
  104. }
  105. /* ==================== 字节写 ======================== */
  106. void byte_write(uint byte_addr, uchar original_data)
  107. {
  108.         ISP_ADDRH = (uchar)(byte_addr >> 8);  /* 取地址  */
  109.         ISP_ADDRL = (uchar)(byte_addr & 0x00ff);
  110.         ISP_CMD  = ISP_CMD & 0xf8;    /* 清低3位 */
  111.         ISP_CMD  = ISP_CMD | PrgCommand;  /* 写命令2 */
  112.         ISP_DATA = original_data;   /* 写入数据准备 */
  113.         ISPgoon();       /* 触发执行  */
  114.         ISP_IAP_disable();     /* 关闭IAP功能 */
  115. }

  116. void main()
  117. {
  118.         uchar i;
  119.         uint j,k;
  120.         LED2=0;//LED2亮
  121.         k=50000;
  122.         while(k--);//延时等待系统稳定
  123.         LED2=1;//LED2灭
  124.         for(i=0;i<6;i++)//循环写6个扇区
  125.         {
  126.                 SectorErase(table1[i]);//擦除扇区
  127.                 for(j=0;j<512;j++)//循环写512个字节
  128.                 {
  129.                         byte_write(table1[i]+j,table[j]);//写入数据
  130.                         if(byte_read(table1[i]+j)!= table[j])//比较对错
  131.                         {
  132.                                 LED1=0;//错误LED1亮
  133.                                 k=50000;
  134.                                 while(k--);//延时便于观察
  135.                                 LED1=1;//LED1灭
  136.                         }       
  137.                 }
  138.         }
  139.         Random=rand()/64;//生成0~511随机数  
  140.         if(byte_read(table1[Random%6]+Random)==table[Random])//随机在6个扇区的随机地址抽样比对
  141.                 LED2=0;//正确LED2亮
  142.         else LED1=0;//错误LED1亮
  143.         while(1);
  144. }
复制代码



回复

使用道具 举报

ID:636538 发表于 2019-11-18 08:33 | 显示全部楼层
wulin 发表于 2019-11-16 17:01
给你写了一个EEPROM连续写3072个字节(6个扇区)的示例程序,经实际电路测试无误,使用芯片STC89C52RC, ...

跪谢大神,大神收下我的膝盖吧!!!!!!
回复

使用道具 举报

ID:636538 发表于 2019-11-18 09:58 | 显示全部楼层
我还有个细节疑问在关闭IAP功能ISP_CONTR =0;与ISP_CONTR =0x7f;实现的作用一样吗,会对其他产生影响吗
回复

使用道具 举报

ID:245004 发表于 2019-11-18 10:24 | 显示全部楼层
SKYLAR123 发表于 2019-11-14 17:46
我是一次性删除7个扇区想依次向里面存入数据,在扇区的0x5200开始依次存入,读取的时候2个字节开始读出,但 ...

最好能说具体一些。
例如我一般存储几个或者几十个数据,从一个扇区的开始依次写入;
当读取的时候从最后往前查找,读取最后一组;
如果再次写入,先查看有没有写满,如果写满了先擦除再写,没写满的话从后面接着写,就不用每次擦除,这样不但擦除少,也数倍延长存储空间的使用寿命。

擦除后为0xff的空间可以直接写数据,不用擦除,非空(x0ff)的空间要改写数据的话必须整个区块擦除再写。
再完善一些就是每组数据带有校验位,再再完善一些就是两个区块配合用,擦除或写数据的过程中掉电没写全也能保留最后一组有效数据...
回复

使用道具 举报

ID:213173 发表于 2019-11-18 10:40 | 显示全部楼层
SKYLAR123 发表于 2019-11-18 09:58
我还有个细节疑问在关闭IAP功能ISP_CONTR =0;与ISP_CONTR =0x7f;实现的作用一样吗,会对其他产生影响吗

对于不支持位操作的寄存器只能用字节操作的方式来控制其中某一位或某几位的状态,并保持其它位的状态不变。
ISP_CONTR = ISP_CONTR & 0x7f;的结果是ISP_CONTR的最高位=0,其它7位保持原来状态不变。
记住:&&是逻辑与,&是按位与。
示例:
a=0xaa;     //10101010
a=a&0x7f;  //00101010 操作后的状态
假设写成a=0x7f; //01111111就把不该变化的位都改变了。
回复

使用道具 举报

ID:644158 发表于 2019-11-18 11:13 | 显示全部楼层
STC89C54的EEPROM有45K, 读写是以扇区为单位,1个扇区一次最多保存512个字节,共80多个扇区。2688字节需要分6个扇区保存。起始地址0x4000,0x4200、0x4400、0x4800........。可以任意保存在任何扇区。 因为每次写之前,哪怕只写一个字节也需要把1整个扇区全部删除再写。如果数据量大,只能分解后删1个扇区写一个扇区,再删下1个扇区写下1个扇区,如此重复,直到写完。读的时候可以按扇区中任意地址读出一个字节或一串字节。最多一次读512个字节。
回复

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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