Hi JC,

On 25-01-2019 14:27, JC Kuo wrote:
> Hi Nagarjuna,
> 1. tegra210_utmi_bias_pad_power_off() should
> set XUSB_PADCTL_USB2_BIAS_PAD_CTL0_PD in order to really power down bias
> pad.
Will update the setting accordingly 
> 2. The XUSB_PADCTL_USB2_BIAS_PAD_CTL0 register programming
> in tegra210_usb2_phy_power_on() can be removed since you have added it
> to tegra210_utmi_bias_pad_power_on().
Will check and remove code from phy_power_on which is used in utmi pad and
bias pad power on API's 
> 3. In tegra210_usb2_phy_power_off(), there is
> + value = padctl_readl(padctl, XUSB_PADCTL_SS_PORT_MAP);
> + value &= ~XUSB_PADCTL_SS_PORT_MAP_PORTX_MAP_MASK(
> + port->usb3_port_fake);
> + value |= XUSB_PADCTL_SS_PORT_MAP_PORTX_MAP_MASK(
> + port->usb3_port_fake);
> + padctl_writel(padctl, value, XUSB_PADCTL_SS_PORT_MAP);
> I think it should be
> + value = padctl_readl(padctl, XUSB_PADCTL_SS_PORT_MAP);
> + value &= ~XUSB_PADCTL_SS_PORT_MAP_PORTX_MAP_MASK(
> + port->usb3_port_fake);
> + value |= XUSB_PADCTL_SS_PORT_MAP_PORTX_MAP(port->usb3_port_fake,
> +                           XUSB_PADCTL_SS_PORT_MAP_PORT_DISABLED );
> + padctl_writel(padctl, value, XUSB_PADCTL_SS_PORT_MAP);
> 
Yes, suggest change is right setting for disabled, will do the same.
> 4. XUSB_PADCTL_USB2_VBUS_ID_OVERRIDE_VAL should be renamed
> with XUSB_PADCTL_USB2_VBUS_ID_OVERRIDE_FLOATING . Another possible ID pin
> state is "grounded" should be defined
> XUSB_PADCTL_USB2_VBUS_ID_OVERRIDE_GROUNDED with value 0x0.
> 
Will do
> Thanks,
> JC
> 
> 
> On Thu, Jan 3, 2019 at 7:43 PM Nagarjuna Kristam <nkris...@nvidia.com>
> wrote:
> 
>> Add support for XUSB device mode controller on Tegra210.
>> Update PADCTL driver to set port cap based on DT config.
>> Add code to handle property "nvidia,usb3-port-fake"
>> Provide API's to control vbus override and utmi pad power control
>>
>> Signed-off-by: Nagarjuna Kristam <nkris...@nvidia.com>
>> ---
>>  drivers/phy/tegra/xusb-tegra210.c | 297
>> +++++++++++++++++++++++++++++++++++++-
>>  drivers/phy/tegra/xusb.c          |  75 +++++++++-
>>  drivers/phy/tegra/xusb.h          |  16 +-
>>  include/linux/phy/tegra/xusb.h    |   7 +-
>>  4 files changed, 386 insertions(+), 9 deletions(-)
>>
>> diff --git a/drivers/phy/tegra/xusb-tegra210.c
>> b/drivers/phy/tegra/xusb-tegra210.c
>> index 05bee32..0289e10 100644
>> --- a/drivers/phy/tegra/xusb-tegra210.c
>> +++ b/drivers/phy/tegra/xusb-tegra210.c
>> @@ -1,5 +1,5 @@
>>  /*
>> - * Copyright (c) 2014, NVIDIA CORPORATION.  All rights reserved.
>> + * Copyright (c) 2014-2018, NVIDIA CORPORATION.  All rights reserved.
>>   * Copyright (C) 2015 Google, Inc.
>>   *
>>   * This program is free software; you can redistribute it and/or modify it
>> @@ -47,7 +47,10 @@
>>  #define XUSB_PADCTL_USB2_PAD_MUX_USB2_BIAS_PAD_XUSB 0x1
>>
>>  #define XUSB_PADCTL_USB2_PORT_CAP 0x008
>> +#define XUSB_PADCTL_USB2_PORT_CAP_PORTX_CAP_DISABLED(x) (0x0 << ((x) * 4))
>>  #define XUSB_PADCTL_USB2_PORT_CAP_PORTX_CAP_HOST(x) (0x1 << ((x) * 4))
>> +#define XUSB_PADCTL_USB2_PORT_CAP_PORTX_CAP_DEVICE(x) (0x2 << ((x) * 4))
>> +#define XUSB_PADCTL_USB2_PORT_CAP_PORTX_CAP_OTG(x) (0x3 << ((x) * 4))
>>  #define XUSB_PADCTL_USB2_PORT_CAP_PORTX_CAP_MASK(x) (0x3 << ((x) * 4))
>>
>>  #define XUSB_PADCTL_SS_PORT_MAP 0x014
>> @@ -55,6 +58,7 @@
>>  #define XUSB_PADCTL_SS_PORT_MAP_PORTX_MAP_SHIFT(x) ((x) * 5)
>>  #define XUSB_PADCTL_SS_PORT_MAP_PORTX_MAP_MASK(x) (0x7 << ((x) * 5))
>>  #define XUSB_PADCTL_SS_PORT_MAP_PORTX_MAP(x, v) (((v) & 0x7) << ((x) * 5))
>> +#define XUSB_PADCTL_SS_PORT_MAP_PORT_DISABLED 0x7
>>
>>  #define XUSB_PADCTL_ELPG_PROGRAM1 0x024
>>  #define XUSB_PADCTL_ELPG_PROGRAM1_AUX_MUX_LP0_VCORE_DOWN (1 << 31)
>> @@ -69,9 +73,14 @@
>>  #define XUSB_PADCTL_USB3_PAD_MUX_PCIE_IDDQ_DISABLE(x) (1 << (1 + (x)))
>>  #define XUSB_PADCTL_USB3_PAD_MUX_SATA_IDDQ_DISABLE(x) (1 << (8 + (x)))
>>
>> +#define XUSB_PADCTL_USB2_BATTERY_CHRG_OTGPADX_CTL0(x) (0x080 + (x) * 0x40)
>> +#define XUSB_PADCTL_USB2_BATTERY_CHRG_OTGPAD_CTL0_ZIP (1 << 18)
>> +#define XUSB_PADCTL_USB2_BATTERY_CHRG_OTGPAD_CTL0_ZIN (1 << 22)
>> +
>>  #define XUSB_PADCTL_USB2_BATTERY_CHRG_OTGPADX_CTL1(x) (0x084 + (x) * 0x40)
>>  #define XUSB_PADCTL_USB2_BATTERY_CHRG_OTGPAD_CTL1_VREG_LEV_SHIFT 7
>>  #define XUSB_PADCTL_USB2_BATTERY_CHRG_OTGPAD_CTL1_VREG_LEV_MASK 0x3
>> +#define XUSB_PADCTL_USB2_BATTERY_CHRG_OTGPAD_CTL1_VREG_LEV_VAL 0x1
>>  #define XUSB_PADCTL_USB2_BATTERY_CHRG_OTGPAD_CTL1_VREG_FIX18 (1 << 6)
>>
>>  #define XUSB_PADCTL_USB2_OTG_PADX_CTL0(x) (0x088 + (x) * 0x40)
>> @@ -230,6 +239,12 @@
>>  #define XUSB_PADCTL_UPHY_USB3_PADX_ECTL6(x) (0xa74 + (x) * 0x40)
>>  #define XUSB_PADCTL_UPHY_USB3_PAD_ECTL6_RX_EQ_CTRL_H_VAL 0xfcf01368
>>
>> +#define XUSB_PADCTL_USB2_VBUS_ID 0xc60
>> +#define XUSB_PADCTL_USB2_VBUS_ID_OVERRIDE_VBUS_ON (1 << 14)
>> +#define XUSB_PADCTL_USB2_VBUS_ID_OVERRIDE_SHIFT 18
>> +#define XUSB_PADCTL_USB2_VBUS_ID_OVERRIDE_MASK 0xf
>> +#define XUSB_PADCTL_USB2_VBUS_ID_OVERRIDE_VAL 8
>> +
>>  struct tegra210_xusb_fuse_calibration {
>>         u32 hs_curr_level[4];
>>         u32 hs_term_range_adj;
>> @@ -905,6 +920,161 @@ static const struct tegra_xusb_lane_ops
>> tegra210_usb2_lane_ops = {
>>         .remove = tegra210_usb2_lane_remove,
>>  };
>>
>> +/* must be called under padctl->lock */
>> +static void tegra210_utmi_bias_pad_power_on(struct tegra_xusb_padctl
>> *padctl,
>> +                               struct tegra_xusb_usb2_pad *pad)
>> +{
>> +       u32 reg;
>> +
>> +       if (pad->enable++ > 0)
>> +               return;
>> +
>> +       dev_dbg(padctl->dev, "power on BIAS PAD & USB2 tracking\n");
>> +
>> +       if (clk_prepare_enable(pad->clk))
>> +               dev_warn(padctl->dev, "failed to enable BIAS PAD & USB2
>> tracking\n");
>> +
>> +       reg = padctl_readl(padctl, XUSB_PADCTL_USB2_BIAS_PAD_CTL1);
>> +
>> +       reg &= ~((XUSB_PADCTL_USB2_BIAS_PAD_CTL1_TRK_START_TIMER_MASK <<
>> +                   XUSB_PADCTL_USB2_BIAS_PAD_CTL1_TRK_START_TIMER_SHIFT) |
>> +
>> (XUSB_PADCTL_USB2_BIAS_PAD_CTL1_TRK_DONE_RESET_TIMER_MASK <<
>> +
>>  XUSB_PADCTL_USB2_BIAS_PAD_CTL1_TRK_DONE_RESET_TIMER_SHIFT));
>> +       reg |= (XUSB_PADCTL_USB2_BIAS_PAD_CTL1_TRK_START_TIMER_VAL <<
>> +                 XUSB_PADCTL_USB2_BIAS_PAD_CTL1_TRK_START_TIMER_SHIFT) |
>> +                (XUSB_PADCTL_USB2_BIAS_PAD_CTL1_TRK_DONE_RESET_TIMER_VAL
>> <<
>> +
>>  XUSB_PADCTL_USB2_BIAS_PAD_CTL1_TRK_DONE_RESET_TIMER_SHIFT);
>> +       padctl_writel(padctl, reg, XUSB_PADCTL_USB2_BIAS_PAD_CTL1);
>> +
>> +       reg = padctl_readl(padctl, XUSB_PADCTL_USB2_BIAS_PAD_CTL0);
>> +       reg &= ~XUSB_PADCTL_USB2_BIAS_PAD_CTL0_PD;
>> +
>> +       reg &= ~((XUSB_PADCTL_USB2_BIAS_PAD_CTL0_HS_SQUELCH_LEVEL_MASK <<
>> +                   XUSB_PADCTL_USB2_BIAS_PAD_CTL0_HS_SQUELCH_LEVEL_SHIFT)
>> |
>> +                  (XUSB_PADCTL_USB2_BIAS_PAD_CTL0_HS_DISCON_LEVEL_MASK <<
>> +                   XUSB_PADCTL_USB2_BIAS_PAD_CTL0_HS_DISCON_LEVEL_SHIFT));
>> +       reg |= (XUSB_PADCTL_USB2_BIAS_PAD_CTL0_HS_DISCON_LEVEL_VAL <<
>> +                 XUSB_PADCTL_USB2_BIAS_PAD_CTL0_HS_DISCON_LEVEL_SHIFT);
>> +
>> +       if (tegra_sku_info.revision < TEGRA_REVISION_A02)
>> +               reg |=
>> (XUSB_PADCTL_USB2_BIAS_PAD_CTL0_HS_SQUELCH_LEVEL_VAL <<
>> +
>>  XUSB_PADCTL_USB2_BIAS_PAD_CTL0_HS_SQUELCH_LEVEL_SHIFT);
>> +       padctl_writel(padctl, reg, XUSB_PADCTL_USB2_BIAS_PAD_CTL0);
>> +
>> +       udelay(1);
>> +
>> +       reg = padctl_readl(padctl, XUSB_PADCTL_USB2_BIAS_PAD_CTL1);
>> +       reg &= ~XUSB_PADCTL_USB2_BIAS_PAD_CTL1_PD_TRK;
>> +       padctl_writel(padctl, reg, XUSB_PADCTL_USB2_BIAS_PAD_CTL1);
>> +}
>> +
>> +/* must be called under padctl->lock */
>> +static void tegra210_utmi_bias_pad_power_off(struct tegra_xusb_padctl
>> *padctl,
>> +                               struct tegra_xusb_usb2_pad *pad)
>> +{
>> +       u32 reg;
>> +
>> +       if (WARN_ON(pad->enable == 0))
>> +               return;
>> +
>> +       if (--pad->enable > 0)
>> +               return;
>> +
>> +       dev_dbg(padctl->dev, "power off USB2 tracking\n");
>> +       reg = padctl_readl(padctl, XUSB_PADCTL_USB2_BIAS_PAD_CTL1);
>> +       reg |= XUSB_PADCTL_USB2_BIAS_PAD_CTL1_PD_TRK;
>> +       padctl_writel(padctl, reg, XUSB_PADCTL_USB2_BIAS_PAD_CTL1);
>> +
>> +       clk_disable_unprepare(pad->clk);
>> +}
>> +
>> +void tegra210_utmi_pad_power_on(struct phy *phy)
>> +{
>> +       struct tegra_xusb_lane *lane;
>> +       struct tegra_xusb_usb2_lane *usb2;
>> +       struct tegra_xusb_usb2_pad *pad;
>> +       struct tegra_xusb_padctl *padctl;
>> +       unsigned int index;
>> +       struct device *dev;
>> +       u32 reg;
>> +
>> +       if (!phy)
>> +               return;
>> +
>> +       lane = phy_get_drvdata(phy);
>> +       usb2 = to_usb2_lane(lane);
>> +       pad = to_usb2_pad(lane->pad);
>> +       padctl = lane->pad->padctl;
>> +       index = lane->index;
>> +       dev = padctl->dev;
>> +
>> +       if (usb2->powered_on)
>> +               return;
>> +
>> +       mutex_lock(&padctl->lock);
>> +
>> +       dev_info(dev, "power on UTMI pads %d\n", index);
>> +
>> +       tegra210_utmi_bias_pad_power_on(padctl, pad);
>> +
>> +       udelay(2);
>> +
>> +       reg = padctl_readl(padctl, XUSB_PADCTL_USB2_OTG_PADX_CTL0(index));
>> +       reg &= ~XUSB_PADCTL_USB2_OTG_PAD_CTL0_PD;
>> +       padctl_writel(padctl, reg, XUSB_PADCTL_USB2_OTG_PADX_CTL0(index));
>> +
>> +       reg = padctl_readl(padctl, XUSB_PADCTL_USB2_OTG_PADX_CTL1(index));
>> +       reg &= ~XUSB_PADCTL_USB2_OTG_PAD_CTL1_PD_DR;
>> +       padctl_writel(padctl, reg, XUSB_PADCTL_USB2_OTG_PADX_CTL1(index));
>> +
>> +       usb2->powered_on = true;
>> +
>> +       mutex_unlock(&padctl->lock);
>> +}
>> +
>> +void tegra210_utmi_pad_power_down(struct phy *phy)
>> +{
>> +       struct tegra_xusb_lane *lane;
>> +       struct tegra_xusb_usb2_lane *usb2;
>> +       struct tegra_xusb_usb2_pad *pad;
>> +       struct tegra_xusb_padctl *padctl;
>> +       unsigned int index;
>> +       struct device *dev;
>> +       u32 reg;
>> +
>> +       if (!phy)
>> +               return;
>> +
>> +       lane = phy_get_drvdata(phy);
>> +       usb2 = to_usb2_lane(lane);
>> +       pad = to_usb2_pad(lane->pad);
>> +       padctl = lane->pad->padctl;
>> +       index = lane->index;
>> +       dev = padctl->dev;
>> +
>> +       if (!usb2->powered_on)
>> +               return;
>> +
>> +       mutex_lock(&padctl->lock);
>> +
>> +       dev_info(dev, "power down UTMI pad %d\n", index);
>> +
>> +       reg = padctl_readl(padctl, XUSB_PADCTL_USB2_OTG_PADX_CTL0(index));
>> +       reg |= XUSB_PADCTL_USB2_OTG_PAD_CTL0_PD;
>> +       padctl_writel(padctl, reg, XUSB_PADCTL_USB2_OTG_PADX_CTL0(index));
>> +
>> +       reg = padctl_readl(padctl, XUSB_PADCTL_USB2_OTG_PADX_CTL1(index));
>> +       reg |= XUSB_PADCTL_USB2_OTG_PAD_CTL1_PD_DR;
>> +       padctl_writel(padctl, reg, XUSB_PADCTL_USB2_OTG_PADX_CTL1(index));
>> +
>> +       udelay(2);
>> +
>> +       tegra210_utmi_bias_pad_power_off(padctl, pad);
>> +       usb2->powered_on = false;
>> +
>> +       mutex_unlock(&padctl->lock);
>> +}
>> +
>>  static int tegra210_usb2_phy_init(struct phy *phy)
>>  {
>>         struct tegra_xusb_lane *lane = phy_get_drvdata(phy);
>> @@ -948,6 +1118,34 @@ static int tegra210_usb2_phy_power_on(struct phy
>> *phy)
>>
>>         priv = to_tegra210_xusb_padctl(padctl);
>>
>> +       if (port->usb3_port_fake != -1) {
>> +               value = padctl_readl(padctl, XUSB_PADCTL_SS_PORT_MAP);
>> +               value &= ~XUSB_PADCTL_SS_PORT_MAP_PORTX_MAP_MASK(
>> +                                       port->usb3_port_fake);
>> +               value |= XUSB_PADCTL_SS_PORT_MAP_PORTX_MAP(
>> +                                       port->usb3_port_fake, index);
>> +               padctl_writel(padctl, value, XUSB_PADCTL_SS_PORT_MAP);
>> +
>> +               value = padctl_readl(padctl, XUSB_PADCTL_ELPG_PROGRAM1);
>> +               value &= ~XUSB_PADCTL_ELPG_PROGRAM1_SSPX_ELPG_VCORE_DOWN(
>> +                                       port->usb3_port_fake);
>> +               padctl_writel(padctl, value, XUSB_PADCTL_ELPG_PROGRAM1);
>> +
>> +               usleep_range(100, 200);
>> +
>> +               value = padctl_readl(padctl, XUSB_PADCTL_ELPG_PROGRAM1);
>> +               value &=
>> ~XUSB_PADCTL_ELPG_PROGRAM1_SSPX_ELPG_CLAMP_EN_EARLY(
>> +                                       port->usb3_port_fake);
>> +               padctl_writel(padctl, value, XUSB_PADCTL_ELPG_PROGRAM1);
>> +
>> +               usleep_range(100, 200);
>> +
>> +               value = padctl_readl(padctl, XUSB_PADCTL_ELPG_PROGRAM1);
>> +               value &= ~XUSB_PADCTL_ELPG_PROGRAM1_SSPX_ELPG_CLAMP_EN(
>> +                                       port->usb3_port_fake);
>> +               padctl_writel(padctl, value, XUSB_PADCTL_ELPG_PROGRAM1);
>> +       }
>> +
>>         value = padctl_readl(padctl, XUSB_PADCTL_USB2_BIAS_PAD_CTL0);
>>         value &= ~((XUSB_PADCTL_USB2_BIAS_PAD_CTL0_HS_SQUELCH_LEVEL_MASK <<
>>                     XUSB_PADCTL_USB2_BIAS_PAD_CTL0_HS_SQUELCH_LEVEL_SHIFT)
>> |
>> @@ -965,7 +1163,14 @@ static int tegra210_usb2_phy_power_on(struct phy
>> *phy)
>>
>>         value = padctl_readl(padctl, XUSB_PADCTL_USB2_PORT_CAP);
>>         value &= ~XUSB_PADCTL_USB2_PORT_CAP_PORTX_CAP_MASK(index);
>> -       value |= XUSB_PADCTL_USB2_PORT_CAP_PORTX_CAP_HOST(index);
>> +       if (port->port_cap == USB_PORT_DISABLED)
>> +               value |=
>> XUSB_PADCTL_USB2_PORT_CAP_PORTX_CAP_DISABLED(index);
>> +       else if (port->port_cap == USB_DEVICE_CAP)
>> +               value |= XUSB_PADCTL_USB2_PORT_CAP_PORTX_CAP_DEVICE(index);
>> +       else if (port->port_cap == USB_HOST_CAP)
>> +               value |= XUSB_PADCTL_USB2_PORT_CAP_PORTX_CAP_HOST(index);
>> +       else if (port->port_cap == USB_OTG_CAP)
>> +               value |= XUSB_PADCTL_USB2_PORT_CAP_PORTX_CAP_OTG(index);
>>         padctl_writel(padctl, value, XUSB_PADCTL_USB2_PORT_CAP);
>>
>>         value = padctl_readl(padctl,
>> XUSB_PADCTL_USB2_OTG_PADX_CTL0(index));
>> @@ -997,7 +1202,12 @@ static int tegra210_usb2_phy_power_on(struct phy
>> *phy)
>>
>>  XUSB_PADCTL_USB2_BATTERY_CHRG_OTGPADX_CTL1(index));
>>         value &= ~(XUSB_PADCTL_USB2_BATTERY_CHRG_OTGPAD_CTL1_VREG_LEV_MASK
>> <<
>>
>>  XUSB_PADCTL_USB2_BATTERY_CHRG_OTGPAD_CTL1_VREG_LEV_SHIFT);
>> -       value |= XUSB_PADCTL_USB2_BATTERY_CHRG_OTGPAD_CTL1_VREG_FIX18;
>> +       if (port->port_cap == USB_HOST_CAP)
>> +               value |=
>> XUSB_PADCTL_USB2_BATTERY_CHRG_OTGPAD_CTL1_VREG_FIX18;
>> +       else
>> +               value |=
>> +
>>  XUSB_PADCTL_USB2_BATTERY_CHRG_OTGPAD_CTL1_VREG_LEV_VAL <<
>> +
>>  XUSB_PADCTL_USB2_BATTERY_CHRG_OTGPAD_CTL1_VREG_LEV_SHIFT;
>>         padctl_writel(padctl, value,
>>                       XUSB_PADCTL_USB2_BATTERY_CHRG_OTGPADX_CTL1(index));
>>
>> @@ -1070,6 +1280,34 @@ static int tegra210_usb2_phy_power_off(struct phy
>> *phy)
>>
>>         mutex_lock(&padctl->lock);
>>
>> +       if (port->usb3_port_fake != -1) {
>> +               value = padctl_readl(padctl, XUSB_PADCTL_ELPG_PROGRAM1);
>> +               value |=
>> XUSB_PADCTL_ELPG_PROGRAM1_SSPX_ELPG_CLAMP_EN_EARLY(
>> +                                       port->usb3_port_fake);
>> +               padctl_writel(padctl, value, XUSB_PADCTL_ELPG_PROGRAM1);
>> +
>> +               usleep_range(100, 200);
>> +
>> +               value = padctl_readl(padctl, XUSB_PADCTL_ELPG_PROGRAM1);
>> +               value |= XUSB_PADCTL_ELPG_PROGRAM1_SSPX_ELPG_CLAMP_EN(
>> +                                       port->usb3_port_fake);
>> +               padctl_writel(padctl, value, XUSB_PADCTL_ELPG_PROGRAM1);
>> +
>> +               usleep_range(250, 350);
>> +
>> +               value = padctl_readl(padctl, XUSB_PADCTL_ELPG_PROGRAM1);
>> +               value |= XUSB_PADCTL_ELPG_PROGRAM1_SSPX_ELPG_VCORE_DOWN(
>> +                                       port->usb3_port_fake);
>> +               padctl_writel(padctl, value, XUSB_PADCTL_ELPG_PROGRAM1);
>> +
>> +               value = padctl_readl(padctl, XUSB_PADCTL_SS_PORT_MAP);
>> +               value &= ~XUSB_PADCTL_SS_PORT_MAP_PORTX_MAP_MASK(
>> +                                       port->usb3_port_fake);
>> +               value |= XUSB_PADCTL_SS_PORT_MAP_PORTX_MAP_MASK(
>> +                                       port->usb3_port_fake);
>> +               padctl_writel(padctl, value, XUSB_PADCTL_SS_PORT_MAP);
>> +       }
>> +
>>         if (WARN_ON(pad->enable == 0))
>>                 goto out;
>>
>> @@ -1953,6 +2191,55 @@ static const struct tegra_xusb_port_ops
>> tegra210_usb3_port_ops = {
>>         .map = tegra210_usb3_port_map,
>>  };
>>
>> +static int tegra210_xusb_padctl_vbus_override(struct tegra_xusb_padctl
>> *padctl,
>> +                                             bool set)
>> +{
>> +       u32 reg;
>> +
>> +       dev_dbg(padctl->dev, "%s vbus override\n", set ? "set" : "clear");
>> +
>> +       reg = padctl_readl(padctl, XUSB_PADCTL_USB2_VBUS_ID);
>> +       if (set) {
>> +               reg |= XUSB_PADCTL_USB2_VBUS_ID_OVERRIDE_VBUS_ON;
>> +               reg &= ~(XUSB_PADCTL_USB2_VBUS_ID_OVERRIDE_MASK <<
>> +                          XUSB_PADCTL_USB2_VBUS_ID_OVERRIDE_SHIFT);
>> +               reg |= XUSB_PADCTL_USB2_VBUS_ID_OVERRIDE_VAL <<
>> +                        XUSB_PADCTL_USB2_VBUS_ID_OVERRIDE_SHIFT;
>> +       } else
>> +               reg &= ~XUSB_PADCTL_USB2_VBUS_ID_OVERRIDE_VBUS_ON;
>> +       padctl_writel(padctl, reg, XUSB_PADCTL_USB2_VBUS_ID);
>> +
>> +       return 0;
>> +}
>> +
>> +static int tegra210_utmi_port_reset_quirk(struct phy *phy)
>> +{
>> +       struct tegra_xusb_padctl *padctl;
>> +       struct tegra_xusb_lane *lane;
>> +       struct device *dev;
>> +       u32 reg;
>> +
>> +       if (!phy)
>> +               return -ENODEV;
>> +
>> +       lane = phy_get_drvdata(phy);
>> +       padctl = lane->pad->padctl;
>> +       dev = padctl->dev;
>> +
>> +       reg = padctl_readl(padctl,
>> +
>>  XUSB_PADCTL_USB2_BATTERY_CHRG_OTGPADX_CTL0(0));
>> +       dev_dbg(dev, "BATTERY_CHRG_OTGPADX_CTL0(0): 0x%x\n", reg);
>> +
>> +       if ((reg & XUSB_PADCTL_USB2_BATTERY_CHRG_OTGPAD_CTL0_ZIP) ||
>> +           (reg & XUSB_PADCTL_USB2_BATTERY_CHRG_OTGPAD_CTL0_ZIN)) {
>> +               dev_dbg(dev, "Toggle vbus\n");
>> +               tegra210_xusb_padctl_vbus_override(padctl, false);
>> +               tegra210_xusb_padctl_vbus_override(padctl, true);
>> +               return 1;
>> +       }
>> +       return 0;
>> +}
>> +
>>  static int
>>  tegra210_xusb_read_fuse_calibration(struct tegra210_xusb_fuse_calibration
>> *fuse)
>>  {
>> @@ -2015,6 +2302,10 @@ static const struct tegra_xusb_padctl_ops
>> tegra210_xusb_padctl_ops = {
>>         .remove = tegra210_xusb_padctl_remove,
>>         .usb3_set_lfps_detect = tegra210_usb3_set_lfps_detect,
>>         .hsic_set_idle = tegra210_hsic_set_idle,
>> +       .vbus_override = tegra210_xusb_padctl_vbus_override,
>> +       .utmi_pad_power_on = tegra210_utmi_pad_power_on,
>> +       .utmi_pad_power_down = tegra210_utmi_pad_power_down,
>> +       .utmi_port_reset_quirk = tegra210_utmi_port_reset_quirk,
>>  };
>>
>>  const struct tegra_xusb_padctl_soc tegra210_xusb_padctl_soc = {
>> diff --git a/drivers/phy/tegra/xusb.c b/drivers/phy/tegra/xusb.c
>> index 5b3b886..9a25b5d 100644
>> --- a/drivers/phy/tegra/xusb.c
>> +++ b/drivers/phy/tegra/xusb.c
>> @@ -1,5 +1,5 @@
>>  /*
>> - * Copyright (c) 2014-2015, NVIDIA CORPORATION.  All rights reserved.
>> + * Copyright (c) 2014-2018, NVIDIA CORPORATION.  All rights reserved.
>>   *
>>   * This program is free software; you can redistribute it and/or modify it
>>   * under the terms and conditions of the GNU General Public License,
>> @@ -546,9 +546,26 @@ static int tegra_xusb_usb2_port_parse_dt(struct
>> tegra_xusb_usb2_port *usb2)
>>  {
>>         struct tegra_xusb_port *port = &usb2->base;
>>         struct device_node *np = port->dev.of_node;
>> +       const char *prop_string;
>> +       u32 value;
>>
>>         usb2->internal = of_property_read_bool(np, "nvidia,internal");
>>
>> +       usb2->port_cap = USB_PORT_DISABLED; /* default */
>> +       if (!of_property_read_string(np, "mode", &prop_string)) {
>> +               if (!strcmp("host", prop_string))
>> +                       usb2->port_cap = USB_HOST_CAP;
>> +               else if (!strcmp("device", prop_string))
>> +                       usb2->port_cap = USB_DEVICE_CAP;
>> +               else if (!strcmp("otg", prop_string))
>> +                       usb2->port_cap = USB_OTG_CAP;
>> +       }
>> +
>> +       if (!of_property_read_u32(np, "nvidia,usb3-port-fake", &value))
>> +               usb2->usb3_port_fake = value;
>> +       else
>> +               usb2->usb3_port_fake = -1; /* default */
>> +
>>         usb2->supply = devm_regulator_get(&port->dev, "vbus");
>>         return PTR_ERR_OR_ZERO(usb2->supply);
>>  }
>> @@ -976,7 +993,7 @@ int tegra_xusb_padctl_usb3_save_context(struct
>> tegra_xusb_padctl *padctl,
>>         if (padctl->soc->ops->usb3_save_context)
>>                 return padctl->soc->ops->usb3_save_context(padctl, port);
>>
>> -       return -ENOSYS;
>> +       return -EINVAL;
>>  }
>>  EXPORT_SYMBOL_GPL(tegra_xusb_padctl_usb3_save_context);
>>
>> @@ -986,7 +1003,7 @@ int tegra_xusb_padctl_hsic_set_idle(struct
>> tegra_xusb_padctl *padctl,
>>         if (padctl->soc->ops->hsic_set_idle)
>>                 return padctl->soc->ops->hsic_set_idle(padctl, port, idle);
>>
>> -       return -ENOSYS;
>> +       return -EINVAL;
>>  }
>>  EXPORT_SYMBOL_GPL(tegra_xusb_padctl_hsic_set_idle);
>>
>> @@ -997,10 +1014,60 @@ int tegra_xusb_padctl_usb3_set_lfps_detect(struct
>> tegra_xusb_padctl *padctl,
>>                 return padctl->soc->ops->usb3_set_lfps_detect(padctl, port,
>>                                                               enable);
>>
>> -       return -ENOSYS;
>> +       return -EINVAL;
>>  }
>>  EXPORT_SYMBOL_GPL(tegra_xusb_padctl_usb3_set_lfps_detect);
>>
>> +int tegra_xusb_padctl_set_vbus_override(struct tegra_xusb_padctl *padctl)
>> +{
>> +       if (padctl->soc->ops->vbus_override)
>> +               return padctl->soc->ops->vbus_override(padctl, true);
>> +
>> +       return -EINVAL;
>> +}
>> +EXPORT_SYMBOL_GPL(tegra_xusb_padctl_set_vbus_override);
>> +
>> +int tegra_xusb_padctl_clear_vbus_override(struct tegra_xusb_padctl
>> *padctl)
>> +{
>> +       if (padctl->soc->ops->vbus_override)
>> +               return padctl->soc->ops->vbus_override(padctl, false);
>> +
>> +       return -EINVAL;
>> +}
>> +EXPORT_SYMBOL_GPL(tegra_xusb_padctl_clear_vbus_override);
>> +
>> +void tegra_phy_xusb_utmi_pad_power_on(struct phy *phy)
>> +{
>> +       struct tegra_xusb_lane *lane = phy_get_drvdata(phy);
>> +       struct tegra_xusb_padctl *padctl = lane->pad->padctl;
>> +
>> +       if (padctl->soc->ops->utmi_pad_power_on)
>> +               padctl->soc->ops->utmi_pad_power_on(phy);
>> +}
>> +EXPORT_SYMBOL_GPL(tegra_phy_xusb_utmi_pad_power_on);
>> +
>> +void tegra_phy_xusb_utmi_pad_power_down(struct phy *phy)
>> +{
>> +       struct tegra_xusb_lane *lane = phy_get_drvdata(phy);
>> +       struct tegra_xusb_padctl *padctl = lane->pad->padctl;
>> +
>> +       if (padctl->soc->ops->utmi_pad_power_down)
>> +               padctl->soc->ops->utmi_pad_power_down(phy);
>> +}
>> +EXPORT_SYMBOL_GPL(tegra_phy_xusb_utmi_pad_power_down);
>> +
>> +int tegra_phy_xusb_utmi_port_reset_quirk(struct phy *phy)
>> +{
>> +       struct tegra_xusb_lane *lane = phy_get_drvdata(phy);
>> +       struct tegra_xusb_padctl *padctl = lane->pad->padctl;
>> +
>> +       if (padctl->soc->ops->utmi_port_reset_quirk)
>> +               return padctl->soc->ops->utmi_port_reset_quirk(phy);
>> +
>> +       return -EINVAL;
>> +}
>> +EXPORT_SYMBOL_GPL(tegra_phy_xusb_utmi_port_reset_quirk);
>> +
>>  MODULE_AUTHOR("Thierry Reding <tred...@nvidia.com>");
>>  MODULE_DESCRIPTION("Tegra XUSB Pad Controller driver");
>>  MODULE_LICENSE("GPL v2");
>> diff --git a/drivers/phy/tegra/xusb.h b/drivers/phy/tegra/xusb.h
>> index b49dbc3..74acc08 100644
>> --- a/drivers/phy/tegra/xusb.h
>> +++ b/drivers/phy/tegra/xusb.h
>> @@ -1,5 +1,5 @@
>>  /*
>> - * Copyright (c) 2014-2015, NVIDIA CORPORATION.  All rights reserved.
>> + * Copyright (c) 2014-2018, NVIDIA CORPORATION.  All rights reserved.
>>   * Copyright (c) 2015, Google Inc.
>>   *
>>   * This program is free software; you can redistribute it and/or modify it
>> @@ -58,6 +58,7 @@ struct tegra_xusb_usb2_lane {
>>         struct tegra_xusb_lane base;
>>
>>         u32 hs_curr_level_offset;
>> +       bool powered_on;
>>  };
>>
>>  static inline struct tegra_xusb_usb2_lane *
>> @@ -267,11 +268,20 @@ struct tegra_xusb_port *
>>  tegra_xusb_find_port(struct tegra_xusb_padctl *padctl, const char *type,
>>                      unsigned int index);
>>
>> +enum tegra_xusb_usb_port_cap {
>> +       USB_PORT_DISABLED = 0,
>> +       USB_HOST_CAP,
>> +       USB_DEVICE_CAP,
>> +       USB_OTG_CAP,
>> +};
>> +
>>  struct tegra_xusb_usb2_port {
>>         struct tegra_xusb_port base;
>>
>>         struct regulator *supply;
>>         bool internal;
>> +       enum tegra_xusb_usb_port_cap port_cap;
>> +       int usb3_port_fake;
>>  };
>>
>>  static inline struct tegra_xusb_usb2_port *
>> @@ -353,6 +363,10 @@ struct tegra_xusb_padctl_ops {
>>                              unsigned int index, bool idle);
>>         int (*usb3_set_lfps_detect)(struct tegra_xusb_padctl *padctl,
>>                                     unsigned int index, bool enable);
>> +       int (*vbus_override)(struct tegra_xusb_padctl *padctl, bool set);
>> +       void (*utmi_pad_power_on)(struct phy *phy);
>> +       void (*utmi_pad_power_down)(struct phy *phy);
>> +       int (*utmi_port_reset_quirk)(struct phy *phy);
>>  };
>>
>>  struct tegra_xusb_padctl_soc {
>> diff --git a/include/linux/phy/tegra/xusb.h
>> b/include/linux/phy/tegra/xusb.h
>> index 8e1a57a..143af44 100644
>> --- a/include/linux/phy/tegra/xusb.h
>> +++ b/include/linux/phy/tegra/xusb.h
>> @@ -1,5 +1,5 @@
>>  /*
>> - * Copyright (c) 2016, NVIDIA CORPORATION.  All rights reserved.
>> + * Copyright (c) 2016-2018, NVIDIA CORPORATION.  All rights reserved.
>>   *
>>   * This program is free software; you can redistribute it and/or modify it
>>   * under the terms and conditions of the GNU General Public License,
>> @@ -26,5 +26,10 @@ int tegra_xusb_padctl_hsic_set_idle(struct
>> tegra_xusb_padctl *padctl,
>>                                     unsigned int port, bool idle);
>>  int tegra_xusb_padctl_usb3_set_lfps_detect(struct tegra_xusb_padctl
>> *padctl,
>>                                            unsigned int port, bool enable);
>> +int tegra_xusb_padctl_set_vbus_override(struct tegra_xusb_padctl *padctl);
>> +int tegra_xusb_padctl_clear_vbus_override(struct tegra_xusb_padctl
>> *padctl);
>>
>> +void tegra_phy_xusb_utmi_pad_power_on(struct phy *phy);
>> +void tegra_phy_xusb_utmi_pad_power_down(struct phy *phy);
>> +int tegra_phy_xusb_utmi_port_reset_quirk(struct phy *phy);
>>  #endif /* PHY_TEGRA_XUSB_H */
>> --
>> 2.7.4
>>
>>
> 

Reply via email to