单片机片上EEPROM读写例程, 本程序在STC89C52RC上测验,运转成功。
//============================================================================
//prjname :STC89C51RC,STC89C52RC,STC89C54RD+ EEPROM例程
//funtion :程序完结P0口,P1口别离驱动LED灯(低电平亮)把两个8位变量count.dat,temp.dat
// 显示出来,这两个个八位约1s自增一次,且变量的值可以掉电维护
//author :viviFire
//阐明 :STC89C51RC,STC89C52RC的EEPROM的榜首扇区开始地址为0x2000
// STC89C54RD+的EEPROM的榜首扇区开始地址为0x8000,需修正本程序
// 其他芯片请参阅手册
// 运用本程序请注明出自 viviFire, http://hi.baidu.com/2vivifire
// 本程序参阅 宏晶公司供给的STC5Axx 系列 EEPROM 比如程序
//============================================================================
#include
#include
typedef unsigned char INT8U;
typedef unsigned int INT16U;
sfr IAP_DATA = 0xE2;
sfr IAP_ADDRH = 0xE3;
sfr IAP_ADDRL = 0xE4;
sfr IAP_CMD = 0xE5;
sfr IAP_TRIG = 0xE6;
sfr IAP_CONTR = 0xE7;
//界说Flash 操作等待时间及答应IAP/ISP/EEPROM 操作的常数
//#define ENABLE_ISP 0x80 //体系作业时钟5MHz 时,对IAP_CONTR 寄存器设置此值
//#define ENABLE_ISP 0x81 //体系作业时钟10MHz 时,对IAP_CONTR 寄存器设置此值
#define ENABLE_ISP 0x82 //体系作业时钟20MHz 时,对IAP_CONTR 寄存器设置此值
//#define ENABLE_ISP 0x83 //体系作业时钟40MHz 时,对IAP_CONTR 寄存器设置此值
#define DATA_FLASH_START_ADDRESS 0x2000 //STC89C51,STC89C52 系列 EEPROM 测验开始地址
//——————————————————————————————
INT8U Byte_Read(INT16U add); //读一字节,调用前需翻开IAP 功用
void Byte_Program(INT16U add, INT8U ch); //字节编程,调用前需翻开IAP 功用
void Sector_Erase(INT16U add); //擦除扇区
void IAP_Disable(); //封闭IAP 功用
void delayms(INT16U z);
void EEPROM_Init();
//——————————————————————————
struct EEP_dat
{
INT16U add;
INT8U dat;
}count,temp;
//——————————————————————————
void main()
{
EEPROM_Init();
while(1)
{
P0=~count.dat;
P1=~temp.dat;
delayms(1000);
count.dat++;
temp.dat++;
Sector_Erase(DATA_FLASH_START_ADDRESS);
Byte_Program(count.add,count.dat);
Byte_Program(temp.add,temp.dat);
}
}//—————————————————————————–
void EEPROM_Init()
{
count.add=0x2000; //把EEPROM变量的地址现在这儿界说好
count.dat=Byte_Read(count.add);
temp.add=0x2001; //把EEPROM变量的地址现在这儿界说好
temp.dat=Byte_Read(temp.add);
}//—————————————————————————–
//读一字节,调用前需翻开IAP 功用,进口:DPTR = 字节地址,回来:A = 读出字节
INT8U Byte_Read(INT16U add)
{
IAP_DATA = 0x00;
IAP_CONTR = ENABLE_ISP; //翻开IAP 功用, 设置Flash 操作等待时间
IAP_CMD = 0x01; //IAP/ISP/EEPROM 字节读指令
IAP_ADDRH = (INT8U)(add>>8); //设置方针单元地址的高8 位地址
IAP_ADDRL = (INT8U)(add0xff); //设置方针单元地址的低8 位地址
EA = 0;
IAP_TRIG = 0x46; //先送 46h,再送B9h 到ISP/IAP 触发寄存器,每次都需如此
IAP_TRIG = 0xB9; //送完 B9h 后,ISP/IAP 指令立即被触发起动
_nop_();
EA = 1;
IAP_Disable(); //封闭IAP 功用, 清相关的特别功用寄存器,使CPU 处于安全状况,
//一次接连的IAP 操作完结之后主张封闭IAP 功用,不需要每次都关
return (IAP_DATA);
}//——————————————————————————
//字节编程,调用前需翻开IAP 功用,进口:DPTR = 字节地址, A= 须编程字节的数据
void Byte_Program(INT16U add,INT8U ch)
{
IAP_CONTR = ENABLE_ISP; //翻开 IAP 功用, 设置Flash 操作等待时间
IAP_CMD = 0x02; //IAP/ISP/EEPROM 字节编程指令
IAP_ADDRH = (INT8U)(add>>8); //设置方针单元地址的高8 位地址
IAP_ADDRL = (INT8U)(add0xff); //设置方针单元地址的低8 位地址
IAP_DATA = ch; //要编程的数据先送进IAP_DATA 寄存器
EA = 0;
IAP_TRIG = 0x46; //先送 46h,再送B9h 到ISP/IAP 触发寄存器,每次都需如此
IAP_TRIG = 0xB9; //送完 B9h 后,ISP/IAP 指令立即被触发起动
_nop_();
EA = 1;
IAP_Disable(); //封闭IAP 功用, 清相关的特别功用寄存器,使CPU 处于安全状况,
//一次接连的IAP 操作完结之后主张封闭IAP 功用,不需要每次都关
}//——————————————————————————
//擦除扇区, 进口:DPTR = 扇区地址
void Sector_Erase(INT16U add)
{
IAP_CONTR = ENABLE_ISP; //翻开IAP 功用, 设置Flash 操作等待时间
IAP_CMD = 0x03; //IAP/ISP/EEPROM 扇区擦除指令
IAP_ADDRH = (INT8U)(add>>8); //设置方针单元地址的高8 位地址
IAP_ADDRL = (INT8U)(add0xff); //设置方针单元地址的低8 位地址
EA = 0;
IAP_TRIG = 0x46; //先送 46h,再送B9h 到ISP/IAP 触发寄存器,每次都需如此
IAP_TRIG = 0xB9; //送完 B9h 后,ISP/IAP 指令立即被触发起动
_nop_();
EA = 1;
IAP_Disable(); //封闭IAP 功用, 清相关的特别功用寄存器,使CPU 处于安全状况,
//一次接连的IAP 操作完结之后主张封闭IAP 功用,不需要每次都关
}//——————————————————————————
void IAP_Disable()
{
//封闭IAP 功用, 清相关的特别功用寄存器,使CPU 处于安全状况,
//一次接连的IAP 操作完结之后主张封闭IAP 功用,不需要每次都关
IAP_CONTR = 0; //封闭IAP 功用
IAP_CMD = 0; //清指令寄存器,使指令寄存器无指令,此句可不必
IAP_TRIG = 0; //清指令触发寄存器,使指令触发寄存器无触发,此句可不必
IAP_ADDRH = 0;
IAP_ADDRL = 0;
}//——————————————————————————
void delayms(INT16U z)
{
INT16U x,y;
for(x=z;x>0;x–)
for(y=125;y>0;y–);
}//——————————————————————————-