STC15系列单片机利用EEPROM统计开机次数

Posted perseverance52

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了STC15系列单片机利用EEPROM统计开机次数相关的知识,希望对你有一定的参考价值。

STC15系列单片机利用EEPROM统计开机次数


  • 📺测试效果:

  • 📌相关篇 《STC15系列单片机EEPROM读写示例

  • ✨采用的是无符号整数(unsigned int)变量计数方式,可统计次数位:65535,如果还觉得不够,还可以定义个无符号长整型变量(unsigned long);

  • 🚩在STC15W408AS上验证没有问题。

📓EEPROM初次写入计数程序

/*---------------------------------------------------------------------*/
/* --- STC MCU Limited ------------------------------------------------*/
/* --- STC15F4K60S4 系列 内部EEPROM举例--------------------------------*/
/* --- Mobile: (86)13922805190 ----------------------------------------*/
/* --- Fax: 86-0513-55012956,55012947,55012969 ------------------------*/
/* --- Tel: 86-0513-55012928,55012929,55012966-------------------------*/
/* --- Web: www.STCMCU.com --------------------------------------------*/
/* --- Web: www.GXWMCU.com --------------------------------------------*/
/* 如果要在程序中使用此代码,请在程序中注明使用了STC的资料及程序        */
/* 如果要在文章中应用此代码,请在文章中注明使用了STC的资料及程序        */
/*---------------------------------------------------------------------*/

//本示例在Keil开发环境下请选择Intel的8058芯片型号进行编译
//若无特别说明,工作频率一般为11.0592MHz


#include "reg51.h"
#include "intrins.h"
//#include <string.h>
//#include <stdio.h>
#define     FOSC    11059200UL
#define     BAUD    9600

typedef unsigned char u8;
typedef unsigned int u16;

sbit P17 = P1 ^ 7;

//-----------------------------------------------

sfr P1M1 = 0x91;    //PxM1.n,PxM0.n     =00--->Standard,    01--->push-pull
sfr P1M0 = 0x92;    //                  =10--->pure input,  11--->open drain
sfr P0M1 = 0x93;
sfr P0M0 = 0x94;
sfr P2M1 = 0x95;
sfr P2M0 = 0x96;
sfr P3M1 = 0xB1;
sfr P3M0 = 0xB2;


sfr IAP_DATA    =   0xC2;           //IAP数据寄存器
sfr IAP_ADDRH   =   0xC3;           //IAP地址寄存器高字节
sfr IAP_ADDRL   =   0xC4;           //IAP地址寄存器低字节
sfr IAP_CMD     =   0xC5;           //IAP命令寄存器
sfr IAP_TRIG    =   0xC6;           //IAP命令触发寄存器
sfr IAP_CONTR   =   0xC7;           //IAP控制寄存器

#define CMD_IDLE    0               //空闲模式
#define CMD_READ    1               //IAP字节读命令
#define CMD_PROGRAM 2               //IAP字节编程命令
#define CMD_ERASE   3               //IAP扇区擦除命令

#define     URMD    0           //0:使用定时器2作为波特率发生器
//1:使用定时器1的模式0(16位自动重载模式)作为波特率发生器
//2:使用定时器1的模式2(8位自动重载模式)作为波特率发生器

sfr T2H   = 0xd6;               //定时器2高8位
sfr T2L   = 0xd7;               //定时器2低8位

sfr  AUXR       =   0x8e;       //辅助寄存器

//#define ENABLE_IAP 0x80           //if SYSCLK<30MHz
//#define ENABLE_IAP 0x81           //if SYSCLK<24MHz
//#define ENABLE_IAP  0x82            //if SYSCLK<20MHz
#define ENABLE_IAP 0x83           //if SYSCLK<12MHz
//#define ENABLE_IAP 0x84           //if SYSCLK<6MHz
//#define ENABLE_IAP 0x85           //if SYSCLK<3MHz
//#define ENABLE_IAP 0x86           //if SYSCLK<2MHz
//#define ENABLE_IAP 0x87           //if SYSCLK<1MHz

//测试地址,一个扇区为512字节
#define IAP_ADDRESS 0x0800// code:549字节,占用第一个扇区;结束地址:0xF3FF
//u8  code  T_Strings[] = "去年今日此门中,人面桃花相映红。人面不知何处去,桃花依旧笑春风。";

//u8    xdata   tmp[70];


u16 cnt = 1;
u8 pt[5];

