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