IIC总线协议---以存储芯片at24c64为例

时间:2021-03-26 00:13:49

IIC总线协议

 

前言:年前给老师做个红外抄表系统,,现在对当中用到的一些模块总结一下.

 

1.只有在总线空闲时才允许启动数据传送.

2.在数据传送过程中,当时钟线为高电平时,数据线必须保持稳定状态,不允许有跳变.时钟线为高电平时,数据线的任何电平变化将被看做总线的起始或停止信号.

3. 任何将数据传送到总线的器件作为发送器任何从总线接收数据的器件为接收器, 主器件和从器件都可以作为发送器或接收器但由主器件控制传送数据.

4.有两根数据线:

SDA : IIC数据传送位

SCL : IIC 时钟控制位

下面对IIC的底层驱动加以说明,并给出汇编和C语言的代码,AT24C64为例.在这之前先定义一些常量和公共代码:

汇编:

    SDA   BIT  P3.4            ;数据端

       SCL   BIT  P3.5            ;时钟端

 

       DELAY : NOP        ;延时子程序,根据单片机的晶振不同,延时长短略有不同

                     NOP        ;12MHZ晶振

                     NOP

                     NOP

                     NOP

                     NOP

                     RET

C语言:

  sbit SDA = P3.4;

       sbit SCL = P3.5;

 

       void delay(void)       //延时子程序

              {

                     unsigned char i;

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

              }

                    

起始信号:

       时钟线保持高电平期间,数据线电平从高到低的跳变作为IIC总线的起始信号.

汇编:

       IIC_ST:                ;IIC起始代码

LCALL DELAY

           CLR SCL

LCALL DELAY

        LCALL DELAY

        SETB SDA

LCALL DELAY

        LCALL DELAY

        SETB SCL

LCALL DELAY

        LCALL DELAY

        CLR SDA

LCALL DELAY

LCALL DELAY

        CLR SCL

LCALL DELAY

        RET

 

C语言:

       void iic_start(void)

       {

           SCL=0;

        delay();

        SDA=1;

        delay();

        SCL=1;

        delay();

        SDA=0;

        delay();

        SCL=0;

        delay();

       }

       

停止信号:

时钟线保持高电平期间,数据线电平从低变高.

       汇编:

              IIC_SP:

LCALL DELAY

               CLR SCL

LCALL DELAY

LCALL DELAY

                     CLR SDA

LCALL DELAY

LCALL DELAY

                     SETB SCL

LCALL DELAY

LCALL DELAY

                     SETB SDA

LCALL DELAY

                     RET

 

       C语言:

              void iic_stop(void)

              {

                     delay();

               SCL=0;

               delay();

                  SDA=0;

               delay();

                     SCL=1

               delay();

                     SDA=1;

               delay();

              }

      

器件寻址:

       首先发送一个起始信号,启动发送过程,然后发送它所需要的寻址的从器件的地址.8从位器件地址的高4位固定为1010,接下来的三位为器件的地址位,最低一位作为读写控制位.

1: 表示从器件进行读;

0: 表示对器件进行写.

 

应答信号:

       IIC总线数据传送时,每成功的传送一个字节数据后,接收器都必须产生一个应答信号.应答的器件在第九个时钟周期时将SDA拉低,表示收到一个8位数据.

 

写操作:(字节模式)

       主器件发送起始信号和从器件地址信息(R/W位清0)给从器件,在从器件送回应答信号后,主器件发送两个8位地址字写入从器件的地址指针,主器件在收到从器件的应答信号后,再发送数据到到被寻址的从器件存储单元,从器件再次应答,并在主器件产生停止信号后开始内部数据的擦写.

 

汇编:

       ;##############写数据到at24c64-ATW2##############

;说明:写一字节数据到at24c64,失败PSW.51

;入口参数:A

;出口参数:

;###############################################

IIC_WB: 

MOV R2,#8

        CLR SCL

J4:     RLC A

LCALL DELAY

LCALL DELAY

              MOV SDA,C

LCALL DELAY

LCALL DELAY

              SETB SCL

LCALL DELAY

LCALL DELAY

              CLR SCL

              DJNZ R2,J4

        SETB SDA

LCALL DELAY

LCALL DELAY

              SETB SCL     ;第九个时钟周期

LCALL DELAY

LCALL DELAY

              JB SDA,J1     ;正常情况下,第九个时钟周期收到的SDA应为低

              CLR SCL      ;释放总线

LCALL DELAY

LCALL DELAY

              CLR PSW.5      ;接收成功

              LJMP edw

J1: CLR SCL  ;释放总线

SETB PSW.5 ;接收失败

edw:  RET

 