void Delay(u8 n);
void  delay_ms(unsigned char ms);
void IapIdle();
u8 IapReadu8(u16 addr);
void IapProgramu8(u16 addr, u8 dat);
void IapEraseSector(u16 addr);
void InitUart();
u8 SendData(u8 dat);
void SendString(unsigned char *pstr);       //串口发送一个字符串
char *_uitoa(unsigned int num, char *str);//整数转字符串
int unsignedstrtoint(char *str);     //开头不带正负号的字符串转整型

void main()

    u16 i;
//	u8 readdata[5];
    P0M0 = 0x00;
    P0M1 = 0x00;
    P1M0 = 0x00;
    P1M1 = 0x00;
    P2M0 = 0x00;
    P2M1 = 0x00;
    P3M0 = 0x00;
    P3M1 = 0x00;
	
		_uitoa(cnt,pt);//整数转字符串	
    InitUart();                     //初始化串口
    Delay(10);                      //延时
    SendString("Hello World");
	//-----------eeprom读------------
//	    for (i = 0; i < 5; i++)      //检测是否擦除成功(全FF检测)
//    
//        readdata[i]=IapReadu8(IAP_ADDRESS + i);
//    
//		value = unsignedstrtoint(readdata);//字符串转无符号整形
//		value +=1;
//		_uitoa(value,readdata);//无符号整形转字符串
		//-----------eeprom擦除------------
    IapEraseSector(IAP_ADDRESS);    //扇区擦除		
    for (i = 0; i < 512; i++)      //检测是否擦除成功(全FF检测)
    
        if (IapReadu8(IAP_ADDRESS + i) != 0xff)
         SendString("ca chu error");            //如果出错,则退出
    
		 SendString("ca chu success \\n\\r"); //1111,1100 擦除成功
		//-----------eeprom写入数据------------
			    for (i = 0; i < 5; i++)      //检测是否擦除成功(全FF检测)
    
        IapProgramu8(IAP_ADDRESS + i,pt[i]);
    
   SendString("write success");
    Delay(10);                      //延时
    while (1)
    

        P17 = 0x00;                     //0xxx,xxxx IAP操作失败
        delay_ms(250);
        delay_ms(250);
        P17 = 0x01;
        delay_ms(250);
        delay_ms(250);
 //       SendString(readdata);


    


/*----------------------------
软件延时
----------------------------*/
void Delay(u8 n)

    u16 x;
    while (n--)
    
        x = 0;
        while (++x);
    

void  delay_ms(unsigned char ms)

    unsigned int i;
    do
    
        i = FOSC / 13000;
        while (--i)    ;  //14T per loop
    
    while (--ms);

/*----------------------------
关闭IAP
----------------------------*/
void IapIdle()

    IAP_CONTR = 0;                  //关闭IAP功能
    IAP_CMD = 0;                    //清除命令寄存器
    IAP_TRIG = 0;                   //清除触发寄存器
    IAP_ADDRH = 0x80;               //将地址设置到非IAP区域
    IAP_ADDRL = 0;


/*----------------------------
从ISP/IAP/EEPROM区域读取一字节
----------------------------*/
u8 IapReadu8(u16 addr)

    u8 dat;                       //数据缓冲区

    IAP_CONTR = ENABLE_IAP;         //使能IAP
    IAP_CMD = CMD_READ;             //设置IAP命令
    IAP_ADDRL = addr;               //设置IAP低地址
    IAP_ADDRH = addr >> 8;          //设置IAP高地址
    IAP_TRIG = 0x5a;                //写触发命令(0x5a)
    IAP_TRIG = 0xa5;                //写触发命令(0xa5)
    _nop_();                        //等待ISP/IAP/EEPROM操作完成
    dat = IAP_DATA;                 //读ISP/IAP/EEPROM数据
    IapIdle();                      //关闭IAP功能

    return dat;                     //返回


/*----------------------------
写一字节数据到ISP/IAP/EEPROM区域
----------------------------*/
void IapProgramu8(u16 addr, u8 dat)

    IAP_CONTR = ENABLE_IAP;         //使能IAP
    IAP_CMD = CMD_PROGRAM;          //设置IAP命令
    IAP_ADDRL = addr;               //设置IAP低地址
    IAP_ADDRH = addr >> 8;          //设置IAP高地址
    IAP_DATA = dat;                 //写ISP/IAP/EEPROM数据
    IAP_TRIG = 0x5a;                //写触发命令(0x5a)
    IAP_TRIG = 0xa5;                //写触发命令(0xa5)
    _nop_();                        //等待ISP/IAP/EEPROM操作完成
    IapIdle();


