1.. _i3c_api:
2
3Improved Inter-Integrated Circuit (I3C) Bus
4###########################################
5
6I3C (Improved Inter-Integrated Circuit) is a two-signal shared
7peripheral interface bus.  Devices on the bus can operate in
8two roles: as a "controller" that initiates transactions and
9controls the clock, or as a "target" that responds to transaction
10commands.
11
12Currently, the API is based on `I3C Specification`_ version 1.1.1.
13
14.. contents::
15    :local:
16    :depth: 2
17
18.. _i3c-controller-api:
19
20I3C Controller API
21******************
22
23Zephyr's I3C controller API is used when an I3C controller controls
24the bus, particularly the start and stop conditions and the clock.
25This is the most common mode, used to interact with I3C target
26devices such as sensors.
27
28Due to the nature of the I3C, there are devices on the bus where
29they may not have addresses when powered on. Therefore, an additional
30dynamic address assignment needs to be carried out by the I3C
31controller. Because of this, the controller needs to maintain
32separate structures to keep track of device status. This can be done
33at build time, for example, by creating arrays of device descriptors
34for both I3C and I\ :sup:`2`\ C devices:
35
36.. code-block:: c
37
38   static struct i3c_device_desc i3c_device_array[] = I3C_DEVICE_ARRAY_DT_INST(inst);
39   static struct i3c_i2c_device_desc i2c_device_array[] = I3C_I2C_DEVICE_ARRAY_DT_INST(inst);
40
41The macros :c:macro:`I3C_DEVICE_ARRAY_DT_INST` and
42:c:macro:`I3C_I2C_DEVICE_ARRAY_DT_INST` are helper macros to aid in
43create arrays of device descriptors corresponding to the devicetree
44nodes under the I3C controller.
45
46Here is a list of generic steps for initializing the I3C
47controller and the I3C bus inside the device driver
48initialization function:
49
50#. Initialize the data structure of the I3C controller device
51   driver instance. The usual device defining macros such as
52   :c:macro:`DEVICE_DT_INST_DEFINE` can be used, and the initialization
53   function provided as a parameter to the macro.
54
55   * The :c:struct:`i3c_addr_slots` and :c:struct:`i3c_dev_list` are
56     structures to aid in address assignments and device list management.
57     If this is being used, this struct needs to be initialized by calling
58     :c:func:`i3c_addr_slots_init`. These two structures can also be used
59     with various helper functions.
60
61   * Initialize the device descriptors if needed by the controller
62     driver.
63
64#. Initialize the hardware, including but not limited to:
65
66   * Setup pin mux and directions.
67
68   * Setup the clock for the controller.
69
70   * Power on the hardware.
71
72   * Configure the hardware (e.g. SCL clock frequency).
73
74#. Perform bus initialization. There is a generic helper function,
75   :c:func:`i3c_bus_init`, which performs the following steps.
76   This function can be used if the controller does not require
77   any special handling during bus initialization.
78
79   #. Do ``RSTDAA`` to reset dynamic addresses of connected devices.
80      If any connected devices have already been assigned an address,
81      the bookkeeping data structures do not have records of these,
82      for example, at power-on. So it is a good idea to reset and
83      assign them new addresses.
84
85   #. Do ``DISEC`` to disable any events from devices.
86
87   #. Do ``SETDASA`` to assign a dynamic address using the static address of the device
88      if so desired.
89
90      * ``SETAASA`` may not be supported for all connected devices
91        to assign static addresses as dynamic addresses.
92
93      * BCR and DCR need to be obtained separately to populate
94        the relevant fields in the I3C target device descriptor
95        struct.
96
97   #. Do ``ENTDAA`` to start dynamic address assignment, if there are
98      still devices without addresses.
99
100      * If there is a device waiting for address, it will send
101        its Provisioned ID, BCR, and DCR back. Match the received
102        Provisioned ID to the list of registered I3C devices.
103
104        * If there is a match, assign an address (either from
105          the stated static address if ``SETDASA`` has not been
106          done, or use a free address).
107
108          * Also, set the BCR and DCR fields in the device descriptor
109            struct.
110
111        * If there is no match, depending on policy, it can be
112          assigned a free address, or the device driver can stop
113          the assignment process and errors out.
114
115          * Note that the I3C API requires device descriptor to
116            function. A device without a device descriptor cannot be
117            accessed through the API.
118
119      * This step can be skipped if there is no connected devices
120        requiring DAA.
121
122   #. These are optional but highly recommended:
123
124      * Do ``GETMRL`` and ``GETMWL`` to get maximum read/write
125        length.
126
127      * Do ``GETMXDS`` to get maximum read/write speed and maximum
128        read turnaround time.
129
130      * The helper function, :c:func:`i3c_bus_init`, would retrieve
131        basic device information such as BCR, DCR, MRL and MWL.
132
133   #. Do ``ENEC`` to re-enable events from devices.
134
135      * The helper function, :c:func:`i3c_bus_init`, only re-enables
136        hot-join events. IBI event should only be enabled when
137        enabling IBI of a device.
138
139In-Band Interrupt (IBI)
140=======================
141
142If a target device can generate In-Band Interrupt (IBI),
143the controller needs to be made aware of it.
144
145* :c:func:`i3c_ibi_enable` to enable IBI of a target device.
146
147  * Some controller hardware have IBI slots which need to be
148    programmed so that the controller can recognize incoming IBIs
149    from a particular target device.
150
151    * If the hardware has IBI slots, :c:func:`i3c_ibi_enable`
152      needs to program those IBI slots.
153
154    * Note that there are usually limited IBI slots on
155      the controller so this operation may fail.
156
157  * The implementation in driver should also send the ``ENEC`` command
158    to enable interrupt of this target device.
159
160* :c:func:`i3c_ibi_disable` to disable IBI of a target device.
161
162  * If controller hardware makes use of IBI slots, this will remove
163    description of the target device from the slots.
164
165  * The implementation in driver should also send the ``DISEC`` command
166    to disable interrupt of this target device.
167
168Device Tree
169===========
170
171Here is an example for defining a I3C controller in device tree:
172
173.. code-block:: devicetree
174
175   i3c0: i3c@10000 {
176           compatible = "vendor,i3c";
177
178           #address-cells = < 0x3 >;
179           #size-cells = < 0x0 >;
180
181           reg = < 0x10000 0x1000 >;
182           interrupts = < 0x1F 0x0 >;
183
184           pinctrl-0 = < &pinmux-i3c >;
185           pinctrl-names = "default";
186
187           i2c-scl-hz = < 400000 >;
188
189           i3c-scl-hz = < 12000000 >;
190
191           status = "okay";
192
193           i3c-dev0: i3c-dev0@420000ABCD12345678 {
194                   compatible = "vendor,i3c-dev";
195
196                   reg = < 0x42 0xABCD 0x12345678 >;
197
198                   status = "okay";
199           };
200
201           i2c-dev0: i2c-dev0@380000000000000050 {
202                   compatible = "vendor-i2c-dev";
203
204                   reg = < 0x38 0x0 0x50 >;
205
206                   status = "okay";
207           };
208   };
209
210I3C Devices
211-----------
212
213For I3C devices, the ``reg`` property has 3 elements:
214
215* The first one is the static address of the device.
216
217  * Can be zero if static address is not used. Address will be
218    assigned during DAA (Dynamic Address Assignment).
219
220  * If non-zero and property ``assigned-address`` is not set,
221    this will be the address of the device after SETDASA
222    (Set Dynamic Address from Static Address) is issued.
223
224* Second element is the upper 16-bit of the Provisioned ID (PID)
225  which contains the manufacturer ID left-shifted by 1. This is
226  the bits 33-47 (zero-based) of the 48-bit Provisioned ID.
227
228* Third element contains the lower 32-bit of the Provisioned ID
229  which is a combination of the part ID (left-shifted by 16,
230  bits 16-31 of the PID) and the instance ID (left-shifted by 12,
231  bits 12-15 of the PID).
232
233Note that the unit-address (the part after ``@``) must match
234the ``reg`` property fully where each element is treated as
23532-bit integer, combining to form a 96-bit integer. This is
236required for properly generating device tree macros.
237
238I\ :sup:`2`\ C Devices
239----------------------
240
241For I\ :sup:`2`\ C devices where the device driver has support for
242working under I3C bus, the device node can be described as
243a child of the I3C controller. If the device driver is written to
244only work with I\ :sup:`2`\ C controllers, define the node under
245the I\ :sup:`2`\ C virtual controller as described below.
246Otherwise, the ``reg`` property, similar to I3C devices,
247has 3 elements:
248
249* The first one is the static address of the device. This must be
250  a valid address as I\ :sup:`2`\ C devices do not support
251  dynamic address assignment.
252
253* Second element is always zero.
254
255  * This is used by various helper macros to determine whether
256    the device tree entry corresponds to a I\ :sup:`2`\ C device.
257
258* Third element is the LVR (Legacy Virtual Register):
259
260  * bit[31:8] are unused.
261
262  * bit[7:5] are the I\ :sup:`2`\ C device index:
263
264    * Index ``0``
265
266      * I3C device has a 50 ns spike filter where it is not
267        affected by high frequency on SCL.
268
269    * Index ``1``
270
271      * I\ :sup:`2`\ C device does not have a 50 ns spike filter but
272        can work with high frequency on SCL.
273
274    * Index ``2``
275
276      * I3C device does not have a 50 ns spike filter and
277        cannot work with high frequency on SCL.
278
279  * bit[4] is the I\ :sup:`2`\ C mode indicator:
280
281    * ``0`` is FM+ mode.
282
283    * ``1`` is FM mode.
284
285Similar to I3C devices, the unit-address must match the ``reg``
286property fully where each element is treated as 32-bit integer,
287combining to form a 96-bit integer.
288
289Device Drivers for I3C Devices
290==============================
291
292All of the transfer functions of I3C controller API require
293the use of device descriptors, :c:struct:`i3c_device_desc`.
294This struct contains runtime information about a I3C device,
295such as, its dynamic address, BCR, DCR, MRL and MWL. Therefore,
296the device driver of a I3C device should grab a pointer to
297this device descriptor from the controller using
298:c:func:`i3c_device_find`. This function takes an ID parameter
299of type :c:struct:`i3c_device_id` for matching. The returned
300pointer can then be used in subsequent API calls to
301the controller.
302
303I\ :sup:`2`\ C Devices under I3C Bus
304====================================
305
306Since I3C is backward compatible with I\ :sup:`2`\ C, the I3C controller
307API can accommodate I2C API calls without modifications if the controller
308device driver implements the I2C API. This has the advantage of using
309existing I2C devices without any modifications to their device drivers.
310However, since the I3C controller API works on device descriptors,
311any calls to I2C API will need to look up the corresponding device
312descriptor from the I2C device address. This adds a bit of processing
313cost to any I2C API calls.
314
315On the other hand, a device driver can be extended to utilize native
316I2C device support via the I3C controller API. During device
317initialization, :c:func:`i3c_i2c_device_find` needs to be called to
318retrieve the pointer to the device descriptor. This pointer can be used
319in subsequent API calls.
320
321Note that, with either methods mentioned above, the devicetree node of
322the I2C device must be declared according to I3C standard:
323
324The I\ :sup:`2`\ C virtual controller device driver provides a way to
325interface I\ :sup:`2`\ C devices on the I3C bus where the associated
326device drivers can be used as-is without modifications. This requires
327adding an intermediate node in the device tree:
328
329.. code-block:: devicetree
330
331   i3c0: i3c@10000 {
332           <... I3C controller related properties ...>
333           <... Nodes of I3C devices, if any ...>
334
335           i2c-dev0: i2c-dev0@420000000000000050 {
336                   compatible = "vendor-i2c-dev";
337
338                   reg = < 0x42 0x0 0x50 >;
339
340                   status = "okay";
341           };
342   };
343
344Configuration Options
345*********************
346
347Related configuration options:
348
349* :kconfig:option:`CONFIG_I3C`
350* :kconfig:option:`CONFIG_I3C_USE_GROUP_ADDR`
351* :kconfig:option:`CONFIG_I3C_USE_IBI`
352* :kconfig:option:`CONFIG_I3C_IBI_MAX_PAYLOAD_SIZE`
353* :kconfig:option:`CONFIG_I3C_CONTROLLER_INIT_PRIORITY`
354
355API Reference
356*************
357
358.. doxygengroup:: i3c_interface
359.. doxygengroup:: i3c_ccc
360.. doxygengroup:: i3c_addresses
361.. doxygengroup:: i3c_target_device
362
363.. _I3C Specification: https://www.mipi.org/specifications/i3c-sensor-specification
364