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