I2C总线相关代码分析

#include<reg52.h>
#include<intrins.h>
sbit SCL=P2^1;
sbit SDA=P2^0;
sbit DU=P2^6;
sbit WE=P2^7;
#define uchar unsigned char
#define At24c02ADDR 0xA0 //AT24c02芯片的地址
#define I2CRead 1
#define I2CWrite 0
bit AckFlag;
uchar num;
//共阴数码管段选表0-9
uchar code SMGduan[]= {0x3F, 0x06, 0x5B, 0x4F, 0x66, 0x6D, 0x7D, 0x07, 0x7F, 0x6F,};
//数码管位选码
uchar code SMGwei[] = {0xfe, 0xfd, 0xfb};
/**************************
IIC通信代码
**************************/
/毫秒级延时函数/
void delay(uchar z)
{
uchar x,y;
for(x=z;x>0;x–)
for(y=114;y>0;y–);
}
/5us延时函数/
void delay5us()
{
/_nop_函数执行一次是一个机械周期,
进入一次这个函数需要三点几个机械周期,
整个函数共占用约5us时间
/
nop();
}
//定时器0初始化
void timer0Init()
{
EA = 1; //打开总中断
ET0 = 1;//打开定时器0中断
TR0 = 1; //启动定时器0
TMOD = 0X01; //定时器工作模式1,16位定时模式
TH0 = 0xED;
TL0 = 0xFF; //定时5ms
}
void display(uchar i)
{
static uchar wei;
P0 = 0XFF;//清除断码
WE = 1;//打开位选锁存器
P0 = SMGwei[wei];
WE = 0;//锁存位选数据
switch(wei)
{
case 0: DU = 1; P0 = SMGduan[i / 100]; DU = 0; break;
case 1: DU = 1; P0 = SMGduan[i % 100 / 10]; DU = 0; break;
case 2: DU = 1; P0 = SMGduan[i % 10]; DU = 0; break;
}
wei++;
if(wei == 3)
wei = 0;
}
/起始信号/
void I2CStart()
{
SCL=1;
SDA=1;
delay5us();
SDA=0;
delay5us();
}
/终止信号/
void I2CStop()
{
SCL=0;
SDA=0;
SCL=1;
delay5us();
SDA=1;
delay5us();
}
/I2C主机读从机应答信号/
bit ReadACK()
{
SCL=0;
SCL=1;
delay5us();
if(SDA) //NOACK
{
SCL=0;
return 1;
}
else //ACK
{
SCL=0;
return 0;
}
}
/I2C主机发送应答/
void SendACK(bit i)
{
SCL=0;
if(i)
SDA=1;
else
SDA=0;
SCL=1;
delay5us();
SCL=0;
SDA=1; //释放数据总线
}
/发送一字节/
void I2CSendByte(uchar DAT)
{
uchar i;
for(i=0;i<8;i++)
{
SCL=0; //时钟线拉低,允许数据发生变化
if(DAT&0x80)
SDA=1;
else
SDA=0;
SCL=1; //时钟总线拉高后接在总线上的At24c02会读SDA送的数据
DAT<<=1; //最高位移出,最低位补零
}
SCL=0;
SDA=1;//释放数据总线
}
/读一字节/
uchar I2CReadByte()
{
uchar i,DAT;
for(i=0;i<8;i++)
{
DAT<<=1;
SCL=0; //允许SDA进行变化
SCL=1; //从机将该数据放在SDA线上
if(SDA)
DAT|=0x01;
}
return DAT;
}
//
void At24c02Write(uchar ADDR,DAT)
{
I2CStart();
I2CSendByte(At24c02ADDR+I2CWrite);//发送一个字节(地址+方向)
if(ReadACK()) //NOACK
AckFlag=1;
else //ACK
AckFlag=0;
I2CSendByte(ADDR);//发送一个字节
if(ReadACK()) //NOACK
AckFlag=1;
else //ACK
AckFlag=0;
I2CSendByte(DAT);//发送数据
if(ReadACK()) //NOACK
AckFlag=1;
else //ACK
AckFlag=0;
I2CStop();
}
//
uchar At24c02Read(uchar ADDR)//读出数据
{
uchar DAT;
I2CStart();
I2CSendByte(At24c02ADDR+I2CWrite);//发送一个字节(地址+方向)
if(ReadACK()) //NOACK
AckFlag=1;
else //ACK
AckFlag=0;
I2CSendByte(ADDR);//发送一个字节
ReadAck();
I2CStart();
I2CSendByte(At24c02ADDR+I2CRead);//读一个字节(地址+方向)
if(ReadACK()) //NOACK
AckFlag=1;
else //ACK
AckFlag=0;
DAT=I2CReadByte(); //读一字节
SendACK(1);//主机发送非应答
I2CStop();
return DAT;

}
void main()
{
timer0Init();//定时器0初始化
EA=0; //屏蔽中段
At24c02Write(2,9);//写数据(第二个单元送入9这个数据)
delay(5);
num=At24c02Read(2);//读第二单元数据
EA=1; //开中段
while(1);
}
//定时器0中断函数
void timer0() interrupt 1
{
TH0 = 0xED;
TL0 = 0xFF; //定时5ms
display(num); //数码管显示函数
}

  • 2
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
i2ctools是一个用于I2C总线通信的工具包,它提供了一组用于读取和写入I2C设备的命令行工具。其源码主要由C语言编写。 i2ctools的主要功能包括: 1. 通过i2c设备节点文件(如/dev/i2c-0)打开和关闭一个I2C设备连接。 2. 通过发送START、STOP、ACK、NACK等信号控制I2C总线。 3. 发送和接收字节数据,支持读取和写入I2C设备。 4. 支持读取和写入I2C设备寄存器的值。 5. 提供了一些用于设置I2C总线传输速度和其他参数的选项。 6. 支持通过SMbus协议进行I2C通信。 7. 支持对I2C设备进行命名,并提供了一些用于查找和识别I2C设备地址的方法。 i2ctools的源码包括多个文件,其中包括用于解析命令行参数、实现I2C通信协议的函数、用于读写I2C设备寄存器和处理I2C设备连接的函数等。通过阅读源码可以更深入地理解和定制化i2ctools的功能。 在源码中,首先会进行一些初始化操作,如解析命令行参数,打开I2C设备连接等。然后,根据用户的输入执行相应的操作,如读写寄存器、发送数据等。在执行这些操作的过程中,可能会使用到I2C通信协议相关的函数来控制I2C总线的信号传输。 总的来说,i2ctools的源码提供了一套实用的工具和函数,使得我们可以通过命令行快速、方便地进行I2C总线通信和与I2C设备的交互。通过阅读和理解源码,可以更好地进行调试、定制和扩展i2ctools的功能。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

月牙 ,呐

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值