blob: 88f74054f35926d5be757231b9868de410e9b636 [file] [log] [blame]
/* Copyright 2019 The Chromium OS Authors. All rights reserved.
* Use of this source code is governed by a BSD-style license that can be
* found in the LICENSE file.
*/
/* Intel TGL-U-RVP-ITE board-specific configuration */
#include "bb_retimer.h"
#include "button.h"
#include "charger.h"
#include "driver/charger/isl9241.h"
#include "extpower.h"
#include "i2c.h"
#include "intc.h"
#include "it83xx_pd.h"
#include "lid_switch.h"
#include "power.h"
#include "power/icelake.h"
#include "power_button.h"
#include "switch.h"
#include "system.h"
#include "tablet_mode.h"
#include "uart.h"
#include "usb_mux.h"
#include "usb_pd_tcpm.h"
#include "gpio_list.h"
#define CPRINTS(format, args...) cprints(CC_COMMAND, format, ## args)
#define CPRINTF(format, args...) cprintf(CC_COMMAND, format, ## args)
/* Mutex for shared NVM access */
static struct mutex bb_nvm_mutex;
/* TCPC gpios */
const struct tcpc_gpio_config_t tcpc_gpios[] = {
[TYPE_C_PORT_0] = {
.vbus = {
.pin = GPIO_USB_C0_VBUS_INT,
.pin_pol = 1,
},
.src = {
.pin = GPIO_USB_C0_SRC_EN,
.pin_pol = 1,
},
.snk = {
.pin = GPIO_USB_C0_SNK_EN_L,
.pin_pol = 0,
},
.vconn = {
.cc1_pin = GPIO_USB_C0_CC1_VCONN_EN,
.cc2_pin = GPIO_USB_C0_CC2_VCONN_EN,
.pin_pol = 1,
},
.src_ilim = {
.pin = GPIO_USB_C0_SRC_HI_ILIM,
.pin_pol = 1,
},
},
[TYPE_C_PORT_1] = {
.vbus = {
.pin = GPIO_USB_C1_VBUS_INT,
.pin_pol = 1,
},
.src = {
.pin = GPIO_USB_C1_SRC_EN,
.pin_pol = 1,
},
.snk = {
.pin = GPIO_USB_C1_SNK_EN_L,
.pin_pol = 0,
},
.vconn = {
.cc1_pin = GPIO_USB_C1_CC1_VCONN_EN,
.cc2_pin = GPIO_USB_C1_CC2_VCONN_EN,
.pin_pol = 1,
},
.src_ilim = {
.pin = GPIO_USB_C1_SRC_HI_ILIM,
.pin_pol = 1,
},
},
};
BUILD_ASSERT(ARRAY_SIZE(tcpc_gpios) == CONFIG_USB_PD_PORT_MAX_COUNT);
/* USB-C TPCP Configuration */
const struct tcpc_config_t tcpc_config[] = {
[TYPE_C_PORT_0] = {
.bus_type = EC_BUS_TYPE_EMBEDDED,
/* TCPC is embedded within EC so no i2c config needed */
.drv = &it83xx_tcpm_drv,
},
[TYPE_C_PORT_1] = {
.bus_type = EC_BUS_TYPE_EMBEDDED,
/* TCPC is embedded within EC so no i2c config needed */
.drv = &it83xx_tcpm_drv,
},
};
BUILD_ASSERT(ARRAY_SIZE(tcpc_config) == CONFIG_USB_PD_PORT_MAX_COUNT);
/* BB Retimers configuration */
const struct bb_usb_control bb_controls[] = {
[TYPE_C_PORT_0] = {
.usb_ls_en_gpio = GPIO_USB_C0_LS_EN,
.retimer_rst_gpio = GPIO_USB_C0_RETIMER_RST,
},
[TYPE_C_PORT_1] = {
.usb_ls_en_gpio = GPIO_USB_C1_LS_EN,
.retimer_rst_gpio = GPIO_USB_C1_RETIMER_RST,
},
};
BUILD_ASSERT(ARRAY_SIZE(bb_controls) == CONFIG_USB_PD_PORT_MAX_COUNT);
/* USB retimer Configuration */
struct usb_mux usbc0_tcss_usb_mux = {
.usb_port = TYPE_C_PORT_0,
.driver = &virtual_usb_mux_driver,
.hpd_update = &virtual_hpd_update,
};
struct usb_mux usbc1_tcss_usb_mux = {
.usb_port = TYPE_C_PORT_1,
.driver = &virtual_usb_mux_driver,
.hpd_update = &virtual_hpd_update,
};
/* USB muxes Configuration */
const struct usb_mux usb_muxes[] = {
[TYPE_C_PORT_0] = {
.usb_port = TYPE_C_PORT_0,
.next_mux = &usbc0_tcss_usb_mux,
.driver = &bb_usb_retimer,
.i2c_port = I2C_PORT0_BB_RETIMER,
.i2c_addr_flags = I2C_PORT0_BB_RETIMER_ADDR,
},
[TYPE_C_PORT_1] = {
.usb_port = TYPE_C_PORT_1,
.next_mux = &usbc1_tcss_usb_mux,
.driver = &bb_usb_retimer,
.i2c_port = I2C_PORT1_BB_RETIMER,
.i2c_addr_flags = I2C_PORT1_BB_RETIMER_ADDR,
},
};
BUILD_ASSERT(ARRAY_SIZE(usb_muxes) == CONFIG_USB_PD_PORT_MAX_COUNT);
/* I2C ports */
const struct i2c_port_t i2c_ports[] = {
/* Flash EC */
[I2C_CHAN_FLASH] = {
.name = "chan-A",
.port = IT83XX_I2C_CH_A,
.kbps = 100,
.scl = GPIO_I2C_A_SCL,
.sda = GPIO_I2C_A_SDA,
},
/*
* Port-80 Display, Charger, Battery, IO-expanders, EEPROM,
* IMVP9, AUX-rail, power-monitor.
*/
[I2C_CHAN_BATT_CHG] = {
.name = "batt_chg",
.port = IT83XX_I2C_CH_B,
.kbps = 100,
.scl = GPIO_I2C_B_SCL,
.sda = GPIO_I2C_B_SDA,
},
/* Retimers, PDs */
[I2C_CHAN_RETIMER] = {
.name = "retimer",
.port = IT83XX_I2C_CH_E,
.kbps = 100,
.scl = GPIO_I2C_E_SCL,
.sda = GPIO_I2C_E_SDA,
},
};
BUILD_ASSERT(ARRAY_SIZE(i2c_ports) == I2C_CHAN_COUNT);
const unsigned int i2c_ports_used = ARRAY_SIZE(i2c_ports);
/* Charger Chips */
const struct charger_config_t chg_chips[] = {
{
.i2c_port = I2C_PORT_CHARGER,
.i2c_addr_flags = ISL9241_ADDR_FLAGS,
.drv = &isl9241_drv,
},
};
/******************************************************************************/
/* PWROK signal configuration */
/*
* On TGLRVP the ALL_SYS_PWRGD, VCCST_PWRGD, PCH_PWROK, and SYS_PWROK
* signals are handled by the board. No EC control needed.
*/
const struct intel_x86_pwrok_signal pwrok_signal_assert_list[] = {};
const int pwrok_signal_assert_count = ARRAY_SIZE(pwrok_signal_assert_list);
const struct intel_x86_pwrok_signal pwrok_signal_deassert_list[] = {};
const int pwrok_signal_deassert_count = ARRAY_SIZE(pwrok_signal_assert_list);
/*
* Returns board information (board id[7:0] and Fab id[15:8]) on success
* -1 on error.
*/
int board_get_version(void)
{
int port0, port1;
int fab_id, board_id, bom_id;
if (ioexpander_read_intelrvp_version(&port0, &port1))
return -1;
/*
* Port0: bit 0 - BOM ID(2)
* bit 2:1 - FAB ID(1:0) + 1
* Port1: bit 7:6 - BOM ID(1:0)
* bit 5:0 - BOARD ID(5:0)
*/
bom_id = ((port1 & 0xC0) >> 6) | ((port0 & 0x01) << 2);
fab_id = ((port0 & 0x06) >> 1) + 1;
board_id = port1 & 0x3F;
CPRINTS("BID:0x%x, FID:0x%x, BOM:0x%x", board_id, fab_id, bom_id);
return board_id | (fab_id << 8);
}
__override void bb_retimer_power_handle(const struct usb_mux *me, int on_off)
{
const struct bb_usb_control *control = &bb_controls[me->usb_port];
/*
* LSx based F/W updating is a POR, however to avoid the rework on
* RVP retain the FORCE_PWR GPIO with EC.
*/
enum gpio_signal force_power_gpio = me->usb_port ?
GPIO_USB_C1_RETIMER_FORCE_PWR : GPIO_USB_C0_RETIMER_FORCE_PWR;
/* handle retimer's power domain */
if (on_off) {
/*
* BB retimer NVM can be shared between multiple ports, hence
* lock enabling the retimer until the current retimer request
* is complete.
*/
mutex_lock(&bb_nvm_mutex);
gpio_set_level(control->usb_ls_en_gpio, 1);
/*
* Tpw, minimum time from VCC to RESET_N de-assertion is 100us.
* For boards that don't provide a load switch control, the
* retimer_init() function ensures power is up before calling
* this function.
*/
msleep(1);
gpio_set_level(control->retimer_rst_gpio, 1);
msleep(10);
gpio_set_level(force_power_gpio, 1);
/* Allow 20ms time for the retimer to be initialized. */
msleep(20);
mutex_unlock(&bb_nvm_mutex);
} else {
gpio_set_level(force_power_gpio, 0);
msleep(1);
gpio_set_level(control->retimer_rst_gpio, 0);
msleep(1);
gpio_set_level(control->usb_ls_en_gpio, 0);
}
}