搜索
写经验 领红包
 > 运动

电子工程师笔记,I2C知识简介之底层协议

I2C知识简介之底层协议

I2C:IIC,Inter-Integrated Circuit,两线式串行总线,由Philips公司开发;是一种半双工通信方式,传输速度由SCL的时钟频率决定:

标准速度100kbps,

快速模式400kpbs,

高速模式可达3.4Mbit/s;

电子工程师笔记,I2C知识简介之底层协议

I2C协议,从以下6个方面去了解:

1)空闲状态;2)开始信号;3)停止信号;4)应答信号;5)数据的有效性;6)数据传输。

1)空闲状态

SDA、SCL同时处于高电平,各个外围器件的输出级场效应管均处于截止状态,释放总线;

2)起始、停止信号

SCL保持高、SDA出现跳变过程时的信号,如下图示:

电子工程师笔记,I2C知识简介之底层协议

因空闲状态时,SCL、SDA都保持高,所以当SDA发生由高变低这个过程时,即表示开始要有动作发生,故此时为起始信号;反之,当SDA由低变高这个过程发生时,又预示着即将进入空闲状态,故表示停止信号(注:此过程中,SCL一直为高)。

//产生IIC起始信号void IIC_Start(void){ SDA_OUT(); //设置SDA IO为输出 IIC_SDA = 1;  IIC_SCL = 1; delay_us(4); IIC_SDA = 0; //START:when CLK is high,DATA change form high to low  delay_us(4); IIC_SCL = 0; //钳住I2C总线,准备发送或接收数据}

//产生IIC停止信号void IIC_Stop(void){ SDA_OUT(); //设置SDA IO为输出 IIC_SCL = 0; IIC_SDA = 0; //STOP:when CLK is high DATA change form low to high delay_us(4); IIC_SCL = 1;  IIC_SDA = 1; //发送I2C总线结束信号 delay_us(4); }

3)应答信号ACK

发送器每发送完一个字节(8bit),都需要接收器反馈一个应答信号,来表示接收是否成功;若反馈0,则表示应答ACK成功;反之为NACK;

电子工程师笔记,I2C知识简介之底层协议

因发送一个字节需要8个时钟脉冲,所以应答信号发生在第9个时钟脉冲期间;

对于一个有效应答ACK信号的要求:

接收器在第9个脉冲之前的低电平期间,将SDA信号拉低,并且确保在该时钟的高电平期间为稳定的低电平。

//等待应答信号到来//返回值:1,接收应答失败// 0,接收应答成功u8 IIC_Wait_Ack (void){ u8 ucErrTime = 0; SDA_IN(); //SDA设置为输入 IIC_SDA = 1; delay_us(1); //将SDA和SCL都拉高  IIC_SCL = 1; delay_us(1); while (READ_SDA) //等待接收器发出应答信号 { ucErrTime++; if(ucErrTime > 250) { IIC_Stop(); return 1; } } IIC_SCL = 0; //时钟输出0  return 0; }

//产生ACK应答void IIC_Ack(void){ IIC_SCL = 0; SDA_OUT(); IIC_SDA = 0; delay_us(2); IIC_SCL = 1; delay_us(2); IIC_SCL = 0;}//不产生ACK应答 void IIC_NAck(void){ IIC_SCL = 0; SDA_OUT(); IIC_SDA = 1; delay_us(2); IIC_SCL = 1; delay_us(2); IIC_SCL = 0;} 

4)数据有效性

I2C传输数据时,只发生在SCL为高电平期间,且SDA数据必需保持稳定;只有在SCL为低电平期间,才允许SDA数据发生变化;

电子工程师笔记,I2C知识简介之底层协议

5)数据传输

数据位的传输是边沿触发,在SCL时钟的同步控制下,逐位地串行传送每一位数据。

//IIC发送一个字节,返回从机有无应答//1,有应答//0,无应答 void IIC_Send_Byte (u8 txd){  u8 t;  SDA_OUT();  IIC_SCL=0; //接低时钟开始数据传输 for (t=0; t<8; t++) {  if ((txd & 0x80) >> 7) IIC_SDA = 1; else IIC_SDA = 0;  txd <<= 1;  delay_us(2);  IIC_SCL = 1; delay_us(2);  IIC_SCL = 0;  delay_us(2); } } 

//读1个字节,ACK=1时,发送ACK;ACK=0时,发送NACKu8 IIC_Read_Byte (unsigned char ack){ unsigned char i, receive = 0;  SDA_IN(); //SDA设置为输入 for (i=0; i<8; i++ ) { IIC_SCL = 0;  Delay_us(2); IIC_SCL = 1; receive <<= 1; if (READ_SDA) receive++;  Delay_Us(1);  } if (!ack) IIC_NAck(); //发送NACK else IIC_Ack(); //发送ACK  return receive;}