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