Bootstrap

I2C总线 24C02芯片的读写应用

摘自:http://blog.sina.com.cn/s/blog_62efabea0100r4qq.html

什么是I2C总线?I2C(InterIntegrated Circuit)总线是一种由PHILIPS公司开发的两线式串行总线,用于连接微控制器及其外围设备。也可以简单地理解为I2C是微控制器与外围芯片的一种通讯协议。在不同的书籍中,可能会称为I2CIIC,或者I平方C,但是概念也是一样的,只是叫法不同。

一﹑I2C总线特点

I2C总线的优点非常多,其中最主要体现在 1:硬件结构上具有相同的接口界面;2:电路接口的简单性;3:软件操作的一致性。I2C总线占用芯片的引脚非常的少,只需要两组信号作为通信的协议,一条为数据线(SDA),另一条为时钟线(SCL)。因此减少了电路板的空间和芯片管脚的数量,所以降低了互联成本。总线的长度可高达25英尺,并且能够以10Kbps的最大传输速率支持40个组件。I2C总线还具备了另一个优点,就是任何能够进行发送和接收数据的设备都可以成为主控机。当然,在任何时间点上只能允许有一个主控机。

 

I2C总线 <wbr>24C02芯片的读写应用

 

5-20(总线连接图)

二﹑I2C总线工作原理

图5-20为I2C总线的连接图。I2C总线是由数据线SDA和时钟线SCL构成的串行总线,可发送和接收数据。在单片机与被控IC之间,最高传送速率100kbps。各种I2C器件均并联在这条总线上,就像电话线网络一样不会互相冲突,要互相通信就必须拨通其电话号码,每一个I2C模块都有唯一地址。并接在I2C总线上的模块,既可以是主控器(或被控器),也可以是发送器(或接收器),这取决于它所要完成的功能。I2C总线在传送数据过程中共有四种类型信号,它们分别是:起始信号、停止信号﹑应答信号与非应答信号。

三﹑I2C总线数据的传送规则

起始信号:在I2C总线工作过程中,当SCL为高电平时,SDA由高电平向低电平跳变,定义为起始信号,起始信号由主控机产生。如图5-21所示

I2C总线 <wbr>24C02芯片的读写应用

 

图5-21(开始信号)

停止信号:当SCL为高电平时,SDA由低电平向高电平跳变,定义为停止信号,此信号也只能由主控机产生。如图5-22所示。

 

I2C总线 <wbr>24C02芯片的读写应用

图5-22(停止信号)

应答信号:I2C总线传送的每个字节为8位,受控的器件在接收到8位数据后,在第9个脉冲必须输出低电平作为应答信号,同时,要求主控器在第9个时钟脉冲位上释放SDA线,以便受控器发出应答信号,将SDA拉低,表示接收数据的应答(如图5-23所示)。若果在第9个脉冲收到受控器的非应答信号(如图5-24所示),则表示停止数据的发送或接收。

I2C总线 <wbr>24C02芯片的读写应用

图5-23(应答信号) 5-24(非应答信号)

其次,每启动一次总线,传输的字节数没有限制。主控件和受控器件都可以工作于接收和发送状态。总线必须由主器件控制,也就是说必须由主控器产生时钟信号﹑起始信号﹑停止信号。在时钟信号为高电平期间,数据线上的数据必须保特稳定,数据线上的数据状态仅在时钟为低电平的期间才能改变(如图5-25),而当时钟线为高电平的期间,数据线状态的改变被用来表示起始和停止条件(如图5-21与5-22所示)。

I2C总线 <wbr>24C02芯片的读写应用

图5-25(数据的有效性)

图5-26为总线的完整时序,在这里有一点要加以说明的,当主控器接收数据时,在最后一个数据字节,必须发送一个非应答信号,使受控器释放数据线,以便主控器产生一个停止信号来终止总线的数据传送。

I2C总线 <wbr>24C02芯片的读写应用

图5-26(总线的完整时序)

