EEPROM 为什么不好使啊,断电后清0

2020-02-05 09:28发布

请大侠们帮忙看下,我的这个EEPROM为什么不好使啊,断电后清0,哪里出了问题,帮忙找下好么谢谢
/********************************************************************************************/
#include <STC12C5A60S2.h>                                                // 包含头文件 //
#include <intrins.h>       //51基本运算(包括_nop_空函数)
#include <stdio.h>
#include <string.h>
/********************************************************************************************/
typedef unsigned char      uint8;          // 无符号8位整型变量 //
#define uchar unsigned char
#define uint  unsigned  int
typedef unsigned char  INT8U;
typedef unsigned int   INT16U;
/********************************************************************************************
// 引脚定义 // (使用者根据实际更改)
/********************************************************************************************/
#define                LCM2402_DB0_DB7                P0                        // 定义LCM2402的数据总线
sbit LCM2402_RS   = P3 ^ 2;                                        // 定义LCM2402的RS控制线
sbit LCM2402_RW   = P3 ^ 3;                                        // 定义LCM2402的RW控制线
sbit LCM2402_E    = P3 ^ 4;                                        // 定义LCM2402的E控制线
sbit LCM2402_Busy = P0 ^ 7;                                        // 定义LCM2402的测忙线(与LCM2402_DB0_DB7关联)
data unsigned char DIS_BIT = 0; //多种信息的切换显示
//data unsigned char cou  = 0;     // 软计数器,对10ms时基信号累加到1s
uint tt;
uchar a,tl,th1,th2;
uint val;

/********************************************************************************************
// 定义LCM2402指令集 // (详细请见技术手册)
/********************************************************************************************/
#define                        CMD_clear                0x01             // 清除屏幕
#define                        CMD_back                0x02             // DDRAM回零位
#define                        CMD_dec1                0x04             // 读入后AC(指针)减1,向左写
#define                        CMD_add1                0x06             // 读入后AC(指针)加1,向右写
#define                        CMD_dis_gb1                0x0f             // 开显示_开光标_开光标闪烁
#define                        CMD_dis_gb2                0x0e             // 开显示_开光标_关光标闪烁
#define                        CMD_dis_gb3                0x0c             // 开显示_关光标_关光标闪烁
#define                        CMD_OFF_dis                0x08             // 关显示_关光标_关光标闪烁
#define                        CMD_set82                0x38             // 8位总线_2行显示
#define                        CMD_set81                0x30             // 8位总线_1行显示(上边行)
#define                        CMD_set42                0x28             // 4位总线_2行显示
#define                        CMD_set41                0x20             // 4位总线_1行显示(上边行)
#define                        lin_1                        0x80             // 4位总线_1行显示(上边行)
#define                        lin_2                        0xc0             // 4位总线_1行显示(上边行)
/*********************************************************************************************/
//定义Flash 操作等待时间及允许IAP/ISP/EEPROM 操作的常数//
//#define ENABLE_ISP 0x80 //系统工作时钟<30MHz 时,对IAP_CONTR 寄存器设置此值
//#define ENABLE_ISP 0x81 //系统工作时钟<24MHz 时,对IAP_CONTR 寄存器设置此值
#define ENABLE_ISP 0x82 //系统工作时钟<20MHz 时,对IAP_CONTR 寄存器设置此值
//#define ENABLE_ISP 0x83 //系统工作时钟<12MHz 时,对IAP_CONTR 寄存器设置此值
//#define ENABLE_ISP 0x84 //系统工作时钟<6MHz 时,对IAP_CONTR 寄存器设置此值
//#define ENABLE_ISP 0x85 //系统工作时钟<3MHz 时,对IAP_CONTR 寄存器设置此值
//#define ENABLE_ISP 0x86 //系统工作时钟<2MHz 时,对IAP_CONTR 寄存器设置此值
//#define ENABLE_ISP 0x87 //系统工作时钟<1MHz 时,对IAP_CONTR 寄存器设置此值
#define WD1        0x46        //使用STC12C2052系列单片机时,先写入0x46,然写入0xb9
#define WD2        0xb9
/*********************************************************************************************/
#define DEBUG_DATA               0x55  //存储在 EEPROM 单元的数值(用户可修改测试)
#define DATA_FLASH_START_ADDRESS 0x00  //EEPROM存入地址(用户可修改测试)
/*********************************************************************************************/
/*********************************************************************************************/
union union_temp16
{
    INT16U un_temp16;
    INT8U  un_temp8[2];
}my_unTemp16;