C语言:

  void iic_write_byte(unsigned char w_byte)

  {

         unsigned char i;

         SCL=0;

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

         {

                delay();

                if((w_byte<<i)&0x80)

                       SDA=1;

                else

                       SDA=0;

                delay();

                SCL=1;

                delay();

                SCL=0;

         }

         SDA=1;

         delay();

         SCL=1;  //第九个时钟周期

         delay();

         if(SDA==1)  //正常情况下,第九个时钟周期时,收到的SDA应该为低电平

         {           //写入出错

                SCL=0;

                PSW.5=1;  //出错标志

         }

         else

         {             //写入成功

                SCL=0;

                PSW.5=0;  //成功标志

         }

  }

 

读操作:

       读操作的初始化方式和写操作一样,仅把R/W位置1.

       读操作的种类:

1.       立即/当前地址读(本例未用)

2.       选择/随机读: 允许对任意字节进行读,首先发送起始信号,从器件地址和它想读的字节数数据地址,执行一个伪写操作,AT24C64应答之后,主器件重新发送起始信号和从器件地址,此时R/W位置1.AT24C64响应并发送应答信号,然后输出所要求的一个8位字节数据.主器件不发送应答信号但产生一个停止信号.

汇编1:

;/** 单字节读 **/

;读出的一个字节存放在A

IIC_RB:  

CLR A

              CLR C

LCALL DELAY

LCALL DELAY

              CLR SCL

LCALL DELAY

LCALL DELAY

        MOV R2,#8

J5:   SETB SCL

LCALL DELAY

LCALL DELAY

              MOV C,SDA

              RLC A

LCALL DELAY

LCALL DELAY

              CLR SCL

LCALL DELAY

LCALL DELAY

              DJNZ R2,J5

              CLR SDA

LCALL DELAY

LCALL DELAY

              SETB SCL   ;第九个时钟周期,应将SDA拉低作为应答信号

LCALL DELAY

LCALL DELAY

              CLR SCL   ;释放总线

LCALL DELAY

LCALL DELAY

        SETB SDA

        RET

 

汇编2:

;################序列读最后一字节###############

;读最后一字节时需要发送停止信号

IIC_RL:  

CLR A

              CLR C

LCALL DELAY

LCALL DELAY

              CLR SCL

LCALL DELAY

LCALL DELAY

        MOV R2,#8

J50: SETB SCL

LCALL DELAY

LCALL DELAY

 

              MOV C,SDA

              RLC A

LCALL DELAY

LCALL DELAY

              CLR SCL

LCALL DELAY

LCALL DELAY

              DJNZ R2,J50

              SETB SDA

LCALL DELAY

LCALL DELAY

              SETB SCL                ;停止信号

LCALL DELAY

LCALL DELAY

        RET

 

C语言1: 单字节读

       unsigned char iic_read_byte(void)

       {

              unsigned char i,j=0;

              delay();

              SCL=0;

              delay();

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

              {

                     SCL=1;

                     delay();

                     j<<=1;

                     if(SDA==1)

                            j+=1;

                     delay();

                     SCL=0;

                     delay();

              }

              SDA=0;

              delay();

              SCL=1;

              delay();

              SCL=0;

delay();

SDA=1;

             

return j;

       }

 

C语言2: 序列读最后一字节

       unsigned char iic_read_last_byte(void)

       {

              unsigned char I,j=0;

              delay();

              SCL=0;

              delay();

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

              {

                     SCL=1;

                     delay();

                     j<<=1;

                     if(SDA==1)

                            j+=1;

                     delay();

                     SCL=0;

                     delay();

              }

              SDA=1;

              delay();

              SCL=1;    //停止信号

              delay();

              return j;

       }

 

举例: AT公司生产的IIC器件EEPROM: at24c64为例,对该器件进行读写

1.      原理图

 

AT24C64器件简介:

2.将位于R0(buf)所指向的地址单元中的COUNT(count)个数据写入AT24C64ADDRH,ADDRL(addrh,dddrl)地址单元中.(括号内为c语言变量)

 

汇编:(会用到上面给出的IIC驱动子程序以及定义的一些变量和接口)

 

ADDRH   EQU 13H         

ADDRL   EQU 14H           ;定义读写AT24C64的首地址

 

STOR1:  MOV COUNT,#4      ;写入的字节数

               MOV  ADDRH,#xx    ; 写入到AT24C64的地址字

               MOV  ADDRL,#xx

         LCALL WRDAT     ;将位于R0所指向的地址单元中的COUNT个数据写入

;AT24C64ADDRH,ADDRL地址单元中

 

;###########所有参数写入AT24C64,4字节#################

;说明:将表号和用户电量共四字节数据写入AT24C64

;入口参数:

;    1.数据间接寻址地址-R0

;    2.写入到AT24C64的地址字-ADDRH,ADDRL

;    3.写入字节数-COUNT

;出口参数:

;#######################################################

 

WRDAT:  LCALL IIC_ST         ;START

       MOV A,#0A0H

       LCALL IIC_WB        ;写器件寻址00H(第一片AT24C64)  A中的字符发送

       JB PSW.5,WRDAT

       MOV A,ADDRH

        LCALL IIC_WB

       JB PSW.5,WRDAT

       MOV A,ADDRL

       LCALL IIC_WB       ;存储单元地址

       JB PSW.5,WRDAT

     MOV R7,COUNT