下面我们来看一下关于I2C总线的读操作与写操作:

I2C总线 <wbr>24C02芯片的读写应用

 图5-27(总线写格式)

写操作就是主控器件向受控器件发送数据,如图5-27所示。首先,主控器会对总线发送起始信号,紧跟应该是第一个字节的8位数据,但是从地址只有7位,所谓从地址就是受控器的地址,而第8位是受控器约定的数据方向位,“0”为写,从图5-26中我们可以清楚地看到发送完一个8位数之后应该是一个受控器的应答信号。应答信号过后就是第二个字节的8位数据,这个数多般是受控器件的寄存器地址,寄存器地址过后就是要发送的数据,当数据发送完后就是一个应答信号,每启动一次总线,传输的字节数没有限制,一个字节地址或数据过后的第9个脉冲是受控器件应答信号,当数据传送完之后由主控器发出停止信号来停止总线。

I2C总线 <wbr>24C02芯片的读写应用 

图5-28(总线读格式)

读操作指受控器件向主控器件发送数,其总线的操作格式如图5-28。首先,由主控器发出起始信号,前两个传送的字节与写操作相同,但是到了第二个字节之后,就要从新启动总线,改变传送数据的方向,前面两个字节数据方向为写,即“0”;第二次启动总线后数据方向为读,即“1”;之后就是要接收的数据。从图5-28的写格式中我们可以看到有两种的应答信号。一种是受控器的,另一种是主控器的。前面三个字节的数据方向均指向受控器件,所以应答信号就由受控器发后出。但是后面要接收的N个数据则是指向主控器件,所以应答信号应由主控器件发出,当N个数据接收完成之后,主控器件应发出一个非应答信号,告知受控器件数据接收完成,不用再发送。最后的停止信号同样也是由主控器发出。

四﹑支持I2C总线的器件

在当今市面上有部分的单片机是内置硬件I2C总线的,用户只需要设置好内部相关的寄存器就可以灵活地运用它。但是如果不内置硬件I2C,我们在使用过程中可以用普通的I/O端口进行模拟。如实验板的AT89S52芯片,如果要用到I2C协议就得要用到软件模拟。

而在非单片机类的芯片当中,如时钟芯片PCF8563,存储器24Cxx系列等都是使用I2C协议进行数据的操作,而我们实验板用的则是24C02。下面我们先来介绍一下24C02的引脚功能和内部结构,然后再介绍如何利用单片机模拟I2C协议与24C02进行数据的传送。

五﹑I2C器件24C02

24C02是一个2K串行CMOS E2PROM,内部含有256个8位字节,该器件通过I2C总线接功能列表。

 

I2C总线 <wbr>24C02芯片的读写应用

图5-29(24Cxx引脚排列)

 

管脚名称

功能

A0﹑ A1 ﹑A2

这三个引脚用于多个器件同时使用时设置区分器件地址,当这些引脚悬空时默认为0。在同一总线中最多可同时使用8个24C02器件。如果总线只

有一个24C02器件被寻址,这三个地址可悬或接地。

SDA

双向串行数据/地址管脚,用于器件所有数据的发送或接收,SDA是一个

开漏输出管脚。

SCL

串行时钟。串行时钟输入管脚用于产生器件所有数据发送或接收的时钟,

这是一个输入管脚

写保护引脚。当WP引脚连接到VCC时芯片里面的内容为只读内容而不能进行写操作。而当WP引脚连接到VSS时芯片里面的内容可进行正常

 

 

 

WP

的读/写操作。

VCC

+1.8V~6.0V工作电压

VSS

 

 

图5-30(24Cxx引脚功能)

六﹑24C02的从地址

 

I2C总线 <wbr>24C02芯片的读写应用

图5-31(24C02从器件地址)

I2C总线 <wbr>24C02芯片的读写应用

图5-32(实验电路)

