1# I3C<a name="1"></a> 2 3## 概述<a name="section1"></a> 4 5### 功能简介<a name="section2"></a> 6 7I3C(Improved Inter Integrated Circuit)总线是由MIPI Alliance开发的一种简单、低成本的双向二线制同步串行总线。 8 9I3C是两线双向串行总线,针对多个传感器从设备进行了优化,并且一次只能由一个I3C主设备控制。相比于I2C,I3C总线拥有更高的速度、更低的功耗,支持带内中断、从设备热接入以及切换当前主设备,同时向后兼容I2C从设备。I3C增加了带内中断(In-Bind Interrupt)功能,支持I3C设备进行热接入操作,弥补了I2C总线需要额外增加中断线来完成中断的不足。I3C总线上允许同时存在I2C设备、I3C从设备和I3C次级主设备。 10 11I3C接口定义了完成I3C传输的通用方法集合,包括: 12 13- I3C控制器管理:打开或关闭I3C控制器。 14 15- I3C控制器配置:获取或配置I3C控制器参数。 16 17- I3C消息传输:通过消息传输结构体数组进行自定义传输。 18 19- I3C带内中断:请求或释放带内中断。 20 21### 基本概念<a name="section3"></a> 22 23- IBI(In-Band Interrupt)<br> 24 25 带内中断。在SCL线没有启动信号时,I3C从设备可以通过拉低SDA线使主设备发出SCL启动信号,从而发出带内中断请求。若有多个从机同时发出中断请求,I3C主机则通过从机地址进行仲裁,低地址优先相应。 26 27- DAA(Dynamic Address Assignment)<br> 28 29 动态地址分配。I3C支持对从设备地址进行动态分配从而避免地址冲突。在分配动态地址之前,连接到I3C总线上的每个I3C设备都应以两种方式之一来唯一标识: 30 31 1)设备可能有一个符合I2C规范的静态地址,主机可以使用此静态地址; 32 33 2)在任何情况下,设备均应具有48位的临时ID。 除非设备具有静态地址且主机使用静态地址,否则主机应使用此48位临时ID。 34 35- CCC(Common Command Code)<br> 36 37 通用命令代码,所有I3C设备均支持CCC,可以直接将其传输到特定的I3C从设备,也可以同时传输到所有I3C从设备。 38 39- BCR(Bus Characteristic Register)<br> 40 41 总线特性寄存器,每个连接到 I3C 总线的 I3C 设备都应具有相关的只读总线特性寄存器 (BCR),该寄存器描述了I3C兼容设备在动态地址分配和通用命令代码中的作用和功能。 42 43- DCR(Device Characteristic Register)<br> 44 45 设备特性寄存器,连接到 I3C 总线的每个 I3C 设备都应具有相关的只读设备特性寄存器 (DCR)。 该寄存器描述了用于动态地址分配和通用命令代码的 I3C 兼容设备类型(例如,加速度计、陀螺仪等)。 46 47### 运作机制<a name="section4"></a> 48 49在HDF框架中,I3C模块接口适配模式采用统一服务模式,这需要一个设备服务来作为I3C模块的管理器,统一处理外部访问,这会在配置文件中有所体现。统一服务模式适合于同类型设备对象较多的情况,如I3C可能同时具备十几个控制器,采用独立服务模式需要配置更多的设备节点,且服务会占据内存资源。相反,采用统一服务模式可以使用一个设备服务作为管理器,统一处理所有同类型对象的外部访问(这会在配置文件中有所体现),实现便捷管理和节约资源的目的。 50 51相比于I2C,I3C总线拥有更高的速度、更低的功耗,支持带内中断、从设备热接入以及切换当前主设备,同时向后兼容I2C从设备。一路I3C总线上,可以连接多个设备,这些设备可以是I2C从设备、I3C从设备和I3C次级主设备,但只能同时存在一个主设备,一般为控制器本身。 52 53**图 1** I3C物理连线示意图<a name="fig1"></a> 54 55 56### 约束与限制<a name="section5"></a> 57 58I3C模块当前仅支持轻量和小型系统内核(LiteOS-A),不支持在用户态使用。 59 60## 使用指导<a name="section6"></a> 61 62### 场景介绍<a name="section7"></a> 63 64I3C可连接单个或多个I3C、I2C从器件,它主要用于: 65 66- 与传感器通信,如陀螺仪、气压计或支持I3C协议的图像传感器等; 67 68- 通过软件或硬件协议转换,与其他接口(如 UART 串口等)的设备进行通信。 69 70### 接口说明<a name="section8"></a> 71 72I3C模块提供的主要接口如表1所示,具体API详见//drivers/hdf_core/framework/include/platform/i3c_if.h。 73 74**表 1** I3C驱动API接口功能介绍 75 76<a name="table1"></a> 77 78| 接口名 | 接口描述 | 79| ------------- | ----------------- | 80| DevHandle I3cOpen(int16_t number) | 打开I3C控制器 | 81| void I3cClose(DevHandle handle) | 关闭I3C控制器 | 82| int32_t I3cTransfer(DevHandle handle, struct I3cMsg \*msg, int16_t count, enum TransMode mode) | 自定义传输 | 83| int32_t I3cSetConfig(DevHandle handle, struct I3cConfig \*config) | 配置I3C控制器 | 84| int32_t I3cGetConfig(DevHandle handle, struct I3cConfig \*config) | 获取I3C控制器配置 | 85| int32_t I3cRequestIbi(DevHandle handle, uint16_t addr, I3cIbiFunc func, uint32_t payload) | 请求带内中断 | 86| int32_t I3cFreeIbi(DevHandle handle, uint16_t addr) | 释放带内中断 | 87 88> **说明:**<br> 89>本文涉及的所有接口,仅限内核态使用,不支持在用户态使用。 90 91### 开发步骤<a name="section9"></a> 92 93I3C的使用流程如图2所示。 94 95**图 2** I3C使用流程图<a name="fig2"></a> 96 97 98#### 打开I3C控制器<a name="section5"></a> 99 100在进行I3C通信前,首先要调用I3cOpen打开I3C控制器。 101```c 102DevHandle I3cOpen(int16_t number); 103``` 104 105**表 2** I3cOpen参数和返回值描述 106 107<a name="table2"></a> 108 109| 参数 | 参数描述 | 110| ---------- | ------------------- | 111| number | int16_t类型,I3C控制器号 | 112| **返回值** | **返回值描述** | 113| NULL | 打开I3C控制器失败 | 114| 控制器句柄 | 打开的I3C控制器句柄 | 115 116假设系统中存在8个I3C控制器,编号从0到7,以下示例代码为打开1号控制器: 117 118```c 119DevHandle i3cHandle = NULL; // I3C控制器句柄 120 121// 打开I3C控制器 122i3cHandle = I3cOpen(1); 123if (i3cHandle == NULL) { 124 HDF_LOGE("I3cOpen: i3c open fail.\n"); 125 return NULL; 126} 127``` 128 129#### 获取I3C控制器配置<a name="section7"></a> 130 131```c 132int32_t I3cGetConfig(DevHandle handle, struct I3cConfig *config); 133``` 134 135**表 3** I3cGetConfig参数和返回值描述 136 137<a name="table3"></a> 138 139| 参数 | 参数描述 | 140| ---------- | -------------- | 141| handle | DevHandle类型,I3C控制器句柄 | 142| config | 结构体指针,I3C控制器配置 | 143| **返回值** | **返回值描述** | 144| HDF_SUCCESS | 获取成功 | 145| 负数 | 获取失败 | 146 147获取I3C控制器配置示例: 148 149```c 150struct I3cConfig config; 151 152int32_t ret = I3cGetConfig(i3cHandle, &config); 153if (ret != HDF_SUCCESS) { 154 HDF_LOGE("I3cGetConfig: get config fail, ret:%d", ret); 155 return ret; 156} 157``` 158 159#### 配置I3C控制器<a name="section8"></a> 160 161```c 162int32_t I3cSetConfig(DevHandle handle, struct I3cConfig *config); 163``` 164 165**表 4** I3cSetConfig参数和返回值描述 166 167<a name="table4"></a> 168 169| 参数 | 参数描述 | 170| ---------- | -------------- | 171| handle | DevHandle类型,I3C控制器句柄 | 172| config | 结构体指针,I3C控制器配置 | 173| **返回值** | **返回值描述** | 174| HDF_SUCCESS | 配置成功 | 175| 负数 | 配置失败 | 176 177配置I3C控制器示例: 178 179```c 180struct I3cConfig config; 181 182config->busMode = I3C_BUS_HDR_MODE; 183config->curMaster = NULL; 184int32_t ret = I3cSetConfig(i3cHandle, &config); 185if (ret != HDF_SUCCESS) { 186 HDF_LOGE("I3cSetConfig: set config fail, ret:%d", ret); 187 return ret; 188} 189``` 190 191#### 进行I3C通信<a name="section6"></a> 192 193消息传输 194```c 195int32_t I3cTransfer(DevHandle handle, struct I3cMsg *msgs, int16_t count, enum TransMode mode); 196``` 197 198**表 5** I3cTransfer参数和返回值描述 199 200<a name="table5"></a> 201 202| 参数 | 参数描述 | 203| ---------- | -------------------------------------------- | 204| handle | DevHandle类型,I3C控制器句柄 | 205| msgs | 结构体指针,待传输数据的消息结构体数组 | 206| count | int16_t类型,消息数组长度 | 207| mode | 枚举类型,传输模式,0:I2C模式;1:I3C模式;2:发送CCC | 208| **返回值** | **返回值描述** | 209| 正整数 | 成功传输的消息结构体数目 | 210| 负数 | 执行失败 | 211 212I3C传输消息类型为I3cMsg,每个传输消息结构体表示一次读或写,通过一个消息数组,可以执行若干次的读写组合操作。 213 214```c 215int32_t ret; 216uint8_t wbuff[2] = { 0x12, 0x13 }; 217uint8_t rbuff[2] = { 0 }; 218struct I3cMsg msgs[2]; // 自定义传输的消息结构体数组 219msgs[0].buf = wbuff; // 写入的数据 220msgs[0].len = 2; // 写入数据长度为2 221msgs[0].addr = 0x3F; // 写入设备地址为0x3F 222msgs[0].flags = 0; // 传输标记为0,默认为写 223msgs[1].buf = rbuff; // 要读取的数据 224msgs[1].len = 2; // 读取数据长度为2 225msgs[1].addr = 0x3F; // 读取设备地址为0x3F 226msgs[1].flags = I3C_FLAG_READ // I3C_FLAG_READ置位 227// 进行一次I2C模式自定义传输,传输的消息个数为2 228ret = I3cTransfer(i3cHandle, msgs, 2, I2C_MODE); 229if (ret != 2) { 230 HDF_LOGE("I3cTransfer: transfer fail, ret:%d\n", ret); 231 return HDF_FAILURE; 232} 233``` 234 235> **注意:** 236>- I3cMsg结构体中的设备地址不包含读写标志位,读写信息由flags成员变量的读写控制位传递。 237>- 本函数不对消息结构体个数做限制,其最大个数度由具体I3C控制器决定。 238>- 本函数不对每个消息结构体中的数据长度做限制,同样由具体I3C控制器决定。 239>- 本函数可能会引起系统休眠,禁止在中断上下文调用。 240 241#### 请求IBI(带内中断)<a name="section9"></a> 242 243```c 244int32_t I3cRequestIbi(DevHandle handle, uint16_t addr, I3cIbiFunc func, uint32_t payload); 245``` 246 247**表 6** I3cRequestIbi参数和返回值描述 248 249<a name="table6"></a> 250 251| 参数 | 参数描述 | 252| ---------- | -------------- | 253| handle | DevHandle类型,I3C控制器句柄 | 254| addr | uint16_t类型,I3C设备地址 | 255| func | 函数指针,IBI回调函数 | 256| payload | IBI有效载荷 | 257| **返回值** | **返回值描述** | 258| HDF_SUCCESS | 请求成功 | 259| 负数 | 请求失败 | 260 261请求带内中断示例: 262 263```c 264static int32_t TestI3cIbiFunc(DevHandle handle, uint16_t addr, struct I3cIbiData data) 265{ 266 (void)handle; 267 (void)addr; 268 HDF_LOGD("TestI3cIbiFunc: %.16s", (char *)data.buf); 269 270 return HDF_SUCCESS; 271} 272 273int32_t I3cTestRequestIbi(void) 274{ 275 DevHandle i3cHandle = NULL; 276 int32_t ret; 277 278 // 打开I3C控制器 279 i3cHandle = I3cOpen(1); 280 if (i3cHandle == NULL) { 281 HDF_LOGE("I3cOpen: i3c open fail.\n"); 282 return; 283 } 284 ret = I3cRequestIbi(i3cHandle, 0x3F, TestI3cIbiFunc, 16); 285 if (ret != HDF_SUCCESS) { 286 HDF_LOGE("%s: Request IBI failed!", __func__); 287 return ret; 288 } 289 290 I3cClose(i3cHandle); 291 HDF_LOGD("I3cTestRequestIbi: done"); 292 293 return HDF_SUCCESS; 294} 295``` 296 297#### 释放IBI(带内中断)<a name="section10"></a> 298 299```c 300int32_t I3cFreeIbi(DevHandle handle, uint16_t addr); 301``` 302 303**表 7** I3cFreeIbi参数和返回值描述 304 305<a name="table7"></a> 306 307| 参数 | 参数描述 | 308| ---------- | -------------- | 309| handle | DevHandle类型,I3C控制器句柄 | 310| addr | uint16_t类型,I3C设备地址 | 311| **返回值** | **返回值描述** | 312| HDF_SUCCESS | 释放成功 | 313| 负数 | 释放失败 | 314 315释放带内中断示例: 316 317```c 318I3cFreeIbi(i3cHandle, 0x3F); // 释放带内中断 319``` 320 321#### 关闭I3C控制器<a name="section11"></a> 322 323I3C通信完成之后,需要关闭I3C控制器,关闭函数如下所示: 324```c 325void I3cClose(DevHandle handle); 326``` 327 328**表 8** I3cClose参数和返回值描述 329 330<a name="table8"></a> 331 332| 参数 | 参数描述 | 333| ---------- | -------------- | 334| handle | DevHandle类型,I3C控制器句柄 | 335 336关闭I3C控制器实例: 337 338```c 339I3cClose(i3cHandle); // 关闭I3C控制器 340``` 341 342## 使用实例<a name="section10"></a> 343 344本例程以操作Hi3516DV300开发板上的I3C虚拟设备为例,详细展示I3C接口的完整使用流程,基本硬件信息如下。 345 346- SOC:hi3516dv300。 347 348- 虚拟I3C设备:I3C地址为0x3f, 寄存器位宽为1字节。 349 350- 硬件连接:虚拟I3C设备挂接在18号和19号I3C控制器下。 351 352本例程进行简单的I3C传输,测试I3C通路是否正常。 353 354示例如下: 355 356```c 357#include "i3c_if.h" // I3C标准接口头文件 358#include "hdf_log.h" // 标准日志打印头文件 359#include "osal_io.h" // 标准IO读写接口头文件 360#include "osal_time.h" // 标准延迟&睡眠接口头文件 361 362// 定义一个表示设备的结构体,存储信息 363struct TestI3cDevice { 364 uint16_t busNum; // I3C总线号 365 uint16_t addr; // I3C设备地址 366 uint16_t regLen; // 寄存器字节宽度 367 DevHandle i3cHandle; // I3C控制器句柄 368}; 369 370// 基于I3cTransfer方法封装一个寄存器读写的辅助函数,通过flag表示读或写 371static int32_t TestI3cReadWrite(struct TestI3cDevice *testDevice, unsigned int regAddr, 372 unsigned char *regData, unsigned int dataLen, uint8_t flag) 373{ 374 int index = 0; 375 unsigned char regBuf[4] = {0}; 376 struct I3cMsg msgs[2] = {0}; 377 378 // 单双字节寄存器长度适配 379 if (testDevice->regLen == 1) { 380 regBuf[index++] = regAddr & 0xFF; 381 } else { 382 regBuf[index++] = (regAddr >> 8) & 0xFF; 383 regBuf[index++] = regAddr & 0xFF; 384 } 385 386 // 填充I3cMsg消息结构 387 msgs[0].addr = testDevice->addr; 388 msgs[0].flags = 0; // 标记为0,表示写入 389 msgs[0].len = testDevice->regLen; 390 msgs[0].buf = regBuf; 391 392 msgs[1].addr = testDevice->addr; 393 msgs[1].flags = (flag == 1) ? I3C_FLAG_READ : 0; // 添加读标记位,表示读取 394 msgs[1].len = dataLen; 395 msgs[1].buf = regData; 396 397 if (I3cTransfer(testDevice->i3cHandle, msgs, 2, I2C_MODE) != 2) { 398 HDF_LOGE("TestI3cReadWrite: i3c transfer err."); 399 return HDF_FAILURE; 400 } 401 return HDF_SUCCESS; 402} 403 404// 寄存器读函数 405static inline int32_t TestI3cReadReg(struct TestI3cDevice *testDevice, unsigned int regAddr, 406 unsigned char *regData, unsigned int dataLen) 407{ 408 return TestI3cReadWrite(testDevice, regAddr, regData, dataLen, 1); 409} 410 411// 寄存器写函数 412static inline int32_t TestI3cWriteReg(struct TestI3cDevice *testDevice, unsigned int regAddr, 413 unsigned char *regData, unsigned int dataLen) 414{ 415 return TestI3cReadWrite(testDevice, regAddr, regData, dataLen, 0); 416} 417 418// I3C例程总入口 419static int32_t TestCaseI3c(void) 420{ 421 int32_t ret; 422 unsigned char bufWrite[7] = { 0xFF, 0xFF, 0xFF, 0xFF, 0xA, 0xB, 0xC }; 423 unsigned char bufRead[7] = {0}; 424 static struct TestI3cDevice testDevice; 425 426 // 设备信息初始化 427 testDevice.busNum = 18; 428 testDevice.addr = 0x3F; 429 testDevice.regLen = 2; 430 testDevice.i3cHandle = NULL; 431 432 // 打开I3C控制器 433 testDevice.i3cHandle = I3cOpen(testDevice.busNum); 434 if (testDevice.i3cHandle == NULL) { 435 HDF_LOGE("TestCaseI3c: open I3c:%u fail!", testDevice.busNum); 436 return HDF_FAILURE; 437 } 438 439 // 向地址为0x3F的设备连续写7字节数据 440 ret = TestI3cWriteReg(&testDevice, 0x3F, bufWrite, 7); 441 if (ret != HDF_SUCCESS) { 442 HDF_LOGE("TestCaseI3c: test i3c write reg fail, ret:%d", ret); 443 I3cClose(testDevice.i3cHandle); 444 return ret; 445 } 446 OsalMSleep(10); 447 448 // 从地址为0x3F的设备连续读7字节数据 449 ret = TestI3cReadReg(&testDevice, 0x3F, bufRead, 7); 450 if (ret != HDF_SUCCESS) { 451 HDF_LOGE("TestCaseI3c: test i3c read reg fail, ret:%d", ret); 452 I3cClose(testDevice.i3cHandle); 453 return ret; 454 } 455 HDF_LOGD("TestCaseI3c: test i3c write&read reg success!"); 456 HDF_LOGD("TestCaseI3c: function tests end."); 457 // 访问完毕关闭I3C控制器 458 I3cClose(testDevice.i3cHandle); 459 460 return HDF_SUCCESS; 461} 462``` 463