标题: STC8H8K64U记忆功能问题 [打印本页]

作者: jianuli    时间: 2025-6-21 13:21
标题: STC8H8K64U记忆功能问题
这几天在学单片机的EEPROM,有几个问题想下大家,1、EEPROM扇区擦除的时候其他的程序能跑吗?是不是会停几ms,为何动态扫描的数码管会抖一下。2、EEPROM按范例程序写入127以下,开机还原关电前状况时数码管显示一致,写入128以上时数码管显示不一样,这是为何?如果我想写入6666,该怎样改范例的程序。搜索了好久始终没找到合适的程序,大家帮帮忙
作者: zhuls    时间: 2025-6-21 14:59
现在请各位来到你猜我猜环节。。。
作者: vkbvfhp    时间: 2025-6-21 16:14
在官方的库里,操作擦写EEPROM是会关中断的.
作者: WL0123    时间: 2025-6-21 16:27
1、EEPROM扇区擦除或读写的时候不能与其他的程序同时跑。要解决EEPROM擦除或读写的时候不影响数码管动态扫描需要学会统筹调度,这要结合具体应用处理,不是三言两语说的清的。
2、读取128以上数据显示出错多为数据类型使用不当。想写入6666需要拆分为两个字节,读取后再合并还原。
测试程序:
  1. //测试条件:STC8H实验板,MCU型号STC8H3K48S2
  2. //注意:测试本示例时,需在ISP下载时将【允许低压复位(禁止低压中断)】关闭
  3. #include <STC8H.H>
  4. #include <intrins.h>                                //库头文件
  5. #define uint unsigned int                         //宏定义数据类型uint
  6. #define uchar unsigned char                 //宏定义数据类型uchar

  7. #define IAP_ADDRESS 0x0000                        //测试地址

  8. //顺序共阴极数码管段码表,段码a-h顺序接PX0-PX7
  9. uchar code table[]={//共阴数码管段码"0~f-."
  10.                 0x3f,0x06,0x5b,0x4f,
  11.                 0x66,0x6d,0x7d,0x07,
  12.                 0x7f,0x6f,0x77,0x7c,
  13.                 0x39,0x5e,0x79,0x71,0x40,0x80};
  14. uchar data dis_buf[8];              //缓存数组
  15. uint  num,sec;
  16. uchar i;
  17. uint  sign;
  18. void  Timer0Init();                 //定时器初始化声明
  19. void  IapIdle();                    //关闭IAP/EEPROM
  20. uchar IapRead(uint addr);           //读取EEPROM数据
  21. void  IapProgram(uint addr, uchar dat);//写入EEPROM数据
  22. void  IapErase(uint addr);          //擦除EEPROM扇区

  23. //初始化单片机端口
  24. void McuInit()
  25. {
  26.         P0M0 = 0xff; P0M1 = 0x00;
  27.         P1M0 = 0x00; P1M1 = 0x00;
  28.         P2M0 = 0x00; P2M1 = 0x00;
  29.         P3M0 = 0x00; P3M1 = 0x00;
  30.         P4M0 = 0x00; P4M1 = 0x00;
  31.         P5M0 = 0x00; P5M1 = 0x00;
  32.         P6M0 = 0x00; P6M1 = 0x00;
  33.         P7M0 = 0x00; P7M1 = 0x00;
  34. }
  35. //关闭单片机端口
  36. void McuSleep()
  37. {
  38.         P0M0 = 0x00; P0M1 = 0xff;
  39.         P1M0 = 0x00; P1M1 = 0xff;
  40.         P2M0 = 0x00; P2M1 = 0xff;
  41.         P3M0 = 0x00; P3M1 = 0xff;
  42.         P4M0 = 0x00; P4M1 = 0xff;
  43.         P5M0 = 0x00; P5M1 = 0xff;
  44.         P6M0 = 0x00; P6M1 = 0xff;
  45.         P7M0 = 0x00; P7M1 = 0xff;
  46. }

  47. void main()                                                       
  48. {
  49.         McuInit();
  50.         if(IapRead(IAP_ADDRESS)==0xff)//如果没有保存过数据
  51.         {
  52.                 IapProgram(IAP_ADDRESS, 0);//扇区首地址写0
  53.                 sec=0;
  54.                 sign=1;
  55.         }
  56.         else
  57.         {
  58. //                for(i=1;i<12;i++)         //测试写10次
  59.                 for(i=1;i<511;i++)        //测试写满510个字节
  60.                 {
  61.                         if(IapRead(IAP_ADDRESS+i)==0xff)//如果遇到没有保存数据的单元
  62.                         {
  63.                                 sec=IapRead(IAP_ADDRESS+i-1);//读取前一个字节保存的数据
  64.                                 sign=i;           //地址缓存
  65.                                 break;            //跳出循环
  66.                         }
  67.                 }
  68.         }
  69. //        if(sign==11)                  //测试写10次
  70.         if(sign==510)                 //如果写满510
  71.         {
  72.                 IapErase(IAP_ADDRESS);    //擦除扇区
  73.                 IapProgram(IAP_ADDRESS, 0);//首地址写0
  74.                 sign=1;
  75.         }
  76.         PCON &= 0xDF;                 //清0掉电标志
  77.         ELVD = 1;                     //开低压中断
  78.         EA   = 1;                     //开总中断

  79.         Timer0Init();                 //初始化定时器

  80.         while(1)
  81.         {
  82.                 if(TF0)                   //查询T0中断请求标志
  83.                 {               
  84.                         TF0=0;                //T0中断请求标志清0
  85.                         if(++num>=1000)       //1秒
  86.                         {
  87.                                 num=0;                               
  88.                                 sec=++sec%250;
  89.                         }
  90.                         dis_buf[0]=table[sec/100%10];
  91.                         dis_buf[1]=table[sec/10%10];
  92.                         dis_buf[2]=table[sec%10];
  93.                         P0=0x00;              //段消隐
  94.                         P2=~(0x01<<i);        //送段码
  95.                         P0=dis_buf[i];        //送位码
  96.                         i=++i%3;              //循环计数
  97.                 }//耗时569us
  98.         }
  99. }

  100. void PowerLost() interrupt 6      //低压中断
  101. {
  102.         EA = 0;                       //关闭总中断
  103.         McuSleep();                   //关闭所有端口(停止所有耗电电路)
  104.         IapProgram(IAP_ADDRESS+sign,sec);//写数据到EEPROM
  105.         while((PCON & 0x20) != 0)     //复查低压标志
  106.         {
  107.                 PCON &= 0xDF;             //清除低压标志
  108.                 _nop_();               
  109.                 _nop_();                  //坐等掉电
  110.         }
  111.         IAP_CONTR = 0x20;             //发现是误报,重启单片机,恢复正常工作
  112. }

  113. void Timer0Init(void)             //1毫秒@11.0592MHz
  114. {
  115.         AUXR |= 0x80;                 //定时器时钟1T模式
  116.         TMOD &= 0xF0;                 //设置定时器模式
  117.         TL0 = 0xCD;                   //设置定时初始值
  118.         TH0 = 0xD4;                   //设置定时初始值
  119.         TF0 = 0;                      //清除TF0标志
  120.         TR0 = 1;                      //定时器0开始计时
  121. }
  122. //关闭IAP/EEPROM
  123. void IapIdle()
  124. {
  125.     IAP_CONTR = 0;                //关闭IAP功能
  126.     IAP_CMD = 0;                  //清除命令寄存器
  127.     IAP_TRIG = 0;                 //清除触发寄存器
  128.     IAP_ADDRH = 0x80;             //将地址设置到非IAP区域
  129.     IAP_ADDRL = 0;
  130. }
  131. //读取EEPROM数据
  132. uchar IapRead(uint addr)
  133. {
  134.     uchar dat;

  135.     IAP_CONTR = 0x80;             //使能IAP
  136.     IAP_TPS = 12;                 //设置等待参数12MHz
  137.     IAP_CMD = 1;                  //设置IAP读命令
  138.     IAP_ADDRL = addr;             //设置IAP低地址
  139.     IAP_ADDRH = addr >> 8;        //设置IAP高地址
  140.     IAP_TRIG = 0x5a;              //写触发命令(0x5a)
  141.     IAP_TRIG = 0xa5;              //写触发命令(0xa5)
  142.     _nop_();
  143.     dat = IAP_DATA;               //读IAP数据
  144.     IapIdle();                    //关闭IAP功能

  145.     return dat;
  146. }
  147. //写入EEPROM数据
  148. void IapProgram(uint addr, uchar dat)
  149. {
  150.     IAP_CONTR = 0x80;             //使能IAP
  151.     IAP_TPS = 12;                 //设置等待参数12MHz
  152.     IAP_CMD = 2;                  //设置IAP写命令
  153.     IAP_ADDRL = addr;             //设置IAP低地址
  154.     IAP_ADDRH = addr >> 8;        //设置IAP高地址
  155.     IAP_DATA = dat;               //写IAP数据
  156.     IAP_TRIG = 0x5a;              //写触发命令(0x5a)
  157.     IAP_TRIG = 0xa5;              //写触发命令(0xa5)
  158.     _nop_();
  159.     IapIdle();                    //关闭IAP功能
  160. }

  161. void IapErase(uint addr)
  162. {
  163.     IAP_CONTR = 0x80;             //使能IAP
  164.     IAP_TPS = 12;                 //设置等待参数12MHz
  165.     IAP_CMD = 3;                  //设置IAP擦除命令
  166.     IAP_ADDRL = addr;             //设置IAP低地址
  167.     IAP_ADDRH = addr >> 8;        //设置IAP高地址
  168.     IAP_TRIG = 0x5a;              //写触发命令(0x5a)
  169.     IAP_TRIG = 0xa5;              //写触发命令(0xa5)
  170.     _nop_();                      //
  171.     IapIdle();                    //关闭IAP功能
  172. }

