I2C总线协议(AT24c02)程序
主:STC89C54
从: AT24C02
电路图

时序图







下面是代码
#define uchar unsigned char
#define addr_x 0xae//写
#define addr_d 0xaf // 读
sbit sda = P2^1;//数据管脚
sbit scl = P2^0;//时钟管脚
bit ack;
void DelayUs2x(unsigned char t)//延时1
{
while(--t);
}
void DelayMs(unsigned char t)//延时2
{
while(t--)
{
//大致延时1mS
DelayUs2x(245);
DelayUs2x(245);
}
}
void delay() //延时大于4μs
{;;}
void i2_qs()//起始信号
{
sda = 1;//拉高数据
scl = 1;//拉高时钟
delay();//延时大于 4μs
sda = 0;//拉低数据产生起始信号(下降沿)
delay();//延时大于 4μs
scl = 0;//拉低时钟
delay();//延时大于 4μs
}
void i2_tz()//停止信号
{
sda = 0;//拉低数据
scl = 1;//拉高时钟
delay();//延时大于 4μs
sda = 1;//拉高时钟产生结束信号(上升沿)
delay();//延时大于 4μs
}
void i2_ack(bit _ack)//入口产生 0 ack 1 nak
{
sda = _ack;//ack或者nak
scl = 1;//拉高时钟
delay();//延时大于 4μs
scl = 0;//拉低时钟
delay();//延时大于 4μs
}
void i2_fs(uchar Data) //发送8位数据
{
uchar i;
for(i=0;i<8;i++)//8位计数
{
Data <<= 1;//把最高位移送到进制标志位中(CY)
sda = CY;//把进制位中的数据赋值给数据线
scl = 1;//拉高时钟
delay();//延时大于 4μs
scl = 0;//拉低时钟
//这里
}
//下面代码是接收ACK的代码
delay();//延时大于 4μs
sda = 1;//拉高数据准备接收ACK
scl = 1;//拉高时钟产生稳定的有效的数据(相对的)
if(sda==1)//确认接收的是ACK还是NAK
ack = 0;//ack
else
ack = 1;//nak
scl = 0;//拉低时钟
delay();//延时大于 4us
}
uchar i2_js()//接收8位数据
{
uchar i,Data = 0;
sda = 1;//使能内部上拉,准备读取数据
for(i=0;i<8;i++)//8位计数器
{
Data <<= 1;//移出数据的最高位
scl = 1;//拉高时钟
delay();//延时大于 4us
Data |= sda;//接收数据
scl = 0;//拉低时钟
delay();//延时大于 4us
}
return Data;
}
void i2_sj_x(uchar addr,uchar Data)//往设备内写入数据(参数 1、寄存器地址 2、写入的数据)
{
i2_qs();//起始信号
i2_fs(addr_x);//设备地址+写信号
i2_fs(addr);//寄存器内部地址
i2_fs(Data);//写入设备的数据
i2_tz();//停止信号
}
uchar i2_sj_d(uchar addr)//读取数据(参数 寄存器地址)
{
uchar Data;
i2_qs();//起始信号
i2_fs(addr_x);//设备地址+写信号
i2_fs(addr);//寄存器内部地址
i2_qs();//起始信号
i2_fs(addr_d);//设备地址+读信号
Data = i2_js();//读取数据
i2_ack(0);//ACK应答
i2_tz();//停止信号
return Data;//返回读取的数据
}
void main(void)
{
uchar dat;
i2_sj_x(3,0x0f); //数据写入24c02
DelayMs(50);
dat = i2_sj_d(3); //从24c02中读取数据
P1 = dat;//使用8个LED显示读出的数据
while(1)
{
;
}
}
以上代码只是简单的实现I2C总线的读写