xref: /openbmc/qemu/include/hw/i3c/dw-i3c.h (revision 5a2a5453bac7daafcdf0601a91009e01345b1dd2)
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 
139c0476a2SJoe Komlodi #include "qemu/fifo32.h"
149c0476a2SJoe 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 
209c0476a2SJoe Komlodi /*
219c0476a2SJoe Komlodi  * Sufficiently large enough to handle configurations with large device address
229c0476a2SJoe Komlodi  * tables.
239c0476a2SJoe Komlodi  */
249c0476a2SJoe Komlodi #define DW_I3C_NR_REGS (0x1000 >> 2)
259c0476a2SJoe Komlodi 
269c0476a2SJoe Komlodi /* From datasheet. */
279c0476a2SJoe Komlodi #define DW_I3C_CMD_ATTR_TRANSFER_CMD 0
289c0476a2SJoe Komlodi #define DW_I3C_CMD_ATTR_TRANSFER_ARG 1
299c0476a2SJoe Komlodi #define DW_I3C_CMD_ATTR_SHORT_DATA_ARG 2
309c0476a2SJoe Komlodi #define DW_I3C_CMD_ATTR_ADDR_ASSIGN_CMD 3
319c0476a2SJoe Komlodi 
329c0476a2SJoe Komlodi /* Enum values from datasheet. */
339c0476a2SJoe Komlodi typedef enum DWI3CRespQueueErr {
349c0476a2SJoe Komlodi     DW_I3C_RESP_QUEUE_ERR_NONE = 0,
359c0476a2SJoe Komlodi     DW_I3C_RESP_QUEUE_ERR_CRC = 1,
369c0476a2SJoe Komlodi     DW_I3C_RESP_QUEUE_ERR_PARITY = 2,
379c0476a2SJoe Komlodi     DW_I3C_RESP_QUEUE_ERR_FRAME = 3,
389c0476a2SJoe Komlodi     DW_I3C_RESP_QUEUE_ERR_BROADCAST_NACK = 4,
399c0476a2SJoe Komlodi     DW_I3C_RESP_QUEUE_ERR_DAA_NACK = 5,
409c0476a2SJoe Komlodi     DW_I3C_RESP_QUEUE_ERR_OVERFLOW = 6,
419c0476a2SJoe Komlodi     DW_I3C_RESP_QUEUE_ERR_ABORTED = 8,
429c0476a2SJoe Komlodi     DW_I3C_RESP_QUEUE_ERR_I2C_NACK = 9,
439c0476a2SJoe Komlodi } DWI3CRespQueueErr;
449c0476a2SJoe Komlodi 
459c0476a2SJoe Komlodi typedef enum DWI3CTransferState {
469c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATE_IDLE = 0x00,
479c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATE_START = 0x01,
489c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATE_RESTART = 0x02,
499c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATE_STOP = 0x03,
509c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATE_START_HOLD = 0x04,
519c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATE_BROADCAST_W = 0x05,
529c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATE_BROADCAST_R = 0x06,
539c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATE_DAA = 0x07,
549c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATE_DAA_GEN = 0x08,
559c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATE_CCC_BYTE = 0x0b,
569c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATE_HDR_CMD = 0x0c,
579c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATE_WRITE = 0x0d,
589c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATE_READ = 0x0e,
599c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATE_IBI_READ = 0x0f,
609c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATE_IBI_DIS = 0x10,
619c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATE_HDR_DDR_CRC = 0x11,
629c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATE_CLK_STRETCH = 0x12,
639c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATE_HALT = 0x13,
649c0476a2SJoe Komlodi } DWI3CTransferState;
659c0476a2SJoe Komlodi 
669c0476a2SJoe Komlodi typedef enum DWI3CTransferStatus {
679c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATUS_IDLE = 0x00,
689c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATUS_BROACAST_CCC = 0x01,
699c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATUS_DIRECT_CCC_W = 0x02,
709c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATUS_DIRECT_CCC_R = 0x03,
719c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATUS_ENTDAA = 0x04,
729c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATUS_SETDASA = 0x05,
739c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATUS_I3C_SDR_W = 0x06,
749c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATUS_I3C_SDR_R = 0x07,
759c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATUS_I2C_SDR_W = 0x08,
769c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATUS_I2C_SDR_R = 0x09,
779c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATUS_HDR_TS_W = 0x0a,
789c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATUS_HDR_TS_R = 0x0b,
799c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATUS_HDR_DDR_W = 0x0c,
809c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATUS_HDR_DDR_R = 0x0d,
819c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATUS_IBI = 0x0e,
829c0476a2SJoe Komlodi     DW_I3C_TRANSFER_STATUS_HALT = 0x0f,
839c0476a2SJoe Komlodi } DWI3CTransferStatus;
849c0476a2SJoe Komlodi 
859c0476a2SJoe Komlodi /*
869c0476a2SJoe Komlodi  * Transfer commands and arguments are 32-bit wide values that the user passes
879c0476a2SJoe Komlodi  * into the command queue. We interpret each 32-bit word based on the cmd_attr
889c0476a2SJoe Komlodi  * field.
899c0476a2SJoe Komlodi  */
909c0476a2SJoe Komlodi typedef struct DWI3CTransferCmd {
919c0476a2SJoe Komlodi     uint8_t cmd_attr:3;
929c0476a2SJoe Komlodi     uint8_t tid:4; /* Transaction ID */
939c0476a2SJoe Komlodi     uint16_t cmd:8;
949c0476a2SJoe Komlodi     uint8_t cp:1; /* Command present */
959c0476a2SJoe Komlodi     uint8_t dev_index:5;
969c0476a2SJoe Komlodi     uint8_t speed:3;
979c0476a2SJoe Komlodi     uint8_t resv0:1;
989c0476a2SJoe Komlodi     uint8_t dbp:1; /* Defining byte present */
999c0476a2SJoe Komlodi     uint8_t roc:1; /* Response on completion */
1009c0476a2SJoe Komlodi     uint8_t sdap:1; /* Short data argument present */
1019c0476a2SJoe Komlodi     uint8_t rnw:1; /* Read not write */
1029c0476a2SJoe Komlodi     uint8_t resv1:1;
1039c0476a2SJoe Komlodi     uint8_t toc:1; /* Termination (I3C STOP) on completion */
1049c0476a2SJoe Komlodi     uint8_t pec:1; /* Parity error check enabled */
1059c0476a2SJoe Komlodi } DWI3CTransferCmd;
1069c0476a2SJoe Komlodi 
1079c0476a2SJoe Komlodi typedef struct DWI3CTransferArg {
1089c0476a2SJoe Komlodi     uint8_t cmd_attr:3;
1099c0476a2SJoe Komlodi     uint8_t resv:5;
1109c0476a2SJoe Komlodi     uint8_t db; /* Defining byte */
1119c0476a2SJoe Komlodi     uint16_t data_len;
1129c0476a2SJoe Komlodi } DWI3CTransferArg;
1139c0476a2SJoe Komlodi 
1149c0476a2SJoe Komlodi typedef struct DWI3CShortArg {
1159c0476a2SJoe Komlodi     uint8_t cmd_attr:3;
1169c0476a2SJoe Komlodi     uint8_t byte_strb:3;
1179c0476a2SJoe Komlodi     uint8_t resv:2;
1189c0476a2SJoe Komlodi     uint8_t byte0;
1199c0476a2SJoe Komlodi     uint8_t byte1;
1209c0476a2SJoe Komlodi     uint8_t byte2;
1219c0476a2SJoe Komlodi } DWI3CShortArg;
1229c0476a2SJoe Komlodi 
1239c0476a2SJoe Komlodi typedef struct DWI3CAddrAssignCmd {
1249c0476a2SJoe Komlodi     uint8_t cmd_attr:3;
1259c0476a2SJoe Komlodi     uint8_t tid:4; /* Transaction ID */
1269c0476a2SJoe Komlodi     uint16_t cmd:8;
1279c0476a2SJoe Komlodi     uint8_t resv0:1;
1289c0476a2SJoe Komlodi     uint8_t dev_index:5;
1299c0476a2SJoe Komlodi     uint16_t dev_count:5;
1309c0476a2SJoe Komlodi     uint8_t roc:1; /* Response on completion */
1319c0476a2SJoe Komlodi     uint8_t resv1:3;
1329c0476a2SJoe Komlodi     uint8_t toc:1; /* Termination (I3C STOP) on completion */
1339c0476a2SJoe Komlodi     uint8_t resv2:1;
1349c0476a2SJoe Komlodi } DWI3CAddrAssignCmd;
1359c0476a2SJoe Komlodi 
1369c0476a2SJoe Komlodi typedef union DWI3CCmdQueueData {
1379c0476a2SJoe Komlodi     uint32_t word;
1389c0476a2SJoe Komlodi     DWI3CTransferCmd transfer_cmd;
1399c0476a2SJoe Komlodi     DWI3CTransferArg transfer_arg;
1409c0476a2SJoe Komlodi     DWI3CShortArg short_arg;
1419c0476a2SJoe Komlodi     DWI3CAddrAssignCmd addr_assign_cmd;
1429c0476a2SJoe Komlodi } DWI3CCmdQueueData;
143c52aaabdSJoe Komlodi 
144*5a2a5453SJoe Komlodi /*
145*5a2a5453SJoe Komlodi  * When we receive an IBI with data, we need to store it temporarily until
146*5a2a5453SJoe Komlodi  * the target is finished sending data. Then we can set the IBI queue status
147*5a2a5453SJoe Komlodi  * appropriately.
148*5a2a5453SJoe Komlodi  */
149*5a2a5453SJoe Komlodi typedef struct DWI3CIBIData {
150*5a2a5453SJoe Komlodi     /* Do we notify the user that an IBI was NACKed? */
151*5a2a5453SJoe Komlodi     bool notify_ibi_nack;
152*5a2a5453SJoe Komlodi     /* Intermediate storage of IBI_QUEUE_STATUS. */
153*5a2a5453SJoe Komlodi     uint32_t ibi_queue_status;
154*5a2a5453SJoe Komlodi     /* Temporary buffer to store IBI data from the target. */
155*5a2a5453SJoe Komlodi     Fifo8 ibi_intermediate_queue;
156*5a2a5453SJoe Komlodi     /* The address we should send a CCC_DISEC to. */
157*5a2a5453SJoe Komlodi     uint8_t disec_addr;
158*5a2a5453SJoe Komlodi     /* The byte we should send along with the CCC_DISEC. */
159*5a2a5453SJoe Komlodi     uint8_t disec_byte;
160*5a2a5453SJoe Komlodi     /* Should we send a direct DISEC CCC? (As opposed to global). */
161*5a2a5453SJoe Komlodi     bool send_direct_disec;
162*5a2a5453SJoe Komlodi     /* Was this IBI NACKed? */
163*5a2a5453SJoe Komlodi     bool ibi_nacked;
164*5a2a5453SJoe Komlodi } DWI3CIBIData;
165*5a2a5453SJoe Komlodi 
166c52aaabdSJoe Komlodi typedef struct DWI3C {
167c52aaabdSJoe Komlodi     /* <private> */
168c52aaabdSJoe Komlodi     SysBusDevice parent;
169c52aaabdSJoe Komlodi 
170c52aaabdSJoe Komlodi     /* <public> */
171c52aaabdSJoe Komlodi     MemoryRegion mr;
172c52aaabdSJoe Komlodi     qemu_irq irq;
1739c0476a2SJoe Komlodi     I3CBus *bus;
174c52aaabdSJoe Komlodi 
1759c0476a2SJoe Komlodi     Fifo32 cmd_queue;
1769c0476a2SJoe Komlodi     Fifo32 resp_queue;
1779c0476a2SJoe Komlodi     Fifo32 tx_queue;
1789c0476a2SJoe Komlodi     Fifo32 rx_queue;
179*5a2a5453SJoe Komlodi     Fifo32 ibi_queue;
180*5a2a5453SJoe Komlodi 
181*5a2a5453SJoe Komlodi     /* Temporary storage for IBI data. */
182*5a2a5453SJoe Komlodi     DWI3CIBIData ibi_data;
1839c0476a2SJoe Komlodi 
1849c0476a2SJoe Komlodi     struct {
185c52aaabdSJoe Komlodi         uint8_t id;
1869c0476a2SJoe Komlodi         uint8_t cmd_resp_queue_capacity_bytes;
1879c0476a2SJoe Komlodi         uint16_t tx_rx_queue_capacity_bytes;
188*5a2a5453SJoe Komlodi         uint8_t ibi_queue_capacity_bytes;
1899c0476a2SJoe Komlodi         uint8_t num_addressable_devices;
1909c0476a2SJoe Komlodi         uint16_t dev_addr_table_pointer;
1919c0476a2SJoe Komlodi         uint16_t dev_addr_table_depth;
1929c0476a2SJoe Komlodi         uint16_t dev_char_table_pointer;
1939c0476a2SJoe Komlodi         uint16_t dev_char_table_depth;
1949c0476a2SJoe Komlodi     } cfg;
195c52aaabdSJoe Komlodi     uint32_t regs[DW_I3C_NR_REGS];
196c52aaabdSJoe Komlodi } DWI3C;
197c52aaabdSJoe Komlodi 
198c52aaabdSJoe Komlodi /* Extern for other controllers that use DesignWare I3C. */
199c52aaabdSJoe Komlodi extern const VMStateDescription vmstate_dw_i3c;
200c52aaabdSJoe Komlodi 
201c52aaabdSJoe Komlodi #endif /* DW_I3C_H */
202