IIC通信协议

Posted EA7_King

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了IIC通信协议相关的知识,希望对你有一定的参考价值。

1.IIC通信协议简介

I2C 通讯协议(Inter-Integrated Circuit)是由Phiilps公司开发的, 由于它引脚少,硬件实现简单,可扩展性强,不需要USART、CAN等通讯协议的外部收发设备, 被广泛地使用在多个集成电路(IC)间的通讯。

1.1 IIC物理层

它的物理层有如下特点:

  • 它是一个支持多设备的总线。“总线”指多个设备共用的信号线。 在一个I2C通讯总线中,可连接多个I2C通讯设备,支持多个通讯主机及多个通讯从机。

  • 一个I2C总线只使用两条总线线路,一条双向串行数据线(SDA) ,一条串行时钟线 (SCL)。 数据线即用来表示数据,时钟线用于数据收发同步。

  • 每个连接到总线的设备都有一个独立的设备地址,主机可以利用这个地址进行不同设备之间的访问。 其中地址是一个七位或十位的数字。

  • 总线通过上拉电阻接到电源。当I2C设备空闲时,会输出高阻态,而当所有设备都空闲, 都输出高阻态时,由上拉电阻把总线拉成高电平。

  • 多个主机同时使用总线时,为了防止数据冲突,会利用仲裁方式决定由哪个设备占用总线。

  • 具有三种传输模式:标准模式传输速率为100kbit/s ,快速模式为400kbit/s , 高速模式下可达 3.4Mbit/s,但目前大多I2C设备尚不支持高速模式。

  • 连接到相同总线的 IC 数量受到总线的最大电容 400pF 限制 。

1.2 IIC协议层

I2C的协议定义了通讯的起始和停止信号、数据有效性、响应、仲裁、时钟同步和地址广播等环节。

  • IIC 3种基本读写过程

这些图表示的是主机和从机通讯时,SDA线的数据包序列。

1.其中S表示由主机的I2C接口产生的传输起始信号(S),这时连接到I2C总线上的所有从机都会接收到这个信号。

2.起始信号产生后,所有从机就开始等待主机紧接下来广播 的从机地址信号 (SLAVE_ADDRESS)。 在I2C总线上,每个设备的地址都是唯一的,当主机广播的地址与某个设备地址相同时,这个设备就被选中了,没被选中的设备将会忽略之后的数据信号。 根据I2C协议,这个从机地址可以是7位或10位。

3.在地址位之后,是传输方向的选择位,该位为0,表示后面的数据传输方向是由主机传输至从机,即主机向从机写数据。该位为1,则相反,即主机由从机读数据。

4.从机接收到匹配的地址后,主机或从机会返回一个应答(ACK)或非应答(NACK)信号,只有接收到应答信号后,主机才能继续发送或接收数据。

写数据方向:

若配置的方向传输位为 “写数据” 方向,即第一幅图的情况,广播完地址,接收到应答信号后, 主机开始正式向从机 传输数据(DATA) ,数据包的大小为8位,主机每发送完一个字节数据, 都要等待从机的应答信号(ACK),重复这个过程,可以向从机传输N个数据,这个N没有大小限制。 当数据传输结束时,主机向从机发送一个停止传输信号(P),表示不再传输数据。

读数据方向:

若配置的方向传输位为 “读数据” 方向,即第二幅图的情况,广播完地址,接收到应答信号后, 从机开始向主机 返回数据(DATA) ,数据包大小也为8位,从机每发送完一个数据, 都会等待主机的应答信号(ACK),重复这个过程,可以返回N个数据,这个N也没有大小限制。 当主机希望停止接收数据时,就向从机返回一个非应答信号(NACK),则从机自动停止数据传输。

复合格式:

