5.1 I2C协议介绍
5.1.1 硬件连接
I2C在硬件上的接法如下所示,主控芯片引出两条线SCL,SDA线,在一条I2C总线上可以接很多I2C设备,我们还会放一个上拉电阻(放一个上拉电阻的原因以后我们再说)。
图5.1 I2C硬件连接
5.1.2 传输数据类比
怎么通过I2C传输数据,我们需要把数据从主设备发送到从设备上去,也需要把数据从从设备传送到主设备上去,数据涉及到双向传输。举个例子:
图5.2 I2C举例示意图
体育老师:可以把球发给学生,也可以把球从学生中接过来。
⚫ 发球:
◼ 老师:开始了(start)
◼ 老师:A!我要发球给你!(地址/方向)
◼ 学生A:到!(回应)
◼ 老师把球发出去(传输)
◼ A收到球之后,应该告诉老师一声(回应)
◼ 老师:结束(停止)
⚫ 接球:
◼ 老师:开始了(start)
◼ 老师:B!把球发给我!(地址/方向)
◼ 学生B:到!
◼ B把球发给老师(传输)
◼ 老师收到球之后,给B说一声,表示收到球了(回应)
◼ 老师:结束(停止)
我们就使用这个简单的例子,来解释一下IIC的传输协议:
⚫ 老师说开始了,表示开始信号(start)
⚫ 老师提醒某个学生要发球,表示发送地址和方向(address/read/write)
⚫ 老师发球/接球,表示数据的传输
⚫ 收到球要回应:回应信号(ACK)
⚫ 老师说结束,表示IIC传输结束(P)
5.1.3 IIC传输数据的格式
1. 写操作
流程如下:
⚫ 主芯片要发出一个start信号
⚫ 然后发出一个设备地址(用来确定是往哪一个芯片写数据),方向(读/写,θ表示写,1表示读)
⚫ 从设备回应(用来确定这个设备是否存在),然后就可以传输数据
⚫ 主设备发送一个字节数据给从设备,并等待回应
⚫ 每传输一字节数据,接收方要有一个回应信号(确定数据是否接受完成),然后再传输下一个数据
⚫ 数据发送完之后,主芯片就会发送一个停止信号
⚫ 下图:白色背景表示"主→从",灰色背景表示"从→主"
图5.3 I2C写传输格式
2. 读操作
流程如下:
⚫ 主芯片要发出一个start信号
⚫ 然后发出一个设备地址(用来确定是往哪一个芯片写数据),方向(读/写,θ表示写,1表示读)
⚫ 从设备回应(用来确定这个设备是否存在),然后就可以传输数据
⚫ 从设备发送一个字节数据给主设备,并等待回应
⚫ 每传输一字节数据,接收方要有一个回应信号(确定数据是否接受完成),然后再传输下一个数据。
⚫ 数据发送完之后,主芯片就会发送一个停止信号。下图:白色背景表示"主→从",灰色背景表示"从→主"
图5.4 I2C读传输格式
3. I2C信号
I2C协议中数据传输的单位是字节,也就是8位。但是要用到9个时钟:前面8个时钟用来传输8数据,第9个时钟用来传输回应信号。传输时,先传输最高位(MSB)。
⚫ 开始信号(S):SCL为高电平时,SDA山高电平向低电平跳变,开始传送数据。
⚫ 结束信号(P):SCL为高电平时,SDA由低电平向高电平跳变,结束传送数据。
⚫ 响应信号(ACK):接收器在接收到8位数据后,在第9个时钟周期,拉低SDA。
⚫ SDA上传输的数据必须在SCL为高电平期间保持稳定,SDA上的数据只能在SCL为低电平期间变化I2C协议信号如下:
图5.5 I2C协议信号
对于IIC协议它只能规定怎么传输数据,数据是什么含义由从设备决定。
5.2 SMBus协议介绍
参考资料:
⚫ Linux内核文档:Documentation\i2c\smbus-protocol.rst
https://smbus.org/specs/SMBus_3_0_20141220.pdf
⚫ SMBus_3_0_20141220.pdf
https://mirrors.edge.kernel.org/pub/software/utils/i2c-tools/
5.2.1 SMBus是I2C协议的一个子集
SMBus: System Management Bus,系统管理总线。SMBus最初的目的是为智能电池、充电电池、其他微控制器之间的通信链路而定义的。SMBus也被用来连接各种设备,包括电源相关设备,系统传感器,EEPROM通讯设备等等。SMBus为系统和电源管理这样的任务提供了一条控制总线,使用SMBus的系统,设备之间发送和接收消息都是通过SMBus,而不是使用单独的控制线,这样可以节省设备的管脚数。SMBus是基于I2C协议的,SMBus要求更严格,SMBus是I2C协议的子集。
图5.6 SMBus是I2C的子集
SMBus有哪些更严格的要求?跟一般的I2C协议有哪些差别?
①VDD的极限值不一样
I2C协议:范围很广,甚至讨论了高达12V的情况
SMBus:1.8V~5V
②最小时钟频率、最大的Clock Stretching
Clock Stretching含义:某个设备需要更多时间进行内部的处理时,它可以把SCL拉低占住I2C总线。对于I2C协议:时钟频率最小值无限制,Clock Stretching时长也没有限制;对于SMBus协议:时钟频率最小值是10KHz,Clock Stretching的最大时间值也有限制。
地址回应(Address Acknowledge):一个I2C设备接收到它的设备地址后,是否必须发出回应信号?对于I2C协议:没有强制要求必须发出回应信号;对于SMBus协议:强制要求必须发出回应信号,这样对方才知道该设备的状态:busy,failed,或是被移除了。
对于数据格式:SMBus协议明确了数据的传输格式;在I2C协议里,它只定义了怎么传输数据,但是并没有定义数据的格式,这完全由设备来定义。
5.2.2 SMBus协议分析
对于I2C协议,它只定义了怎么传输数据,但是并没有定义数据的格式,这完全由设备来定义。
对于SMBus协议,它定义了几种数据格式。
注意:下面文档中的Functionality flag是Linux的某个I2C控制器驱动所支持的功能。比如Functionalityflag:I2C_FUNC_SMBUS_QUICK,表示需要I2C控制器支持SMBus Quick Command
1. symbols(符号)
S (1 bit) : Start bit(开始位)
Sr (1 bit) : 重复的开始位
P (1 bit) : Stop bit(停止位)
R/W# (1 bit) : Read/Write bit. Rd equals 1, Wr equals 0.(读写位)
A, N (1 bit) : Accept and reverse accept bit.(回应位)
Address(7 bits): I2C 7 bit address. Note that this can be expanded as usual to
get a 10 bit I2C address.
(地址位,7 位地址)
Command Code (8 bits): Command byte, a data byte which often selects a register on
the device.
(命令字节,一般用来选择芯片内部的寄存器)
Data Byte (8 bits): A plain data byte. Sometimes, I write DataLow, DataHigh
for 16 bit data.
(数据字节,8 位;如果是 16 位数据的话,用 2 个字节来表示:DataLow、DataHigh)
Count (8 bits): A data byte containing the length of a block operation.
(在 block 操作总,表示数据长度)
[..]: Data sent by I2C device, as opposed to data sent by the host
adapter.
(中括号表示 I2C 设备发送的数据,没有中括号表示 host adapter 发送的数据)
2. SMBus Quick Command
图5.7 2 SMBus Quick Command
只是用来发送一位数据:R/W#本意是用来表示读或写,但是在SMBus里可以用来表示其他含义。比如某些开关设备,可以根据这一位来决定是打开还是关闭。
Functionality flag: I2C_FUNC_SMBUS_QUICK
3. SMBus Receive Byte
图5.8 SMBus Receive Byte
I2C-tools中的函数:i2c_smbus_read_byte()。读取一个字节,Host adapter接收到一个字节后不需要发出回应信号(上图中N表示不回应)。
Functionality flag: I2C_FUNC_SMBUS_READ_BYTE
4. SMBus Send Byte
图5.9 SMBus Send Byte
I2C-tools中的函数:i2c_smbus_write_byte()。发送一个字节。
Functionality flag: I2C_FUNC_SMBUS_WRITE_BYTE
5. SMBus Read Byte
图5.10 5 SMBus Read Byte
I2C-tools中的函数:i2c_smbus_read_byte_data()。先发出Command Code(它一般表示芯片内部的寄存器地址),再读取一个字节的数据。上面介绍的SMBus Receive Byte是不发送Comand,直接读取数据。
Functionality flag: I2C_FUNC_SMBUS_READ_BYTE_DATA
6. SMBus Read Word
图5.11 SMBus Read Word
I2C-tools中的函数:i2c_smbus_read_word_data()。先发出Command Code(它一般表示芯片内部的寄存器地址),再读取2个字节的数据。
Functionality flag: I2C_FUNC_SMBUS_READ_WORD_DATA
7. SMBus Write Byte
图5.12 SMBus Write Byte
I2C-tools中的函数:i2c_smbus_write_byte_data()。先发出Command Code(它一般表示芯片内部的寄存器地址),再发出1个字节的数据。
Functionality flag: I2C_FUNC_SMBUS_WRITE_BYTE_DATA
8. SMBus Write Word
图5.13 8 SMBus Write Word
I2C-tools中的函数:i2c_smbus_write_word_data()。先发出Command Code(它一般表示芯片内部的寄存器地址),再发出1个字节的数据。
Functionality flag: I2C_FUNC_SMBUS_WRITE_WORD_DATA
9. SMBus Block Read
图5.14 9 SMBus Block Read
I2C-tools中的函数:i2c_smbus_read_block_data()。先发出Command Code(它一般表示芯片内部的寄存器地址),再发起度操作:
⚫ 先读到一个字节(Block Count),表示后续要读的字节数
⚫ 然后读取全部数据
Functionality flag: I2C_FUNC_SMBUS_READ_BLOCK_DATA
10. SMBus Block Write
图5.15 10 SMBus Block Write
I2C-tools中的函数:i2c_smbus_write_block_data()。先发出Command Code(它一般表示芯片内部的寄存器地址),再发出1个字节的Byte Conut(表示后续要发出的数据字节数),最后发出全部数据。
Functionality flag: I2C_FUNC_SMBUS_WRITE_BLOCK_DATA
11. I2C Block Read
在一般的I2C协议中,也可以连续读出多个字节。它跟SMBus Block Read的差别在于设备发出的第1个数据不是长度N,如下图所示:
图5.16 11 I2C Block Read
I2C-tools中的函数:i2c_smbus_read_i2c_block_data()。先发出Command Code(它一般表示芯片内部的寄存器地址),再发出1个字节的Byte Conut(表示后续要发出的数据字节数),最后发出全部数据。
Functionality flag: I2C_FUNC_SMBUS_READ_I2C_BLOCK
12. I2C Block Write
在一般的I2C协议中,也可以连续发出多个字节。它跟SMBus Block Write的差别在于发出的第1个数据不是长度N,如下图所示:
图5.17 12 I2C Block Write
I2C-tools中的函数:i2c_smbus_write_i2c_block_data()。先发出Command Code(它一般表示芯片内部的寄存器地址),再发出1个字节的Byte Conut(表示后续要发出的数据字节数),最后发出全部数据。
Functionality flag: I2C_FUNC_SMBUS_WRITE_I2C_BLOCK
13. SMBus Block Write - Block Read Process Call
图5.18 13 SMBus Block Write - Block Read Process Call
先写一块数据,再读一块数据。
Functionality flag: I2C_FUNC_SMBUS_BLOCK_PROC_CALL
14. Packet Error Checking (PEC)
PEC是一种错误校验码,如果使用PEC,那么在P信号之前,数据发送方要发送一个字节的PEC码(它是CRC-8码)。以SMBus Send Byte为例,下图中,一个未使用PEC,另一个使用PEC:
图5.19 Packet Error Checking
5.2.3 SMBus和I2C的建议
因为很多设备都实现了SMBus,而不是更宽泛的I2C协议,所以优先使用SMBus。即使I2C控制器没有实现SMBus,软件方面也是可以使用I2C协议来模拟SMBus。所以:Linux建议优先使用SMBus。
如您在使用瑞萨MCU/MPU产品中有任何问题,可识别下方二维码或复制网址到浏览器中打开,进入瑞萨技术论坛寻找答案或获取在线技术支持。
https://community-ja.renesas.com/zh/forums-groups/mcu-mpu/
未完待续
推荐阅读
文件IO - RZ MPU工业控制教程连载(4)
开发板SDK和工具链-RZ MPU工业控制教程连载(3)
配置桥接网卡-RZ MPU工业控制教程连载(2)