/*----------------------------
扇区擦除
----------------------------*/
void IapEraseSector(u16 addr)

    IAP_CONTR = ENABLE_IAP;         //使能IAP
    IAP_CMD = CMD_ERASE;            //设置IAP命令
    IAP_ADDRL = addr;               //设置IAP低地址
    IAP_ADDRH = addr >> 8;          //设置IAP高地址
    IAP_TRIG = 0x5a;                //写触发命令(0x5a)
    IAP_TRIG = 0xa5;                //写触发命令(0xa5)
    _nop_();                        //等待ISP/IAP/EEPROM操作完成
    IapIdle();


/*----------------------------
初始化串口
----------------------------*/
void InitUart()

    SCON = 0x5a;                //设置串口为8位可变波特率
#if URMD == 0
    T2L = (65536 - (FOSC / 4 / BAUD));
    T2H = (65536 - (FOSC / 4 / BAUD)) >> 8;
    AUXR = 0x14;                //T2为1T模式, 并启动定时器2
    AUXR |= 0x01;               //选择定时器2为串口1的波特率发生器
#elif URMD == 1
    AUXR = 0x40;                //定时器1为1T模式
    TMOD = 0x00;                //定时器1为模式0(16位自动重载)
    TL1 = (65536 - (FOSC / 4 / BAUD));
    TH1 = (65536 - (FOSC / 4 / BAUD)) >> 8;
    TR1 = 1;                    //定时器1开始启动
#else
    TMOD = 0x20;                //设置定时器1为8位自动重装载模式
    AUXR = 0x40;                //定时器1为1T模式
    TH1 = TL1 = (256 - (FOSC / 32 / BAUD));
    TR1 = 1;
#endif



/*----------------------------
发送串口数据
----------------------------*/
u8 SendData(u8 dat)

    while (!TI);                    //等待前一个数据发送完成
    TI = 0;                         //清除发送标志
    SBUF = dat;                     //发送当前数据

    return dat;


//----------------------------------------------------------
//  函数名称:void SendString(unsigned char *pstr)
//  函数功能:串口发送一个字符串
//----------------------------------------------------------
void SendString(unsigned char *pstr)

    while (*pstr != '\\0')          //字符串是否发完
    
        SendData(*pstr);           //发送字符串数据
        pstr++;                    //指向下一个字符
    


char *_uitoa(unsigned int num, char *str)

    unsigned char i = 0, index = 0;
    unsigned char temp = 0;
    unsigned int bottom = 10000;
    for (i = 0; i < 5; i++)
    
        temp = num / bottom;
        if (temp > 0 || index > 0) str[index++] = 0x30 + temp;
        num %= bottom;
        bottom /= 10;
    
    if (index == 0)
        str[index++] = 0x30;

    str[index] = '\\0';
    return str;




int unsignedstrtoint(char *str)     //开头不带正负号的字符串转整型

    int num = 0;
    if (0 == *str)   //字符串为空(NULL)
        return 0;
    else
    
        while (str)
        
            if (*str < '0' || *str > '9')   //只要当前字符不在数字字符的范围内,直接跳出循环
                break;
            else
                num = num * 10 + (*str - '0');      //括号里面的是二者字符对应ASCII值的差值,即为当前位置的字符对应的整型值
            str++;
        
    
    return num;


📖读取开机次数程序

/*---------------------------------------------------------------------*/
/* --- STC MCU Limited ------------------------------------------------*/
/* --- STC15F4K60S4 系列 内部EEPROM举例--------------------------------*/
/* --- Mobile: (86)13922805190 ----------------------------------------*/
/* --- Fax: 86-0513-55012956,55012947,55012969 ------------------------*/
/* --- Tel: 86-0513-55012928,55012929,55012966-------------------------*/
/* --- Web: www.STCMCU.com --------------------------------------------*/
/* --- Web: www.GXWMCU.com --------------------------------------------*/
/* 如果要在程序中使用此代码,请在程序中注明使用了STC的资料及程序        */
/* 如果要在文章中应用此代码,请在文章中注明使用了STC的资料及程序        */
/*---------------------------------------------------------------------*/

//本示例在Keil开发环境下请选择Intel的8058芯片型号进行编译
//若无特别说明,工作频率一般为11.0592MHz


#include "reg51.h"
#include "intrins.h"
//#include <string.h>
//#include <stdio.h>
#define     FOSC    11059200UL
#define     BAUD    9600

typedef unsigned char u8;
typedef unsigned int u16;

sbit P17 = P1 ^ 7;

//-----------------------------------------------

