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" /******************定义命令字节******************/ #define read_cmd 0x01 //字节读数据命令 #define wirte_cmd 0x02 //字节编程数据命令 #define erase_cmd 0x03 //扇区擦除数据命令 /****************特殊功能寄存器声明****************/ sfr ISP_DATA = 0xe2; sfr ISP_ADDRH = 0xe3; sfr ISP_ADDRL = 0xe4; sfr ISP_CMD = 0xe5; sfr ISP_TRIG = 0xe6; sfr ISP_CONTR = 0xe7; /*定义Flash 操作等待时间及允许IAP/ISP/EEPROM 操作的常数******************/ //#define enable_waitTime 0x80 //系统工作时钟<30MHz 时,对IAP_CONTR 寄存器设置此值 //#define enable_waitTime 0x81 //系统工作时钟<24MHz 时,对IAP_CONTR 寄存器设置此值 //#define enable_waitTime 0x82 //系统工作时钟<20MHz 时,对IAP_CONTR 寄存器设置此值 #define enable_waitTime 0x83 //系统工作时钟<12MHz 时,对IAP_CONTR 寄存器设置此值 //#define enable_waitTime 0x84 //系统工作时钟<6MHz 时,对IAP_CONTR 寄存器设置此值 void ISP_IAP_disable(void) //关闭ISP_IAP { EA=1; //恢复中断 ISP_CONTR = 0x00; ISP_CMD = 0x00; ISP_TRIG = 0x00; } void ISP_IAP_trigger() //触发 { EA=0; //下面的2条指令必须连续执行,故关中断 ISP_TRIG = 0x46; //送触发命令字0x46 ISP_TRIG = 0xB9; //送触发命令字0xB9 } void ISP_IAP_readData(u16 beginAddr, u8* pBuf, u16 dataSize) //读取数据 { 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功能 } void ISP_IAP_writeData(u16 beginAddr,u8* pDat,u16 dataSize) //写数据 { 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(); //关闭 } void ISP_IAP_sectorErase(u16 sectorAddr) //扇区擦除 { 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功能 } void main() //测试 { u8 buf[3]={0}; //接收数据缓冲区 u8 dat[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 <reg52.h> //#include <math.h> #include <intrins.h> #include <stdio.h> #include "mytype.h" /*************二进制输入宏****************************/ #ifndef _LongToBin_ #define _LongToBin_ #define LongToBin(n) \ ( \ ((n >> 21) & 0x80) | \ ((n >> 18) & 0x40) | \ ((n >> 15) & 0x20) | \ ((n >> 12) & 0x10) | \ ((n >> 9) & 0x08) | \ ((n >> 6) & 0x04) | \ ((n >> 3) & 0x02) | \ ((n ) & 0x01) \ ) #define bin(n) LongToBin(0x##n##l) #define BIN(n) bin(n) #define B(n) bin(n) #define b(n) bin(n) #endif /*************单个数据位的置位宏*********************/ #ifndef _BIT_ #define _BIT_ #define BIT(n) (1<<n) #define bit(n) BIT(n) #endif #define high 1 //高电平 #define low 0 //低电平 #define led P1 //灯总线控制 sbit led0=P1^0; //8个led灯,阴极送低电平点亮 sbit led1=P1^1; sbit led2=P1^2; sbit led3=P1^3; sbit led4=P1^4; sbit led5=P1^5; sbit led6=P1^6; sbit led7=P1^7; sbit ledLock=P2^5; //led锁存的状态,0锁定 ,1不锁定 sbit beep=P2^3; //蜂鸣器 void delayms(u16 ms); //void delayXus(u8 us); //函数执行(8+6x)个机器周期, 即t=(8+6x)*1.085 ///////////////////////////////////////////////////////////////////////////// #endif
相关推荐
stc89c52rc单片机手册.pdfstc89c52rc单片机手册.pdfstc89c52rc单片机手册.pdfstc89c52rc单片机手册.pdfstc89c52rc单片机手册.pdfstc89c52rc单片机手册.pdf
51单片机STC89C52RC开发板例程之数码管显示0-F。1、单片机型号:STC89C52RC。2、开发环境:KEIL。3、编程语言:C语言。4、提供配套PDF格式51单片机STC89C52RC开发板电路原理图。
STC89C52RC单片机开发板PDF原理图,包括常用的接口及外围电路,可以做为你的学习设计参考。
本文给出一个STC89C52单片机内部EEPROM驱动的程序,感兴趣的朋友可以看看。
2、程序对应处理器是:STC89C52RC(51单片机); 3、智能小车电机驱动芯片是:L293D; 4、智能小车电机为:TT直流减速电机; 5、需要用到的超声波测距模块(HC-SR04)。 该程序源代码在本人STC89C52RC单片机智能小车...
51单片机STC89C52RC开发板例程之按键计数器。1、单片机型号:STC89C52RC。2、开发环境:KEIL。3、编程语言:C语言。4、提供配套PDF格式51单片机STC89C52RC开发板电路原理图。5、功能:每按下按键一次,累加一次计数...
STC89C51、52内部都自带有2K字节的EEPROM,54、55和58都自带有16K字节的EEPROM,STC单片机是利用IAP技术实现的EEPROM,内部Flash擦写次数可达100,000 次以上,先来介绍下ISP与IAP的区别和特点。
51单片机(STC89C52RC)代码记录.zip51单片机(STC89C52RC)代码记录.zip 51单片机(STC89C52RC)代码记录.zip51单片机(STC89C52RC)代码记录.zip 51单片机(STC89C52RC)代码记录.zip51单片机(STC89C52RC)代码记录.zip 51...
51单片机STC89C52RC开发板例程之AT24C02(EEPROM)记忆大于255的数程序源代码。 1、单片机型号:STC89C52RC。 2、开发环境:KEIL。 3、编程语言:C语言。 4、提供配套PDF格式51单片机STC89C52RC开发板电路原理图。 5...
2、程序对应处理器是:STC89C52RC(51单片机); 3、智能小车电机驱动芯片是:L293D; 4、智能小车电机为:TT直流减速电机; 5、需要用到测速模块; 6、红外避障(避开障碍物)。 该程序源代码在本人STC89C52RC单片机...
STC89C52RC TQFP44封装 51单片机最小系统板AD设计硬件原理图+PCB+3D封装库文件,2层板设计,大小为59*82mm,包括AD设计的原理图和PCB及封装库文件,主要如下: Library Component Count : 17 Name Description -----...
STC89C52RC单片机开发板PDF原理图
STC89C52RC单片机学习板PDF原理图+配套试验例程KEIL源码程序(105例): 10 LED循环左移 100.12864液晶基础显示 101.1602动态显示 102.红外解码1602液晶显示 103.红外解码数码管显示 105 NRF24L01开发板一 11 LED循环...
用STC89C52单片机内部EEPROM保存数据的应用例子程序 内部资料,官方权威程序
2、程序对应处理器是:STC89C52RC(51单片机); 3、智能小车电机驱动芯片是:L293D; 4、智能小车电机为:TT直流减速电机; 5、需要用到测速模块; 6、需要用到液晶1602。 该程序源代码在本人STC89C52RC单片机智能...
采用C语言,STC89C52为主控,控制RC522模块对IC卡(M1)进行读写卡操作,持续读写卡,快捷高效
单片机设计电路之STC89C52RC实验板原理图及PCB图
STC单片机 STC89C52RC 烧写软件,分享分享,含烧写方法。。。。。
51单片机STC89C52RC开发板例程之AT24C02(EEPROM)保存计时数程序源代码。 1、单片机型号:STC89C52RC。 2、开发环境:KEIL。 3、编程语言:C语言。 4、提供配套PDF格式51单片机STC89C52RC开发板电路原理图。 5、...
基于STC89C52RC单片机的DA/AD转换C程序