当前位置:首页 > 单片机 > 单片机
[导读]单片机片上EEPROM读写例程, 本程序在STC89C52RC上测试,运行成功。

单片机片上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)(add&0xff); //设置目标单元地址的低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)(add&0xff); //设置目标单元地址的低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)(add&0xff); //设置目标单元地址的低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--);

}//-------------------------------------------------------------------------------

本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除( 邮箱:macysun@21ic.com )。
换一批
延伸阅读

在单片机通信系统中,数据帧是实现设备间可靠对话的核心载体。不同于网络通信中成熟的TCP/IP协议,单片机通信往往需要自定义数据帧格式,而帧头、帧尾与校验机制则是保障数据传输准确性的三大关键。

关键字: 单片机 数据帧

随着嵌入式技术的不断发展,时序分析工具和方法也在不断进步,未来将朝着智能化、自动化的方向发展,为开发者提供更高效的调试手段。但无论技术如何发展,扎实的时序分析基础都是嵌入式开发者不可或缺的能力,只有深入理解通信时序的本质...

关键字: 时序 单片机

在嵌入式系统设计中,不同架构、不同厂商的单片机协同工作早已成为常态。从8位的51系列到32位的STM32,从精简指令集的PIC到复杂指令集的AVR,这些性能各异的单片机如何突破硬件差异实现数据交互,是嵌入式开发中的核心课...

关键字: 嵌入式 单片机

在嵌入式系统开发中,单片机的时钟系统是整个系统的"心脏",所有的指令执行、外设操作、定时器中断都依赖于精准的时钟信号。但在实际开发过程中,很多开发者都会遇到单片机时钟不准的问题,表现为定时器计时偏差、UART通信波特率错...

关键字: 控制系统 单片机

在单片机开发与调试过程中,复位电路作为保障芯片正常启动的核心模块,其稳定性直接影响程序烧录与系统运行。实际应用中,不少开发者会遇到“接稳压电源可正常烧录,接入电池后却无法烧录程序”的故障,此类问题多与复位电路设计、电池供...

关键字: 单片机 复位电路 时序匹配

在单片机的世界里,“字节”(Byte)是一个贯穿始终的核心概念。从存储数据到执行指令,从变量定义到外设通信,字节无处不在。很多初学者在学习单片机时,往往更关注复杂的程序逻辑和外设驱动,却忽略了字节这个基础知识点,导致在后...

关键字: 单片机 字节

在单片机的数字逻辑中,我们通常最关注的是高电平和低电平两种状态,它们构成了二进制数字世界的基础。然而,除了这两种状态之外,还有一种至关重要但常常被忽视的状态——高阻态(High Impedance State)。高阻态就...

关键字: 单片机 高阻态

对于PIC入门者来说,不需要盲目追求高端开发板,一块功能均衡、资料丰富的入门款就能满足需求。比如Microchip官方推出的PIC16F84A开发板,它搭载经典的8位PIC内核,引脚布局清晰,自带LED、按键等基础外设,...

关键字: PIC 单片机

该低功耗器件支持5V运行,在实现高性能的同时,能有效保障系统简洁性与成本效益

关键字: MCU 单片机 工业自动化
关闭