除了基本的读写,I2C通讯更常用的是 复合格式 ,即第三幅图的情况,该传输过程有 两次起始信号(S) 。 一般在第一次传输中,主机通过SLAVE_ADDRESS寻找到从设备后,发送一段“数据”, 这段数据通常用于表示从设备内部的寄存器或存储器地址(注意区分它与SLAVE_ADDRESS的区别); 在第二次的传输中,对该地址的内容进行读或写。也就是说,第一次通讯是告诉从机读写地址,第二次则是读写的实际内容。

通信协议:IIC协议的原理和模拟IIC的实现

1、IIC协议的由来

IIC协议最早是在1982年由飞利浦公司设计开发的,它是一种两线制(SDL + SCL)的串行通行方式,它也是主从机之间通信的方式,在今天也是被广泛的应用在很多的产品设备上。

使用IIC协议进行数据通信的设备,它既可以作为主机又可以作为从机(支持多主多从),并且它是一种半双工的通信方式

另外,IIC协议还是带有总线仲裁功能的一种通信协议!


2、IIC 协议的一些参数

IIC 作为一种通信的协议,它是包含了几个相关的特征参数的,如下所示:

通信协议:IIC协议的原理和模拟IIC的实现_通信协议

注:IIC协议是一种半双工、同步的通信方式!


3、IIC 协议的通信速率

IIC可以支持的通信速率范围较大,可以很好的满足多种设备对于不同的通信速度的要求,常见的IIC支持的速率有以下几个:

1)普通模式(100kHz即100kbps)

2)快速模式(Fm)(400kHz)

3)快速模式+(Fs+)(1MHz)

4)高速模式(Hs)(3.4MHz)

5)超高速模式(UFm)(5MHz)

当然,以上标明的速率一般指的是硬件IIC的速率,对于通过软件模拟实现的IIC,它的速率是受到所使用的CPU的处理速度和性能影响的,不可以一概而论!


4、IIC 协议的接口

IIC 协议的接口有两个:一个是用于时钟同步的时钟线 SCL,另一个是用于数据收发的数据线 SDL。一般我们也称为串行数据线 SDA 和串行时钟线 SCL。

连接到IIC总线上的设备通过这两根线互相传递信息,SDA 和 SCL 都是双向线,可以互相之间进行信息的交互,但是这样的交互是半双工的,同一时刻只能有一个方向进行数据的操作,不能同时进行

IIC设备上的两根通信线一般示意如下:

通信协议:IIC协议的原理和模拟IIC的实现_通信协议_02

重要:一般IIC总线上都会接两个上拉电阻(如上图的R27、R28),阻值一般选择为4.7K,这两个上拉电阻有很大的用处:

1)提高总线的驱动能力

2)让总线在空闲时处于高电平,保证下次能够快速启动传输


5、IIC 协议的时序图

IIC 协议的一次通信过程如下图所示:

通信协议:IIC协议的原理和模拟IIC的实现_通信协议_03

IIC 协议的每部分分解介绍如下:

(1)IIC总线的起始信号

IIC总线发送启动信号的时序如图:

通信协议:IIC协议的原理和模拟IIC的实现_通信协议_04

使用IIC协议进行通信是,首先要发送起始信号。

起始信号是需要时钟线SCL稳定的保持在高电位,SDA由高电位变化为低电位。启动信号发送完成之后就可以进行数据的发送了。

如果在一次通信过程中,有两台以上的设备同时发出了起始信号,都希望获得总线控制权的话,那么第一个发出起始信号的设备将获得总线控制权,作为主设备开始传输数据。这就是IIC的总线仲裁!

注意:起始信号由主机负责产生。


(2)停止信号

IIC总线发送停止信号的时序如图:

通信协议:IIC协议的原理和模拟IIC的实现_通信协议_05

停止信号是在时钟线SCL为高电位的时候,数据线SDA由低电平变化为高电平。停止信号一般是在通信完成之后或者通信失败退出之后发送的。

注意:停止信号由主机负责产生。


(3)数据传输的有效性

数据的传输是在发送完成了启动信号之后便可以进行数据的传输了。

