STC15W408AS单片机IAP/EEPROM的相关寄存器

控制/MCU

1878人已加入

描述

STC15系列单片机内部集成了大容量的EEPROM,与其程序空间是分开的。利用ISP/IAP技术可将内部Data Flash当EEPROM,擦写次数在10W次以上。EEPROM可分为若干个扇区,每个扇区包含512字节。使用时,建议同一次修改的数据放在同一个扇区,不是同一次修改的数据放在不同扇区,不一定要用满。数据存储器的擦除操作是按扇区进行的。

EEPROM可用于保存一些需要在应用中修改并且掉电不易丢失的参数数据。在用户程序中,可以对EEPROM进行子节读/字节编程/扇区擦除操作。在工作电压Vcc偏低时,建议不要进行EEPROM/IAP操作。

一、IAP及EEPROM相关寄存器

单片机

1.1 数据寄存器IAP_DATA

IAP_DATA是ISP/IAP操作时的数据寄存器。ISP/IAP从Flash读出的数据放在此处,向Flash写的数据也许放在此处。

1.2 地址寄存器IAP_ADDRH和IAP_ADDRL

IAP_ADDRH,ISP/IAP操作时的地址寄存器高八位。

IAP_ADDRL, ISP/IAP操作时的地址寄存器高低位。

1.3 命令寄存器IAP_CMD

单片机

1.4 命令触发寄存器IAP_TRIG

IAP_TRIG是ISP/IAP操场时的命令触发寄存器。在IAPEN(IAP_CONTER.7) = 1时,对IAP_TRIG先写入5Ah,再写入A5h,ISP/IAP命令才会生效。ISP/IAP操作完成后,IAP地址高八位IAP_ADDRH、IAP地址低八位寄存器IAP_ADDRL和IAP命令寄存器IAP_CMD的内容不变。如果接下来要对下一个地址数据进行ISP/IAP操作,需手动将地址的高8位和低8位分别写入IAP_ADDRH和IAP_ADDRL寄存器。

每次IAP操作时,IAP_TRIG先写入5Ah,再写入A5h,ISP/IAP命令才会生效。每次在触发钱,需要重新发送字节读/字节编程/扇区擦除命令,在命令不改变时,不需要重新送命令。

1.5 命令寄存器IAP_CONTR

单片机

IAPEN:ISP/IAP功能允许位。

0:禁止IAP读/写/擦除Data Flash/EEPROM

1:允许IAP读/写/擦除Data Flash/EEPROM

SWBS:

0:软件选择复位后从用户应用程序区启动

1:还是从系统ISP检测程序区启动

SWRST:

0:不操作

1:软件控制产生复位,单片机自动复位

CMD_FAIL:如果IAP地址(由IAP地址寄存器IAP_ADDRH和IAP_ADDRL的值决定)指向了非法地址或者无效地址,并送了ISP/IAP命令,并对IAP_TRIG送5Ah/A5h触发失败,则CMD_FAIL为1,需要软件清零。

WT0~WT2控制等待时间:

单片机

1.6 工作电压过低判断 此时不需要EEPROM/IAP操作

PCON,电源控制寄存器:

单片机

LVDF:低压检测标志位,当工作电压Vcc低于低压检测门槛电压时,该位置1。该位要软件清0。当低压检测电路发现工作电压Vcc偏低时,不要进行EEPROM/IAP操作。

单片机

二、STC15单片机EEPROM空间大小及地址

单片机

这里,我们暂时只关注STC15W408AS即可。型号不一样记得看手册。

三、测试程序

#include "stc15.h"
#include "intrins.h"


void IapIdle();
uchar IapReadByte(uchar addr);
void IapProgramByte(uchar addr, uchar dat);
void IapEraseSector(uchar addr);


#define IAP_ADDRESS 0x0001


void main()
{
  uchar ret;
  uint i;
  P1M0 = 0x02;
  P1M1 = 0x00;
  P11 = 0;   // 先 关闭LED
  // 这里最好加一点延时 方便查看现象
  IapEraseSector(IAP_ADDRESS); // 擦除地址的数据 恢复初始值
    IapProgramByte(IAP_ADDRESS,1); // 写入1
    if(IapReadByte(IAP_ADDRESS) == 1) 
    {
        P11 = 1; // 如果读取的数据是1 就点亮LED
    }
  while(1);
}


// 关闭IAP  寄存器设置成默认值
void IapIdle()
{
  IAP_CONTR = 0;       // 禁止IAP读/写/擦除Data Flash/EEPROM
  IAP_CMD   = 0;       // 待机 无ISP操作
  IAP_TRIG  = 0;       // 清除命令寄存器
  IAP_ADDRH = 0x80;    // 将地址位设置到非IAP区域
  IAP_ADDRL = 0;
}
// 从ISP/IAP/EEPROM区域读取一字节
uchar IapReadByte(uchar addr)
{
  uchar dat;
  IAP_CMD = 0x01;                 //设置IAP命令 读
  IAP_ADDRL = addr;               //设置IAP低地址
  IAP_ADDRH = addr > > 8;          //设置IAP高地址
  IAP_TRIG = 0x5a;                //写触发命令(0x5a)
  IAP_TRIG = 0xa5;                //写触发命令(0xa5)
  _nop_();                        //等待ISP/IAP/EEPROM操作完成
  dat = IAP_DATA;                 //读ISP/IAP/EEPROM数据
  IapIdle();                      //关闭IAP功能
  return dat;                     //返回
}
// 写一字节数据到ISP/IAP/EEPROM区域
void IapProgramByte(uchar addr, uchar dat)
{
  IAP_CONTR = 0x80;               //使能IAP
  IAP_CMD = 0x02;                 //设置IAP命令 编程
  IAP_ADDRL = addr;               //设置IAP低地址
  IAP_ADDRH = addr > > 8;          //设置IAP高地址
  IAP_DATA = dat;                 //写ISP/IAP/EEPROM数据
  IAP_TRIG = 0x5a;                //写触发命令(0x5a)
  IAP_TRIG = 0xa5;                //写触发命令(0xa5)
  _nop_();                        //等待ISP/IAP/EEPROM操作完成
  IapIdle();
}


// 擦除某一地址的数据
void IapEraseSector(uchar addr)
{
  IAP_CONTR = 0x80;               //使能IAP
  IAP_CMD = 0x03;                 //设置IAP命令 擦除
  IAP_ADDRL = addr;               //设置IAP低地址
  IAP_ADDRH = addr > > 8;          //设置IAP高地址
  IAP_TRIG = 0x5a;                //写触发命令(0x5a)
  IAP_TRIG = 0xa5;                //写触发命令(0xa5)
  _nop_();                        //等待ISP/IAP/EEPROM操作完成
  IapIdle();
}
打开APP阅读更多精彩内容
声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉

全部0条评论

快来发表一下你的评论吧 !

×
20
完善资料,
赚取积分