|
本帖最后由 pull1121 于 2020-2-22 11:00 編輯
Protues 仿真51單片機 用AT24C2數據不保存,重啟后,數據歸零。
假如 數碼管上顯示 XX,(XX為任意數)重新上電后數碼管變成 00效果圖:如下
#include <reg51.h>
#include <intrins.h>
#define uint unsigned int
#define uchar unsigned char
uchar code table[]={ //數碼管顯示0到F
0x3f,0x06,0x5b,0x4f,
0x66,0x6d,0x7d,0x07,
0x7f,0x6f,0x77,0x7c,
0x39,0x5e,0x79,0x71};
sbit dula=P2^6;
sbit wela=P2^7;
sbit SDA=P2^0;
sbit SCL=P2^1;
bit write=0;
uchar sec,num1;
void delay(uint cc) //ms級別的延時
{
uint aa,bb;
for(aa=cc;aa>0;aa--)
for(bb=110;bb>0;bb--);
}
void delay_ws() //ns級別的延時
{
_nop_();
_nop_();
_nop_();
_nop_();
_nop_();
}
void display(uchar bai_c,uchar sh_c)
{
//shi=flag/10;
//ge=flag%10;
P0=0x00;
dula=1;
P0=table[bai_c];
dula=0;
P0=0xff;
wela=1;
P0=0x7e;
wela=0;
delay(5);
P0=0x00;
dula=1;
P0=table[sh_c];
dula=0;
P0=0xff;
wela=1;
P0=0x7d;
wela=0;
delay(5);
}
void init ()
{
/* TMOD=0x11; //T0定時器初始化
TH0=(65536-45872)/256;
TL0=(65536-45872)%256;
TR0=1;
ET0=1;
EA=1;
*/
SCL=1; //I2C的2條總線初始化
delay_ws();
SDA=1;
delay_ws();
}
void start() //單片機開始寫信號
{ //嚴格按照時序圖來寫,SCK=1時候,SDA從1到0,一個下降沿表示開始
SDA=1;
delay_ws();
SCL=1;
delay_ws();
SDA=0;
delay_ws();
SCL=0; //等待下一步開始 寫數據 或者 讀數據
delay_ws();
}
void respons() //從機應答信號
{ //嚴格按照時序圖來寫
uchar i=0;
SCL=1;
delay_ws();
while((SDA==1)&&(i<255)) // SDA=0時候表示應答,或者等待一下無應答
i++; // 都表示結束
SCL=0;
delay_ws();
}
void stop() //單片機結束寫信號
{ //嚴格按照時序圖來寫,SCK=1時候,SDA從0到1,一個上升沿表示結束
SDA=0;
delay_ws();
SCL=1;
delay_ws();
SDA=1;
delay_ws();
}
void write_byte(uchar date)//單片機寫一個字節信號,從1個字節的最高位到最低位依次寫入
{ //和讀數據一樣 最高到最低
uchar i,temp;
temp=date;
for(i=0;i<8;i++) //將date的最高位到最低位依次賦給SDA
{
temp=temp<<1; //將數據 date 最高位右移一位賦給變量 temp
SCL=0;
delay_ws();
SDA=CY; //變量 temp移出來的一位通過PSW寄存器的CY位賦給SDA
delay(5);
SCL=1; //SDA數據的變化 需要SCK從0到1變化
delay_ws();
}
SCL=0; //等待下次sda上數據變化,個人理解是等待從機相應
delay_ws();
SDA=1; //釋放數據總線
delay_ws();
}
uchar read_byte()//單片機讀一個字節信號,從1個字節的最高位到最低位依次讀出
{ //此子函數帶返回值 return k
uchar i,k;
SCL=0; //置0,等待SDA上的數據被讀取
delay_ws();
SDA=1; //置1,釋放SDA,等待SDA上的數據被讀取
delay_ws();
for(i=0;i<8;i++)
{
SCL=1; //SCL從0到1,SDA上的一個字節數據被依次放入返回值 k
delay_ws();
k=(k<<1)|SDA; //將第一位SDA數據置于最高位,k0 | X(0或者1)
// 0|X都為X,所以將第一位SDA的數據置于K的最高位
//然后依次循環8次,將SDA從最高位放到最低位
SCL=0;
delay_ws();
}
return k; //SDA上的數據從第一位到第八位依次放入K的最高位到最低位
}
void write_add(uchar address,uchar date) //從E2Prom任意地址寫任意數據
{
start(); //開始信號
write_byte(0xa0); // 1010 0000 單片機發地址,在I2C上面找
// 設備(低4位前3位代表設備地址000,后1位的0代表方向主機到從機)
respons(); // E2Prom設備應答
write_byte(address); // 從E2Prom設備的address地址開始寫數據
respons(); // E2Prom設備應答
write_byte(date); //給E2Prom設備開始寫數據 date
respons(); // E2Prom設備應答
stop(); //停止
} //從E2Prom任意地址寫任意數據的流程是:1.開始 2.設備地址
//3.被尋址的設備應答 4.給被尋址的設備要寫數據的位置
//5.被尋址的設備應答 6.給被尋址的設備寫數據
//7.應答 8.停止
uchar read_add(uchar address) //從E2Prom任意地址讀任意數據
{
uchar date;
start(); //開始信號
write_byte(0xa0); // 1010 0000 單片機發地址,在I2C上面找
//設備(低4位前3位代表設備地址000,后1位的0代表方向主機到從機)
respons();
write_byte(address);//從E2Prom設備的address地址開始讀數據
respons(); // E2Prom設備應答
start(); //開始信號
write_byte(0xa1); // 1010 0001 單片機發地址,在I2C上面找
// 設備(低4位前3位代表設備地址000,后1位的1代表方向從機到主機)
respons(); // E2Prom設備應答
date=read_byte(); //把讀到的數據賦給date
stop(); //停止
return date; //返回讀到的數據 date
}
//從E2Prom任意地址讀數據的流程是:1.開始 2.設備地址(主機到從機)
//3.被尋址的設備應答 4.給被尋址的設備要讀數據的位置
//5.開始 6. 設備地址(從機到主機)7.被尋址的設備應答
//8.從被尋址的設備讀數據賦給date 9.停止 10.將date返回
void Time0 () interrupt 1 //T0中斷子函數
{
TH0=(65536-45872)/256; //頻率為11.0592MHZ下的 高8位 初值
TL0=(65536-45872)%256; //頻率為11.0592MHZ下的 低8位 初值
num1++; //每50ms加一次
if(num1==20) //T0計滿20次(1秒)時
{
num1=0; //清零,繼續計數
sec++; //數碼管上的數1s加一次
write=1; //一秒寫一次AT24C02(E2PROM)
if(sec==100)//如果計時到了100S就重新計時
sec=0; //清零
}
}
void main()
{
init(); //初始化 ,其實就是SDA和SCL釋放
sec=read_add(2);//讀出保存的數據賦給sec
if(sec>99)//讀出的E2PROM如果大于99,只有2位數碼管顯示最多顯示2位數
sec=0; //清零
TMOD=0x01;//設置T0定時器工作方式為1
TH0=(65536-45872)/256; //頻率為11.0592MHZ下的 高8位 初值
TL0=(65536-45872)%256; //頻率為11.0592MHZ下的 低8位 初值
TR0=1; //T0的運行位打開,開始計時
ET0=1; //T0中斷打開
EA=1; //總中斷打開
while(1)
{
display(sec/10,sec%10); //將數sec顯示在數碼管上
if(write==1) //判斷計時器是否計時1S
{
write=0; //清零
write_add(2,sec);//在E2PROM的列地址2中寫入數據sec
delay(5);
}
}
}
|
|