/* Copyright 2022 The ChromiumOS Authors * Use of this source code is governed by a BSD-style license that can be * found in the LICENSE file. * * Cypress CCGXXF PD chip driver source */ #include "ccgxxf.h" #include "console.h" #include "tcpm/tcpci.h" /* * TODO (b/236994474): Once the PD negotiation completes, CCGXXF chip stops * responding over I2C for about 10 seconds. As DRP is enabled, TCPM algorithm * constantly looks for any CC status changes even after negotiation completes. * Hence, cache the CC state and return the cached values in case of I2C * failures. This workaround will be removed once the fix is added in the * physical layer firmware of CCGXXF. */ struct ccgxxf_cc { bool good_cc; enum tcpc_cc_voltage_status cc1; enum tcpc_cc_voltage_status cc2; }; static struct ccgxxf_cc ccgxxf_cc_cache[CONFIG_USB_PD_PORT_MAX_COUNT]; static int ccgxxf_tcpci_tcpm_get_cc(int port, enum tcpc_cc_voltage_status *cc1, enum tcpc_cc_voltage_status *cc2) { int rv = tcpci_tcpm_get_cc(port, cc1, cc2); if (rv) { if (!ccgxxf_cc_cache[port].good_cc) return rv; *cc1 = ccgxxf_cc_cache[port].cc1; *cc2 = ccgxxf_cc_cache[port].cc2; } else { ccgxxf_cc_cache[port].good_cc = true; ccgxxf_cc_cache[port].cc1 = *cc1; ccgxxf_cc_cache[port].cc2 = *cc2; } return EC_SUCCESS; } static int ccgxxf_tcpci_tcpm_init(int port) { ccgxxf_cc_cache[port].good_cc = false; return tcpci_tcpm_init(port); } #ifdef CONFIG_USB_PD_TCPM_SBU static int ccgxxf_tcpc_set_sbu(int port, bool enable) { return tcpc_write(port, CCGXXF_REG_SBU_MUX_CTL, enable); } #endif #ifdef CONFIG_CMD_TCPC_DUMP static void ccgxxf_dump_registers(int port) { int fw_ver, fw_build; tcpc_dump_std_registers(port); /* Get the F/W version and build ID */ if (!tcpc_read16(port, CCGXXF_REG_FW_VERSION, &fw_ver) && !tcpc_read16(port, CCGXXF_REG_FW_VERSION_BUILD, &fw_build)) { ccprintf(" FW_VERSION(build.major.minor) = %d.%d.%d\n", fw_build & 0xFF, (fw_ver >> 8) & 0xFF, fw_ver & 0xFF); } } #endif int ccgxxf_reset(int port) { return tcpc_write16(port, CCGXXF_REG_FWU_COMMAND, CCGXXF_FWU_CMD_RESET); } const struct tcpm_drv ccgxxf_tcpm_drv = { .init = &ccgxxf_tcpci_tcpm_init, .release = &tcpci_tcpm_release, .get_cc = &ccgxxf_tcpci_tcpm_get_cc, #ifdef CONFIG_USB_PD_VBUS_DETECT_TCPC .check_vbus_level = &tcpci_tcpm_check_vbus_level, #endif .select_rp_value = &tcpci_tcpm_select_rp_value, .set_cc = &tcpci_tcpm_set_cc, .set_polarity = &tcpci_tcpm_set_polarity, #ifdef CONFIG_USB_PD_DECODE_SOP .sop_prime_enable = &tcpci_tcpm_sop_prime_enable, #endif .set_vconn = &tcpci_tcpm_set_vconn, .set_msg_header = &tcpci_tcpm_set_msg_header, .set_rx_enable = &tcpci_tcpm_set_rx_enable, .get_message_raw = &tcpci_tcpm_get_message_raw, .transmit = &tcpci_tcpm_transmit, .tcpc_alert = &tcpci_tcpc_alert, #ifdef CONFIG_USB_PD_DISCHARGE_TCPC .tcpc_discharge_vbus = &tcpci_tcpc_discharge_vbus, #endif .tcpc_enable_auto_discharge_disconnect = &tcpci_tcpc_enable_auto_discharge_disconnect, #ifdef CONFIG_USB_PD_DUAL_ROLE_AUTO_TOGGLE .drp_toggle = &tcpci_tcpc_drp_toggle, #endif .get_chip_info = &tcpci_get_chip_info, #ifdef CONFIG_USB_PD_PPC .get_snk_ctrl = &tcpci_tcpm_get_snk_ctrl, .set_snk_ctrl = &tcpci_tcpm_set_snk_ctrl, .get_src_ctrl = &tcpci_tcpm_get_src_ctrl, .set_src_ctrl = &tcpci_tcpm_set_src_ctrl, #endif #ifdef CONFIG_USB_PD_TCPM_SBU .set_sbu = &ccgxxf_tcpc_set_sbu, #endif #ifdef CONFIG_USB_PD_TCPC_LOW_POWER .enter_low_power_mode = &tcpci_enter_low_power_mode, #endif .set_bist_test_mode = &tcpci_set_bist_test_mode, .get_bist_test_mode = &tcpci_get_bist_test_mode, #ifdef CONFIG_CMD_TCPC_DUMP .dump_registers = &ccgxxf_dump_registers, #endif };