从图5-31中我们可以看到从地址的高4位是固定的,而低4位是可根据A0﹑ A1 ﹑A2引脚的接法与数据的方向位来确定的,其中R/W为“0”时表示写,为“1”时表示读。而图5-32为实验板电路,A0﹑ A1 ﹑A2均接地,即为0﹑0﹑0。所以在实验板中24C02器件的写从地址为1010 0000(转为十六进制数为0xa0),读从地址为1010 0001(转为十六进制数为0xa1)。

 

下面是测试过可以读取和写入的程序,只供参考、个人保留:

 

#include<reg52.h>

#include<intrins.h>

 

#define uchar unsigned char

#define uint unsigned int

 

sbit SCL=P3^3;

sbit SDA=P3^4;

 

bit ask;

 

void Delay_Us(uchar us)

{

    while(us--)

    {

 

 

    }

}

 

void Delay_Ms(uchar ms)

{

    uchar i;

    while(ms--)

    {

       for(i=0;i<110;i++);

    }

}

 

void I2C_Start()

{

    SDA=1;

    Delay_Us(1);

    SCL=1;

    Delay_Us(1);

    SDA=0;

    Delay_Us(1);

    SCL=0;

    Delay_Us(1);

}

 

void I2C_Stop()

{

    SDA=0;

    Delay_Us(1);

    SCL=1;

    Delay_Us(1);

    SDA=1;

    Delay_Us(1); 

}

 

void Ack()

{

    SDA=1;

    Delay_Us(1);

    SCL=1;

    Delay_Us(1);

    if(SDA==1)

       ask=1;

    else

       ask=0;

    SCL=0;

    Delay_Us(1);

    SDA=1;

    Delay_Us(1);

}

 

void NoAck()

{

    SCL=0;

    SDA=1;

    Delay_Us(1);

    SCL=1;

    Delay_Us(1);

    while(SDA==1);

    Delay_Us(1);

    SCL=0;

    Delay_Us(1);

}

 

void I2C_Write_Byte(uchar c)

{

    uchar i,temp;

    temp=c;

    for(i=0;i<8;i++)

    {

       temp=temp<<1;

       SDA=CY;

       Delay_Us(1);

       SCL=1;

       Delay_Us(1);

       SCL=0;

       Delay_Us(1);

    }

}

 

uchar I2C_Read_Byte()

{

    uchar i,temp;

    SDA=1;

    for(i=0;i<8;i++)

    {

       SCL=1;

       Delay_Us(1);

       temp=temp<<1;

       if(SDA==1)

           temp=temp|0x01;

       SCL=0;

    }

    return temp; 

}

 

void I2C_Ack()

{

    uchar i=0;

    SCL=1;

    Delay_Us(1);

    while((SDA==1)&&(i<200)  )i++;

    SCL=0;

    Delay_Us(1);

}

 

void I2C_Write_Dat(uchar add,uchar subadd,uchar c)

{

    I2C_Start();

    I2C_Write_Byte(add);

    I2C_Ack();

    I2C_Write_Byte(subadd);

    I2C_Ack();

    I2C_Write_Byte(c);

    I2C_Ack();

    I2C_Stop();

}

 

uchar I2C_Read_Dat(uchar add,uchar subadd)

{

    uchar c;

    I2C_Start();

    I2C_Write_Byte(add);

    Ack();

    I2C_Write_Byte(subadd);

    Ack();

   

    I2C_Start();

    I2C_Write_Byte(add+1);

    Ack();

    c=I2C_Read_Byte();

 

    return c;

}

 

void main()

{

    uchar temp,i,c=0x55;

     while(1)

     {

        temp=I2C_Read_Dat(0xa0,i);

       P2=temp;

       i++;

       Delay_Ms(200);

       Delay_Ms(200);

       Delay_Ms(200);

     } 

//     while(1)

//     {

//        I2C_Write_Dat(0xa0,i,c);

//       c=~c;

//       P2=c;

//       i++;

//       Delay_Ms(200);

//       Delay_Ms(200);

//       Delay_Ms(200);

//    }       

}           

;