sfr P1M1 = 0x91;    //PxM1.n,PxM0.n     =00--->Standard,    01--->push-pull
sfr P1M0 = 0x92;    //                  =10--->pure input,  11--->open drain
sfr P0M1 = 0x93;
sfr P0M0 = 0x94;
sfr P2M1 = 0x95;
sfr P2M0 = 0x96;
sfr P3M1 = 0xB1;
sfr P3M0 = 0xB2;


sfr IAP_DATA    =   0xC2;           //IAP数据寄存器
sfr IAP_ADDRH   =   0xC3;           //IAP地址寄存器高字节
sfr IAP_ADDRL   =   0xC4;           //IAP地址寄存器低字节
sfr IAP_CMD     =   0xC5;           //IAP命令寄存器
sfr IAP_TRIG    =   0xC6;           //IAP命令触发寄存器
sfr IAP_CONTR   =   0xC7;           //IAP控制寄存器

#define CMD_IDLE    0               //空闲模式
#define CMD_READ    1               //IAP字节读命令
#define CMD_PROGRAM 2               //IAP字节编程命令
#define CMD_ERASE   3               //IAP扇区擦除命令

#define     URMD    0           //0:使用定时器2作为波特率发生器
//1:使用定时器1的模式0(16位自动重载模式)作为波特率发生器
//2:使用定时器1的模式2(8位自动重载模式)作为波特率发生器

sfr T2H   = 0xd6;               //定时器2高8位
sfr T2L   = 0xd7;               //定时器2低8位

sfr  AUXR       =   0x8e;       //辅助寄存器

//#define ENABLE_IAP 0x80           //if SYSCLK<30MHz
//#define ENABLE_IAP 0x81           //if SYSCLK<24MHz
//#define ENABLE_IAP  0x82            //if SYSCLK<20MHz
#define ENABLE_IAP 0x83           //if SYSCLK<12MHz
//#define ENABLE_IAP 0x84           //if SYSCLK<6MHz
//#define ENABLE_IAP 0x85           //if SYSCLK<3MHz
//#define ENABLE_IAP 0x86           //if SYSCLK<2MHz
//#define ENABLE_IAP 0x87           //if SYSCLK<1MHz

//测试地址,一个扇区为512字节
#define IAP_ADDRESS 0x0800// code:549字节,占用第一个扇区;结束地址:0xF3FF
//u8  code  T_Strings[] = "去年今日此门中,人面桃花相映红。人面不知何处去,桃花依旧笑春风。";

//u8    xdata   tmp[70];


//u16 cnt = 1;
//u8 pt[5];

void Delay(u8 n);
void  delay_ms(unsigned char ms);
void IapIdle();
u8 IapReadu8(u16 addr);
void IapProgramu8(u16 addr, u8 dat);
void IapEraseSector(u16 addr);
void InitUart();
u8 SendData(u8 dat);
void SendString(unsigned char *pstr);       //串口发送一个字符串
char *_uitoa(unsigned int num, char *str);//整数转字符串
int unsignedstrtoint(char *str);     //开头不带正负号的字符串转整型




void main()

    u16 i,value;
	u8 readdata[5];
    P0M0 = 0x00;
    P0M1 = 0x00;
    P1M0 = 0x00;
    P1M1 = 0x00;
    P2M0 = 0x00;
    P2M1 = 0x00;
    P3M0 = 0x00;
    P3M1 = 0x00;
	
//		_uitoa(cnt,pt);//整数转字符串
	
    InitUart();                     //初始化串口
    Delay(10);                      //延时
    SendString("Hello World");
	//-----------eeprom读------------
	    for (i = 0; i < 5; i++)      //检测是否擦除成功(全FF检测)
    
        readdata[i]=IapReadu8(IAP_ADDRESS + i);
    
		value = unsignedstrtoint(readdata);//字符串转无符号整形
		value +=1;
		_uitoa(value,readdata);//无符号整形转字符串
		//-----------eeprom擦除------------
    IapEraseSector(IAP_ADDRESS);    //扇区擦除		
    for (i = 0; i < 512; i++)      //检测是否擦除成功(全FF检测)
    
        if (IapReadu8(IAP_ADDRESS + i) != 0xff)
         SendString("ca chu error");     

以上是关于STC15系列单片机利用EEPROM统计开机次数的主要内容,如果未能解决你的问题,请参考以下文章

STC15系列单片机EEPROM读写示例

STC15系列单片机EEPROM读写示例

9.STC15W408AS单片机EEPROM

9.STC15W408AS单片机EEPROM

9.STC15W408AS单片机EEPROM

STC15系列单片机通过串口多字节数据读写EEPROM操作