YSZDH_20240116 发表于 2025-6-16 10:26:32

eerom和flash如何定位




使用flash存储设置参数,软件下到STC15W4K32S4,运行正常,参数能存,软件能运行;
软件下到IAP15W4K61S4,运行开始正常,参数存后,软件不能正常运行;

参数地址是如何选择的,我还想用IAP15W4K61S4这个仿真,还的再占用6K,
如何分配地址都能兼顾呐?

YSZDH_20240116 发表于 2025-6-16 10:51:30

这句话是否输错了

ercircle 发表于 2025-6-16 10:52:08

这个型号IAP15W4K61S4标的IAP但是ISP软件貌似没分配EEPROM大小,默认全区?
IAP15W4K61S4 IAP操作主动偏移+0x8000试下,应该可以和STC15W4K32S4对齐

天宁宁 发表于 2025-6-18 18:40:11

上次把400H写成4000H了
看说明书文档EEPROM地址

YSZDH_20240116 发表于 7 天前

看文档
STC有专门的EEROM
IAP系列没有专门的EEROM,需要flash作为eerom,直接改地址,效果不行。还在看手册

C_wolf 发表于 7 天前

做个参考吧

#include        "STC15Fxxxx.H"

#define CMD_IDLE    0               //空闲模式
#define CMD_READ    1               //IAP字节读命令
#define CMD_PROGRAM 2               //IAP字节编程命令
#define CMD_ERASE   3               //IAP扇区擦除命令

//#define ENABLE_IAP 0x80         //if SYSCLK<30MHz
//#define ENABLE_IAP 0x81         //if SYSCLK<24MHz
#define ENABLE_IAP0x82            //if SYSCLK<20MHz
//#define ENABLE_IAP 0x83         //if SYSCLK<12MHz
//#define ENABLE_IAP 0x84         //if SYSCLK<6MHz
//#define ENABLE_IAP 0x85         //if SYSCLK<3MHz
//#define ENABLE_IAP 0x86         //if SYSCLK<2MHz
//#define ENABLE_IAP 0x87         //if SYSCLK<1MHz

#define IAP_ADDRESS 0x8000

void IapIdle()
{
    IAP_CONTR = 0;                  //关闭IAP功能
    IAP_CMD = 0;                  //清除命令寄存器
    IAP_TRIG = 0;                   //清除触发寄存器
    IAP_ADDRH = 0x80;               //将地址设置到非IAP区域
    IAP_ADDRL = 0;
}

u8 IapReadByte(u16 addr)
{
    u8 dat;                     //数据缓冲区

    IAP_CONTR = ENABLE_IAP;         //使能IAP
    IAP_CMD = CMD_READ;             //设置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;                     //返回
}

void IapProgramByte(u16 addr, u8 dat)
{
    IAP_CONTR = ENABLE_IAP;         //使能IAP
    IAP_CMD = CMD_PROGRAM;          //设置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(u16 addr)
{
    IAP_CONTR = ENABLE_IAP;         //使能IAP
    IAP_CMD = CMD_ERASE;            //设置IAP命令
    IAP_ADDRL = addr;               //设置IAP低地址
    IAP_ADDRH = addr >> 8;          //设置IAP高地址
    IAP_TRIG = 0x5a;                //写触发命令(0x5a)
    IAP_TRIG = 0xa5;                //写触发命令(0xa5)
    _nop_();                        //等待ISP/IAP/EEPROM操作完成
    IapIdle();
}

void main()
{
    WORD i;

    P1 = 0xfe;                      //1111,1110 系统OK
    Delay(10);                      //延时
    IapEraseSector(IAP_ADDRESS);    //扇区擦除
    for (i=0; i<512; i++)         //检测是否擦除成功(全FF检测)
    {
      if (IapReadByte(IAP_ADDRESS+i) != 0xff)
            goto Error;             //如果出错,则退出
    }
    P1 = 0xfc;                      //1111,1100 擦除成功
    Delay(10);                      //延时
    for (i=0; i<512; i++)         //编程512字节
    {
      IapProgramByte(IAP_ADDRESS+i, (BYTE)i);
    }
    P1 = 0xf8;                      //1111,1000 编程完成
    Delay(10);                      //延时
    for (i=0; i<512; i++)         //校验512字节
    {
      if (IapReadByte(IAP_ADDRESS+i) != (BYTE)i)
            goto Error;             //如果校验错误,则退出
    }
    P1 = 0xf0;                      //1111,0000 测试完成
    while (1);
Error:
    P1 &= 0x7f;                     //0xxx,xxxx IAP操作失败
    while (1);
}
页: [1]
查看完整版本: eerom和flash如何定位