IIC数据传输的协议如下图:

通信协议:IIC协议的原理和模拟IIC的实现_通信协议_06

IIC 协议要求在时钟信号SCL为高电位的期间,数据线SDA上的数据要保持稳定,不能发生变化(上图中1的位置)。只有在时钟信号SCL电位变低的时候,数据线SDA上的电平状态才能发生跳变。

每一个数据的bit位传输需要一个时钟脉冲,一次传输最多是8bit。

一个完整的传输过程的通信时序如图:

通信协议:IIC协议的原理和模拟IIC的实现_通信协议_07

通信开始时,最开始发送的都是地址帧。比如,一个7Bit的地址,首先发出的是最高位,即读写位(1-读,0-写),用于指示当前的通信是读操作还是写操作。


(4)应答(ACK / NACK)

IIC 协议帧的第9位是应答位(ACK/NACK)。所有帧(数据或地址)都是一样的。一旦发送帧的前8位,接收设备就可以控制数据线SDA进行应答。

如果接收设备在第9个时钟脉冲没有将SDA线拉低进行应答,则可能是接收设备没有接收到数据,或者出现错误。在这种情况下,主机需要决定该做什么样的处理(一般考虑重发或者退出)。

注意:SCL时钟信号由主机负责产生。数据的发送是高位先发的!


6、IIC 总线的仲裁

IIC 总线仲裁指的是什么呢?

IIC总线支持多个主机同时在总线上发送数据,但是同一时刻只能有一个主机传送数据。因此必须要通过某些手段来决定哪个主机获得总线的控制权,其它的没有获得主机控制权的设备就只能进行等待,直到获得总线控制权才能进行数据的传输。

IIC总线仲裁的方式有两种:时钟同步、仲裁。它们分别如下:

(1)时钟同步

时钟同步是通过时钟线SCL来实现的。在时钟信号SCL由高到低的切换过程中,IIC器件会开始数自身的低电平周期。当主器件的时钟信号变为低电平的时候,它会使SCL线保持这个电平状态直到达到高电平。假如这个时候有另外一个器件的时钟依然是处于低电平的周期,这个时钟的低到高的变化不会改变SCL线的状态。

因此,SCL线被有着最长的低电平周期的器件占有总线的控制权,而这个时候低电平周期短的器件会进入高电平的等待状态,直到当前的主器件释放总线控制权,自身能够获得总线控制权才会改变这些状态。

时钟同步的时序示意图如下:

通信协议:IIC协议的原理和模拟IIC的实现_通信协议_08

(2)仲裁

仲裁和同步一样,都是为了解决多主机情况下的总线控制冲突。仲裁的过程与从机无关。

只有在总线空闲的时候主机才可以启动传输。两个主机可能在比较短的时间内在总线上同时产生一个有效的起始信号,这种情况下需要仲裁来决定由哪个主机占有总线控制权来完成数据传输。

仲裁是逐位进行,在每一位数据的仲裁期间,当时钟线SCL为高电平时,每个主机都检查数据总线SDA上的电平是否和自己要发送的相同。

这个过程需要持续很多位。理论上讲,如果两个主机所传输的内容完全相同,那么他们能够成功传输而不出现错误。但是,如果一个主机发送高电平但检测到SDA总线上的电平为低时,则认为自己仲裁失败并关闭自己的SDA数据线上的数据传输,而另一个主机则继续完成自己的传输。

IIC总线仲裁的时序示意图如下:

通信协议:IIC协议的原理和模拟IIC的实现_通信协议_09

7、IIC通信的流程

每个IIC设备都通过唯一的器件地址进行识别,根据设备功能,他们既可以是发送器也可作为接收器。通信的流程如下:

1)IIC从机检测到IIC总线上的起始信号之后,就开始从总线上接收地址,之后会把从总线接收到的地址和自身的器件地址(通过软件编程)进行比较,一旦两个地址相同,IIC从机将发送一个确认应答(ACK),并响应总线的后续命令;

