Inter IC Bus 由PHILIPS公司推出的,是近年来微电子通信控制领域广泛采用的一种新型总线标准,他是同步通信的一种特殊形式。
/* 利用定时器产生一个0~99秒变化的秒表,并且显示在数码管上,每过一秒将这个变化的数写入AT24C02内部,当关闭电源,并再次打开
电源,单片机先从AT24C02中将原来写入的数读取出来,接着在数码管上显示*/
#include <reg52.h>
#define uchar unsigned char
#define uint unsigned int
bit write=0; //写24C02的标志
sbit sda=P2^0;
sbit scl=P2^1;
sbit dula=P2^6;
sbit wela=P2^7;
uchar sec,tcnt;
uchar code table[]={
0x3f,0x06,0x5b,0x4f,
0x66,0x6d,0x7d,0x07,
0x7f,0x6f,0x77,0x7c,
0x39,0x5e,0x79,0x71
};
void delay()
{
;
;
}
void delay1ms(uint z)
{
uint x,y;
for(x=z;x>0;x--)
for(y=110;y>0;y--);
}
//将总线都拉高以释放总线
void start() //开始信号
{
sda=1;
delay();
scl=1;
delay();
sda=0;
delay();
}
//SCL在高电平期间,SDA一个下降沿启动信号
void stop() //停止信号
{
sda=0;
delay();
scl=1;
delay();
sda=1;
delay();
}
void respons() //应答信号
{
uchar i;
scl=1;
delay();
while((sda==1)&&(i<250))
//SCL在高电平期间,SDA被从设备拉为低电平表示应答,(sda=1)和(i<255)相与的关系表示若在一段时间内没有收到从器件
//的应答则主器件默认从器件已经收到数据而不在等待应答信号
i++;
scl=0;
delay();
}
void init()
{
sda=1;
delay();
scl=1;
delay();
}
void write_byte(uchar date)
{
uchar i,temp;
temp=date;
for(i=0;i<8;i++)
{
temp=temp<<1;
scl=0;
delay();
sda=CY;
delay();
scl=1;
delay();
}
scl=0;
delay();
sda=1;
delay();
}
//串行发送一个字节是,需要把这个字节中的8为一位一位地发送出去,"temp=temp<<1;"表示将temp左移一位,最高位将
//移入PSW寄存器的CY位中,然后将CY赋给SDA进而在SCL的控制下发送出去
uchar read_byte()
{
uchar i,k;
scl=0;
delay();
sda=1;
for(i=0;i<8;i++)
{
scl=1;
delay();
k=(k<<1)|sda;
scl=0;
delay();
}
delay();
return k;
}
//串行接受一个字节时须将8位一位一位地接受,然后再组合成一个字节,上面的代码中定义一个临时变量k,将k左移一位后与SDA进行"或“
//运算 ,依次把8个独立的位放入一个字节中来完成接受
void write_add(uchar address,uchar date)
{
start();
write_byte(0xa0);
respons();
write_byte(address);
respons();
write_byte(date);
respons();
stop();
}
uchar read_addr(uchar address)
{
uchar date;
start();
write_byte(0xa0);
respons();
write_byte(address);
respons();
start();
write_byte(0xa1);
respons();
date=read_byte();
stop();
return date;
}
void display(uchar bai_c,uchar sh_c)
{
dula=0;
//关闭段选
P0=table[bai_c];
//显示第一位
dula=1;
//打开段选
dula=0;
//关闭段选
wela=0;
//关闭位选
P0=0x7e;
//P0口赋值
wela=1;
wela=0;
delay1ms(5);
dula=0;
P0=table[sh_c];
//显示第二位
dula=1;
dula=0;
wela=0;
P0=0x7d;
wela=1;
wela=0;
delay1ms(5);
}
void main()
{
init();
sec=read_addr(2);
//读出保存的数据赋予sec
if(sec>100)
//防止首次读取出错误的数据
sec=0;
TMOD=0x01;
//定时器工作在方式1
ET0=1;
EA=1;
TH0=(65536-50000)/256;
//对TH0 TL0赋值
TL0=(65536-50000)%256;
//使定时器0.05秒中断一次
TR0=1;
//开始计时
while(1)
{
display(sec/10,sec%10);
if(write==1)
{
write=0;
//清0
write_add(2,sec);
//在24c02的地址2中写入数据sec
}
}
}
void t0() interrupt 1 //定时器中断服务函数
{
TH0=(65536-50000)/256;
//对TH0 TL0赋值
TL0=(65536-50000)%256;
//重装计数初值
tcnt++;
//每过50ms tcnt加一
if(tcnt==20)
//计满20次1秒时
{
tcnt=0;
//重新再计
sec++;
write=1;
//1秒写一次24C02
if(sec==100)
//定时器100秒,再从零开始计数
sec=0;
}
}
在主程序的开始处先读取上次写入的AT24C02的数据,后面两句是为了第一次操作AT24C02时出现意外而加的
原文地址:http://blog.csdn.net/u011046042/article/details/37958229