WRDAT0: MOV A,@R0

       LCALL IIC_WB        ;写入数据

       JB PSW.5,WRDAT

        INC R0

        DJNZ R7,WRDAT0

        LCALL IIC_SP     ;停止

        LCALL DL20MS   ;延时,等待将数据擦写到at24c64

        RET

 

C语言:

 

unsigned char count=0x04;

unsigned char addrh=0xxx;

unsigned char addrl=0xxx;

unsigned char t_buf[];

 

while(!write_byte(t_buf,addrh,addrl,count));

 

/*###########所有参数写入AT24C64,4字节#################

//说明:将表号和用户电量共四字节数据写入AT24C64

//入口参数:

;    1.数据间接寻址地址-buf

;    2.写入到AT24C64的地址字-addh,addrl

;    3.写入字节数-count

;出口参数:1表示写成功,0表示写失败

;#######################################################*/

bit write_byte(unsigned char * buf,             //

            unsigned char addrh,

                     unsigned char addrl,

            unsigned char count)

{

       unsigned char i;

       iic_start();             //start

       iic_write_byte(0x0a0);          //写器件寻址0x00(第一片at24c64)

       if(PSW.5)

              return 0;

       iic_write_byte(addrh);

       if(PSW.5)

              return 0;

       iic_write_byte(addrl);

       if(PSW.5)

              return 0;

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

{

              iic_write_byte(*buf);

              if(PSW.5)

                     return 0;

              buf++;

       }

       iic_stop();

       return 1;

}

 

2.      at24c64ADDRH,ADDRL(addrh,addrl)地址单元中读出COUNT(count)个数据存放到单片机R0(buf)指定的地址单元中.(括号中为C语言变量)

 

汇编: (会用到上面给出的IIC驱动子程序以及定义的一些变量和接口)

ADDRH   EQU 13H         

ADDRL   EQU 14H           ;定义读写AT24C64的首地址

 

STOR1:  MOV COUNT,#4      ;读出入的字节数

               MOV  ADDRH,#xx    ; 读入的AT24C64的地址字

               MOV  ADDRL,#xx

         LCALL LDDAT      ; at24c64中读数据

 

;/*############at24c64中读数据#########################

; 入口参数:

LDDAT:  LCALL IIC_ST         ;START

       MOV A,#0A0H                ;对器件进行写

       LCALL IIC_WB        ;虚写操作器件寻址00H(第一片AT24C64)

       JB PSW.5,LDDAT

       MOV A,ADDRH

        LCALL IIC_WB       ;

       JB PSW.5,LDDAT

       MOV A,ADDRL

        LCALL IIC_WB        ;虚写存储单元地址

       JB PSW.5,LDDAT

        ;以上为先初始化一个读数据地址

        LCALL IIC_ST         ;START

       MOV A,#0A1H

       LCALL IIC_WB        ;读操作器件寻址00H(第一片AT24C64)

       JB PSW.5,LDDAT

        MOV R7,COUNT

        CJNE R7,#01,LD01

        LJMP LDDAT1

LD01:   DEC R7

LDDAT0: LCALL IIC_RB        ;现行地址读(0000H)

        MOV @R0,A

        INC R0

        DJNZ R7,LDDAT0

LDDAT1: LCALL IIC_RL            ;读最后一字节,读完后给AT24C64发停止信号

        MOV @R0,A

        inc r0

        LCALL IIC_SP                    ;停止信号

        RET

 

 

 

C语言:

unsigned char count=0x04;

unsigned char addrh=0xxx;

unsigned char addrl=0xxx;

unsigned char r_buf[];

while(!read_byte(r_buf,addrh,addrl,count));

/*            */

bit read_byte(unsigned char *buf,        //存放读到的数据

                 unsigned char addrh,

                 unsigned char addrl,        //要读的数据地址

                 unsigned char count)              //需要读的个数

{

       unsigned char i;

       iic_start();

       iic_write_byte(0x0a0);                 //对器件进行寻址,虚写操作器件 寻址00H(第一片)

       if(PSW.5)

              return 0;

       iic_write_byte(addrh);                  //虚写存储单元地址

       if(PSW.5)

              return 0;

       iic_write_byte(addrl);                   //虚写存储单元地址

       if(PSW.5)

              return 0;

//以上为初始化一个读数据地址

       iic_start();                           //重发起始信号

       iic_write_byte(0x0a1);        //对器件读命令

       if(PSW.5)

              return 0;

       if(count==1)

       {

*buf=iic_read_last_byte();

              buf++;

       }

       else

       {

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

              {

                     *buf=iic_read_byte();

                     buf++;

              }

       }

       iic_stop();

       return 1;

}