xref: /openbmc/qemu/include/hw/i3c/dw-i3c.h (revision 9c0476a21da3b8b027399f93be86e8c06c92590c)
1c52aaabdSJoe Komlodi /*
2c52aaabdSJoe Komlodi  * DesignWare I3C Controller
3c52aaabdSJoe Komlodi  *
4c52aaabdSJoe Komlodi  * Copyright (C) 2021 ASPEED Technology Inc.
5c52aaabdSJoe Komlodi  * Copyright (C) 2025 Google, LLC.
6c52aaabdSJoe Komlodi  *
7c52aaabdSJoe Komlodi  * SPDX-License-Identifier: GPL-2.0-or-later
8c52aaabdSJoe Komlodi  */
9c52aaabdSJoe Komlodi 
10c52aaabdSJoe Komlodi #ifndef DW_I3C_H
11c52aaabdSJoe Komlodi #define DW_I3C_H
12c52aaabdSJoe Komlodi 
13*9c0476a2SJoe Komlodi #include "qemu/fifo32.h"
14*9c0476a2SJoe Komlodi #include "hw/i3c/i3c.h"
15c52aaabdSJoe Komlodi #include "hw/sysbus.h"
16c52aaabdSJoe Komlodi 
17c52aaabdSJoe Komlodi #define TYPE_DW_I3C "dw.i3c"
18c52aaabdSJoe Komlodi OBJECT_DECLARE_SIMPLE_TYPE(DWI3C, DW_I3C)
19c52aaabdSJoe Komlodi 
20*9c0476a2SJoe Komlodi /*
21*9c0476a2SJoe Komlodi  * Sufficiently large enough to handle configurations with large device address
22*9c0476a2SJoe Komlodi  * tables.
23*9c0476a2SJoe Komlodi  */
24*9c0476a2SJoe Komlodi #define DW_I3C_NR_REGS (0x1000 >> 2)
25*9c0476a2SJoe Komlodi 
26*9c0476a2SJoe Komlodi /* From datasheet. */
27*9c0476a2SJoe Komlodi #define DW_I3C_CMD_ATTR_TRANSFER_CMD 0
28*9c0476a2SJoe Komlodi #define DW_I3C_CMD_ATTR_TRANSFER_ARG 1
29*9c0476a2SJoe Komlodi #define DW_I3C_CMD_ATTR_SHORT_DATA_ARG 2
30*9c0476a2SJoe Komlodi #define DW_I3C_CMD_ATTR_ADDR_ASSIGN_CMD 3
31*9c0476a2SJoe Komlodi 
32*9c0476a2SJoe Komlodi /* Enum values from datasheet. */
33*9c0476a2SJoe Komlodi typedef enum DWI3CRespQueueErr {
34*9c0476a2SJoe Komlodi     DW_I3C_RESP_QUEUE_ERR_NONE = 0,
35*9c0476a2SJoe Komlodi     DW_I3C_RESP_QUEUE_ERR_CRC = 1,
36*9c0476a2SJoe Komlodi     DW_I3C_RESP_QUEUE_ERR_PARITY = 2,
37*9c0476a2SJoe Komlodi     DW_I3C_RESP_QUEUE_ERR_FRAME = 3,
38*9c0476a2SJoe Komlodi     DW_I3C_RESP_QUEUE_ERR_BROADCAST_NACK = 4,
39*9c0476a2SJoe Komlodi     DW_I3C_RESP_QUEUE_ERR_DAA_NACK = 5,
40*9c0476a2SJoe Komlodi     DW_I3C_RESP_QUEUE_ERR_OVERFLOW = 6,
41*9c0476a2SJoe Komlodi     DW_I3C_RESP_QUEUE_ERR_ABORTED = 8,
42*9c0476a2SJoe Komlodi     DW_I3C_RESP_QUEUE_ERR_I2C_NACK = 9,
43*9c0476a2SJoe Komlodi } DWI3CRespQueueErr;
44*9c0476a2SJoe Komlodi 
45*9c0476a2SJoe Komlodi typedef enum DWI3CTransferState {
46*9c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATE_IDLE = 0x00,
47*9c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATE_START = 0x01,
48*9c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATE_RESTART = 0x02,
49*9c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATE_STOP = 0x03,
50*9c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATE_START_HOLD = 0x04,
51*9c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATE_BROADCAST_W = 0x05,
52*9c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATE_BROADCAST_R = 0x06,
53*9c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATE_DAA = 0x07,
54*9c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATE_DAA_GEN = 0x08,
55*9c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATE_CCC_BYTE = 0x0b,
56*9c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATE_HDR_CMD = 0x0c,
57*9c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATE_WRITE = 0x0d,
58*9c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATE_READ = 0x0e,
59*9c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATE_IBI_READ = 0x0f,
60*9c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATE_IBI_DIS = 0x10,
61*9c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATE_HDR_DDR_CRC = 0x11,
62*9c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATE_CLK_STRETCH = 0x12,
63*9c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATE_HALT = 0x13,
64*9c0476a2SJoe Komlodi } DWI3CTransferState;
65*9c0476a2SJoe Komlodi 
66*9c0476a2SJoe Komlodi typedef enum DWI3CTransferStatus {
67*9c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATUS_IDLE = 0x00,
68*9c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATUS_BROACAST_CCC = 0x01,
69*9c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATUS_DIRECT_CCC_W = 0x02,
70*9c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATUS_DIRECT_CCC_R = 0x03,
71*9c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATUS_ENTDAA = 0x04,
72*9c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATUS_SETDASA = 0x05,
73*9c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATUS_I3C_SDR_W = 0x06,
74*9c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATUS_I3C_SDR_R = 0x07,
75*9c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATUS_I2C_SDR_W = 0x08,
76*9c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATUS_I2C_SDR_R = 0x09,
77*9c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATUS_HDR_TS_W = 0x0a,
78*9c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATUS_HDR_TS_R = 0x0b,
79*9c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATUS_HDR_DDR_W = 0x0c,
80*9c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATUS_HDR_DDR_R = 0x0d,
81*9c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATUS_IBI = 0x0e,
82*9c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATUS_HALT = 0x0f,
83*9c0476a2SJoe Komlodi } DWI3CTransferStatus;
84*9c0476a2SJoe Komlodi 
85*9c0476a2SJoe Komlodi /*
86*9c0476a2SJoe Komlodi  * Transfer commands and arguments are 32-bit wide values that the user passes
87*9c0476a2SJoe Komlodi  * into the command queue. We interpret each 32-bit word based on the cmd_attr
88*9c0476a2SJoe Komlodi  * field.
89*9c0476a2SJoe Komlodi  */
90*9c0476a2SJoe Komlodi typedef struct DWI3CTransferCmd {
91*9c0476a2SJoe Komlodi     uint8_t cmd_attr:3;
92*9c0476a2SJoe Komlodi     uint8_t tid:4; /* Transaction ID */
93*9c0476a2SJoe Komlodi     uint16_t cmd:8;
94*9c0476a2SJoe Komlodi     uint8_t cp:1; /* Command present */
95*9c0476a2SJoe Komlodi     uint8_t dev_index:5;
96*9c0476a2SJoe Komlodi     uint8_t speed:3;
97*9c0476a2SJoe Komlodi     uint8_t resv0:1;
98*9c0476a2SJoe Komlodi     uint8_t dbp:1; /* Defining byte present */
99*9c0476a2SJoe Komlodi     uint8_t roc:1; /* Response on completion */
100*9c0476a2SJoe Komlodi     uint8_t sdap:1; /* Short data argument present */
101*9c0476a2SJoe Komlodi     uint8_t rnw:1; /* Read not write */
102*9c0476a2SJoe Komlodi     uint8_t resv1:1;
103*9c0476a2SJoe Komlodi     uint8_t toc:1; /* Termination (I3C STOP) on completion */
104*9c0476a2SJoe Komlodi     uint8_t pec:1; /* Parity error check enabled */
105*9c0476a2SJoe Komlodi } DWI3CTransferCmd;
106*9c0476a2SJoe Komlodi 
107*9c0476a2SJoe Komlodi typedef struct DWI3CTransferArg {
108*9c0476a2SJoe Komlodi     uint8_t cmd_attr:3;
109*9c0476a2SJoe Komlodi     uint8_t resv:5;
110*9c0476a2SJoe Komlodi     uint8_t db; /* Defining byte */
111*9c0476a2SJoe Komlodi     uint16_t data_len;
112*9c0476a2SJoe Komlodi } DWI3CTransferArg;
113*9c0476a2SJoe Komlodi 
114*9c0476a2SJoe Komlodi typedef struct DWI3CShortArg {
115*9c0476a2SJoe Komlodi     uint8_t cmd_attr:3;
116*9c0476a2SJoe Komlodi     uint8_t byte_strb:3;
117*9c0476a2SJoe Komlodi     uint8_t resv:2;
118*9c0476a2SJoe Komlodi     uint8_t byte0;
119*9c0476a2SJoe Komlodi     uint8_t byte1;
120*9c0476a2SJoe Komlodi     uint8_t byte2;
121*9c0476a2SJoe Komlodi } DWI3CShortArg;
122*9c0476a2SJoe Komlodi 
123*9c0476a2SJoe Komlodi typedef struct DWI3CAddrAssignCmd {
124*9c0476a2SJoe Komlodi     uint8_t cmd_attr:3;
125*9c0476a2SJoe Komlodi     uint8_t tid:4; /* Transaction ID */
126*9c0476a2SJoe Komlodi     uint16_t cmd:8;
127*9c0476a2SJoe Komlodi     uint8_t resv0:1;
128*9c0476a2SJoe Komlodi     uint8_t dev_index:5;
129*9c0476a2SJoe Komlodi     uint16_t dev_count:5;
130*9c0476a2SJoe Komlodi     uint8_t roc:1; /* Response on completion */
131*9c0476a2SJoe Komlodi     uint8_t resv1:3;
132*9c0476a2SJoe Komlodi     uint8_t toc:1; /* Termination (I3C STOP) on completion */
133*9c0476a2SJoe Komlodi     uint8_t resv2:1;
134*9c0476a2SJoe Komlodi } DWI3CAddrAssignCmd;
135*9c0476a2SJoe Komlodi 
136*9c0476a2SJoe Komlodi typedef union DWI3CCmdQueueData {
137*9c0476a2SJoe Komlodi     uint32_t word;
138*9c0476a2SJoe Komlodi     DWI3CTransferCmd transfer_cmd;
139*9c0476a2SJoe Komlodi     DWI3CTransferArg transfer_arg;
140*9c0476a2SJoe Komlodi     DWI3CShortArg short_arg;
141*9c0476a2SJoe Komlodi     DWI3CAddrAssignCmd addr_assign_cmd;
142*9c0476a2SJoe Komlodi } DWI3CCmdQueueData;
143c52aaabdSJoe Komlodi 
144c52aaabdSJoe Komlodi typedef struct DWI3C {
145c52aaabdSJoe Komlodi     /* <private> */
146c52aaabdSJoe Komlodi     SysBusDevice parent;
147c52aaabdSJoe Komlodi 
148c52aaabdSJoe Komlodi     /* <public> */
149c52aaabdSJoe Komlodi     MemoryRegion mr;
150c52aaabdSJoe Komlodi     qemu_irq irq;
151*9c0476a2SJoe Komlodi     I3CBus *bus;
152c52aaabdSJoe Komlodi 
153*9c0476a2SJoe Komlodi     Fifo32 cmd_queue;
154*9c0476a2SJoe Komlodi     Fifo32 resp_queue;
155*9c0476a2SJoe Komlodi     Fifo32 tx_queue;
156*9c0476a2SJoe Komlodi     Fifo32 rx_queue;
157*9c0476a2SJoe Komlodi 
158*9c0476a2SJoe Komlodi     struct {
159c52aaabdSJoe Komlodi         uint8_t id;
160*9c0476a2SJoe Komlodi         uint8_t cmd_resp_queue_capacity_bytes;
161*9c0476a2SJoe Komlodi         uint16_t tx_rx_queue_capacity_bytes;
162*9c0476a2SJoe Komlodi         uint8_t num_addressable_devices;
163*9c0476a2SJoe Komlodi         uint16_t dev_addr_table_pointer;
164*9c0476a2SJoe Komlodi         uint16_t dev_addr_table_depth;
165*9c0476a2SJoe Komlodi         uint16_t dev_char_table_pointer;
166*9c0476a2SJoe Komlodi         uint16_t dev_char_table_depth;
167*9c0476a2SJoe Komlodi     } cfg;
168c52aaabdSJoe Komlodi     uint32_t regs[DW_I3C_NR_REGS];
169c52aaabdSJoe Komlodi } DWI3C;
170c52aaabdSJoe Komlodi 
171c52aaabdSJoe Komlodi /* Extern for other controllers that use DesignWare I3C. */
172c52aaabdSJoe Komlodi extern const VMStateDescription vmstate_dw_i3c;
173c52aaabdSJoe Komlodi 
174c52aaabdSJoe Komlodi #endif /* DW_I3C_H */
175