51单片机I2C总线
I2C总线是飞利浦公司推出的一种串行总线,所有器件共用两根信号线,实现数据的传输。
总线接口接了上拉电阻,默认为高电平,所以就可以用“当低电平出现”来标记出一种起始信号。我个人把它想象成:许多人在一条走廊上的不同房间(器件)里,大家都把门打开,连出两根长长的听筒(小时候玩的那种),每个人都从两根大主线上各接一根到自己房间里。两根听筒平时都是安静的(1)。如果有某房间的人叫了一声(0),那剩下的人就知道,我们准备开始通话了。
为了保证秩序,大家选出一个人当领队,由他来主导通话的过程。这就是总线的主机,其他人就是从机。从机有多个,主机只有一个。
两根信号线,一根叫数据线SDA,一根叫时钟线SCL。顾名思义,数据线用来传输数据,时钟线用来管理顺序。
怎么样表示开始,怎么样表示结束,用下面的图表示。注意有严格的时间规定。
首先,我们知道怎么样算是通话的开始和结束(起始信号和终止信号)。然后,规定怎样算是回答“可以”,怎么算是回答“不可以”(应答和非应答)。
接着,我们要知道谁向谁喊话,所以要给每个房间的人都赋予一个名字,也就是地址。再用一个0或1表示方向,从谁到谁。
因此,数据传输的过程,大体就是如此:领队喊出“开始”,说出一个房间名,同时,所有房间的人确认是不是自己的。领队表明目的,说出是向他传数据,还是从他那读数据;然后,确认是自己房间的队员给出应答,可以就开始传输数据。完成后,主机/从机给出应答,表明收到了没有。
下面就是SDA上传送的数据格式。
(a)主机向从机发送数据
S表示起始信号。阴影表示主机发送。A表示应答,上加划线表示非应答。P表示停止。
(b)主机发送数据后,从从机读数据
(c)传输过程中,想改变方向
方法是,重复一次起始信号和从机地址,加一个方向位来改变方向。
SCL是用来管秩序的。只要SCL保持高电平状态,SDA正在传的数据就不能乱动,只有把它拉低以后,SDA才能变化。这样确保数据传输不会乱套,所以在实际的传输过程中,SCL会不断地翻转。
另外,如果存在许多一样的器件,怎么区分呢?方法是把前几位固定不能动,表示是同一种器件,后面的几位可以动(可编程)。假如后面空出3位,那么就是可编程8个,也就是允许有8个同种器件接到总线上。前面的地址叫“器件地址”,后面的地址叫“首地址”。所以,每次主从通信时,要先传器件地址,加方向位,等器件应答;再传首地址来寻找特定的器件,再加方向位,等待它应答。接着开始数据传输。
写入过程:
读出过程:
下面是对于I2C总线模拟的一些关键函数的注释。
//延时10微秒函数
void Delay10us(void)
{
unsigned char a,b;
for (b=1;b>0;b--)
for (a=2;a>0;a--)
;
} //I2C起始信号模拟
void I2cStart()
{
SDA = 1;
Delay10us();
SCL =1;
Delay10us();
SDA = 0;
Delay10us();
SCL =0;
Delay10us(); } //I2C停止信号模拟
void I2cStop()
{
SDA = 0;
Delay10us();
SCL =1;
Delay10us();
SDA = 1;
Delay10us(); } //I2C发送数据函数
unsigned char I2cSendByte(unsigned char dat)
{
unsigned char a = 0,b;
for(a=0;a<8;a++) //一位一位传输数据
{
SDA = dat>>7; //右移7位,最高位送给SDA
dat = dat<<1; //左移一位,次高位变成最高位
Delay10us();
SCL = 1;
Delay10us();
SCL = 0; //翻转SCL,SCL为低电平时传输的数据才能改变
Delay10us();
}
SDA = 1;
Delay10us();
SCL = 1; //释放数据线和时钟线
while(SDA) //等待从机应答,如应答则SDA拉低跳出循环
{
b++; //一段时间没有应答就认定为失败
if(b>200)
{
SCL = 0;
Delay10us();
return 0; //数据发送失败
}
}
SCL = 0;
Delay10us();
return 1; //数据发送成功 } //I2C读取数据函数
unsigned char I2cReadByte()
{
unsigned char a = 0;
SDA = 1; //拉高数据线,保持空闲等待数据
for(a=0;a<8;a++) //一位一位读取数据
{
SCL = 1; //拉高时钟线,保持数据稳定,准备接收
Delay10us();
dat<<=1; //左移一位,空出一位准备读数据
dat |= SDA; //或运算,dat空出的位为0,如SDA也为0则为0,SDA为1就为1,相当于保存SDA数据
Delay10us();
SCL = 0; //翻转时钟线,使下位数据能够改变
Delay10us();
}
return dat; //返回读取的数据
} //向At24C02芯片写数据函数
void At24c02Write(unsigned char addr,unsigned char dat)
{
I2cStart(); //起始信号
I2cSendByte(0xa0); //发送器件地址(固定)
I2cSendByte(addr); //发送首地址(自定)
I2cSendByte(dat); //发送数据
I2cStop(); //停止信号 } //读数据函数
unsigned char At24c02Read(unsigned char addr)
{
unsigned char num;
I2cStart(); //起始信号
I2cSendByte(0xa0); //发送器件地址(固定)
I2cSendByte(addr); //发送首地址(自定) I2cStart(); //加一个起始信号,用于改变数据传送方向
I2cSendByte(0xa1); //读取器件地址,最后一位表示方向
num = I2cReadByte(); //保存读取的数据
I2cStop(); //停止信号 return num;
}
最新文章
- 登录FTP,下载并读取文件内容
- Java总结篇系列:java.lang.Object
- Mockito学习资料
- 【概率】poj 2096:Collecting Bugs
- python偏函数(functool.partail)
- 【HDOJ】2473 Junk-Mail Filter
- angularjs现学现记之—$apply()和$digest()
- NSRunLoop的利用
- JSF页面中使用js函数回调后台action方法
- 工作流引擎Activiti 专题
- vue-cli 构建的项目中 如何使用less
- RMAN-06900 RMAN-06901 ORA-19921
- Linux静态设置CentOS 7虚拟机的IP
- Wireshark抓包分析TCP 3次握手、4次挥手过程
- JavaScript使用注意事项
- Hibernate常见面试题(转)
- Java代码审计连载之—添油加醋
- PocketMoney
- C++ 容器元素的存储和获取
- HighCharts终极版本