IIC总线协议

时间:2022-03-26 02:08:48

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

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

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

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

4.有两根数据线:

SDA : IIC数据传送位

SCL : IIC 时钟控制位

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

汇编:

SDA   BIT  P3.            ;数据端
SCL   BIT  P3.            ;时钟端

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

C语言:

sbit SDA = P3.;
sbit SCL = P3.;

void delay(void)       //延时子程序
{
    unsigned char i;
    ;i<;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=;
    delay();
    SDA=;
    delay();
    SCL=;
    delay();
    SDA=;
    delay();
    SCL=;
    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=;
    delay();
    SDA=;
    delay();
    SCL=
    delay();
    SDA=;
    delay();
}

器件寻址:

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

1: 表示从器件进行读;

0: 表示对器件进行写.

应答信号:

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

写操作:(字节模式)

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

汇编:

 ;##############写数据到at24c64-ATW2##############
;说明:写一字节数据到at24c64,失败PSW.5置1
;入口参数:A
;出口参数:无
;###############################################
IIC_WB: 

    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.      ;接收成功
    LJMP edw
J1:
    CLR SCL  ;释放总线
     ;接收失败
edw:  RET

C语言:

void iic_write_byte(unsigned char w_byte)
{
    unsigned char i;
    SCL=;
    ;i<;i++)
    {
        delay();
        if((w_byte<<i)&0x80)
            SDA=;
        else
            SDA=;
        delay();
        SCL=;
        delay();
        SCL=;
    }
    SDA=;
    delay();
    SCL=;  //第九个时钟周期
    delay();
    )  //正常情况下,第九个时钟周期时,收到的SDA应该为低电平
    {           //写入出错
        SCL=;
        PSW.=;  //出错标志
    }
    else
    {             //写入成功
        SCL=;
        PSW.=;  //成功标志
    }
}

读操作:

读操作的初始化方式和写操作一样,仅把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

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

    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 ;
    delay();
    SCL=;
    delay();
    ;i<;i++)
    {
        SCL=;
        delay();
        j<<=;
        )
            j+=;
        delay();
        SCL=;
        delay();
    }
    SDA=;
    delay();
    SCL=;
    delay();
    SCL=;
    delay();
    SDA=;

    return j;
}

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

unsigned char iic_read_last_byte(void)
{
    unsigned ;
    delay();
    SCL=;
    delay();
    ;i<;i++)
    {
        SCL=;
        delay();
        j<<=;
        )
            j+=;
        delay();
        SCL=;
        delay();
    }
    SDA=;
    delay();
    SCL=;    //停止信号
    delay();
    return j;
}

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

1.      原理图

AT24C64器件简介:

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

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

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

STOR1:
          ;写入的字节数
    MOV  ADDRH,#xx    ; 写入到AT24C64的地址字
    MOV  ADDRL,#xx
    LCALL WRDAT     ;将位于R0所指向的地址单元中的COUNT个数据写入
;AT24C64的ADDRH,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中的字符发送
    ,WRDAT
    MOV A,ADDRH
    LCALL IIC_WB
    ,WRDAT
    MOV A,ADDRL
    LCALL IIC_WB       ;存储单元地址
    ,WRDAT
    MOV R7,COUNT
WRDAT0:
    MOV A,@R0
    LCALL IIC_WB        ;写入数据
    ,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)
    )
        ;
    iic_write_byte(addrh);
    )
        ;
    iic_write_byte(addrl);
    )
        ;
    ;i<count;i++)
    {
        iic_write_byte(*buf);
        )
            ;
        buf++;
    }
    iic_stop();
    ;
}

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

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

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

STOR1:
              ;读出入的字节数
        MOV  ADDRH,#xx    ; 读入的AT24C64的地址字
        MOV  ADDRL,#xx
        LCALL LDDAT      ; 从at24c64中读数据

;/*############从at24c64中读数据#########################
; 入口参数:
LDDAT:
        LCALL IIC_ST         ;START
        MOV A,#0A0H                ;对器件进行写
        LCALL IIC_WB        ;虚写操作器件寻址00H(第一片AT24C64)
        ,LDDAT
        MOV A,ADDRH
        LCALL IIC_WB       ;
        ,LDDAT
        MOV A,ADDRL
        LCALL IIC_WB        ;虚写存储单元地址
        ,LDDAT
        ;以上为先初始化一个读数据地址
        LCALL IIC_ST         ;START
        MOV A,#0A1H
        LCALL IIC_WB        ;读操作器件寻址00H(第一片AT24C64)
        ,LDDAT
        MOV R7,COUNT
        CJNE R7,#,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(第一片)
    )
        ;
    iic_write_byte(addrh);                  //虚写存储单元地址
    )
        ;
    iic_write_byte(addrl);                   //虚写存储单元地址
    )
        ;
    //以上为初始化一个读数据地址
    iic_start();                            //重发起始信号
    iic_write_byte(0x0a1);                    //对器件读命令
    )
           ;
    )
    {
        *buf=iic_read_last_byte();
        buf++;
    }
    else
    {
        ;i<count;i++)
        {
            *buf=iic_read_byte();
            buf++;
        }
    }
    iic_stop();
    ;
}