2)发送或接收所数据;

3)发送或接收完成之后,在收到应答信号ACK之后结束数据的传输。

此外,如果软件开启了广播呼叫,则IIC从机始终对一个广播地址 (0x00)发送确认应答。I2C模块始终支持7位和10位的地址。

(1)有关地址帧的发送

1)7 位地址的 IIC 通讯流程

通信协议:IIC协议的原理和模拟IIC的实现_通信协议_10

7Bit地址的通信中,开始信号之后的第一帧是地址帧+读写位,刚好是8Bit的数据,直接发送,等待从设备应答之后便可以进行数据的通信。


2)10 位地址的 IIC 通讯流程(主机发送)

通信协议:IIC协议的原理和模拟IIC的实现_通信协议_11

对于10-bit地的址设备,需要使用两个帧来传输10Bit的slave地址。

第一个帧的前5个bit固定为b11110xx,后接slave地址的高2位,第8位仍然是读写(R/W)位,接着是一个ACK位,由于总线上可能有多个10 Bit 从设备地址的高2bit相同,因此这个ACK可能由多有slave设备返回。

第二个帧紧接着第一帧发送,包含slave地址的低8位(7:0),接着该地址的slave回复一个ACK(或NACK)。

注意:10-bit地址的设备和7-bit地址的设备在总线中是可以并存的,因为7-bit地址的高5位不可能是b11110。


3)10 位地址的 I2C 通讯流程(主机接收)

通信协议:IIC协议的原理和模拟IIC的实现_通信协议_12


8、模拟IIC的实现

注意:本文的代码仅用于个人学习使用,请勿擅自用于商业用途!

模拟IIC的实现是使用单片机的IO口模拟IIC的协议时序,实现IIC的通信。既然要使用单片机的IO口进行模拟,所以需要先进行一些定义,如下:

// 头条:嵌入式之入坑笔记

#define IIC_WRITE 0x00 // 从机写入
#define IIC_READ 0X01 // 从机读取
#define IIC_ACK 0 // I2C器件应答,拉低总线

// 设置数据线 SDA -- PB.6
#define IIC_SDA_INPUT() GPIOB->MODER&=~(3<<(6*2));GPIOB->MODER|=0<<(6*2); delay_us(2);
#define IIC_SDA_OUTPUT() GPIOB->MODER&=~(3<<(6*2));GPIOB->MODER|=1<<(6*2); delay_us(2);
#define IIC_SDA_HIGH() GPIOB->BSRRL |= 1<<6 ; delay_us(2);
#define IIC_SDA_LOW() GPIOB->BSRRH |= 1<<6; delay_us(2);
#define IIC_SDA_IO() ( GPIOB->IDR & (1<<6) )

// 设置时钟线 SCL -- PB.7
#define IIC_SCL_INPUT() GPIOB->MODER&=~(3<<(7*2));GPIOB->MODER|=0<<(7*2); delay_us(2);
#define IIC_SCL_OUTPUT() GPIOB->MODER&=~(3<<(7*2));GPIOB->MODER|=1<<(7*2); delay_us(2);
#define IIC_SCL_HIGH() GPIOB->BSRRL |= 1<<7 ; delay_us(2);
#define IIC_SCL_LOW() GPIOB->BSRRH |= 1<<7; delay_us(2);
#define IIC_SCL_IO() ( GPIOB->IDR & (1<<7) )

注:SDA、SCL修改为自己所使用的单片机IO进行设置即可。


(1)发送IIC起始信号

IIC 起始信号的代码实现如下:

// 头条:嵌入式之入坑笔记

void I2C_Start(void)

uint32_t num;
IIC_SDA_OUTPUT();
IIC_SCL_OUTPUT();
IIC_SDA_HIGH();
IIC_SCL_HIGH();
Delay();
num=2000; /* 用于判断IIC从机是否空闲 */
while(num--)