复制代码





作者: 太阳雨ZW    时间: 2025-6-21 16:54
x=6666;
y=x/256;
z=x%256;
IapProgramByte(0x0000,y);
IapProgramByte(0x03ff,z);
------------------------
y=IapReadByte(0x0000);
z=IapReadByte(0x03ff);
x=y*256+z;
6666=x;
作者: qiuqiu12138    时间: 2025-6-21 19:18
1,擦除扇区的时候需要4~6ms。2,eeprom按字节写入,范围-127-127,如果要写入大于127的数值需要分成高8位和低8位写入
作者: xiaobendan001    时间: 2025-6-21 19:24
6666是int,两个字节
作者: a399288395    时间: 2025-6-22 07:58
1;先擦再写,
2;数据是按一个字节写入的,如果大于一个字节需要拆分
3;写入地址从0000写入, 如果超过地址范围 在下载软件的时候需要在ISP软件里设置分配空间大小(这个很重要,很多人第一次都不会,官方默认空间是0.5K;写入多个数据读出就会出错,因为空间溢出)
4;写EEPROM 确实会关一下中断,所以这个写入需要自己规划好,如果数据量大可以先存到缓存 然后分多次写入;
作者: cocolala    时间: 2025-6-23 10:24
    我以艾克姆科技的STC8H8K64U来把需要注意的几点和你沟通下吧:
    1、实验连接图如下:




    2、因为STC8H8K64U这个型号是没有片内EEPROM的(都是FLASH),所以要先分割出空间作为EEPROM用。




    3、下载附件的例子,可以打开串口调试助手观察到下面实验现象。




    4、如果是项目用,还是建议选择出厂自带片内EEPROM的芯片,比如STC8H8K48U,会给自己带来便利。




EEPROM.rar

81 KB, 下载次数: 0






欢迎光临 (http://www.51hei.com/bbs/) Powered by Discuz! X3.1