1、引言
I2C总线是PHILIPS公司推出的一种具备多主机系统所需的包括裁决和高低速设备同步等功能的高性能串行总线。它使用两条线:串行数据线(SDA)和串行时钟线(SCL),使连接到该总线上可访问的器件之间传送信息,属于多主控制总线。总线上的每个器件都有唯一的地址识别,而且都可以作为一个发送器或接收器。由于I2C总线的使用可以简化电路,省掉了很多常规电路中的接口器件,进步产品的可靠性,在很多领域尤其在目前使用的IC卡获得了广泛的应用。
现今,8051系列的单片机应用很广,但是它们都没有I2C总线接口。但是分析I2C总线的时序可知, I2C总线规定其时钟线(SCL线)和和数据线(SDA线)是各设备对应输出状态相“与”的结果,任一设备都可以用输出低电平的方法延长SCL低电平时间,迫使高速设备进进等待状态,实现不同速度设备间的时钟同步。因此,即使时钟脉冲的高、低电平时间是非不一,也能实现数据的可靠传送,因此我们可以通过软件控制单片机的普通I/O口做I2C接口,实现I2C总线功能。本文以ATMEL公司的具有I2C总线的AT24C256为例介绍I2C总线在8051单片机系统中的应用并给出C语言的软件实现。
2、I2C总线构成及时序
2.1 I2C总线组成
I2C总线是一个多主机总线,即可以连接多于一个能控制总线的器件到总线。I2C串行总线有两根信号线:一根双向的数据线SDA;另一根是时钟线SCL。SDA和SCL都是双向I/O口线,当总线空闲时这两条线路都是高电平。所有接到I2C总线上的器件的串行数据线都接到总线的SDA线,各器件的时钟线都接到SCL总线上。其结构如图一所示。
2.2 I2C总线时序
所有主机在SCL线上产生自己的时钟来传输I2C总线上的数据,数据只在时钟的高电平周期有效,每传输一个数据位就产生一个时钟脉冲,数据线的高或低电平状态只有在SCL线的时钟信号是低电平时才能改变,当SCL为高电平时,SDA的改变表示“开始”和“停止”状态,即SDA由高电平转进低电平表示开始,该命令必须在其它命令前执行;SDA由低电平转进高电平表示停止状态,该命令可终止所有通讯。在开始条件后,SCL低电平期间,SDA答应变化,每位数据需一个时钟脉冲,当SCL为高时,SDA必须稳定,主控器在应答时钟脉冲高电平期间开释SDA线,转由接收器控制。相同总线上的设备在收到数据后,以置SDA为低电平的方式对其确认。总线不忙时,数据线和时钟线保持为高电平。I2C的时序如图二所示。
3、8051单片机系统中I2C总线的实现
8051系列单片机上并不带I2C总线,因此必须根据I2C总线的时序用单片机上的I/O口模拟I2C总线时序实现其功能。现以单片机应用系统中较为常见的E2PROM中AT24C256为例,介绍在8051上利用普通I/O口实现I2C串行总线的方法和软件设计。
AT24C256是ATMEL公司256kbit串行电可擦的可编程存储器,8引脚双排直插式封装,具有结构紧凑、存储容量大等特点,其引脚排列如图三所示。
其中A0、A1为地址选择输进端。A0、A1用来区分各芯片地址,地址分布从0到3。A0、A1悬空时为0。SCL为串行时钟输进,上升沿将SDA上的数据写进存储器,下降沿从存储器读出数据送SDA上。SDA是双向串行数据输进输出口,用于存储器与单片机之间的数据交换。WP为写保护输进。此引脚与地相连时,答应写操纵;与VCC相连时,所有的写存储器操纵被禁止。假如不连,芯片内部下拉到地;VCC接+5V电源;GND接地;NC悬空。
AT24C256在开始状态后需紧接一个8位器件地址,以进行相应的读写操纵。设备寻址码的高4位为1、0、1、0,寻址码高4位后面的三位是器件寻址码,与它们的硬连线管脚相对应。最低应是读写选择位,置0时可激发写操纵,置1激发读操纵。器件的具体的格式为:1010A2A1A0R/W,其中当R/W为0时,为写AT24C256,当为1时为读操纵。
由于AT24C256片内地址是以8位为单元的,其总存储容量为256K,所以其片内地址为从0到32767的任意数值。因此在编写软件时设置其片内地址数据类型时要设置为unsigned int型,而不能为unsigned char型。
图四所示为AT89S51单片机和AT24C256的接口图,分别用I/O口P1.6、P1.7连接SDA和SCL总线。通过控制P1.7口的高低电平得到I2C的虚拟时钟,P1.6口作为数据传输的双向端口,作为单片机和AT24C256之间的数据线。
4、虚拟I2C总线软件结构
对于虚拟I2C总线来说要实现其功能主要是要软件要留意I2C器件的操纵时序,总线的运行由主机控制。所谓主机即启动数据的传送即发出启动信号,发出时钟信号,传送结束时发出停止信号的设备,通常主机是微处理器。被主机寻访的设备都称为从机。为了进行通讯,每个接到I2C总线的设备都有一个唯一的地址,以便于主机寻访。主机和从机的数据传送,可以由主机发送数据到从机,也可以是从机发到主机。凡是发送数据到总线的设备称为发送器,从总线上接收数据的设备被称为接受器。根据I2C的时序,软件分为开始、停止、应答、检查应答、不产生应答、读操纵、写操纵。以下为C程序清单调试成功。
#include
Sbit sda=P1^7;
Sbit scl=P1^6;
启动函数,在SCL为高时,SDA的下降沿为启动信号。
void Start(void)
{ scl=0; //SCL处于低电平时,SDA才能改变
sda=1;// 一个"开始"状态,该状态必须在其他命令之前执行
scl=1; // 当scl为高电平时sda的下降沿表示开始状态
_nop_(); _nop_(); _nop_(); //给一个延时
sda=0; //给下降沿表示开始
_nop_(); _nop_(); _nop_(); _nop_();
scl=0; //恢复低电平以改变sda的值
sda=1; }
停止函数,在SCL为高时,SDA的上升沿为停止信号。
void Stop(void)
{scl=0; //SCL处于低电平时,SDA改变数值 */
sda=0; //scl为高电平时,sda的上升沿表示停止,
scl=1; //scl为高电平时改变sda的状态表示启动,停止
_nop_(); _nop_(); _nop_(); // 延时
sda=1;
_nop_(); _nop_(); _nop_();
scl=0; }
/* * * * * 检查应答位 * * * * */
bit RecAck(void)
{ scl=0; //在scl为0的时候改变sda的值
sda=1;
scl=1; //在scl为1的时候等待sda值的变化,在器件接受到数据后会把sda拉低。
_nop_(); _nop_(); _nop_(); _nop_();
CY=sda; // 由于返回值总是放在CY中的
scl=0;
_nop_();
return(CY); //假如为CY为低则表示接受成功,假如为高,则表示接受失败。
}
/* * * * *对I2C总线产生应答(一般用在读操纵中) * * * * */
void Ack(void)
{
sda=0;// EEPROM通过在收到每个地址或数据之后,
_nop_();_nop_();
scl=1;//置SDA低电平的方式确认表示收到读SDA口状态
_nop_(); _nop_(); _nop_(); _nop_();
scl=0;
_nop_();
sda=1;
}
/* * * * * * * * * 不对I2C总线产生应答 * * * * */
void NoAck(void)
{ sda=1;
scl=1;
_nop_(); _nop_(); _nop_(); _nop_();
scl=0;
}
向I2C总线写数据,每次写8位数据。
void Send(uchar sendbyte)
{uchar j=8;
for(;j>0;j--)
{scl=0;//拉低scl预备给上升沿
delay(5);//延时
sendbyte <<= 1;// 使CY=sendbyte^7;
sda=CY; // CY 进位标志位
scl=1; //给上升沿,发出sda的状态值
}
scl=0;
}
从I2C总线上读数据子程序 ,每次读8位数据。
uchar Receive(void)
{ register receivebyte,i=8;
scl=0;
while(i--)
{ scl=1; //拉高scl预备给下降沿
receivebyte=(receivebyte <<1)|sda; //接受值左移一位把低位和sda相或得到sda的状态值
scl=0; //给下降沿发出sda的状态值
}
return(receivebyte);
}
5、结论
本文介绍了I2C总线的组成及时序,并以ATMEL公司的AT24C256为例给出了用8051C语言模拟I2C总线的时序的起始、停止及CPU向I2C总线的发送和接收8位字节的程序。