I²C总线接口¶
概要¶
本节教程为大家简要介绍I²C总线协议,并以一个简单的例程教会大家在MicroPython下使用I²C
什么是I²C¶
I²C(Inter-integrated Circuit)最早是飞利浦在1982年开发设计的一种总线协议。I²C总线支持设备之间的短距离通信,用于处理器和一些外围设备之间的接口,它只需要两根信号线来完成信息交换。
通信原理¶
物理接线¶
I²C最少只需要两根线,和异步串口类似,但可以支持多个从(slave)设备,和SPI不同的是,I²C可以支持多主机(mul-master)系统,允许有多个master并且每个master都可以与所有的slaves通信(master之间不可通过I²C通信,并且每个master只能轮流使用I²C总线)。master是指启动数据传输的设备并在总线上生成时钟信号以驱动该传输,而被寻址的通信设备都作为slaves。
I²C通讯只需要2条双向总线:
功能编号 | 含义 |
---|---|
SDA(serial data:串行数据线) | 传输数据,SDA线传输数据是大端传输,每次传输一个字节 |
SCL(serial clock:串行时钟线) | 同步数据收发 |
我们以最为简单的一个主机对应多个从机为例,进行接下来的原理科普。
数据有效性¶
SDA 线上的数据必须在时钟的高电平周期保持稳定,数据线的高或低电平状态只有在 SCL 线的时钟信号是低电平时才能改变。
换言之,SCL为高电平时表示有效数据,SDA为高电平表示“1”,低电平表示“0”;SCL为低电平时表示无效数据,此时SDA会进行电平切换,为下次数据表示做准备。如图13-1为数据有效性的时序图。
起始条件S和停止条件P¶
-
起始条件S:当SCL高电平时,SDA由高电平向低电平转换;
-
停止条件P:当SCL高电平时,SDA由低电平向高电平转换。
起始和停止条件一般由主机产生。总线在起始条件后处于busy的状态,在停止条件的某段时间后,总线才再次处于空闲状态。如下图为起始和停止条件的信号产生时序图。
数据格式¶
I2C传输的数据以字节为单位,每个字节必须为8位,可以传输任意多个字节,上图中以一个字节的数据为例进行分析,I2C的数据格式具有以下特点:
-
每个字节后必须跟一个响应位 ACK(如上图中的SCL上ACK),因此实际上传输一个字节(8位)的数据需要花费9位的时间。
-
SDA上首先传输字节的最高位,从上图中我们可以看出,位数编号的发送顺序从左至右 是
Bit7-Bit0
响应ACK¶
数据接收方收到传输的一个字节数据后,需要给出响应,此时处在第九个时钟,发送端释放SDA线控制权 ,将SDA电平拉高,由接收方控制。
接收方表示成功的接收到了8位一个字节的数据,便将SDA拉低为低电平,即ACK信号,表示应答
PDU¶
当你理解了时序图之后,接下来我们为大家贴出I2C的PDU(Protocol Data Unit:协议数据单元,即数据格式):
以下图片均来自网络
硬件资源¶
ESP32 本身拥有 2 个 I²C 总线接口,根据用户的配置,总线接口可以用作 I²C 主机或从机模式。
然而遗憾的是,MicroPython ESP32上的I²C是软件模拟的,并没有充分利用到ESP32的硬件资源。
理论上来讲,大部分同时支持输入与输出的GPIO都能够被配置为I²C的管脚资源。
I2C API文档¶
类¶
class machine.I2C(scl, sda, freq)¶
scl
: I²C设备时钟引脚对象
sda
: I²C设备数据线引脚对象
freq
: SCL时钟频率 0 <freq
≤ 500000(Hz)
定义I2C¶
示例:
from machine import I2C, Pin I2C = I2C(scl=Pin(5), sda=Pin(4), freq=100000)
常用类函数¶
I2C.init(scl, sda, freq)¶
函数说明:初始化构造I²C总线。
scl
:SCL信号线的I/O口
sda
:SDA信号线的I/O口
freq
:SCL时钟频率
示例:
I2C.init(scl=Pin(5), sda=Pin(4), freq=100000)
I2C.scan()¶
函数说明:扫描0x08到0x77之间的I²C地址,并返回设备列表。
示例:
I2C.scan()
I2C.start()¶
函数说明:在总线上触发START状态(SCL为高电平时,SDA转为低电平)。
示例:
I2C.start()
I2C.stop()¶
函数说明:在总线上触发STOP状态 (SCL为高电平时,SDA转为高电平)。
示例:
I2C.stop()
I2C.write(buf)¶
函数说明:buf中的数据写入到总线,并返回写入的字节数。
buf:存储数据的缓冲区
注意:
使用write()函数时要与start函数一起使用,否则无法返回写入的字节数。
示例:
buf = b'123' I2C.start() I2C.write(buf)
I2C.readinto(buf, nack=True)¶
函数说明:从总线上读取数据并存放到buf,无返回值。
buf:存储数据的缓冲区
注意:
读取的字节数是buf的长度。在接收到最后一个字节之前,总线将发送ACK信号。在接收到最后一个字节后,如果nack为True,那么将发送一个NACK信号,否则将发送一个ACK信号。 示例:
buf=bytearray(3) I2C.readinto(buf)
标准总线操作¶
下面介绍的函数是标准的I²C主模式读写操作。
I2C.readfrom(addr, nbytes)¶
函数说明:从指定地址设备读取数据,返回读取对象,这个对象与I²C设备有关。
addr
:I²C设备地址(可由scan函数读取出来)
nbytes
:要读取数据的大小
示例:
>>> print(I2C.scan()) [24] >>> data = I2C.readfrom(24, 8) >>> print(data) b'\x00\x02\x00\x00\xe8\x03\xe8\x03'
I2C.readfrom_into(addr, buf)¶
函数说明:从指定地址设备读取buf.len()个数据到buf。
addr
:I²C设备地址(可由scan函数读取出来)
buf
:存储数据的缓冲区
示例:
>>> buf = bytearray(8) >>> I2C.readfrom_into(24, buf) >>> print(buf) bytearray(b'\x00\x02\x00\x00\xe8\x03\xe8\x03')
I2C.writeto(addr, buf)¶
函数说明:将buf中的数据写入设备,并返回写入数据的大小。
addr
:I²C设备地址(可由scan函数读取出来)
buf
:存储数据的缓冲区
示例:
>>> b = bytearray(3) >>> b[0] = 24 >>> b[1] = 111 >>> b[2] = 107 >>> i = I2C.writeto(24,b) 3
内存操作¶
某些 I²C 设备作为存储设备 (或一组寄存器) ,可以读取或者写入。这种情况下,有两个地址和 I²C 事务相关: 从设备地址和内存地址。下面方法用于和这些设备进行通信。
I2C.readfrom_mem(addr, memaddr, nbytes, addrsize=8)¶
函数说明:从I²C设备的寄存器中读取并返回数据。
addr
:I²C设备地址(可由scan函数读取出来)
memaddr
:寄存器地址
nbytes
:要读取的字节大小
addrsize
:指定地址大小,默认为8位(在ESP8266上这个参数无效,地址大小总是8位)
示例:
b = I2C. readfrom_mem(24, 0x58, 3) print(b)
运行结果:
b'\x00\x02\x01'
I2C.readfrom_mem_into(addr, memaddr, buf, addrsize=8)¶
函数说明:从I²C设备的寄存器中读取buf.len()个数据到buf,无返回值。
addr
:I²C设备地址(可由scan函数读取出来)
memaddr
:寄存器地址
buf
:存储数据的缓冲区
addrsize
:指定地址大小,默认为8位(在ESP8266上这个参数无效,地址大小总是8位),读取数据数量是buf的长度。
示例:
buf=bytearray(8) I2C.readfrom_mem_into(24, 0x58, buf)
I2C.writeto_mem(addr, memaddr, buf, addrsize=8)¶
函数说明: 将buf 中的数据全部写入到从设备 addr 的内存 memaddr。
addr
:I²C设备地址(可由scan函数读取出来)
memaddr
:寄存器地址
buf
:存储数据的缓冲区
addrsize
:指定地址大小,默认为8位(在ESP8266上这个参数无效,地址大小总是8位),读取数据数量是buf的长度。
示例:
buf = b'123' I2C.writeto_mem(24, 0x58, buf)
综合示例¶
TODO