📄 eepdemo.c
字号:
/*========================================================================
IAP操作使用片内flash作为eeprom使用的驱动程序。参考来源:www.mcu-memory.com
本演示代码通过对用户数据的读写操作,具体实现数据存储的组织方法,最大限度的
利用eeprom的空间。这只是对eeprom进行分配轮换使用的一种方法。
实现分配的思路:用户数组(需要记忆的部分)应该在4,8,16,32,64等字节以内,将第
一字节留作索引头:如果读到该字节为0x55(也可以定义为其他数),则认为已经找到
了有效的最后纪录,将本小区域数据读出。如果不是则继续跳到下一个小区域继续上
述操作,直到找到上一次的纪录为止,如果超出了eeprom的有效范围还没有找到,则
认为这是一块新的芯片,将默认的数据写入。
保存的时候,首先将刚才读到的那个数组的对应eeprom中的第一字节清零,以便下次
跳过,然后再推后一个小区域(记录长度),将数据写入。如果出现写入失败,接着到
再下一个区域,直到写入正确为止。
这里有一个问题就是,写到下一个小区域的时候会遇到跨扇区,只要判断出跨区,就
将上一个扇区擦除,以备下一轮循环再使用。
程序利用stc提供的下载板上的LED显示当前读或者写的地址,你可以看到led的"递减"
过程,演示读写256次,从第一次的全亮,到演示完毕也全亮,刚好256次。如果中途
断电,即会出现不同的结果,但再进行完演示的话,会发现结束时的led状态与开始
时的一样,否则就说明你的芯片内部eeprom有损坏。
演示代码仅针对stc89c54rd+进行了调试,但可以给其他芯片编程时作为参考。
晓奇工作室 http://www.xiao-qi.com/
---- xiaoqi 2005.12
======================================================================= */
#include "src51rd.h"
#include <intrins.h>
#define uchar unsigned char
#define uint unsigned int
/* STC89C54RD+的flash空间从0x4000~0xf3ff 共90个扇区,每扇区512字节 */
#define BaseAddr 0x4000
#define EndSectoraddr 0xf200
#define EndAddr 0xf3ff
/* ------------- 定义扇区大小 ------------- */
#define PerSector 512
/* 用户程序需要记忆的数组, 用户实际使用了n-1个数据, */
uchar Ttotal[16] =
{
0x55, /* 作为判别引导头使用,用户程序请不要修改它 */
/* 用户保存记忆的数据 */
0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08,
0x09, 0x0a, 0x0b, 0x0c, 0x0d, 0x0e, 0x0f,
};
uint timerForDelay, /* 专供延时用的变量 */
i, /* 循环变量 */
EepromPtr; /* eeprom读写指针 */
/* =========================== 外部函数列表 ============================*/
extern uchar byte_read(uint byte_addr); // byte读
extern void byte_write(uint byte_addr, uchar original_data);
extern void SectorErase(uint sector_addr); // 扇区擦除
extern uchar byte_write_verify(uint byte_addr, uchar original_data); // byte写并校验
extern uchar ArrayWrite(uint begin_addr, uint len, uchar *array); // byte数组写并校验
extern void ArrayRead(uint begin_addr, uchar len); // 读出, 保存在Ttotal[]中
/* ================ 短延时 =============== */
void delay(uint counter)
{
timerForDelay = counter;
while(timerForDelay);
}
//////////////////////////////////////////////////////////////////
// 初始化开启定时器2 //
//////////////////////////////////////////////////////////////////
void initCT2()
{
RCAP2H = 0xd8;
RCAP2L = 0xf0;
TH2 = 0xd8; // 定时器初值
TL2 = 0xf0;
ES = 0; // 关闭通信中断
ET2 = 1; // 允许T2中断
T2CON = 4; // 自动重装的定时器
TR2 = 1; // 启动
EA = 1;
}
//////////////////////////////////////////////////////////////////
// 定时器CT2中断服务程序 //
//////////////////////////////////////////////////////////////////
void timer2Int(void) interrupt 5
{
TF2 = 0; // 溢出标志必须由软件清零
EXF2 = 0; // 捕获标志必须由软件清零
if(timerForDelay)timerForDelay--; // 定时变量处理
}
//////////////////////////////////////////////////////////////////
// 从eeprom中读取数据 //
//////////////////////////////////////////////////////////////////
void DataRestore()
{
EepromPtr = BaseAddr; /* 指向eeprom的起始点 */
while(EepromPtr < EndAddr) /* 在eeprom的可用区域内 */
{
if(byte_read(EepromPtr) == 0x55)/* 找到了上一次纪录的数据块 */
{
break; /* 寻找完成 */
}
EepromPtr += 0x10; /* 指向下一个小区 */
}
if(EepromPtr >= EndAddr) /* 如果照遍都没有,就是新芯片 */
{
EepromPtr = BaseAddr; /* 指向eeprom的起始点 */
while(ArrayWrite(EepromPtr, 0x10, Ttotal)) /* 初始化为默认数据 */
{
byte_write(EepromPtr, 0); /* 该单元已经失效 */
if(EepromPtr < EndAddr)
{
EepromPtr += 0x10; /* 换一块新的小区 */
}
else
{
P1=0; /* 指示芯片内eeprom已经全部实效 */
EA= 0; /* 不再做任何事 */
while(1); /* 死机 */
}
}
}
ArrayRead(EepromPtr, 15);
}
//////////////////////////////////////////////////////////////////
// 将需要记忆的数据保存到eeprom //
//////////////////////////////////////////////////////////////////
void DataSave()
{
uint wrPtr; /* 临时指针 */
NextArea:
byte_write_verify(EepromPtr, 0); /* 将原来的标记清除 */
EepromPtr += 0x10; /* 目标存入地址 */
/* -------------- 判断是否启用新的扇区 --------------- */
if((EepromPtr & 0x1ff)==0)
{
wrPtr = EepromPtr-0x200; /* 上一个扇区的起始地址 */
SectorErase(wrPtr); /* 将上一个扇区擦除,为以后重新启用准备 */
if(EepromPtr>=0xf400) /* 已经用完了最后一个区域 */
{
EepromPtr=0x4000; /* 从头开始 */
}
}
/* ----------- 数据存入前的准备 ----------- */
/* 。。。。。。。。。。。。。。转移、处理 */
Ttotal[0] = 0x55; /* 重申本区域启用标记 */
if(ArrayWrite(EepromPtr, 0x10, Ttotal)) /* 数据写入,如果有错换一块 */
{
goto NextArea;
}
}
void main()
{
/* 指示灯开始指示程序将要开始 */
P1 = 0xff; /* 所有LED熄灭 */
initCT2(); /* CT2初始化,用来做定时器 */
/* --------------- 演示 256 次读写操作 ----------------- */
for(i=0;i<256;i++)
{
DataRestore(); /* 从eeprom中读取记忆数据 */
P1 = (uchar)((EepromPtr >> 4) & 0xff); /* 显示读取地址*/
delay(200); /* 延时保留显示状态2秒 */
DataSave(); /* 数据再次保存 */
P1 = (uchar)((EepromPtr >> 4) & 0xff); /* 显示写入地址*/
delay(300); /* 延时保留显示状态3秒 */
}
while(1); /* 演示完毕 */
}
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -