usb: gadget: tegra-xudc: add port_speed_quirk
authorNagarjuna Kristam <nkristam@nvidia.com>
Mon, 4 May 2020 06:34:41 +0000 (12:04 +0530)
committerFelipe Balbi <balbi@kernel.org>
Mon, 25 May 2020 08:09:40 +0000 (11:09 +0300)
OTG port on Tegra194 supports GEN1 speeds when in device mode and GEN2
speeds when in host mode. dd port_speed_quirk that configures port to
GEN1/GEN2 speds, corresponding to the mode.

Based on work by WayneChang <waynec@nvidia.com>

Signed-off-by: Nagarjuna Kristam <nkristam@nvidia.com>
Signed-off-by: Felipe Balbi <balbi@kernel.org>
drivers/usb/gadget/udc/tegra-xudc.c

index 8ab1091ebc0e1756699eed2300eacbcee58b709c..c40e98dacd98c6b2c558aaab352a06930c39b2cb 100644 (file)
 #define  SSPX_CORE_CNT32_POLL_TBURST_MAX_MASK GENMASK(7, 0)
 #define  SSPX_CORE_CNT32_POLL_TBURST_MAX(x) ((x) & \
                                        SSPX_CORE_CNT32_POLL_TBURST_MAX_MASK)
+#define SSPX_CORE_CNT56 0x6fc
+#define  SSPX_CORE_CNT56_SCD_BIT0_TRPT_MAX_MASK GENMASK(19, 0)
+#define  SSPX_CORE_CNT56_SCD_BIT0_TRPT_MAX(x) ((x) & \
+                               SSPX_CORE_CNT56_SCD_BIT0_TRPT_MAX_MASK)
+#define SSPX_CORE_CNT57 0x700
+#define  SSPX_CORE_CNT57_SCD_BIT1_TRPT_MAX_MASK GENMASK(19, 0)
+#define  SSPX_CORE_CNT57_SCD_BIT1_TRPT_MAX(x) ((x) & \
+                               SSPX_CORE_CNT57_SCD_BIT1_TRPT_MAX_MASK)
+#define SSPX_CORE_CNT65 0x720
+#define  SSPX_CORE_CNT65_TX_SCD_END_TRPT_MID_MASK GENMASK(19, 0)
+#define  SSPX_CORE_CNT65_TX_SCD_END_TRPT_MID(x) ((x) & \
+                               SSPX_CORE_CNT65_TX_SCD_END_TRPT_MID_MASK)
+#define SSPX_CORE_CNT66 0x724
+#define  SSPX_CORE_CNT66_TX_SCD_BIT0_TRPT_MID_MASK GENMASK(19, 0)
+#define  SSPX_CORE_CNT66_TX_SCD_BIT0_TRPT_MID(x) ((x) & \
+                               SSPX_CORE_CNT66_TX_SCD_BIT0_TRPT_MID_MASK)
+#define SSPX_CORE_CNT67 0x728
+#define  SSPX_CORE_CNT67_TX_SCD_BIT1_TRPT_MID_MASK GENMASK(19, 0)
+#define  SSPX_CORE_CNT67_TX_SCD_BIT1_TRPT_MID(x) ((x) & \
+                               SSPX_CORE_CNT67_TX_SCD_BIT1_TRPT_MID_MASK)
+#define SSPX_CORE_CNT72 0x73c
+#define  SSPX_CORE_CNT72_SCD_LFPS_TIMEOUT_MASK GENMASK(19, 0)
+#define  SSPX_CORE_CNT72_SCD_LFPS_TIMEOUT(x) ((x) & \
+                               SSPX_CORE_CNT72_SCD_LFPS_TIMEOUT_MASK)
 #define SSPX_CORE_PADCTL4 0x750
 #define  SSPX_CORE_PADCTL4_RXDAT_VLD_TIMEOUT_U3_MASK GENMASK(19, 0)
 #define  SSPX_CORE_PADCTL4_RXDAT_VLD_TIMEOUT_U3(x) ((x) & \
@@ -531,6 +555,7 @@ struct tegra_xudc_soc {
        bool invalid_seq_num;
        bool pls_quirk;
        bool port_reset_quirk;
+       bool port_speed_quirk;
        bool has_ipfs;
 };
 
@@ -600,6 +625,78 @@ static inline void dump_trb(struct tegra_xudc *xudc, const char *type,
                trb->control);
 }
 