INT8U Byte_Read(INT16U add);              //读一字节,调用前需打开IAP 功能
void Byte_Program(INT16U add, INT8U ch);  //字节编程,调用前需打开IAP 功能
void Sector_Erase(INT16U add);            //擦除扇区
void IAP_Disable();                       //关闭IAP 功能
void Delay();
/*********************************************************************************************/
/*********************************************************************************************
函数名:毫秒级CPU延时函数
调  用:DELAY_MS (?);
参  数:1~65535(参数不可为0)
返回值:无
结  果:占用CPU方式延时与参数数值相同的毫秒时间
备  注:应用于1T单片机时i<600,应用于12T单片机时i<125
/*********************************************************************************************/
void DELAY_MS (unsigned int a){
        unsigned int i;
        while( --a != 0){
                for(i = 0; i < 600; i++);
        }
}
/*********************************************************************************************/

// 读LCM忙程序 [底层协议] // (所有底层协议都无需关注)
// LCM2402测忙,若LCM2402处于忙状态,本函数将等待至非忙状态 //
/********************************************************************************************/
void LCM2402_TestBusy(void){
           LCM2402_DB0_DB7 = 0xff;                //设备读状态
           LCM2402_RS = 0;
           LCM2402_RW = 1;
           LCM2402_E = 1;
           while(LCM2402_Busy);                //等待LCM不忙
           LCM2402_E = 0;                                //
}
/********************************************************************************************
// 写指令程序 //
// 向LCM2402写命令 本函数需要1个指令集的入口参数 //
/********************************************************************************************/
void LCM2402_WriteCMD(uint8 LCM2402_command) {
          LCM2402_TestBusy();
   LCM2402_RS = 0;
          LCM2402_RW = 0;
          LCM2402_DB0_DB7 =  (LCM2402_command/16)<<4;//0x45 0x40
          LCM2402_E = 1;
          LCM2402_E = 0;
        LCM2402_DB0_DB7 =  (LCM2402_command%16)<<4;//0x45 0x50
          LCM2402_E = 1;
          LCM2402_E = 0;
}
/********************************************************************************************
// 写数据程序 //
// 向LCM2402写数据 //
/********************************************************************************************/
void LCM2402_WriteData(uint8 LCM2402_data){
    LCM2402_TestBusy();
          LCM2402_RS = 1;
          LCM2402_RW = 0;
          LCM2402_DB0_DB7 =  (LCM2402_data/16)*16;
        LCM2402_E = 1;
          LCM2402_E = 0;
        LCM2402_DB0_DB7 =  (LCM2402_data%16)*16;
        LCM2402_E = 1;
          LCM2402_E = 0;
}
/********************************************************************************************
// 打印字符串程序 // (本函数调用指针函数)
// 向LCM发送一个字符串,长度48字符之内
// 第一行位置 0x00~0x17  第二行位置 0x40~0x57
// 应用举例:print(0x80,"doyoung.net"); //在第一行第一位处从左向右打印doyoung.net字符串
/********************************************************************************************/
void print(uint8 a,uint8 *str){
        LCM2402_WriteCMD(a | 0x80);
        while(*str != ''){
                LCM2402_WriteData(*str++);
        }
        *str = 0;
}
/********************************************************************************************
// 打印单字符程序 //
// 第一行位置 0x00~0x17  第二行位置 0x40~0x57
// 向LCM发送一个字符,以十六进制(0x00)表示
// 应用举例:print(0xc0,0x30); //在第二行第一位处打印字符“0”
/********************************************************************************************/
void print2(uint8 a,uint8 t){
                LCM2402_WriteCMD(a | 0x80);
                LCM2402_WriteData(t);
}
/********************************************************************************************
// 定义小汉字 //
// 可写入8个自字义字符,写入后可用其CGRAM代码直接提取显示。
// 字符定义方法请参考技术手册
/********************************************************************************************/
void CgramWrite(void) {        // 装入CGRAM //
    uint8 i;
        LCM2402_WriteCMD(0x06);                        // CGRAM地址自动加1
        LCM2402_WriteCMD(0x40);                        // CGRAM地址设为00处
    for(i=0;i<64;i++) {
   
    }
}
/********************************************************************************************
// LCM2402初始化 //(使用者可自定义,加 * 号程序行必须保留但可修改)
/********************************************************************************************/
void LCM2402_Init(void){
          LCM2402_WriteCMD(CMD_set42);        //* 显示模式设置:显示2行,每个字符为5*7个像素
        LCM2402_WriteCMD(CMD_set42);        //* 显示模式设置:显示2行,每个字符为5*7个像素
          LCM2402_WriteCMD(CMD_clear);        //  显示清屏
          LCM2402_WriteCMD(CMD_back);                //* 数据指针指向第1行第1个字符位置
          LCM2402_WriteCMD(CMD_add1);                //  显示光标移动设置:文字不动,光标右移
          LCM2402_WriteCMD(CMD_dis_gb3);         //  显示开及光标设置:显示开,光标开,闪烁开
        CgramWrite();                                        //  向CGRAM写入自定义字符
}
/********************************************************************************************/
//                        以上是LCM2402驱动程序                        //
/**********************************************************************************************/       
void init (void){ //上电初始化
    TMOD=0x51;
    TH0 = 0x3C;
    TL0 = 0xB0;
         TL1=0;
    TH1=0;
    TR0=1;
    EA = 1;
    ET0 = 1;
    TR1 = 1;
        P1M1 = 0x0e;//            // 开闭定时/计数器0  =
}
/********************************************************************************************
*********************************************************************************************/   
/*********************************************************************************************
函数名:10位A/D转换初始化函数
调  用:Read_init (?);
参  数:输入的端口(0000 0XXX 其中XXX是设置输入端口号,可用十进制0~7表示,0表示P1.0,7表示P1.7)
返回值:无
结  果:开启ADC功能并设置ADC的输入端口
备  注:使用ADC功能时需要将对应的IO接口设置为高阻输入方式(例如:P1M1 = 0x01;)
/**********************************************************************************************/
void Read_init (unsigned char CHA){
        unsigned char AD_FIN=0; //存储A/D转换标志
    CHA &= 0x07;            //选择ADC的8个接口中的一个(0000 0111 清0高5位)
    ADC_CONTR = 0x40;                //ADC转换的速度(0XX0 0000 其中XX控制速度,请根据数据手册设置)
    _nop_();
    ADC_CONTR |= CHA;       //选择A/D当前通道
    _nop_();
    ADC_CONTR |= 0x80;      //启动A/D电源
    DELAY_MS(1);            //使输入电压达到稳定(1ms即可)
}
/**********************************************************************************************/
/*********************************************************************************************
函数名:10位A/D转换函数
调  用:? = ADC_Read();
参  数:无
返回值:10位ADC数据高(从0到1023(十进制))
结  果:读出指定ADC接口的A/D转换值,并返回数值
备  注:适用于STC12C5A60S2系列单片机(必须使用STC12C5A60S2.h头文件)
/**********************************************************************************************/
unsigned int ADC_Read (void){
        unsigned char AD_FIN=0; //存储A/D转换标志
    ADC_CONTR |= 0x08;      //启动A/D转换(0000 1000 令ADCS = 1)
    _nop_();
    _nop_();
    _nop_();
    _nop_();
    while (AD_FIN ==0){     //等待A/D转换结束
    AD_FIN = (ADC_CONTR & 0x10); //0001 0000测试A/D转换结束否
    }
    ADC_CONTR &= 0xE7;      //1111 0111 清ADC_FLAG位, 关闭A/D转换,
return (ADC_RES*4+ADC_RESL);//返回A/D转换结果(10位ADC数据高8位在ADC_RES中,低2位在ADC_RESL中)
}
/**********************************************************************************************/

