51单片机EEPROM的读写.docx
《51单片机EEPROM的读写.docx》由会员分享,可在线阅读,更多相关《51单片机EEPROM的读写.docx(7页珍藏版)》请在冰豆网上搜索。
51单片机EEPROM的读写
51单片机EEPROM的读写
STC单片机的内部EEPROM是用DATAFLASH模拟出来的,不是真正的EEPROM存储器,不能用普通的方法来操作
下面是一些注意点:
1.字节写之前要先将这个字节所在扇区的其它有效数据读取到RAM暂存(这步不是必须的)
2.暂存完之后再对整个扇区(512字节)进行擦除操作,擦拭完后,整个扇区每个地址中数据都变成0xFF
3.将欲写入的N个字节数据,用字节写函数写入EEPROM
4.将暂存到RAM的其它有用的EEPROM值再用字节写函数写回EEPROM
5.STC用FLASH模拟出来的EEPROM的字节写功能只能将1变成0,而不能将0变成1,
只有扇区擦除后数据才是全1,
例如:
在地址0x21f0处第1次写11010110,第2次写111010,读出结果是这2个值的相与10010
所以如果一个地址处的值不是0xff时写入新的数据是不对的,要先执行扇区擦除,变为0xff,
对于单个字节的写入,我们可以先检查该地址处的数据是否为0xff,是的话就不用擦除扇区了
----------------------------------------------------------------------
STC89C52单片机内部EEPROM的读写过程
1 配置ISP_CONTR寄存器,使能第7位ISPEN,让ISP_IAP功能生效,并配置低3位的等待时间
2 写指令:
读/写/擦除扇区这3个命令
3 赋值:
ISP_ADDRH和ISP_ADDRL的地址值
4 关闭总中断EA,因为下面要写的2个触发指令必须是连续操作的,不能被中断
5 执行公用的 ISP_IAP触发指令,触发后读写操作才能进行
6 打开中断EA, 关闭ISP_IAP功能:
清相关寄存器
#include"my51.h"
/******************定义命令字节******************/
#defineread_cmd0x01//字节读数据命令
#definewirte_cmd0x02//字节编程数据命令
#defineerase_cmd0x03//扇区擦除数据命令
/****************特殊功能寄存器声明****************/
sfrISP_DATA=0xe2;
sfrISP_ADDRH=0xe3;
sfrISP_ADDRL=0xe4;
sfrISP_CMD=0xe5;
sfrISP_TRIG=0xe6;
sfrISP_CONTR=0xe7;
/*定义Flash操作等待时间及允许IAP/ISP/EEPROM操作的常数******************/
//#defineenable_waitTime0x80//系统工作时钟<30MHz时,对IAP_CONTR寄存器设置此值
//#defineenable_waitTime0x81//系统工作时钟<24MHz时,对IAP_CONTR寄存器设置此值
//#defineenable_waitTime0x82//系统工作时钟<20MHz时,对IAP_CONTR寄存器设置此值
#defineenable_waitTime0x83//系统工作时钟<12MHz时,对IAP_CONTR寄存器设置此值
//#defineenable_waitTime0x84//系统工作时钟<6MHz时,对IAP_CONTR寄存器设置此值
voidISP_IAP_disable(void)//关闭ISP_IAP
{
EA=1;//恢复中断
ISP_CONTR=0x00;
ISP_CMD=0x00;
ISP_TRIG=0x00;
}
voidISP_IAP_trigger()//触发
{
EA=0;//下面的2条指令必须连续执行,故关中断
ISP_TRIG=0x46;//送触发命令字0x46
ISP_TRIG=0xB9;//送触发命令字0xB9
}
voidISP_IAP_readData(u16beginAddr,u8*pBuf,u16dataSize)//读取数据
{
ISP_DATA=0;//清零,不清也可以
ISP_CMD=read_cmd;//指令:
读取
ISP_CONTR=enable_waitTime;//开启ISP_IAP,并送等待时间
while(dataSize--)//循环读取
{
ISP_ADDRH=(u8)(beginAddr>>8);//送地址高字节
ISP_ADDRL=(u8)(beginAddr&0x00ff);//送地址低字节
ISP_IAP_trigger();//触发
beginAddr++;//地址++
*pBuf++=ISP_DATA;//将数据保存到接收缓冲区
}
ISP_IAP_disable();//关闭ISP_IAP功能
}
voidISP_IAP_writeData(u16beginAddr,u8*pDat,u16dataSize)//写数据
{
ISP_CONTR=enable_waitTime;//开启ISP_IAP,并送等待时间
ISP_CMD=wirte_cmd;//送字节编程命令字
while(dataSize--)
{
ISP_ADDRH=(u8)(beginAddr>>8);//送地址高字节
ISP_ADDRL=(u8)(beginAddr&0x00ff);//送地址低字节
ISP_DATA=*pDat++;//送数据
beginAddr++;
ISP_IAP_trigger();//触发
}
ISP_IAP_disable();//关闭
}
voidISP_IAP_sectorErase(u16sectorAddr)//扇区擦除
{
ISP_CONTR=enable_waitTime;//开启ISP_IAP;并送等待时间
ISP_CMD=erase_cmd;//送扇区擦除命令字
ISP_ADDRH=(u8)(sectorAddr>>8);//送地址高字节
ISP_ADDRL=(u8)(sectorAddr&0X00FF);//送地址低字节
ISP_IAP_trigger();//触发
ISP_IAP_disable();//关闭ISP_IAP功能
}
voidmain()//测试
{
u8buf[3]={0};//接收数据缓冲区
u8dat[5]={b(111010),b(1001),b
(1),b(1011),b(1110)};//我写成二进制是为观察led灯
ISP_IAP_sectorErase(0x2000);//扇区擦除,一块512字节
ISP_IAP_writeData(0x21f0,dat,sizeof(dat));//写EEPROM
ISP_IAP_readData(0x21f0,buf,sizeof(buf));//读取
P1=buf[2];//在地址0x21f0处第1次写11010110,第2次写111010,读出结果是这2个值的相与10010
while
(1);//所以如果一个地址处的值不是0xff时写入新的数据是不对的,要先擦除为0xff
}
#ifndef_MY51_H
#define_MY51_H
#include
//#include
#include
#include
#include"mytype.h"
/*************二进制输入宏****************************/
#ifndef_LongToBin_
#define_LongToBin_
#defineLongToBin(n)\
(\
((n>>21)&0x80)|\
((n>>18)&0x40)|\
((n>>15)&0x20)|\
((n>>12)&0x10)|\
((n>>9)&0x08)|\
((n>>6)&0x04)|\
((n>>3)&0x02)|\
((n)&0x01)\
)
#definebin(n)LongToBin(0x##n##l)
#defineBIN(n)bin(n)
#defineB(n)bin(n)
#defineb(n)bin(n)
#endif
/*************单个数据位的置位宏*********************/
#ifndef_BIT_
#define_BIT_
#defineBIT(n)(1<#definebit(n)BIT(n)
#endif
#definehigh1//高电平
#definelow0//低电平
#defineledP1//灯总线控制
sbitled0=P1^0;//8个led灯,阴极送低电平点亮
sbitled1=P1^1;
sbitled2=P1^2;
sbitled3=P1^3;
sbitled4=P1^4;
sbitled5=P1^5;
sbitled6=P1^6;
sbitled7=P1^7;
sbitledLock=P2^5;//led锁存的状态,0锁定,1不锁定
sbitbeep=P2^3;//蜂鸣器
voiddelayms(u16ms);
//voiddelayXus(u8us);//函数执行(8+6x)个机器周期,即t=(8+6x)*1.085
/////////////////////////////////////////////////////////////////////////////
#endif