+static void tegra_xudc_limit_port_speed(struct tegra_xudc *xudc)
+{
+       u32 val;
+
+       /* limit port speed to gen 1 */
+       val = xudc_readl(xudc, SSPX_CORE_CNT56);
+       val &= ~(SSPX_CORE_CNT56_SCD_BIT0_TRPT_MAX_MASK);
+       val |= SSPX_CORE_CNT56_SCD_BIT0_TRPT_MAX(0x260);
+       xudc_writel(xudc, val, SSPX_CORE_CNT56);
+
+       val = xudc_readl(xudc, SSPX_CORE_CNT57);
+       val &= ~(SSPX_CORE_CNT57_SCD_BIT1_TRPT_MAX_MASK);
+       val |= SSPX_CORE_CNT57_SCD_BIT1_TRPT_MAX(0x6D6);
+       xudc_writel(xudc, val, SSPX_CORE_CNT57);
+
+       val = xudc_readl(xudc, SSPX_CORE_CNT65);
+       val &= ~(SSPX_CORE_CNT65_TX_SCD_END_TRPT_MID_MASK);
+       val |= SSPX_CORE_CNT65_TX_SCD_END_TRPT_MID(0x4B0);
+       xudc_writel(xudc, val, SSPX_CORE_CNT66);
+
+       val = xudc_readl(xudc, SSPX_CORE_CNT66);
+       val &= ~(SSPX_CORE_CNT66_TX_SCD_BIT0_TRPT_MID_MASK);
+       val |= SSPX_CORE_CNT66_TX_SCD_BIT0_TRPT_MID(0x4B0);
+       xudc_writel(xudc, val, SSPX_CORE_CNT66);
+
+       val = xudc_readl(xudc, SSPX_CORE_CNT67);
+       val &= ~(SSPX_CORE_CNT67_TX_SCD_BIT1_TRPT_MID_MASK);
+       val |= SSPX_CORE_CNT67_TX_SCD_BIT1_TRPT_MID(0x4B0);
+       xudc_writel(xudc, val, SSPX_CORE_CNT67);
+
+       val = xudc_readl(xudc, SSPX_CORE_CNT72);
+       val &= ~(SSPX_CORE_CNT72_SCD_LFPS_TIMEOUT_MASK);
+       val |= SSPX_CORE_CNT72_SCD_LFPS_TIMEOUT(0x10);
+       xudc_writel(xudc, val, SSPX_CORE_CNT72);
+}
+
+static void tegra_xudc_restore_port_speed(struct tegra_xudc *xudc)
+{
+       u32 val;
+
+       /* restore port speed to gen2 */
+       val = xudc_readl(xudc, SSPX_CORE_CNT56);
+       val &= ~(SSPX_CORE_CNT56_SCD_BIT0_TRPT_MAX_MASK);
+       val |= SSPX_CORE_CNT56_SCD_BIT0_TRPT_MAX(0x438);
+       xudc_writel(xudc, val, SSPX_CORE_CNT56);
+
+       val = xudc_readl(xudc, SSPX_CORE_CNT57);
+       val &= ~(SSPX_CORE_CNT57_SCD_BIT1_TRPT_MAX_MASK);
+       val |= SSPX_CORE_CNT57_SCD_BIT1_TRPT_MAX(0x528);
+       xudc_writel(xudc, val, SSPX_CORE_CNT57);
+
+       val = xudc_readl(xudc, SSPX_CORE_CNT65);
+       val &= ~(SSPX_CORE_CNT65_TX_SCD_END_TRPT_MID_MASK);
+       val |= SSPX_CORE_CNT65_TX_SCD_END_TRPT_MID(0xE10);
+       xudc_writel(xudc, val, SSPX_CORE_CNT66);
+
+       val = xudc_readl(xudc, SSPX_CORE_CNT66);
+       val &= ~(SSPX_CORE_CNT66_TX_SCD_BIT0_TRPT_MID_MASK);
+       val |= SSPX_CORE_CNT66_TX_SCD_BIT0_TRPT_MID(0x348);
+       xudc_writel(xudc, val, SSPX_CORE_CNT66);
+
+       val = xudc_readl(xudc, SSPX_CORE_CNT67);
+       val &= ~(SSPX_CORE_CNT67_TX_SCD_BIT1_TRPT_MID_MASK);
+       val |= SSPX_CORE_CNT67_TX_SCD_BIT1_TRPT_MID(0x5a0);
+       xudc_writel(xudc, val, SSPX_CORE_CNT67);
+
+       val = xudc_readl(xudc, SSPX_CORE_CNT72);
+       val &= ~(SSPX_CORE_CNT72_SCD_LFPS_TIMEOUT_MASK);
+       val |= SSPX_CORE_CNT72_SCD_LFPS_TIMEOUT(0x1c21);
+       xudc_writel(xudc, val, SSPX_CORE_CNT72);
+}
+
 static void tegra_xudc_device_mode_on(struct tegra_xudc *xudc)
 {
        int err;
@@ -632,6 +729,9 @@ static void tegra_xudc_device_mode_off(struct tegra_xudc *xudc)
 
        reinit_completion(&xudc->disconnect_complete);
 
+       if (xudc->soc->port_speed_quirk)
+               tegra_xudc_restore_port_speed(xudc);
+
        phy_set_mode_ext(xudc->curr_utmi_phy, PHY_MODE_USB_OTG, USB_ROLE_NONE);
 
        pls = (xudc_readl(xudc, PORTSC) & PORTSC_PLS_MASK) >>
@@ -3291,6 +3391,9 @@ static void tegra_xudc_device_params_init(struct tegra_xudc *xudc)
                xudc_writel(xudc, val, BLCG);
        }
 
+       if (xudc->soc->port_speed_quirk)
+               tegra_xudc_limit_port_speed(xudc);
+
        /* Set a reasonable U3 exit timer value. */
        val = xudc_readl(xudc, SSPX_CORE_PADCTL4);
        val &= ~(SSPX_CORE_PADCTL4_RXDAT_VLD_TIMEOUT_U3_MASK);
@@ -3523,6 +3626,7 @@ static struct tegra_xudc_soc tegra210_xudc_soc_data = {
        .invalid_seq_num = true,
        .pls_quirk = true,
        .port_reset_quirk = true,
+       .port_speed_quirk = false,
        .has_ipfs = true,
 };
 
@@ -3536,6 +3640,7 @@ static struct tegra_xudc_soc tegra186_xudc_soc_data = {
        .invalid_seq_num = false,
        .pls_quirk = false,
        .port_reset_quirk = false,
+       .port_speed_quirk = false,
        .has_ipfs = false,
 };
 
@@ -3549,6 +3654,7 @@ static struct tegra_xudc_soc tegra194_xudc_soc_data = {
        .invalid_seq_num = false,
        .pls_quirk = false,
        .port_reset_quirk = false,
+       .port_speed_quirk = true,
        .has_ipfs = false,
 };