/********************************************************************************************/
// 测试用函数 //
void main (void){
unsigned int ADC1,ADC3,ADC4,ADC5;
INT16U eeprom_address;
    INT8U  read_eeprom;
        init();//初始化                           
        LCM2402_Init();//LCM2402初始化
    ADC1=val;
                        print2(0x8c,ADC1/1000+0x30);
                        print2(0x8d,(ADC1%1000)/100+0x30);
                        print2(0x8e,(ADC1%100)/10+0x30);
                        print2(0x8f,ADC1%10+0x30);
                        //将EEPROM 测试起始地址单元的内容读出
    eeprom_address = DATA_FLASH_START_ADDRESS;  //将测试起始地址送eeprom_address
    read_eeprom = Byte_Read(eeprom_address);    //读EEPROM的值,存到read_eeprom
       
   print(0x80,"W");
        a = 1;                           
        while(1){ //主线程//
                if(1){
                        ADC1=val;
                        print2(0x81,ADC1/1000+0x30);
                        print2(0x82,(ADC1%1000)/100+0x30);
                        print2(0x83,(ADC1%100)/10+0x30);
                        print2(0x84,ADC1%10+0x30);
                        }

                print(0x8b,"T");
               
                       
                       
               
               
                print(0x40,"C");
                if(a == 3){
                        Read_init (a);        
                        ADC3 = ADC_Read ();
               
                        print2(0x41,(ADC3%1000)/100+0x30);//
                        print2(0x42,(ADC3%100)/10+0x30);//
                        print2(0x43,ADC3%10+0x30);//
                               
                }
                        print(0x46,"G");
                if(a == 4){
                        Read_init (a);        
                        ADC4 = ADC_Read ();
                       
                        print2(0x47,(ADC4%1000)/100+0x30);//
                        print2(0x48,(ADC4%100)/10+0x30);//
                        print2(0x49,ADC4%10+0x30);//
                        }
                        print(0x4c,"H");
                if(a == 5){
                        Read_init (a);        
                        ADC5 = ADC_Read ();
                       
                        print2(0x4d,(ADC4%1000)/100+0x30);//
                        print2(0x4e,(ADC4%100)/10+0x30);//
                        print2(0x4f,ADC4%10+0x30);//
                                                                       
     if(DEBUG_DATA == read_eeprom){
      
                        ADC1=val;
                       
                        print2(0x8c,ADC1/1000+0x30);//
                        print2(0x8d,(ADC1%1000)/100+0x30);//
                        print2(0x8e,(ADC1%100)/10+0x30);//
                        print2(0x8f,ADC1%10+0x30);//
                        Delay()    ;
                        ADC1 = ~read_eeprom;
                 
                }
                else
    {   ADC1 = ~0x08;
        Delay()    ;                            //延时
        ADC1 = ~read_eeprom;
        Delay()    ;                            //延时
        Sector_Erase(eeprom_address);           //擦除整个扇区
        Byte_Program(eeprom_address, DEBUG_DATA);//将 DEBUG_DATA 写入 EEPROM

        ADC1=val;
                        print2(0x8c,ADC1/1000+0x30);
                        print2(0x8d,(ADC1%1000)/100+0x30);
                        print2(0x8e,(ADC1%100)/10+0x30);
                        print2(0x8f,ADC1%10+0x30);
                         ADC1 = ~0x20;
                         }
                }
                a ++;                   
        }
}
/********************************************************************************************/   
/*********************************************************************************************/
void tiem1(void) interrupt 1{   // T/C0中断服务程序(产生50ms时基信号)     
    TH0 = 0x3C;          // 预置产生50ms时基信号   
    TL0 = 0xB0;
    tt++;
    if(tt == 20){
      tt = 0;

     while(1){
        th1=TH1;
        tl=TL1;
        th2=TH1;
        if(th1==th2)
           break;
      }       
      val=th1*256+tl;

      TH1=0;
      TL1=0;
    }     
}
/*********************************************************************************************/
/*********************************************************************************************/
//读一字节,调用前需打开IAP 功能,入口:DPTR = 字节地址,返回:A = 读出字节
INT8U Byte_Read(INT16U add)
{
    IAP_DATA = 0x00;
    IAP_CONTR = ENABLE_ISP;         //打开IAP 功能, 设置Flash 操作等待时间
    IAP_CMD = 0x01;                 //IAP/ISP/EEPROM 字节读命令

    my_unTemp16.un_temp16 = add;
    IAP_ADDRH = my_unTemp16.un_temp8[0];    //设置目标单元地址的高8 位地址
    IAP_ADDRL = my_unTemp16.un_temp8[1];    //设置目标单元地址的低8 位地址

    //EA = 0;
    IAP_TRIG = WD1;   //先送 WD1,再送WD2 到ISP/IAP 触发寄存器,每次都需如此
    IAP_TRIG = WD2;   //送完WD2 后,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 字节编程命令

    my_unTemp16.un_temp16 = add;
    IAP_ADDRH = my_unTemp16.un_temp8[0];    //设置目标单元地址的高8 位地址
    IAP_ADDRL = my_unTemp16.un_temp8[1];    //设置目标单元地址的低8 位地址

    IAP_DATA = ch;                  //要编程的数据先送进IAP_DATA 寄存器
    //EA = 0;
    IAP_TRIG = WD1;   //先送 WD1,再送WD2 到ISP/IAP 触发寄存器,每次都需如此
    IAP_TRIG = WD2;   //送完WD2 后,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 扇区擦除命令

    my_unTemp16.un_temp16 = add;
    IAP_ADDRH = my_unTemp16.un_temp8[0];    //设置目标单元地址的高8 位地址
    IAP_ADDRL = my_unTemp16.un_temp8[1];    //设置目标单元地址的低8 位地址

    //EA = 0;
    IAP_TRIG = WD1;   //先送 WD1,再送WD2 到ISP/IAP 触发寄存器,每次都需如此
    IAP_TRIG = WD2;   //送完WD2 后,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 Delay() //延时程序
{
    INT8U i;
    INT16U d=5000;
    while (d--)
    {
        i=255;
        while (i--);
    }
}
友情提示: 此问题已得到解决,问题已经关闭,关闭后问题禁止继续编辑,回答。