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