if(IIC_SCL_IO()) /* 根据IIC协议,时钟线拉高空闲 */

break;


Delay();
IIC_SDA_LOW();
Delay();
Delay();
IIC_SCL_LOW();
Delay();


(2)发送IIC停止信号

IIC 停止信号的代码实现如下:

// 头条:嵌入式之入坑笔记

void I2C_Stop()

uint32_t num;
IIC_SDA_LOW();
Delay();
IIC_SCL_LOW();
Delay();
IIC_SCL_HIGH();
Delay();
num=2000;
while(num--)

if(IIC_SCL_IO())

break;


IIC_SDA_HIGH();
Delay();


(3)IIC的应答信号

// 头条:嵌入式之入坑笔记

// IIC发送成功应答
uint32_t I2C_SetACK(void)

uint32_t ack;
uint32_t num;
IIC_SCL_LOW();
Delay();
IIC_SDA_HIGH();
IIC_SDA_OUTPUT();
Delay();
IIC_SDA_LOW();
Delay();
IIC_SCL_HIGH();
num=2000;
while(num--) /* 需要判断IIC时钟是否拉高 */

if(IIC_SCL_IO())

break;


Delay();
IIC_SCL_LOW();
Delay();
return ack;


// IIC发送失败应答
uint32_t I2C_SetNoACK(void)

uint32_t ack,num;
IIC_SCL_LOW();
Delay();
IIC_SDA_HIGH();
IIC_SDA_OUTPUT();
Delay();
IIC_SCL_HIGH();
num = 2000;
while(num--)

if(IIC_SCL_IO()) /* 需要判断IIC时钟是否拉高 */

break;


Delay();
IIC_SCL_LOW();
Delay();
return ack;


// IIC 获取应答信号
uint32_t I2C_GetACK(void)

uint32_t ack;
IIC_SCL_LOW();
Delay();
IIC_SDA_INPUT();
Delay();
IIC_SCL_HIGH();
Delay();
if(IIC_SDA_IO()) //读取SDA的电平状态
ack = 1; //不响应
else
ack = 0; //响应

Delay();
IIC_SCL_LOW();
IIC_SDA_LOW();
IIC_SDA_OUTPUT();
Delay();
return ack;


(4)IIC 读取一个字节

// 头条:嵌入式之入坑笔记

// IIC 读取一个字节
uint8_t I2C_Read(void)

uint8_t t = 8;
uint8_t dat = 0;
uint32_t num;
IIC_SCL_LOW();
IIC_SDA_INPUT();
while(t--)

Delay();
Delay();
IIC_SCL_HIGH();
num = 2000;
while(num--) /* 需要判断IIC时钟是否拉高 */

if(IIC_SCL_IO())
break;

IIC_SCL_HIGH();
Delay();
dat <<= 1;

if( IIC_SDA_IO() )
dat++;
IIC_SCL_LOW();

Delay();
return dat;


(5)IIC 写一个字节

// 头条:嵌入式之入坑笔记

// IIC写一个字节
void I2C_Write(uint8_t dat)

uint8_t t = 8;
uint32_t num;
IIC_SDA_LOW();
IIC_SDA_OUTPUT();
Delay();
while(t--)

if(dat & 0x80) //高位先发 MSB

IIC_SDA_HIGH();

else

IIC_SDA_LOW();

dat <<= 1;
Delay();
IIC_SCL_HIGH();
Delay();
num=2000;
while(num--) /* 需要判断IIC时钟是否拉高 */

if(IIC_SCL_IO() )

break;


Delay();
IIC_SCL_LOW();
Delay();

Delay();



通信协议:IIC协议的原理和模拟IIC的实现_通信协议_13

以上是关于IIC通信协议的主要内容,如果未能解决你的问题,请参考以下文章

IIC通信协议总结

IIC通信协议总结

IIC通信协议总结

IIC通信协议总结

SPI与IIC通信协议比对篇

IIC通信协议