18c2ecf20Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0 28c2ecf20Sopenharmony_ci/* 38c2ecf20Sopenharmony_ci * Copyright (c) 2018 Synopsys, Inc. and/or its affiliates. 48c2ecf20Sopenharmony_ci * 58c2ecf20Sopenharmony_ci * Author: Vitor Soares <vitor.soares@synopsys.com> 68c2ecf20Sopenharmony_ci */ 78c2ecf20Sopenharmony_ci 88c2ecf20Sopenharmony_ci#include <linux/bitops.h> 98c2ecf20Sopenharmony_ci#include <linux/clk.h> 108c2ecf20Sopenharmony_ci#include <linux/completion.h> 118c2ecf20Sopenharmony_ci#include <linux/err.h> 128c2ecf20Sopenharmony_ci#include <linux/errno.h> 138c2ecf20Sopenharmony_ci#include <linux/i3c/master.h> 148c2ecf20Sopenharmony_ci#include <linux/interrupt.h> 158c2ecf20Sopenharmony_ci#include <linux/ioport.h> 168c2ecf20Sopenharmony_ci#include <linux/iopoll.h> 178c2ecf20Sopenharmony_ci#include <linux/list.h> 188c2ecf20Sopenharmony_ci#include <linux/module.h> 198c2ecf20Sopenharmony_ci#include <linux/of.h> 208c2ecf20Sopenharmony_ci#include <linux/platform_device.h> 218c2ecf20Sopenharmony_ci#include <linux/reset.h> 228c2ecf20Sopenharmony_ci#include <linux/slab.h> 238c2ecf20Sopenharmony_ci 248c2ecf20Sopenharmony_ci#define DEVICE_CTRL 0x0 258c2ecf20Sopenharmony_ci#define DEV_CTRL_ENABLE BIT(31) 268c2ecf20Sopenharmony_ci#define DEV_CTRL_RESUME BIT(30) 278c2ecf20Sopenharmony_ci#define DEV_CTRL_HOT_JOIN_NACK BIT(8) 288c2ecf20Sopenharmony_ci#define DEV_CTRL_I2C_SLAVE_PRESENT BIT(7) 298c2ecf20Sopenharmony_ci 308c2ecf20Sopenharmony_ci#define DEVICE_ADDR 0x4 318c2ecf20Sopenharmony_ci#define DEV_ADDR_DYNAMIC_ADDR_VALID BIT(31) 328c2ecf20Sopenharmony_ci#define DEV_ADDR_DYNAMIC(x) (((x) << 16) & GENMASK(22, 16)) 338c2ecf20Sopenharmony_ci 348c2ecf20Sopenharmony_ci#define HW_CAPABILITY 0x8 358c2ecf20Sopenharmony_ci#define COMMAND_QUEUE_PORT 0xc 368c2ecf20Sopenharmony_ci#define COMMAND_PORT_TOC BIT(30) 378c2ecf20Sopenharmony_ci#define COMMAND_PORT_READ_TRANSFER BIT(28) 388c2ecf20Sopenharmony_ci#define COMMAND_PORT_SDAP BIT(27) 398c2ecf20Sopenharmony_ci#define COMMAND_PORT_ROC BIT(26) 408c2ecf20Sopenharmony_ci#define COMMAND_PORT_SPEED(x) (((x) << 21) & GENMASK(23, 21)) 418c2ecf20Sopenharmony_ci#define COMMAND_PORT_DEV_INDEX(x) (((x) << 16) & GENMASK(20, 16)) 428c2ecf20Sopenharmony_ci#define COMMAND_PORT_CP BIT(15) 438c2ecf20Sopenharmony_ci#define COMMAND_PORT_CMD(x) (((x) << 7) & GENMASK(14, 7)) 448c2ecf20Sopenharmony_ci#define COMMAND_PORT_TID(x) (((x) << 3) & GENMASK(6, 3)) 458c2ecf20Sopenharmony_ci 468c2ecf20Sopenharmony_ci#define COMMAND_PORT_ARG_DATA_LEN(x) (((x) << 16) & GENMASK(31, 16)) 478c2ecf20Sopenharmony_ci#define COMMAND_PORT_ARG_DATA_LEN_MAX 65536 488c2ecf20Sopenharmony_ci#define COMMAND_PORT_TRANSFER_ARG 0x01 498c2ecf20Sopenharmony_ci 508c2ecf20Sopenharmony_ci#define COMMAND_PORT_SDA_DATA_BYTE_3(x) (((x) << 24) & GENMASK(31, 24)) 518c2ecf20Sopenharmony_ci#define COMMAND_PORT_SDA_DATA_BYTE_2(x) (((x) << 16) & GENMASK(23, 16)) 528c2ecf20Sopenharmony_ci#define COMMAND_PORT_SDA_DATA_BYTE_1(x) (((x) << 8) & GENMASK(15, 8)) 538c2ecf20Sopenharmony_ci#define COMMAND_PORT_SDA_BYTE_STRB_3 BIT(5) 548c2ecf20Sopenharmony_ci#define COMMAND_PORT_SDA_BYTE_STRB_2 BIT(4) 558c2ecf20Sopenharmony_ci#define COMMAND_PORT_SDA_BYTE_STRB_1 BIT(3) 568c2ecf20Sopenharmony_ci#define COMMAND_PORT_SHORT_DATA_ARG 0x02 578c2ecf20Sopenharmony_ci 588c2ecf20Sopenharmony_ci#define COMMAND_PORT_DEV_COUNT(x) (((x) << 21) & GENMASK(25, 21)) 598c2ecf20Sopenharmony_ci#define COMMAND_PORT_ADDR_ASSGN_CMD 0x03 608c2ecf20Sopenharmony_ci 618c2ecf20Sopenharmony_ci#define RESPONSE_QUEUE_PORT 0x10 628c2ecf20Sopenharmony_ci#define RESPONSE_PORT_ERR_STATUS(x) (((x) & GENMASK(31, 28)) >> 28) 638c2ecf20Sopenharmony_ci#define RESPONSE_NO_ERROR 0 648c2ecf20Sopenharmony_ci#define RESPONSE_ERROR_CRC 1 658c2ecf20Sopenharmony_ci#define RESPONSE_ERROR_PARITY 2 668c2ecf20Sopenharmony_ci#define RESPONSE_ERROR_FRAME 3 678c2ecf20Sopenharmony_ci#define RESPONSE_ERROR_IBA_NACK 4 688c2ecf20Sopenharmony_ci#define RESPONSE_ERROR_ADDRESS_NACK 5 698c2ecf20Sopenharmony_ci#define RESPONSE_ERROR_OVER_UNDER_FLOW 6 708c2ecf20Sopenharmony_ci#define RESPONSE_ERROR_TRANSF_ABORT 8 718c2ecf20Sopenharmony_ci#define RESPONSE_ERROR_I2C_W_NACK_ERR 9 728c2ecf20Sopenharmony_ci#define RESPONSE_PORT_TID(x) (((x) & GENMASK(27, 24)) >> 24) 738c2ecf20Sopenharmony_ci#define RESPONSE_PORT_DATA_LEN(x) ((x) & GENMASK(15, 0)) 748c2ecf20Sopenharmony_ci 758c2ecf20Sopenharmony_ci#define RX_TX_DATA_PORT 0x14 768c2ecf20Sopenharmony_ci#define IBI_QUEUE_STATUS 0x18 778c2ecf20Sopenharmony_ci#define QUEUE_THLD_CTRL 0x1c 788c2ecf20Sopenharmony_ci#define QUEUE_THLD_CTRL_RESP_BUF_MASK GENMASK(15, 8) 798c2ecf20Sopenharmony_ci#define QUEUE_THLD_CTRL_RESP_BUF(x) (((x) - 1) << 8) 808c2ecf20Sopenharmony_ci 818c2ecf20Sopenharmony_ci#define DATA_BUFFER_THLD_CTRL 0x20 828c2ecf20Sopenharmony_ci#define DATA_BUFFER_THLD_CTRL_RX_BUF GENMASK(11, 8) 838c2ecf20Sopenharmony_ci 848c2ecf20Sopenharmony_ci#define IBI_QUEUE_CTRL 0x24 858c2ecf20Sopenharmony_ci#define IBI_MR_REQ_REJECT 0x2C 868c2ecf20Sopenharmony_ci#define IBI_SIR_REQ_REJECT 0x30 878c2ecf20Sopenharmony_ci#define IBI_REQ_REJECT_ALL GENMASK(31, 0) 888c2ecf20Sopenharmony_ci 898c2ecf20Sopenharmony_ci#define RESET_CTRL 0x34 908c2ecf20Sopenharmony_ci#define RESET_CTRL_IBI_QUEUE BIT(5) 918c2ecf20Sopenharmony_ci#define RESET_CTRL_RX_FIFO BIT(4) 928c2ecf20Sopenharmony_ci#define RESET_CTRL_TX_FIFO BIT(3) 938c2ecf20Sopenharmony_ci#define RESET_CTRL_RESP_QUEUE BIT(2) 948c2ecf20Sopenharmony_ci#define RESET_CTRL_CMD_QUEUE BIT(1) 958c2ecf20Sopenharmony_ci#define RESET_CTRL_SOFT BIT(0) 968c2ecf20Sopenharmony_ci 978c2ecf20Sopenharmony_ci#define SLV_EVENT_CTRL 0x38 988c2ecf20Sopenharmony_ci#define INTR_STATUS 0x3c 998c2ecf20Sopenharmony_ci#define INTR_STATUS_EN 0x40 1008c2ecf20Sopenharmony_ci#define INTR_SIGNAL_EN 0x44 1018c2ecf20Sopenharmony_ci#define INTR_FORCE 0x48 1028c2ecf20Sopenharmony_ci#define INTR_BUSOWNER_UPDATE_STAT BIT(13) 1038c2ecf20Sopenharmony_ci#define INTR_IBI_UPDATED_STAT BIT(12) 1048c2ecf20Sopenharmony_ci#define INTR_READ_REQ_RECV_STAT BIT(11) 1058c2ecf20Sopenharmony_ci#define INTR_DEFSLV_STAT BIT(10) 1068c2ecf20Sopenharmony_ci#define INTR_TRANSFER_ERR_STAT BIT(9) 1078c2ecf20Sopenharmony_ci#define INTR_DYN_ADDR_ASSGN_STAT BIT(8) 1088c2ecf20Sopenharmony_ci#define INTR_CCC_UPDATED_STAT BIT(6) 1098c2ecf20Sopenharmony_ci#define INTR_TRANSFER_ABORT_STAT BIT(5) 1108c2ecf20Sopenharmony_ci#define INTR_RESP_READY_STAT BIT(4) 1118c2ecf20Sopenharmony_ci#define INTR_CMD_QUEUE_READY_STAT BIT(3) 1128c2ecf20Sopenharmony_ci#define INTR_IBI_THLD_STAT BIT(2) 1138c2ecf20Sopenharmony_ci#define INTR_RX_THLD_STAT BIT(1) 1148c2ecf20Sopenharmony_ci#define INTR_TX_THLD_STAT BIT(0) 1158c2ecf20Sopenharmony_ci#define INTR_ALL (INTR_BUSOWNER_UPDATE_STAT | \ 1168c2ecf20Sopenharmony_ci INTR_IBI_UPDATED_STAT | \ 1178c2ecf20Sopenharmony_ci INTR_READ_REQ_RECV_STAT | \ 1188c2ecf20Sopenharmony_ci INTR_DEFSLV_STAT | \ 1198c2ecf20Sopenharmony_ci INTR_TRANSFER_ERR_STAT | \ 1208c2ecf20Sopenharmony_ci INTR_DYN_ADDR_ASSGN_STAT | \ 1218c2ecf20Sopenharmony_ci INTR_CCC_UPDATED_STAT | \ 1228c2ecf20Sopenharmony_ci INTR_TRANSFER_ABORT_STAT | \ 1238c2ecf20Sopenharmony_ci INTR_RESP_READY_STAT | \ 1248c2ecf20Sopenharmony_ci INTR_CMD_QUEUE_READY_STAT | \ 1258c2ecf20Sopenharmony_ci INTR_IBI_THLD_STAT | \ 1268c2ecf20Sopenharmony_ci INTR_TX_THLD_STAT | \ 1278c2ecf20Sopenharmony_ci INTR_RX_THLD_STAT) 1288c2ecf20Sopenharmony_ci 1298c2ecf20Sopenharmony_ci#define INTR_MASTER_MASK (INTR_TRANSFER_ERR_STAT | \ 1308c2ecf20Sopenharmony_ci INTR_RESP_READY_STAT) 1318c2ecf20Sopenharmony_ci 1328c2ecf20Sopenharmony_ci#define QUEUE_STATUS_LEVEL 0x4c 1338c2ecf20Sopenharmony_ci#define QUEUE_STATUS_IBI_STATUS_CNT(x) (((x) & GENMASK(28, 24)) >> 24) 1348c2ecf20Sopenharmony_ci#define QUEUE_STATUS_IBI_BUF_BLR(x) (((x) & GENMASK(23, 16)) >> 16) 1358c2ecf20Sopenharmony_ci#define QUEUE_STATUS_LEVEL_RESP(x) (((x) & GENMASK(15, 8)) >> 8) 1368c2ecf20Sopenharmony_ci#define QUEUE_STATUS_LEVEL_CMD(x) ((x) & GENMASK(7, 0)) 1378c2ecf20Sopenharmony_ci 1388c2ecf20Sopenharmony_ci#define DATA_BUFFER_STATUS_LEVEL 0x50 1398c2ecf20Sopenharmony_ci#define DATA_BUFFER_STATUS_LEVEL_TX(x) ((x) & GENMASK(7, 0)) 1408c2ecf20Sopenharmony_ci 1418c2ecf20Sopenharmony_ci#define PRESENT_STATE 0x54 1428c2ecf20Sopenharmony_ci#define CCC_DEVICE_STATUS 0x58 1438c2ecf20Sopenharmony_ci#define DEVICE_ADDR_TABLE_POINTER 0x5c 1448c2ecf20Sopenharmony_ci#define DEVICE_ADDR_TABLE_DEPTH(x) (((x) & GENMASK(31, 16)) >> 16) 1458c2ecf20Sopenharmony_ci#define DEVICE_ADDR_TABLE_ADDR(x) ((x) & GENMASK(7, 0)) 1468c2ecf20Sopenharmony_ci 1478c2ecf20Sopenharmony_ci#define DEV_CHAR_TABLE_POINTER 0x60 1488c2ecf20Sopenharmony_ci#define VENDOR_SPECIFIC_REG_POINTER 0x6c 1498c2ecf20Sopenharmony_ci#define SLV_PID_VALUE 0x74 1508c2ecf20Sopenharmony_ci#define SLV_CHAR_CTRL 0x78 1518c2ecf20Sopenharmony_ci#define SLV_MAX_LEN 0x7c 1528c2ecf20Sopenharmony_ci#define MAX_READ_TURNAROUND 0x80 1538c2ecf20Sopenharmony_ci#define MAX_DATA_SPEED 0x84 1548c2ecf20Sopenharmony_ci#define SLV_DEBUG_STATUS 0x88 1558c2ecf20Sopenharmony_ci#define SLV_INTR_REQ 0x8c 1568c2ecf20Sopenharmony_ci#define DEVICE_CTRL_EXTENDED 0xb0 1578c2ecf20Sopenharmony_ci#define SCL_I3C_OD_TIMING 0xb4 1588c2ecf20Sopenharmony_ci#define SCL_I3C_PP_TIMING 0xb8 1598c2ecf20Sopenharmony_ci#define SCL_I3C_TIMING_HCNT(x) (((x) << 16) & GENMASK(23, 16)) 1608c2ecf20Sopenharmony_ci#define SCL_I3C_TIMING_LCNT(x) ((x) & GENMASK(7, 0)) 1618c2ecf20Sopenharmony_ci#define SCL_I3C_TIMING_CNT_MIN 5 1628c2ecf20Sopenharmony_ci 1638c2ecf20Sopenharmony_ci#define SCL_I2C_FM_TIMING 0xbc 1648c2ecf20Sopenharmony_ci#define SCL_I2C_FM_TIMING_HCNT(x) (((x) << 16) & GENMASK(31, 16)) 1658c2ecf20Sopenharmony_ci#define SCL_I2C_FM_TIMING_LCNT(x) ((x) & GENMASK(15, 0)) 1668c2ecf20Sopenharmony_ci 1678c2ecf20Sopenharmony_ci#define SCL_I2C_FMP_TIMING 0xc0 1688c2ecf20Sopenharmony_ci#define SCL_I2C_FMP_TIMING_HCNT(x) (((x) << 16) & GENMASK(23, 16)) 1698c2ecf20Sopenharmony_ci#define SCL_I2C_FMP_TIMING_LCNT(x) ((x) & GENMASK(15, 0)) 1708c2ecf20Sopenharmony_ci 1718c2ecf20Sopenharmony_ci#define SCL_EXT_LCNT_TIMING 0xc8 1728c2ecf20Sopenharmony_ci#define SCL_EXT_LCNT_4(x) (((x) << 24) & GENMASK(31, 24)) 1738c2ecf20Sopenharmony_ci#define SCL_EXT_LCNT_3(x) (((x) << 16) & GENMASK(23, 16)) 1748c2ecf20Sopenharmony_ci#define SCL_EXT_LCNT_2(x) (((x) << 8) & GENMASK(15, 8)) 1758c2ecf20Sopenharmony_ci#define SCL_EXT_LCNT_1(x) ((x) & GENMASK(7, 0)) 1768c2ecf20Sopenharmony_ci 1778c2ecf20Sopenharmony_ci#define SCL_EXT_TERMN_LCNT_TIMING 0xcc 1788c2ecf20Sopenharmony_ci#define BUS_FREE_TIMING 0xd4 1798c2ecf20Sopenharmony_ci#define BUS_I3C_MST_FREE(x) ((x) & GENMASK(15, 0)) 1808c2ecf20Sopenharmony_ci 1818c2ecf20Sopenharmony_ci#define BUS_IDLE_TIMING 0xd8 1828c2ecf20Sopenharmony_ci#define I3C_VER_ID 0xe0 1838c2ecf20Sopenharmony_ci#define I3C_VER_TYPE 0xe4 1848c2ecf20Sopenharmony_ci#define EXTENDED_CAPABILITY 0xe8 1858c2ecf20Sopenharmony_ci#define SLAVE_CONFIG 0xec 1868c2ecf20Sopenharmony_ci 1878c2ecf20Sopenharmony_ci#define DEV_ADDR_TABLE_LEGACY_I2C_DEV BIT(31) 1888c2ecf20Sopenharmony_ci#define DEV_ADDR_TABLE_DYNAMIC_ADDR(x) (((x) << 16) & GENMASK(23, 16)) 1898c2ecf20Sopenharmony_ci#define DEV_ADDR_TABLE_STATIC_ADDR(x) ((x) & GENMASK(6, 0)) 1908c2ecf20Sopenharmony_ci#define DEV_ADDR_TABLE_LOC(start, idx) ((start) + ((idx) << 2)) 1918c2ecf20Sopenharmony_ci 1928c2ecf20Sopenharmony_ci#define MAX_DEVS 32 1938c2ecf20Sopenharmony_ci 1948c2ecf20Sopenharmony_ci#define I3C_BUS_SDR1_SCL_RATE 8000000 1958c2ecf20Sopenharmony_ci#define I3C_BUS_SDR2_SCL_RATE 6000000 1968c2ecf20Sopenharmony_ci#define I3C_BUS_SDR3_SCL_RATE 4000000 1978c2ecf20Sopenharmony_ci#define I3C_BUS_SDR4_SCL_RATE 2000000 1988c2ecf20Sopenharmony_ci#define I3C_BUS_I2C_FM_TLOW_MIN_NS 1300 1998c2ecf20Sopenharmony_ci#define I3C_BUS_I2C_FMP_TLOW_MIN_NS 500 2008c2ecf20Sopenharmony_ci#define I3C_BUS_THIGH_MAX_NS 41 2018c2ecf20Sopenharmony_ci 2028c2ecf20Sopenharmony_ci#define XFER_TIMEOUT (msecs_to_jiffies(1000)) 2038c2ecf20Sopenharmony_ci 2048c2ecf20Sopenharmony_cistruct dw_i3c_master_caps { 2058c2ecf20Sopenharmony_ci u8 cmdfifodepth; 2068c2ecf20Sopenharmony_ci u8 datafifodepth; 2078c2ecf20Sopenharmony_ci}; 2088c2ecf20Sopenharmony_ci 2098c2ecf20Sopenharmony_cistruct dw_i3c_cmd { 2108c2ecf20Sopenharmony_ci u32 cmd_lo; 2118c2ecf20Sopenharmony_ci u32 cmd_hi; 2128c2ecf20Sopenharmony_ci u16 tx_len; 2138c2ecf20Sopenharmony_ci const void *tx_buf; 2148c2ecf20Sopenharmony_ci u16 rx_len; 2158c2ecf20Sopenharmony_ci void *rx_buf; 2168c2ecf20Sopenharmony_ci u8 error; 2178c2ecf20Sopenharmony_ci}; 2188c2ecf20Sopenharmony_ci 2198c2ecf20Sopenharmony_cistruct dw_i3c_xfer { 2208c2ecf20Sopenharmony_ci struct list_head node; 2218c2ecf20Sopenharmony_ci struct completion comp; 2228c2ecf20Sopenharmony_ci int ret; 2238c2ecf20Sopenharmony_ci unsigned int ncmds; 2248c2ecf20Sopenharmony_ci struct dw_i3c_cmd cmds[]; 2258c2ecf20Sopenharmony_ci}; 2268c2ecf20Sopenharmony_ci 2278c2ecf20Sopenharmony_cistruct dw_i3c_master { 2288c2ecf20Sopenharmony_ci struct i3c_master_controller base; 2298c2ecf20Sopenharmony_ci u16 maxdevs; 2308c2ecf20Sopenharmony_ci u16 datstartaddr; 2318c2ecf20Sopenharmony_ci u32 free_pos; 2328c2ecf20Sopenharmony_ci struct { 2338c2ecf20Sopenharmony_ci struct list_head list; 2348c2ecf20Sopenharmony_ci struct dw_i3c_xfer *cur; 2358c2ecf20Sopenharmony_ci spinlock_t lock; 2368c2ecf20Sopenharmony_ci } xferqueue; 2378c2ecf20Sopenharmony_ci struct dw_i3c_master_caps caps; 2388c2ecf20Sopenharmony_ci void __iomem *regs; 2398c2ecf20Sopenharmony_ci struct reset_control *core_rst; 2408c2ecf20Sopenharmony_ci struct clk *core_clk; 2418c2ecf20Sopenharmony_ci char version[5]; 2428c2ecf20Sopenharmony_ci char type[5]; 2438c2ecf20Sopenharmony_ci u8 addrs[MAX_DEVS]; 2448c2ecf20Sopenharmony_ci}; 2458c2ecf20Sopenharmony_ci 2468c2ecf20Sopenharmony_cistruct dw_i3c_i2c_dev_data { 2478c2ecf20Sopenharmony_ci u8 index; 2488c2ecf20Sopenharmony_ci}; 2498c2ecf20Sopenharmony_ci 2508c2ecf20Sopenharmony_cistatic u8 even_parity(u8 p) 2518c2ecf20Sopenharmony_ci{ 2528c2ecf20Sopenharmony_ci p ^= p >> 4; 2538c2ecf20Sopenharmony_ci p &= 0xf; 2548c2ecf20Sopenharmony_ci 2558c2ecf20Sopenharmony_ci return (0x9669 >> p) & 1; 2568c2ecf20Sopenharmony_ci} 2578c2ecf20Sopenharmony_ci 2588c2ecf20Sopenharmony_cistatic bool dw_i3c_master_supports_ccc_cmd(struct i3c_master_controller *m, 2598c2ecf20Sopenharmony_ci const struct i3c_ccc_cmd *cmd) 2608c2ecf20Sopenharmony_ci{ 2618c2ecf20Sopenharmony_ci if (cmd->ndests > 1) 2628c2ecf20Sopenharmony_ci return false; 2638c2ecf20Sopenharmony_ci 2648c2ecf20Sopenharmony_ci switch (cmd->id) { 2658c2ecf20Sopenharmony_ci case I3C_CCC_ENEC(true): 2668c2ecf20Sopenharmony_ci case I3C_CCC_ENEC(false): 2678c2ecf20Sopenharmony_ci case I3C_CCC_DISEC(true): 2688c2ecf20Sopenharmony_ci case I3C_CCC_DISEC(false): 2698c2ecf20Sopenharmony_ci case I3C_CCC_ENTAS(0, true): 2708c2ecf20Sopenharmony_ci case I3C_CCC_ENTAS(0, false): 2718c2ecf20Sopenharmony_ci case I3C_CCC_RSTDAA(true): 2728c2ecf20Sopenharmony_ci case I3C_CCC_RSTDAA(false): 2738c2ecf20Sopenharmony_ci case I3C_CCC_ENTDAA: 2748c2ecf20Sopenharmony_ci case I3C_CCC_SETMWL(true): 2758c2ecf20Sopenharmony_ci case I3C_CCC_SETMWL(false): 2768c2ecf20Sopenharmony_ci case I3C_CCC_SETMRL(true): 2778c2ecf20Sopenharmony_ci case I3C_CCC_SETMRL(false): 2788c2ecf20Sopenharmony_ci case I3C_CCC_ENTHDR(0): 2798c2ecf20Sopenharmony_ci case I3C_CCC_SETDASA: 2808c2ecf20Sopenharmony_ci case I3C_CCC_SETNEWDA: 2818c2ecf20Sopenharmony_ci case I3C_CCC_GETMWL: 2828c2ecf20Sopenharmony_ci case I3C_CCC_GETMRL: 2838c2ecf20Sopenharmony_ci case I3C_CCC_GETPID: 2848c2ecf20Sopenharmony_ci case I3C_CCC_GETBCR: 2858c2ecf20Sopenharmony_ci case I3C_CCC_GETDCR: 2868c2ecf20Sopenharmony_ci case I3C_CCC_GETSTATUS: 2878c2ecf20Sopenharmony_ci case I3C_CCC_GETMXDS: 2888c2ecf20Sopenharmony_ci case I3C_CCC_GETHDRCAP: 2898c2ecf20Sopenharmony_ci return true; 2908c2ecf20Sopenharmony_ci default: 2918c2ecf20Sopenharmony_ci return false; 2928c2ecf20Sopenharmony_ci } 2938c2ecf20Sopenharmony_ci} 2948c2ecf20Sopenharmony_ci 2958c2ecf20Sopenharmony_cistatic inline struct dw_i3c_master * 2968c2ecf20Sopenharmony_cito_dw_i3c_master(struct i3c_master_controller *master) 2978c2ecf20Sopenharmony_ci{ 2988c2ecf20Sopenharmony_ci return container_of(master, struct dw_i3c_master, base); 2998c2ecf20Sopenharmony_ci} 3008c2ecf20Sopenharmony_ci 3018c2ecf20Sopenharmony_cistatic void dw_i3c_master_disable(struct dw_i3c_master *master) 3028c2ecf20Sopenharmony_ci{ 3038c2ecf20Sopenharmony_ci writel(readl(master->regs + DEVICE_CTRL) & ~DEV_CTRL_ENABLE, 3048c2ecf20Sopenharmony_ci master->regs + DEVICE_CTRL); 3058c2ecf20Sopenharmony_ci} 3068c2ecf20Sopenharmony_ci 3078c2ecf20Sopenharmony_cistatic void dw_i3c_master_enable(struct dw_i3c_master *master) 3088c2ecf20Sopenharmony_ci{ 3098c2ecf20Sopenharmony_ci writel(readl(master->regs + DEVICE_CTRL) | DEV_CTRL_ENABLE, 3108c2ecf20Sopenharmony_ci master->regs + DEVICE_CTRL); 3118c2ecf20Sopenharmony_ci} 3128c2ecf20Sopenharmony_ci 3138c2ecf20Sopenharmony_cistatic int dw_i3c_master_get_addr_pos(struct dw_i3c_master *master, u8 addr) 3148c2ecf20Sopenharmony_ci{ 3158c2ecf20Sopenharmony_ci int pos; 3168c2ecf20Sopenharmony_ci 3178c2ecf20Sopenharmony_ci for (pos = 0; pos < master->maxdevs; pos++) { 3188c2ecf20Sopenharmony_ci if (addr == master->addrs[pos]) 3198c2ecf20Sopenharmony_ci return pos; 3208c2ecf20Sopenharmony_ci } 3218c2ecf20Sopenharmony_ci 3228c2ecf20Sopenharmony_ci return -EINVAL; 3238c2ecf20Sopenharmony_ci} 3248c2ecf20Sopenharmony_ci 3258c2ecf20Sopenharmony_cistatic int dw_i3c_master_get_free_pos(struct dw_i3c_master *master) 3268c2ecf20Sopenharmony_ci{ 3278c2ecf20Sopenharmony_ci if (!(master->free_pos & GENMASK(master->maxdevs - 1, 0))) 3288c2ecf20Sopenharmony_ci return -ENOSPC; 3298c2ecf20Sopenharmony_ci 3308c2ecf20Sopenharmony_ci return ffs(master->free_pos) - 1; 3318c2ecf20Sopenharmony_ci} 3328c2ecf20Sopenharmony_ci 3338c2ecf20Sopenharmony_cistatic void dw_i3c_master_wr_tx_fifo(struct dw_i3c_master *master, 3348c2ecf20Sopenharmony_ci const u8 *bytes, int nbytes) 3358c2ecf20Sopenharmony_ci{ 3368c2ecf20Sopenharmony_ci writesl(master->regs + RX_TX_DATA_PORT, bytes, nbytes / 4); 3378c2ecf20Sopenharmony_ci if (nbytes & 3) { 3388c2ecf20Sopenharmony_ci u32 tmp = 0; 3398c2ecf20Sopenharmony_ci 3408c2ecf20Sopenharmony_ci memcpy(&tmp, bytes + (nbytes & ~3), nbytes & 3); 3418c2ecf20Sopenharmony_ci writesl(master->regs + RX_TX_DATA_PORT, &tmp, 1); 3428c2ecf20Sopenharmony_ci } 3438c2ecf20Sopenharmony_ci} 3448c2ecf20Sopenharmony_ci 3458c2ecf20Sopenharmony_cistatic void dw_i3c_master_read_rx_fifo(struct dw_i3c_master *master, 3468c2ecf20Sopenharmony_ci u8 *bytes, int nbytes) 3478c2ecf20Sopenharmony_ci{ 3488c2ecf20Sopenharmony_ci readsl(master->regs + RX_TX_DATA_PORT, bytes, nbytes / 4); 3498c2ecf20Sopenharmony_ci if (nbytes & 3) { 3508c2ecf20Sopenharmony_ci u32 tmp; 3518c2ecf20Sopenharmony_ci 3528c2ecf20Sopenharmony_ci readsl(master->regs + RX_TX_DATA_PORT, &tmp, 1); 3538c2ecf20Sopenharmony_ci memcpy(bytes + (nbytes & ~3), &tmp, nbytes & 3); 3548c2ecf20Sopenharmony_ci } 3558c2ecf20Sopenharmony_ci} 3568c2ecf20Sopenharmony_ci 3578c2ecf20Sopenharmony_cistatic struct dw_i3c_xfer * 3588c2ecf20Sopenharmony_cidw_i3c_master_alloc_xfer(struct dw_i3c_master *master, unsigned int ncmds) 3598c2ecf20Sopenharmony_ci{ 3608c2ecf20Sopenharmony_ci struct dw_i3c_xfer *xfer; 3618c2ecf20Sopenharmony_ci 3628c2ecf20Sopenharmony_ci xfer = kzalloc(struct_size(xfer, cmds, ncmds), GFP_KERNEL); 3638c2ecf20Sopenharmony_ci if (!xfer) 3648c2ecf20Sopenharmony_ci return NULL; 3658c2ecf20Sopenharmony_ci 3668c2ecf20Sopenharmony_ci INIT_LIST_HEAD(&xfer->node); 3678c2ecf20Sopenharmony_ci xfer->ncmds = ncmds; 3688c2ecf20Sopenharmony_ci xfer->ret = -ETIMEDOUT; 3698c2ecf20Sopenharmony_ci 3708c2ecf20Sopenharmony_ci return xfer; 3718c2ecf20Sopenharmony_ci} 3728c2ecf20Sopenharmony_ci 3738c2ecf20Sopenharmony_cistatic void dw_i3c_master_free_xfer(struct dw_i3c_xfer *xfer) 3748c2ecf20Sopenharmony_ci{ 3758c2ecf20Sopenharmony_ci kfree(xfer); 3768c2ecf20Sopenharmony_ci} 3778c2ecf20Sopenharmony_ci 3788c2ecf20Sopenharmony_cistatic void dw_i3c_master_start_xfer_locked(struct dw_i3c_master *master) 3798c2ecf20Sopenharmony_ci{ 3808c2ecf20Sopenharmony_ci struct dw_i3c_xfer *xfer = master->xferqueue.cur; 3818c2ecf20Sopenharmony_ci unsigned int i; 3828c2ecf20Sopenharmony_ci u32 thld_ctrl; 3838c2ecf20Sopenharmony_ci 3848c2ecf20Sopenharmony_ci if (!xfer) 3858c2ecf20Sopenharmony_ci return; 3868c2ecf20Sopenharmony_ci 3878c2ecf20Sopenharmony_ci for (i = 0; i < xfer->ncmds; i++) { 3888c2ecf20Sopenharmony_ci struct dw_i3c_cmd *cmd = &xfer->cmds[i]; 3898c2ecf20Sopenharmony_ci 3908c2ecf20Sopenharmony_ci dw_i3c_master_wr_tx_fifo(master, cmd->tx_buf, cmd->tx_len); 3918c2ecf20Sopenharmony_ci } 3928c2ecf20Sopenharmony_ci 3938c2ecf20Sopenharmony_ci thld_ctrl = readl(master->regs + QUEUE_THLD_CTRL); 3948c2ecf20Sopenharmony_ci thld_ctrl &= ~QUEUE_THLD_CTRL_RESP_BUF_MASK; 3958c2ecf20Sopenharmony_ci thld_ctrl |= QUEUE_THLD_CTRL_RESP_BUF(xfer->ncmds); 3968c2ecf20Sopenharmony_ci writel(thld_ctrl, master->regs + QUEUE_THLD_CTRL); 3978c2ecf20Sopenharmony_ci 3988c2ecf20Sopenharmony_ci for (i = 0; i < xfer->ncmds; i++) { 3998c2ecf20Sopenharmony_ci struct dw_i3c_cmd *cmd = &xfer->cmds[i]; 4008c2ecf20Sopenharmony_ci 4018c2ecf20Sopenharmony_ci writel(cmd->cmd_hi, master->regs + COMMAND_QUEUE_PORT); 4028c2ecf20Sopenharmony_ci writel(cmd->cmd_lo, master->regs + COMMAND_QUEUE_PORT); 4038c2ecf20Sopenharmony_ci } 4048c2ecf20Sopenharmony_ci} 4058c2ecf20Sopenharmony_ci 4068c2ecf20Sopenharmony_cistatic void dw_i3c_master_enqueue_xfer(struct dw_i3c_master *master, 4078c2ecf20Sopenharmony_ci struct dw_i3c_xfer *xfer) 4088c2ecf20Sopenharmony_ci{ 4098c2ecf20Sopenharmony_ci unsigned long flags; 4108c2ecf20Sopenharmony_ci 4118c2ecf20Sopenharmony_ci init_completion(&xfer->comp); 4128c2ecf20Sopenharmony_ci spin_lock_irqsave(&master->xferqueue.lock, flags); 4138c2ecf20Sopenharmony_ci if (master->xferqueue.cur) { 4148c2ecf20Sopenharmony_ci list_add_tail(&xfer->node, &master->xferqueue.list); 4158c2ecf20Sopenharmony_ci } else { 4168c2ecf20Sopenharmony_ci master->xferqueue.cur = xfer; 4178c2ecf20Sopenharmony_ci dw_i3c_master_start_xfer_locked(master); 4188c2ecf20Sopenharmony_ci } 4198c2ecf20Sopenharmony_ci spin_unlock_irqrestore(&master->xferqueue.lock, flags); 4208c2ecf20Sopenharmony_ci} 4218c2ecf20Sopenharmony_ci 4228c2ecf20Sopenharmony_cistatic void dw_i3c_master_dequeue_xfer_locked(struct dw_i3c_master *master, 4238c2ecf20Sopenharmony_ci struct dw_i3c_xfer *xfer) 4248c2ecf20Sopenharmony_ci{ 4258c2ecf20Sopenharmony_ci if (master->xferqueue.cur == xfer) { 4268c2ecf20Sopenharmony_ci u32 status; 4278c2ecf20Sopenharmony_ci 4288c2ecf20Sopenharmony_ci master->xferqueue.cur = NULL; 4298c2ecf20Sopenharmony_ci 4308c2ecf20Sopenharmony_ci writel(RESET_CTRL_RX_FIFO | RESET_CTRL_TX_FIFO | 4318c2ecf20Sopenharmony_ci RESET_CTRL_RESP_QUEUE | RESET_CTRL_CMD_QUEUE, 4328c2ecf20Sopenharmony_ci master->regs + RESET_CTRL); 4338c2ecf20Sopenharmony_ci 4348c2ecf20Sopenharmony_ci readl_poll_timeout_atomic(master->regs + RESET_CTRL, status, 4358c2ecf20Sopenharmony_ci !status, 10, 1000000); 4368c2ecf20Sopenharmony_ci } else { 4378c2ecf20Sopenharmony_ci list_del_init(&xfer->node); 4388c2ecf20Sopenharmony_ci } 4398c2ecf20Sopenharmony_ci} 4408c2ecf20Sopenharmony_ci 4418c2ecf20Sopenharmony_cistatic void dw_i3c_master_dequeue_xfer(struct dw_i3c_master *master, 4428c2ecf20Sopenharmony_ci struct dw_i3c_xfer *xfer) 4438c2ecf20Sopenharmony_ci{ 4448c2ecf20Sopenharmony_ci unsigned long flags; 4458c2ecf20Sopenharmony_ci 4468c2ecf20Sopenharmony_ci spin_lock_irqsave(&master->xferqueue.lock, flags); 4478c2ecf20Sopenharmony_ci dw_i3c_master_dequeue_xfer_locked(master, xfer); 4488c2ecf20Sopenharmony_ci spin_unlock_irqrestore(&master->xferqueue.lock, flags); 4498c2ecf20Sopenharmony_ci} 4508c2ecf20Sopenharmony_ci 4518c2ecf20Sopenharmony_cistatic void dw_i3c_master_end_xfer_locked(struct dw_i3c_master *master, u32 isr) 4528c2ecf20Sopenharmony_ci{ 4538c2ecf20Sopenharmony_ci struct dw_i3c_xfer *xfer = master->xferqueue.cur; 4548c2ecf20Sopenharmony_ci int i, ret = 0; 4558c2ecf20Sopenharmony_ci u32 nresp; 4568c2ecf20Sopenharmony_ci 4578c2ecf20Sopenharmony_ci if (!xfer) 4588c2ecf20Sopenharmony_ci return; 4598c2ecf20Sopenharmony_ci 4608c2ecf20Sopenharmony_ci nresp = readl(master->regs + QUEUE_STATUS_LEVEL); 4618c2ecf20Sopenharmony_ci nresp = QUEUE_STATUS_LEVEL_RESP(nresp); 4628c2ecf20Sopenharmony_ci 4638c2ecf20Sopenharmony_ci for (i = 0; i < nresp; i++) { 4648c2ecf20Sopenharmony_ci struct dw_i3c_cmd *cmd; 4658c2ecf20Sopenharmony_ci u32 resp; 4668c2ecf20Sopenharmony_ci 4678c2ecf20Sopenharmony_ci resp = readl(master->regs + RESPONSE_QUEUE_PORT); 4688c2ecf20Sopenharmony_ci 4698c2ecf20Sopenharmony_ci cmd = &xfer->cmds[RESPONSE_PORT_TID(resp)]; 4708c2ecf20Sopenharmony_ci cmd->rx_len = RESPONSE_PORT_DATA_LEN(resp); 4718c2ecf20Sopenharmony_ci cmd->error = RESPONSE_PORT_ERR_STATUS(resp); 4728c2ecf20Sopenharmony_ci if (cmd->rx_len && !cmd->error) 4738c2ecf20Sopenharmony_ci dw_i3c_master_read_rx_fifo(master, cmd->rx_buf, 4748c2ecf20Sopenharmony_ci cmd->rx_len); 4758c2ecf20Sopenharmony_ci } 4768c2ecf20Sopenharmony_ci 4778c2ecf20Sopenharmony_ci for (i = 0; i < nresp; i++) { 4788c2ecf20Sopenharmony_ci switch (xfer->cmds[i].error) { 4798c2ecf20Sopenharmony_ci case RESPONSE_NO_ERROR: 4808c2ecf20Sopenharmony_ci break; 4818c2ecf20Sopenharmony_ci case RESPONSE_ERROR_PARITY: 4828c2ecf20Sopenharmony_ci case RESPONSE_ERROR_IBA_NACK: 4838c2ecf20Sopenharmony_ci case RESPONSE_ERROR_TRANSF_ABORT: 4848c2ecf20Sopenharmony_ci case RESPONSE_ERROR_CRC: 4858c2ecf20Sopenharmony_ci case RESPONSE_ERROR_FRAME: 4868c2ecf20Sopenharmony_ci ret = -EIO; 4878c2ecf20Sopenharmony_ci break; 4888c2ecf20Sopenharmony_ci case RESPONSE_ERROR_OVER_UNDER_FLOW: 4898c2ecf20Sopenharmony_ci ret = -ENOSPC; 4908c2ecf20Sopenharmony_ci break; 4918c2ecf20Sopenharmony_ci case RESPONSE_ERROR_I2C_W_NACK_ERR: 4928c2ecf20Sopenharmony_ci case RESPONSE_ERROR_ADDRESS_NACK: 4938c2ecf20Sopenharmony_ci default: 4948c2ecf20Sopenharmony_ci ret = -EINVAL; 4958c2ecf20Sopenharmony_ci break; 4968c2ecf20Sopenharmony_ci } 4978c2ecf20Sopenharmony_ci } 4988c2ecf20Sopenharmony_ci 4998c2ecf20Sopenharmony_ci xfer->ret = ret; 5008c2ecf20Sopenharmony_ci complete(&xfer->comp); 5018c2ecf20Sopenharmony_ci 5028c2ecf20Sopenharmony_ci if (ret < 0) { 5038c2ecf20Sopenharmony_ci dw_i3c_master_dequeue_xfer_locked(master, xfer); 5048c2ecf20Sopenharmony_ci writel(readl(master->regs + DEVICE_CTRL) | DEV_CTRL_RESUME, 5058c2ecf20Sopenharmony_ci master->regs + DEVICE_CTRL); 5068c2ecf20Sopenharmony_ci } 5078c2ecf20Sopenharmony_ci 5088c2ecf20Sopenharmony_ci xfer = list_first_entry_or_null(&master->xferqueue.list, 5098c2ecf20Sopenharmony_ci struct dw_i3c_xfer, 5108c2ecf20Sopenharmony_ci node); 5118c2ecf20Sopenharmony_ci if (xfer) 5128c2ecf20Sopenharmony_ci list_del_init(&xfer->node); 5138c2ecf20Sopenharmony_ci 5148c2ecf20Sopenharmony_ci master->xferqueue.cur = xfer; 5158c2ecf20Sopenharmony_ci dw_i3c_master_start_xfer_locked(master); 5168c2ecf20Sopenharmony_ci} 5178c2ecf20Sopenharmony_ci 5188c2ecf20Sopenharmony_cistatic int dw_i3c_clk_cfg(struct dw_i3c_master *master) 5198c2ecf20Sopenharmony_ci{ 5208c2ecf20Sopenharmony_ci unsigned long core_rate, core_period; 5218c2ecf20Sopenharmony_ci u32 scl_timing; 5228c2ecf20Sopenharmony_ci u8 hcnt, lcnt; 5238c2ecf20Sopenharmony_ci 5248c2ecf20Sopenharmony_ci core_rate = clk_get_rate(master->core_clk); 5258c2ecf20Sopenharmony_ci if (!core_rate) 5268c2ecf20Sopenharmony_ci return -EINVAL; 5278c2ecf20Sopenharmony_ci 5288c2ecf20Sopenharmony_ci core_period = DIV_ROUND_UP(1000000000, core_rate); 5298c2ecf20Sopenharmony_ci 5308c2ecf20Sopenharmony_ci hcnt = DIV_ROUND_UP(I3C_BUS_THIGH_MAX_NS, core_period) - 1; 5318c2ecf20Sopenharmony_ci if (hcnt < SCL_I3C_TIMING_CNT_MIN) 5328c2ecf20Sopenharmony_ci hcnt = SCL_I3C_TIMING_CNT_MIN; 5338c2ecf20Sopenharmony_ci 5348c2ecf20Sopenharmony_ci lcnt = DIV_ROUND_UP(core_rate, I3C_BUS_TYP_I3C_SCL_RATE) - hcnt; 5358c2ecf20Sopenharmony_ci if (lcnt < SCL_I3C_TIMING_CNT_MIN) 5368c2ecf20Sopenharmony_ci lcnt = SCL_I3C_TIMING_CNT_MIN; 5378c2ecf20Sopenharmony_ci 5388c2ecf20Sopenharmony_ci scl_timing = SCL_I3C_TIMING_HCNT(hcnt) | SCL_I3C_TIMING_LCNT(lcnt); 5398c2ecf20Sopenharmony_ci writel(scl_timing, master->regs + SCL_I3C_PP_TIMING); 5408c2ecf20Sopenharmony_ci 5418c2ecf20Sopenharmony_ci if (!(readl(master->regs + DEVICE_CTRL) & DEV_CTRL_I2C_SLAVE_PRESENT)) 5428c2ecf20Sopenharmony_ci writel(BUS_I3C_MST_FREE(lcnt), master->regs + BUS_FREE_TIMING); 5438c2ecf20Sopenharmony_ci 5448c2ecf20Sopenharmony_ci lcnt = DIV_ROUND_UP(I3C_BUS_TLOW_OD_MIN_NS, core_period); 5458c2ecf20Sopenharmony_ci scl_timing = SCL_I3C_TIMING_HCNT(hcnt) | SCL_I3C_TIMING_LCNT(lcnt); 5468c2ecf20Sopenharmony_ci writel(scl_timing, master->regs + SCL_I3C_OD_TIMING); 5478c2ecf20Sopenharmony_ci 5488c2ecf20Sopenharmony_ci lcnt = DIV_ROUND_UP(core_rate, I3C_BUS_SDR1_SCL_RATE) - hcnt; 5498c2ecf20Sopenharmony_ci scl_timing = SCL_EXT_LCNT_1(lcnt); 5508c2ecf20Sopenharmony_ci lcnt = DIV_ROUND_UP(core_rate, I3C_BUS_SDR2_SCL_RATE) - hcnt; 5518c2ecf20Sopenharmony_ci scl_timing |= SCL_EXT_LCNT_2(lcnt); 5528c2ecf20Sopenharmony_ci lcnt = DIV_ROUND_UP(core_rate, I3C_BUS_SDR3_SCL_RATE) - hcnt; 5538c2ecf20Sopenharmony_ci scl_timing |= SCL_EXT_LCNT_3(lcnt); 5548c2ecf20Sopenharmony_ci lcnt = DIV_ROUND_UP(core_rate, I3C_BUS_SDR4_SCL_RATE) - hcnt; 5558c2ecf20Sopenharmony_ci scl_timing |= SCL_EXT_LCNT_4(lcnt); 5568c2ecf20Sopenharmony_ci writel(scl_timing, master->regs + SCL_EXT_LCNT_TIMING); 5578c2ecf20Sopenharmony_ci 5588c2ecf20Sopenharmony_ci return 0; 5598c2ecf20Sopenharmony_ci} 5608c2ecf20Sopenharmony_ci 5618c2ecf20Sopenharmony_cistatic int dw_i2c_clk_cfg(struct dw_i3c_master *master) 5628c2ecf20Sopenharmony_ci{ 5638c2ecf20Sopenharmony_ci unsigned long core_rate, core_period; 5648c2ecf20Sopenharmony_ci u16 hcnt, lcnt; 5658c2ecf20Sopenharmony_ci u32 scl_timing; 5668c2ecf20Sopenharmony_ci 5678c2ecf20Sopenharmony_ci core_rate = clk_get_rate(master->core_clk); 5688c2ecf20Sopenharmony_ci if (!core_rate) 5698c2ecf20Sopenharmony_ci return -EINVAL; 5708c2ecf20Sopenharmony_ci 5718c2ecf20Sopenharmony_ci core_period = DIV_ROUND_UP(1000000000, core_rate); 5728c2ecf20Sopenharmony_ci 5738c2ecf20Sopenharmony_ci lcnt = DIV_ROUND_UP(I3C_BUS_I2C_FMP_TLOW_MIN_NS, core_period); 5748c2ecf20Sopenharmony_ci hcnt = DIV_ROUND_UP(core_rate, I3C_BUS_I2C_FM_PLUS_SCL_RATE) - lcnt; 5758c2ecf20Sopenharmony_ci scl_timing = SCL_I2C_FMP_TIMING_HCNT(hcnt) | 5768c2ecf20Sopenharmony_ci SCL_I2C_FMP_TIMING_LCNT(lcnt); 5778c2ecf20Sopenharmony_ci writel(scl_timing, master->regs + SCL_I2C_FMP_TIMING); 5788c2ecf20Sopenharmony_ci 5798c2ecf20Sopenharmony_ci lcnt = DIV_ROUND_UP(I3C_BUS_I2C_FM_TLOW_MIN_NS, core_period); 5808c2ecf20Sopenharmony_ci hcnt = DIV_ROUND_UP(core_rate, I3C_BUS_I2C_FM_SCL_RATE) - lcnt; 5818c2ecf20Sopenharmony_ci scl_timing = SCL_I2C_FM_TIMING_HCNT(hcnt) | 5828c2ecf20Sopenharmony_ci SCL_I2C_FM_TIMING_LCNT(lcnt); 5838c2ecf20Sopenharmony_ci writel(scl_timing, master->regs + SCL_I2C_FM_TIMING); 5848c2ecf20Sopenharmony_ci 5858c2ecf20Sopenharmony_ci writel(BUS_I3C_MST_FREE(lcnt), master->regs + BUS_FREE_TIMING); 5868c2ecf20Sopenharmony_ci writel(readl(master->regs + DEVICE_CTRL) | DEV_CTRL_I2C_SLAVE_PRESENT, 5878c2ecf20Sopenharmony_ci master->regs + DEVICE_CTRL); 5888c2ecf20Sopenharmony_ci 5898c2ecf20Sopenharmony_ci return 0; 5908c2ecf20Sopenharmony_ci} 5918c2ecf20Sopenharmony_ci 5928c2ecf20Sopenharmony_cistatic int dw_i3c_master_bus_init(struct i3c_master_controller *m) 5938c2ecf20Sopenharmony_ci{ 5948c2ecf20Sopenharmony_ci struct dw_i3c_master *master = to_dw_i3c_master(m); 5958c2ecf20Sopenharmony_ci struct i3c_bus *bus = i3c_master_get_bus(m); 5968c2ecf20Sopenharmony_ci struct i3c_device_info info = { }; 5978c2ecf20Sopenharmony_ci u32 thld_ctrl; 5988c2ecf20Sopenharmony_ci int ret; 5998c2ecf20Sopenharmony_ci 6008c2ecf20Sopenharmony_ci switch (bus->mode) { 6018c2ecf20Sopenharmony_ci case I3C_BUS_MODE_MIXED_FAST: 6028c2ecf20Sopenharmony_ci case I3C_BUS_MODE_MIXED_LIMITED: 6038c2ecf20Sopenharmony_ci ret = dw_i2c_clk_cfg(master); 6048c2ecf20Sopenharmony_ci if (ret) 6058c2ecf20Sopenharmony_ci return ret; 6068c2ecf20Sopenharmony_ci fallthrough; 6078c2ecf20Sopenharmony_ci case I3C_BUS_MODE_PURE: 6088c2ecf20Sopenharmony_ci ret = dw_i3c_clk_cfg(master); 6098c2ecf20Sopenharmony_ci if (ret) 6108c2ecf20Sopenharmony_ci return ret; 6118c2ecf20Sopenharmony_ci break; 6128c2ecf20Sopenharmony_ci default: 6138c2ecf20Sopenharmony_ci return -EINVAL; 6148c2ecf20Sopenharmony_ci } 6158c2ecf20Sopenharmony_ci 6168c2ecf20Sopenharmony_ci thld_ctrl = readl(master->regs + QUEUE_THLD_CTRL); 6178c2ecf20Sopenharmony_ci thld_ctrl &= ~QUEUE_THLD_CTRL_RESP_BUF_MASK; 6188c2ecf20Sopenharmony_ci writel(thld_ctrl, master->regs + QUEUE_THLD_CTRL); 6198c2ecf20Sopenharmony_ci 6208c2ecf20Sopenharmony_ci thld_ctrl = readl(master->regs + DATA_BUFFER_THLD_CTRL); 6218c2ecf20Sopenharmony_ci thld_ctrl &= ~DATA_BUFFER_THLD_CTRL_RX_BUF; 6228c2ecf20Sopenharmony_ci writel(thld_ctrl, master->regs + DATA_BUFFER_THLD_CTRL); 6238c2ecf20Sopenharmony_ci 6248c2ecf20Sopenharmony_ci writel(INTR_ALL, master->regs + INTR_STATUS); 6258c2ecf20Sopenharmony_ci writel(INTR_MASTER_MASK, master->regs + INTR_STATUS_EN); 6268c2ecf20Sopenharmony_ci writel(INTR_MASTER_MASK, master->regs + INTR_SIGNAL_EN); 6278c2ecf20Sopenharmony_ci 6288c2ecf20Sopenharmony_ci ret = i3c_master_get_free_addr(m, 0); 6298c2ecf20Sopenharmony_ci if (ret < 0) 6308c2ecf20Sopenharmony_ci return ret; 6318c2ecf20Sopenharmony_ci 6328c2ecf20Sopenharmony_ci writel(DEV_ADDR_DYNAMIC_ADDR_VALID | DEV_ADDR_DYNAMIC(ret), 6338c2ecf20Sopenharmony_ci master->regs + DEVICE_ADDR); 6348c2ecf20Sopenharmony_ci 6358c2ecf20Sopenharmony_ci memset(&info, 0, sizeof(info)); 6368c2ecf20Sopenharmony_ci info.dyn_addr = ret; 6378c2ecf20Sopenharmony_ci 6388c2ecf20Sopenharmony_ci ret = i3c_master_set_info(&master->base, &info); 6398c2ecf20Sopenharmony_ci if (ret) 6408c2ecf20Sopenharmony_ci return ret; 6418c2ecf20Sopenharmony_ci 6428c2ecf20Sopenharmony_ci writel(IBI_REQ_REJECT_ALL, master->regs + IBI_SIR_REQ_REJECT); 6438c2ecf20Sopenharmony_ci writel(IBI_REQ_REJECT_ALL, master->regs + IBI_MR_REQ_REJECT); 6448c2ecf20Sopenharmony_ci 6458c2ecf20Sopenharmony_ci /* For now don't support Hot-Join */ 6468c2ecf20Sopenharmony_ci writel(readl(master->regs + DEVICE_CTRL) | DEV_CTRL_HOT_JOIN_NACK, 6478c2ecf20Sopenharmony_ci master->regs + DEVICE_CTRL); 6488c2ecf20Sopenharmony_ci 6498c2ecf20Sopenharmony_ci dw_i3c_master_enable(master); 6508c2ecf20Sopenharmony_ci 6518c2ecf20Sopenharmony_ci return 0; 6528c2ecf20Sopenharmony_ci} 6538c2ecf20Sopenharmony_ci 6548c2ecf20Sopenharmony_cistatic void dw_i3c_master_bus_cleanup(struct i3c_master_controller *m) 6558c2ecf20Sopenharmony_ci{ 6568c2ecf20Sopenharmony_ci struct dw_i3c_master *master = to_dw_i3c_master(m); 6578c2ecf20Sopenharmony_ci 6588c2ecf20Sopenharmony_ci dw_i3c_master_disable(master); 6598c2ecf20Sopenharmony_ci} 6608c2ecf20Sopenharmony_ci 6618c2ecf20Sopenharmony_cistatic int dw_i3c_ccc_set(struct dw_i3c_master *master, 6628c2ecf20Sopenharmony_ci struct i3c_ccc_cmd *ccc) 6638c2ecf20Sopenharmony_ci{ 6648c2ecf20Sopenharmony_ci struct dw_i3c_xfer *xfer; 6658c2ecf20Sopenharmony_ci struct dw_i3c_cmd *cmd; 6668c2ecf20Sopenharmony_ci int ret, pos = 0; 6678c2ecf20Sopenharmony_ci 6688c2ecf20Sopenharmony_ci if (ccc->id & I3C_CCC_DIRECT) { 6698c2ecf20Sopenharmony_ci pos = dw_i3c_master_get_addr_pos(master, ccc->dests[0].addr); 6708c2ecf20Sopenharmony_ci if (pos < 0) 6718c2ecf20Sopenharmony_ci return pos; 6728c2ecf20Sopenharmony_ci } 6738c2ecf20Sopenharmony_ci 6748c2ecf20Sopenharmony_ci xfer = dw_i3c_master_alloc_xfer(master, 1); 6758c2ecf20Sopenharmony_ci if (!xfer) 6768c2ecf20Sopenharmony_ci return -ENOMEM; 6778c2ecf20Sopenharmony_ci 6788c2ecf20Sopenharmony_ci cmd = xfer->cmds; 6798c2ecf20Sopenharmony_ci cmd->tx_buf = ccc->dests[0].payload.data; 6808c2ecf20Sopenharmony_ci cmd->tx_len = ccc->dests[0].payload.len; 6818c2ecf20Sopenharmony_ci 6828c2ecf20Sopenharmony_ci cmd->cmd_hi = COMMAND_PORT_ARG_DATA_LEN(ccc->dests[0].payload.len) | 6838c2ecf20Sopenharmony_ci COMMAND_PORT_TRANSFER_ARG; 6848c2ecf20Sopenharmony_ci 6858c2ecf20Sopenharmony_ci cmd->cmd_lo = COMMAND_PORT_CP | 6868c2ecf20Sopenharmony_ci COMMAND_PORT_DEV_INDEX(pos) | 6878c2ecf20Sopenharmony_ci COMMAND_PORT_CMD(ccc->id) | 6888c2ecf20Sopenharmony_ci COMMAND_PORT_TOC | 6898c2ecf20Sopenharmony_ci COMMAND_PORT_ROC; 6908c2ecf20Sopenharmony_ci 6918c2ecf20Sopenharmony_ci dw_i3c_master_enqueue_xfer(master, xfer); 6928c2ecf20Sopenharmony_ci if (!wait_for_completion_timeout(&xfer->comp, XFER_TIMEOUT)) 6938c2ecf20Sopenharmony_ci dw_i3c_master_dequeue_xfer(master, xfer); 6948c2ecf20Sopenharmony_ci 6958c2ecf20Sopenharmony_ci ret = xfer->ret; 6968c2ecf20Sopenharmony_ci if (xfer->cmds[0].error == RESPONSE_ERROR_IBA_NACK) 6978c2ecf20Sopenharmony_ci ccc->err = I3C_ERROR_M2; 6988c2ecf20Sopenharmony_ci 6998c2ecf20Sopenharmony_ci dw_i3c_master_free_xfer(xfer); 7008c2ecf20Sopenharmony_ci 7018c2ecf20Sopenharmony_ci return ret; 7028c2ecf20Sopenharmony_ci} 7038c2ecf20Sopenharmony_ci 7048c2ecf20Sopenharmony_cistatic int dw_i3c_ccc_get(struct dw_i3c_master *master, struct i3c_ccc_cmd *ccc) 7058c2ecf20Sopenharmony_ci{ 7068c2ecf20Sopenharmony_ci struct dw_i3c_xfer *xfer; 7078c2ecf20Sopenharmony_ci struct dw_i3c_cmd *cmd; 7088c2ecf20Sopenharmony_ci int ret, pos; 7098c2ecf20Sopenharmony_ci 7108c2ecf20Sopenharmony_ci pos = dw_i3c_master_get_addr_pos(master, ccc->dests[0].addr); 7118c2ecf20Sopenharmony_ci if (pos < 0) 7128c2ecf20Sopenharmony_ci return pos; 7138c2ecf20Sopenharmony_ci 7148c2ecf20Sopenharmony_ci xfer = dw_i3c_master_alloc_xfer(master, 1); 7158c2ecf20Sopenharmony_ci if (!xfer) 7168c2ecf20Sopenharmony_ci return -ENOMEM; 7178c2ecf20Sopenharmony_ci 7188c2ecf20Sopenharmony_ci cmd = xfer->cmds; 7198c2ecf20Sopenharmony_ci cmd->rx_buf = ccc->dests[0].payload.data; 7208c2ecf20Sopenharmony_ci cmd->rx_len = ccc->dests[0].payload.len; 7218c2ecf20Sopenharmony_ci 7228c2ecf20Sopenharmony_ci cmd->cmd_hi = COMMAND_PORT_ARG_DATA_LEN(ccc->dests[0].payload.len) | 7238c2ecf20Sopenharmony_ci COMMAND_PORT_TRANSFER_ARG; 7248c2ecf20Sopenharmony_ci 7258c2ecf20Sopenharmony_ci cmd->cmd_lo = COMMAND_PORT_READ_TRANSFER | 7268c2ecf20Sopenharmony_ci COMMAND_PORT_CP | 7278c2ecf20Sopenharmony_ci COMMAND_PORT_DEV_INDEX(pos) | 7288c2ecf20Sopenharmony_ci COMMAND_PORT_CMD(ccc->id) | 7298c2ecf20Sopenharmony_ci COMMAND_PORT_TOC | 7308c2ecf20Sopenharmony_ci COMMAND_PORT_ROC; 7318c2ecf20Sopenharmony_ci 7328c2ecf20Sopenharmony_ci dw_i3c_master_enqueue_xfer(master, xfer); 7338c2ecf20Sopenharmony_ci if (!wait_for_completion_timeout(&xfer->comp, XFER_TIMEOUT)) 7348c2ecf20Sopenharmony_ci dw_i3c_master_dequeue_xfer(master, xfer); 7358c2ecf20Sopenharmony_ci 7368c2ecf20Sopenharmony_ci ret = xfer->ret; 7378c2ecf20Sopenharmony_ci if (xfer->cmds[0].error == RESPONSE_ERROR_IBA_NACK) 7388c2ecf20Sopenharmony_ci ccc->err = I3C_ERROR_M2; 7398c2ecf20Sopenharmony_ci dw_i3c_master_free_xfer(xfer); 7408c2ecf20Sopenharmony_ci 7418c2ecf20Sopenharmony_ci return ret; 7428c2ecf20Sopenharmony_ci} 7438c2ecf20Sopenharmony_ci 7448c2ecf20Sopenharmony_cistatic int dw_i3c_master_send_ccc_cmd(struct i3c_master_controller *m, 7458c2ecf20Sopenharmony_ci struct i3c_ccc_cmd *ccc) 7468c2ecf20Sopenharmony_ci{ 7478c2ecf20Sopenharmony_ci struct dw_i3c_master *master = to_dw_i3c_master(m); 7488c2ecf20Sopenharmony_ci int ret = 0; 7498c2ecf20Sopenharmony_ci 7508c2ecf20Sopenharmony_ci if (ccc->id == I3C_CCC_ENTDAA) 7518c2ecf20Sopenharmony_ci return -EINVAL; 7528c2ecf20Sopenharmony_ci 7538c2ecf20Sopenharmony_ci if (ccc->rnw) 7548c2ecf20Sopenharmony_ci ret = dw_i3c_ccc_get(master, ccc); 7558c2ecf20Sopenharmony_ci else 7568c2ecf20Sopenharmony_ci ret = dw_i3c_ccc_set(master, ccc); 7578c2ecf20Sopenharmony_ci 7588c2ecf20Sopenharmony_ci return ret; 7598c2ecf20Sopenharmony_ci} 7608c2ecf20Sopenharmony_ci 7618c2ecf20Sopenharmony_cistatic int dw_i3c_master_daa(struct i3c_master_controller *m) 7628c2ecf20Sopenharmony_ci{ 7638c2ecf20Sopenharmony_ci struct dw_i3c_master *master = to_dw_i3c_master(m); 7648c2ecf20Sopenharmony_ci struct dw_i3c_xfer *xfer; 7658c2ecf20Sopenharmony_ci struct dw_i3c_cmd *cmd; 7668c2ecf20Sopenharmony_ci u32 olddevs, newdevs; 7678c2ecf20Sopenharmony_ci u8 p, last_addr = 0; 7688c2ecf20Sopenharmony_ci int ret, pos; 7698c2ecf20Sopenharmony_ci 7708c2ecf20Sopenharmony_ci olddevs = ~(master->free_pos); 7718c2ecf20Sopenharmony_ci 7728c2ecf20Sopenharmony_ci /* Prepare DAT before launching DAA. */ 7738c2ecf20Sopenharmony_ci for (pos = 0; pos < master->maxdevs; pos++) { 7748c2ecf20Sopenharmony_ci if (olddevs & BIT(pos)) 7758c2ecf20Sopenharmony_ci continue; 7768c2ecf20Sopenharmony_ci 7778c2ecf20Sopenharmony_ci ret = i3c_master_get_free_addr(m, last_addr + 1); 7788c2ecf20Sopenharmony_ci if (ret < 0) 7798c2ecf20Sopenharmony_ci return -ENOSPC; 7808c2ecf20Sopenharmony_ci 7818c2ecf20Sopenharmony_ci master->addrs[pos] = ret; 7828c2ecf20Sopenharmony_ci p = even_parity(ret); 7838c2ecf20Sopenharmony_ci last_addr = ret; 7848c2ecf20Sopenharmony_ci ret |= (p << 7); 7858c2ecf20Sopenharmony_ci 7868c2ecf20Sopenharmony_ci writel(DEV_ADDR_TABLE_DYNAMIC_ADDR(ret), 7878c2ecf20Sopenharmony_ci master->regs + 7888c2ecf20Sopenharmony_ci DEV_ADDR_TABLE_LOC(master->datstartaddr, pos)); 7898c2ecf20Sopenharmony_ci } 7908c2ecf20Sopenharmony_ci 7918c2ecf20Sopenharmony_ci xfer = dw_i3c_master_alloc_xfer(master, 1); 7928c2ecf20Sopenharmony_ci if (!xfer) 7938c2ecf20Sopenharmony_ci return -ENOMEM; 7948c2ecf20Sopenharmony_ci 7958c2ecf20Sopenharmony_ci pos = dw_i3c_master_get_free_pos(master); 7968c2ecf20Sopenharmony_ci cmd = &xfer->cmds[0]; 7978c2ecf20Sopenharmony_ci cmd->cmd_hi = 0x1; 7988c2ecf20Sopenharmony_ci cmd->cmd_lo = COMMAND_PORT_DEV_COUNT(master->maxdevs - pos) | 7998c2ecf20Sopenharmony_ci COMMAND_PORT_DEV_INDEX(pos) | 8008c2ecf20Sopenharmony_ci COMMAND_PORT_CMD(I3C_CCC_ENTDAA) | 8018c2ecf20Sopenharmony_ci COMMAND_PORT_ADDR_ASSGN_CMD | 8028c2ecf20Sopenharmony_ci COMMAND_PORT_TOC | 8038c2ecf20Sopenharmony_ci COMMAND_PORT_ROC; 8048c2ecf20Sopenharmony_ci 8058c2ecf20Sopenharmony_ci dw_i3c_master_enqueue_xfer(master, xfer); 8068c2ecf20Sopenharmony_ci if (!wait_for_completion_timeout(&xfer->comp, XFER_TIMEOUT)) 8078c2ecf20Sopenharmony_ci dw_i3c_master_dequeue_xfer(master, xfer); 8088c2ecf20Sopenharmony_ci 8098c2ecf20Sopenharmony_ci newdevs = GENMASK(master->maxdevs - cmd->rx_len - 1, 0); 8108c2ecf20Sopenharmony_ci newdevs &= ~olddevs; 8118c2ecf20Sopenharmony_ci 8128c2ecf20Sopenharmony_ci for (pos = 0; pos < master->maxdevs; pos++) { 8138c2ecf20Sopenharmony_ci if (newdevs & BIT(pos)) 8148c2ecf20Sopenharmony_ci i3c_master_add_i3c_dev_locked(m, master->addrs[pos]); 8158c2ecf20Sopenharmony_ci } 8168c2ecf20Sopenharmony_ci 8178c2ecf20Sopenharmony_ci dw_i3c_master_free_xfer(xfer); 8188c2ecf20Sopenharmony_ci 8198c2ecf20Sopenharmony_ci i3c_master_disec_locked(m, I3C_BROADCAST_ADDR, 8208c2ecf20Sopenharmony_ci I3C_CCC_EVENT_HJ | 8218c2ecf20Sopenharmony_ci I3C_CCC_EVENT_MR | 8228c2ecf20Sopenharmony_ci I3C_CCC_EVENT_SIR); 8238c2ecf20Sopenharmony_ci 8248c2ecf20Sopenharmony_ci return 0; 8258c2ecf20Sopenharmony_ci} 8268c2ecf20Sopenharmony_ci 8278c2ecf20Sopenharmony_cistatic int dw_i3c_master_priv_xfers(struct i3c_dev_desc *dev, 8288c2ecf20Sopenharmony_ci struct i3c_priv_xfer *i3c_xfers, 8298c2ecf20Sopenharmony_ci int i3c_nxfers) 8308c2ecf20Sopenharmony_ci{ 8318c2ecf20Sopenharmony_ci struct dw_i3c_i2c_dev_data *data = i3c_dev_get_master_data(dev); 8328c2ecf20Sopenharmony_ci struct i3c_master_controller *m = i3c_dev_get_master(dev); 8338c2ecf20Sopenharmony_ci struct dw_i3c_master *master = to_dw_i3c_master(m); 8348c2ecf20Sopenharmony_ci unsigned int nrxwords = 0, ntxwords = 0; 8358c2ecf20Sopenharmony_ci struct dw_i3c_xfer *xfer; 8368c2ecf20Sopenharmony_ci int i, ret = 0; 8378c2ecf20Sopenharmony_ci 8388c2ecf20Sopenharmony_ci if (!i3c_nxfers) 8398c2ecf20Sopenharmony_ci return 0; 8408c2ecf20Sopenharmony_ci 8418c2ecf20Sopenharmony_ci if (i3c_nxfers > master->caps.cmdfifodepth) 8428c2ecf20Sopenharmony_ci return -ENOTSUPP; 8438c2ecf20Sopenharmony_ci 8448c2ecf20Sopenharmony_ci for (i = 0; i < i3c_nxfers; i++) { 8458c2ecf20Sopenharmony_ci if (i3c_xfers[i].rnw) 8468c2ecf20Sopenharmony_ci nrxwords += DIV_ROUND_UP(i3c_xfers[i].len, 4); 8478c2ecf20Sopenharmony_ci else 8488c2ecf20Sopenharmony_ci ntxwords += DIV_ROUND_UP(i3c_xfers[i].len, 4); 8498c2ecf20Sopenharmony_ci } 8508c2ecf20Sopenharmony_ci 8518c2ecf20Sopenharmony_ci if (ntxwords > master->caps.datafifodepth || 8528c2ecf20Sopenharmony_ci nrxwords > master->caps.datafifodepth) 8538c2ecf20Sopenharmony_ci return -ENOTSUPP; 8548c2ecf20Sopenharmony_ci 8558c2ecf20Sopenharmony_ci xfer = dw_i3c_master_alloc_xfer(master, i3c_nxfers); 8568c2ecf20Sopenharmony_ci if (!xfer) 8578c2ecf20Sopenharmony_ci return -ENOMEM; 8588c2ecf20Sopenharmony_ci 8598c2ecf20Sopenharmony_ci for (i = 0; i < i3c_nxfers; i++) { 8608c2ecf20Sopenharmony_ci struct dw_i3c_cmd *cmd = &xfer->cmds[i]; 8618c2ecf20Sopenharmony_ci 8628c2ecf20Sopenharmony_ci cmd->cmd_hi = COMMAND_PORT_ARG_DATA_LEN(i3c_xfers[i].len) | 8638c2ecf20Sopenharmony_ci COMMAND_PORT_TRANSFER_ARG; 8648c2ecf20Sopenharmony_ci 8658c2ecf20Sopenharmony_ci if (i3c_xfers[i].rnw) { 8668c2ecf20Sopenharmony_ci cmd->rx_buf = i3c_xfers[i].data.in; 8678c2ecf20Sopenharmony_ci cmd->rx_len = i3c_xfers[i].len; 8688c2ecf20Sopenharmony_ci cmd->cmd_lo = COMMAND_PORT_READ_TRANSFER | 8698c2ecf20Sopenharmony_ci COMMAND_PORT_SPEED(dev->info.max_read_ds); 8708c2ecf20Sopenharmony_ci 8718c2ecf20Sopenharmony_ci } else { 8728c2ecf20Sopenharmony_ci cmd->tx_buf = i3c_xfers[i].data.out; 8738c2ecf20Sopenharmony_ci cmd->tx_len = i3c_xfers[i].len; 8748c2ecf20Sopenharmony_ci cmd->cmd_lo = 8758c2ecf20Sopenharmony_ci COMMAND_PORT_SPEED(dev->info.max_write_ds); 8768c2ecf20Sopenharmony_ci } 8778c2ecf20Sopenharmony_ci 8788c2ecf20Sopenharmony_ci cmd->cmd_lo |= COMMAND_PORT_TID(i) | 8798c2ecf20Sopenharmony_ci COMMAND_PORT_DEV_INDEX(data->index) | 8808c2ecf20Sopenharmony_ci COMMAND_PORT_ROC; 8818c2ecf20Sopenharmony_ci 8828c2ecf20Sopenharmony_ci if (i == (i3c_nxfers - 1)) 8838c2ecf20Sopenharmony_ci cmd->cmd_lo |= COMMAND_PORT_TOC; 8848c2ecf20Sopenharmony_ci } 8858c2ecf20Sopenharmony_ci 8868c2ecf20Sopenharmony_ci dw_i3c_master_enqueue_xfer(master, xfer); 8878c2ecf20Sopenharmony_ci if (!wait_for_completion_timeout(&xfer->comp, XFER_TIMEOUT)) 8888c2ecf20Sopenharmony_ci dw_i3c_master_dequeue_xfer(master, xfer); 8898c2ecf20Sopenharmony_ci 8908c2ecf20Sopenharmony_ci ret = xfer->ret; 8918c2ecf20Sopenharmony_ci dw_i3c_master_free_xfer(xfer); 8928c2ecf20Sopenharmony_ci 8938c2ecf20Sopenharmony_ci return ret; 8948c2ecf20Sopenharmony_ci} 8958c2ecf20Sopenharmony_ci 8968c2ecf20Sopenharmony_cistatic int dw_i3c_master_reattach_i3c_dev(struct i3c_dev_desc *dev, 8978c2ecf20Sopenharmony_ci u8 old_dyn_addr) 8988c2ecf20Sopenharmony_ci{ 8998c2ecf20Sopenharmony_ci struct dw_i3c_i2c_dev_data *data = i3c_dev_get_master_data(dev); 9008c2ecf20Sopenharmony_ci struct i3c_master_controller *m = i3c_dev_get_master(dev); 9018c2ecf20Sopenharmony_ci struct dw_i3c_master *master = to_dw_i3c_master(m); 9028c2ecf20Sopenharmony_ci int pos; 9038c2ecf20Sopenharmony_ci 9048c2ecf20Sopenharmony_ci pos = dw_i3c_master_get_free_pos(master); 9058c2ecf20Sopenharmony_ci 9068c2ecf20Sopenharmony_ci if (data->index > pos && pos > 0) { 9078c2ecf20Sopenharmony_ci writel(0, 9088c2ecf20Sopenharmony_ci master->regs + 9098c2ecf20Sopenharmony_ci DEV_ADDR_TABLE_LOC(master->datstartaddr, data->index)); 9108c2ecf20Sopenharmony_ci 9118c2ecf20Sopenharmony_ci master->addrs[data->index] = 0; 9128c2ecf20Sopenharmony_ci master->free_pos |= BIT(data->index); 9138c2ecf20Sopenharmony_ci 9148c2ecf20Sopenharmony_ci data->index = pos; 9158c2ecf20Sopenharmony_ci master->addrs[pos] = dev->info.dyn_addr; 9168c2ecf20Sopenharmony_ci master->free_pos &= ~BIT(pos); 9178c2ecf20Sopenharmony_ci } 9188c2ecf20Sopenharmony_ci 9198c2ecf20Sopenharmony_ci writel(DEV_ADDR_TABLE_DYNAMIC_ADDR(dev->info.dyn_addr), 9208c2ecf20Sopenharmony_ci master->regs + 9218c2ecf20Sopenharmony_ci DEV_ADDR_TABLE_LOC(master->datstartaddr, data->index)); 9228c2ecf20Sopenharmony_ci 9238c2ecf20Sopenharmony_ci master->addrs[data->index] = dev->info.dyn_addr; 9248c2ecf20Sopenharmony_ci 9258c2ecf20Sopenharmony_ci return 0; 9268c2ecf20Sopenharmony_ci} 9278c2ecf20Sopenharmony_ci 9288c2ecf20Sopenharmony_cistatic int dw_i3c_master_attach_i3c_dev(struct i3c_dev_desc *dev) 9298c2ecf20Sopenharmony_ci{ 9308c2ecf20Sopenharmony_ci struct i3c_master_controller *m = i3c_dev_get_master(dev); 9318c2ecf20Sopenharmony_ci struct dw_i3c_master *master = to_dw_i3c_master(m); 9328c2ecf20Sopenharmony_ci struct dw_i3c_i2c_dev_data *data; 9338c2ecf20Sopenharmony_ci int pos; 9348c2ecf20Sopenharmony_ci 9358c2ecf20Sopenharmony_ci pos = dw_i3c_master_get_free_pos(master); 9368c2ecf20Sopenharmony_ci if (pos < 0) 9378c2ecf20Sopenharmony_ci return pos; 9388c2ecf20Sopenharmony_ci 9398c2ecf20Sopenharmony_ci data = kzalloc(sizeof(*data), GFP_KERNEL); 9408c2ecf20Sopenharmony_ci if (!data) 9418c2ecf20Sopenharmony_ci return -ENOMEM; 9428c2ecf20Sopenharmony_ci 9438c2ecf20Sopenharmony_ci data->index = pos; 9448c2ecf20Sopenharmony_ci master->addrs[pos] = dev->info.dyn_addr ? : dev->info.static_addr; 9458c2ecf20Sopenharmony_ci master->free_pos &= ~BIT(pos); 9468c2ecf20Sopenharmony_ci i3c_dev_set_master_data(dev, data); 9478c2ecf20Sopenharmony_ci 9488c2ecf20Sopenharmony_ci writel(DEV_ADDR_TABLE_DYNAMIC_ADDR(master->addrs[pos]), 9498c2ecf20Sopenharmony_ci master->regs + 9508c2ecf20Sopenharmony_ci DEV_ADDR_TABLE_LOC(master->datstartaddr, data->index)); 9518c2ecf20Sopenharmony_ci 9528c2ecf20Sopenharmony_ci return 0; 9538c2ecf20Sopenharmony_ci} 9548c2ecf20Sopenharmony_ci 9558c2ecf20Sopenharmony_cistatic void dw_i3c_master_detach_i3c_dev(struct i3c_dev_desc *dev) 9568c2ecf20Sopenharmony_ci{ 9578c2ecf20Sopenharmony_ci struct dw_i3c_i2c_dev_data *data = i3c_dev_get_master_data(dev); 9588c2ecf20Sopenharmony_ci struct i3c_master_controller *m = i3c_dev_get_master(dev); 9598c2ecf20Sopenharmony_ci struct dw_i3c_master *master = to_dw_i3c_master(m); 9608c2ecf20Sopenharmony_ci 9618c2ecf20Sopenharmony_ci writel(0, 9628c2ecf20Sopenharmony_ci master->regs + 9638c2ecf20Sopenharmony_ci DEV_ADDR_TABLE_LOC(master->datstartaddr, data->index)); 9648c2ecf20Sopenharmony_ci 9658c2ecf20Sopenharmony_ci i3c_dev_set_master_data(dev, NULL); 9668c2ecf20Sopenharmony_ci master->addrs[data->index] = 0; 9678c2ecf20Sopenharmony_ci master->free_pos |= BIT(data->index); 9688c2ecf20Sopenharmony_ci kfree(data); 9698c2ecf20Sopenharmony_ci} 9708c2ecf20Sopenharmony_ci 9718c2ecf20Sopenharmony_cistatic int dw_i3c_master_i2c_xfers(struct i2c_dev_desc *dev, 9728c2ecf20Sopenharmony_ci const struct i2c_msg *i2c_xfers, 9738c2ecf20Sopenharmony_ci int i2c_nxfers) 9748c2ecf20Sopenharmony_ci{ 9758c2ecf20Sopenharmony_ci struct dw_i3c_i2c_dev_data *data = i2c_dev_get_master_data(dev); 9768c2ecf20Sopenharmony_ci struct i3c_master_controller *m = i2c_dev_get_master(dev); 9778c2ecf20Sopenharmony_ci struct dw_i3c_master *master = to_dw_i3c_master(m); 9788c2ecf20Sopenharmony_ci unsigned int nrxwords = 0, ntxwords = 0; 9798c2ecf20Sopenharmony_ci struct dw_i3c_xfer *xfer; 9808c2ecf20Sopenharmony_ci int i, ret = 0; 9818c2ecf20Sopenharmony_ci 9828c2ecf20Sopenharmony_ci if (!i2c_nxfers) 9838c2ecf20Sopenharmony_ci return 0; 9848c2ecf20Sopenharmony_ci 9858c2ecf20Sopenharmony_ci if (i2c_nxfers > master->caps.cmdfifodepth) 9868c2ecf20Sopenharmony_ci return -ENOTSUPP; 9878c2ecf20Sopenharmony_ci 9888c2ecf20Sopenharmony_ci for (i = 0; i < i2c_nxfers; i++) { 9898c2ecf20Sopenharmony_ci if (i2c_xfers[i].flags & I2C_M_RD) 9908c2ecf20Sopenharmony_ci nrxwords += DIV_ROUND_UP(i2c_xfers[i].len, 4); 9918c2ecf20Sopenharmony_ci else 9928c2ecf20Sopenharmony_ci ntxwords += DIV_ROUND_UP(i2c_xfers[i].len, 4); 9938c2ecf20Sopenharmony_ci } 9948c2ecf20Sopenharmony_ci 9958c2ecf20Sopenharmony_ci if (ntxwords > master->caps.datafifodepth || 9968c2ecf20Sopenharmony_ci nrxwords > master->caps.datafifodepth) 9978c2ecf20Sopenharmony_ci return -ENOTSUPP; 9988c2ecf20Sopenharmony_ci 9998c2ecf20Sopenharmony_ci xfer = dw_i3c_master_alloc_xfer(master, i2c_nxfers); 10008c2ecf20Sopenharmony_ci if (!xfer) 10018c2ecf20Sopenharmony_ci return -ENOMEM; 10028c2ecf20Sopenharmony_ci 10038c2ecf20Sopenharmony_ci for (i = 0; i < i2c_nxfers; i++) { 10048c2ecf20Sopenharmony_ci struct dw_i3c_cmd *cmd = &xfer->cmds[i]; 10058c2ecf20Sopenharmony_ci 10068c2ecf20Sopenharmony_ci cmd->cmd_hi = COMMAND_PORT_ARG_DATA_LEN(i2c_xfers[i].len) | 10078c2ecf20Sopenharmony_ci COMMAND_PORT_TRANSFER_ARG; 10088c2ecf20Sopenharmony_ci 10098c2ecf20Sopenharmony_ci cmd->cmd_lo = COMMAND_PORT_TID(i) | 10108c2ecf20Sopenharmony_ci COMMAND_PORT_DEV_INDEX(data->index) | 10118c2ecf20Sopenharmony_ci COMMAND_PORT_ROC; 10128c2ecf20Sopenharmony_ci 10138c2ecf20Sopenharmony_ci if (i2c_xfers[i].flags & I2C_M_RD) { 10148c2ecf20Sopenharmony_ci cmd->cmd_lo |= COMMAND_PORT_READ_TRANSFER; 10158c2ecf20Sopenharmony_ci cmd->rx_buf = i2c_xfers[i].buf; 10168c2ecf20Sopenharmony_ci cmd->rx_len = i2c_xfers[i].len; 10178c2ecf20Sopenharmony_ci } else { 10188c2ecf20Sopenharmony_ci cmd->tx_buf = i2c_xfers[i].buf; 10198c2ecf20Sopenharmony_ci cmd->tx_len = i2c_xfers[i].len; 10208c2ecf20Sopenharmony_ci } 10218c2ecf20Sopenharmony_ci 10228c2ecf20Sopenharmony_ci if (i == (i2c_nxfers - 1)) 10238c2ecf20Sopenharmony_ci cmd->cmd_lo |= COMMAND_PORT_TOC; 10248c2ecf20Sopenharmony_ci } 10258c2ecf20Sopenharmony_ci 10268c2ecf20Sopenharmony_ci dw_i3c_master_enqueue_xfer(master, xfer); 10278c2ecf20Sopenharmony_ci if (!wait_for_completion_timeout(&xfer->comp, XFER_TIMEOUT)) 10288c2ecf20Sopenharmony_ci dw_i3c_master_dequeue_xfer(master, xfer); 10298c2ecf20Sopenharmony_ci 10308c2ecf20Sopenharmony_ci ret = xfer->ret; 10318c2ecf20Sopenharmony_ci dw_i3c_master_free_xfer(xfer); 10328c2ecf20Sopenharmony_ci 10338c2ecf20Sopenharmony_ci return ret; 10348c2ecf20Sopenharmony_ci} 10358c2ecf20Sopenharmony_ci 10368c2ecf20Sopenharmony_cistatic int dw_i3c_master_attach_i2c_dev(struct i2c_dev_desc *dev) 10378c2ecf20Sopenharmony_ci{ 10388c2ecf20Sopenharmony_ci struct i3c_master_controller *m = i2c_dev_get_master(dev); 10398c2ecf20Sopenharmony_ci struct dw_i3c_master *master = to_dw_i3c_master(m); 10408c2ecf20Sopenharmony_ci struct dw_i3c_i2c_dev_data *data; 10418c2ecf20Sopenharmony_ci int pos; 10428c2ecf20Sopenharmony_ci 10438c2ecf20Sopenharmony_ci pos = dw_i3c_master_get_free_pos(master); 10448c2ecf20Sopenharmony_ci if (pos < 0) 10458c2ecf20Sopenharmony_ci return pos; 10468c2ecf20Sopenharmony_ci 10478c2ecf20Sopenharmony_ci data = kzalloc(sizeof(*data), GFP_KERNEL); 10488c2ecf20Sopenharmony_ci if (!data) 10498c2ecf20Sopenharmony_ci return -ENOMEM; 10508c2ecf20Sopenharmony_ci 10518c2ecf20Sopenharmony_ci data->index = pos; 10528c2ecf20Sopenharmony_ci master->addrs[pos] = dev->addr; 10538c2ecf20Sopenharmony_ci master->free_pos &= ~BIT(pos); 10548c2ecf20Sopenharmony_ci i2c_dev_set_master_data(dev, data); 10558c2ecf20Sopenharmony_ci 10568c2ecf20Sopenharmony_ci writel(DEV_ADDR_TABLE_LEGACY_I2C_DEV | 10578c2ecf20Sopenharmony_ci DEV_ADDR_TABLE_STATIC_ADDR(dev->addr), 10588c2ecf20Sopenharmony_ci master->regs + 10598c2ecf20Sopenharmony_ci DEV_ADDR_TABLE_LOC(master->datstartaddr, data->index)); 10608c2ecf20Sopenharmony_ci 10618c2ecf20Sopenharmony_ci return 0; 10628c2ecf20Sopenharmony_ci} 10638c2ecf20Sopenharmony_ci 10648c2ecf20Sopenharmony_cistatic void dw_i3c_master_detach_i2c_dev(struct i2c_dev_desc *dev) 10658c2ecf20Sopenharmony_ci{ 10668c2ecf20Sopenharmony_ci struct dw_i3c_i2c_dev_data *data = i2c_dev_get_master_data(dev); 10678c2ecf20Sopenharmony_ci struct i3c_master_controller *m = i2c_dev_get_master(dev); 10688c2ecf20Sopenharmony_ci struct dw_i3c_master *master = to_dw_i3c_master(m); 10698c2ecf20Sopenharmony_ci 10708c2ecf20Sopenharmony_ci writel(0, 10718c2ecf20Sopenharmony_ci master->regs + 10728c2ecf20Sopenharmony_ci DEV_ADDR_TABLE_LOC(master->datstartaddr, data->index)); 10738c2ecf20Sopenharmony_ci 10748c2ecf20Sopenharmony_ci i2c_dev_set_master_data(dev, NULL); 10758c2ecf20Sopenharmony_ci master->addrs[data->index] = 0; 10768c2ecf20Sopenharmony_ci master->free_pos |= BIT(data->index); 10778c2ecf20Sopenharmony_ci kfree(data); 10788c2ecf20Sopenharmony_ci} 10798c2ecf20Sopenharmony_ci 10808c2ecf20Sopenharmony_cistatic irqreturn_t dw_i3c_master_irq_handler(int irq, void *dev_id) 10818c2ecf20Sopenharmony_ci{ 10828c2ecf20Sopenharmony_ci struct dw_i3c_master *master = dev_id; 10838c2ecf20Sopenharmony_ci u32 status; 10848c2ecf20Sopenharmony_ci 10858c2ecf20Sopenharmony_ci status = readl(master->regs + INTR_STATUS); 10868c2ecf20Sopenharmony_ci 10878c2ecf20Sopenharmony_ci if (!(status & readl(master->regs + INTR_STATUS_EN))) { 10888c2ecf20Sopenharmony_ci writel(INTR_ALL, master->regs + INTR_STATUS); 10898c2ecf20Sopenharmony_ci return IRQ_NONE; 10908c2ecf20Sopenharmony_ci } 10918c2ecf20Sopenharmony_ci 10928c2ecf20Sopenharmony_ci spin_lock(&master->xferqueue.lock); 10938c2ecf20Sopenharmony_ci dw_i3c_master_end_xfer_locked(master, status); 10948c2ecf20Sopenharmony_ci if (status & INTR_TRANSFER_ERR_STAT) 10958c2ecf20Sopenharmony_ci writel(INTR_TRANSFER_ERR_STAT, master->regs + INTR_STATUS); 10968c2ecf20Sopenharmony_ci spin_unlock(&master->xferqueue.lock); 10978c2ecf20Sopenharmony_ci 10988c2ecf20Sopenharmony_ci return IRQ_HANDLED; 10998c2ecf20Sopenharmony_ci} 11008c2ecf20Sopenharmony_ci 11018c2ecf20Sopenharmony_cistatic const struct i3c_master_controller_ops dw_mipi_i3c_ops = { 11028c2ecf20Sopenharmony_ci .bus_init = dw_i3c_master_bus_init, 11038c2ecf20Sopenharmony_ci .bus_cleanup = dw_i3c_master_bus_cleanup, 11048c2ecf20Sopenharmony_ci .attach_i3c_dev = dw_i3c_master_attach_i3c_dev, 11058c2ecf20Sopenharmony_ci .reattach_i3c_dev = dw_i3c_master_reattach_i3c_dev, 11068c2ecf20Sopenharmony_ci .detach_i3c_dev = dw_i3c_master_detach_i3c_dev, 11078c2ecf20Sopenharmony_ci .do_daa = dw_i3c_master_daa, 11088c2ecf20Sopenharmony_ci .supports_ccc_cmd = dw_i3c_master_supports_ccc_cmd, 11098c2ecf20Sopenharmony_ci .send_ccc_cmd = dw_i3c_master_send_ccc_cmd, 11108c2ecf20Sopenharmony_ci .priv_xfers = dw_i3c_master_priv_xfers, 11118c2ecf20Sopenharmony_ci .attach_i2c_dev = dw_i3c_master_attach_i2c_dev, 11128c2ecf20Sopenharmony_ci .detach_i2c_dev = dw_i3c_master_detach_i2c_dev, 11138c2ecf20Sopenharmony_ci .i2c_xfers = dw_i3c_master_i2c_xfers, 11148c2ecf20Sopenharmony_ci}; 11158c2ecf20Sopenharmony_ci 11168c2ecf20Sopenharmony_cistatic int dw_i3c_probe(struct platform_device *pdev) 11178c2ecf20Sopenharmony_ci{ 11188c2ecf20Sopenharmony_ci struct dw_i3c_master *master; 11198c2ecf20Sopenharmony_ci int ret, irq; 11208c2ecf20Sopenharmony_ci 11218c2ecf20Sopenharmony_ci master = devm_kzalloc(&pdev->dev, sizeof(*master), GFP_KERNEL); 11228c2ecf20Sopenharmony_ci if (!master) 11238c2ecf20Sopenharmony_ci return -ENOMEM; 11248c2ecf20Sopenharmony_ci 11258c2ecf20Sopenharmony_ci master->regs = devm_platform_ioremap_resource(pdev, 0); 11268c2ecf20Sopenharmony_ci if (IS_ERR(master->regs)) 11278c2ecf20Sopenharmony_ci return PTR_ERR(master->regs); 11288c2ecf20Sopenharmony_ci 11298c2ecf20Sopenharmony_ci master->core_clk = devm_clk_get(&pdev->dev, NULL); 11308c2ecf20Sopenharmony_ci if (IS_ERR(master->core_clk)) 11318c2ecf20Sopenharmony_ci return PTR_ERR(master->core_clk); 11328c2ecf20Sopenharmony_ci 11338c2ecf20Sopenharmony_ci master->core_rst = devm_reset_control_get_optional_exclusive(&pdev->dev, 11348c2ecf20Sopenharmony_ci "core_rst"); 11358c2ecf20Sopenharmony_ci if (IS_ERR(master->core_rst)) 11368c2ecf20Sopenharmony_ci return PTR_ERR(master->core_rst); 11378c2ecf20Sopenharmony_ci 11388c2ecf20Sopenharmony_ci ret = clk_prepare_enable(master->core_clk); 11398c2ecf20Sopenharmony_ci if (ret) 11408c2ecf20Sopenharmony_ci goto err_disable_core_clk; 11418c2ecf20Sopenharmony_ci 11428c2ecf20Sopenharmony_ci reset_control_deassert(master->core_rst); 11438c2ecf20Sopenharmony_ci 11448c2ecf20Sopenharmony_ci spin_lock_init(&master->xferqueue.lock); 11458c2ecf20Sopenharmony_ci INIT_LIST_HEAD(&master->xferqueue.list); 11468c2ecf20Sopenharmony_ci 11478c2ecf20Sopenharmony_ci writel(INTR_ALL, master->regs + INTR_STATUS); 11488c2ecf20Sopenharmony_ci irq = platform_get_irq(pdev, 0); 11498c2ecf20Sopenharmony_ci ret = devm_request_irq(&pdev->dev, irq, 11508c2ecf20Sopenharmony_ci dw_i3c_master_irq_handler, 0, 11518c2ecf20Sopenharmony_ci dev_name(&pdev->dev), master); 11528c2ecf20Sopenharmony_ci if (ret) 11538c2ecf20Sopenharmony_ci goto err_assert_rst; 11548c2ecf20Sopenharmony_ci 11558c2ecf20Sopenharmony_ci platform_set_drvdata(pdev, master); 11568c2ecf20Sopenharmony_ci 11578c2ecf20Sopenharmony_ci /* Information regarding the FIFOs/QUEUEs depth */ 11588c2ecf20Sopenharmony_ci ret = readl(master->regs + QUEUE_STATUS_LEVEL); 11598c2ecf20Sopenharmony_ci master->caps.cmdfifodepth = QUEUE_STATUS_LEVEL_CMD(ret); 11608c2ecf20Sopenharmony_ci 11618c2ecf20Sopenharmony_ci ret = readl(master->regs + DATA_BUFFER_STATUS_LEVEL); 11628c2ecf20Sopenharmony_ci master->caps.datafifodepth = DATA_BUFFER_STATUS_LEVEL_TX(ret); 11638c2ecf20Sopenharmony_ci 11648c2ecf20Sopenharmony_ci ret = readl(master->regs + DEVICE_ADDR_TABLE_POINTER); 11658c2ecf20Sopenharmony_ci master->datstartaddr = ret; 11668c2ecf20Sopenharmony_ci master->maxdevs = ret >> 16; 11678c2ecf20Sopenharmony_ci master->free_pos = GENMASK(master->maxdevs - 1, 0); 11688c2ecf20Sopenharmony_ci 11698c2ecf20Sopenharmony_ci ret = i3c_master_register(&master->base, &pdev->dev, 11708c2ecf20Sopenharmony_ci &dw_mipi_i3c_ops, false); 11718c2ecf20Sopenharmony_ci if (ret) 11728c2ecf20Sopenharmony_ci goto err_assert_rst; 11738c2ecf20Sopenharmony_ci 11748c2ecf20Sopenharmony_ci return 0; 11758c2ecf20Sopenharmony_ci 11768c2ecf20Sopenharmony_cierr_assert_rst: 11778c2ecf20Sopenharmony_ci reset_control_assert(master->core_rst); 11788c2ecf20Sopenharmony_ci 11798c2ecf20Sopenharmony_cierr_disable_core_clk: 11808c2ecf20Sopenharmony_ci clk_disable_unprepare(master->core_clk); 11818c2ecf20Sopenharmony_ci 11828c2ecf20Sopenharmony_ci return ret; 11838c2ecf20Sopenharmony_ci} 11848c2ecf20Sopenharmony_ci 11858c2ecf20Sopenharmony_cistatic int dw_i3c_remove(struct platform_device *pdev) 11868c2ecf20Sopenharmony_ci{ 11878c2ecf20Sopenharmony_ci struct dw_i3c_master *master = platform_get_drvdata(pdev); 11888c2ecf20Sopenharmony_ci int ret; 11898c2ecf20Sopenharmony_ci 11908c2ecf20Sopenharmony_ci ret = i3c_master_unregister(&master->base); 11918c2ecf20Sopenharmony_ci if (ret) 11928c2ecf20Sopenharmony_ci return ret; 11938c2ecf20Sopenharmony_ci 11948c2ecf20Sopenharmony_ci reset_control_assert(master->core_rst); 11958c2ecf20Sopenharmony_ci 11968c2ecf20Sopenharmony_ci clk_disable_unprepare(master->core_clk); 11978c2ecf20Sopenharmony_ci 11988c2ecf20Sopenharmony_ci return 0; 11998c2ecf20Sopenharmony_ci} 12008c2ecf20Sopenharmony_ci 12018c2ecf20Sopenharmony_cistatic const struct of_device_id dw_i3c_master_of_match[] = { 12028c2ecf20Sopenharmony_ci { .compatible = "snps,dw-i3c-master-1.00a", }, 12038c2ecf20Sopenharmony_ci {}, 12048c2ecf20Sopenharmony_ci}; 12058c2ecf20Sopenharmony_ciMODULE_DEVICE_TABLE(of, dw_i3c_master_of_match); 12068c2ecf20Sopenharmony_ci 12078c2ecf20Sopenharmony_cistatic struct platform_driver dw_i3c_driver = { 12088c2ecf20Sopenharmony_ci .probe = dw_i3c_probe, 12098c2ecf20Sopenharmony_ci .remove = dw_i3c_remove, 12108c2ecf20Sopenharmony_ci .driver = { 12118c2ecf20Sopenharmony_ci .name = "dw-i3c-master", 12128c2ecf20Sopenharmony_ci .of_match_table = of_match_ptr(dw_i3c_master_of_match), 12138c2ecf20Sopenharmony_ci }, 12148c2ecf20Sopenharmony_ci}; 12158c2ecf20Sopenharmony_cimodule_platform_driver(dw_i3c_driver); 12168c2ecf20Sopenharmony_ci 12178c2ecf20Sopenharmony_ciMODULE_AUTHOR("Vitor Soares <vitor.soares@synopsys.com>"); 12188c2ecf20Sopenharmony_ciMODULE_DESCRIPTION("DesignWare MIPI I3C driver"); 12198c2ecf20Sopenharmony_ciMODULE_LICENSE("GPL v2"); 1220