858 lines
23 KiB
C
858 lines
23 KiB
C
/*
|
|
* Copyright (c) 2012-2014, The Linux Foundation. All rights reserved.
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License version 2 and
|
|
* only version 2 as published by the Free Software Foundation.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
*/
|
|
|
|
#include <linux/module.h>
|
|
#include <linux/kernel.h>
|
|
#include <linux/err.h>
|
|
#include <linux/slab.h>
|
|
#include <linux/clk.h>
|
|
#include <linux/clk/msm-clk.h>
|
|
#include <linux/delay.h>
|
|
#include <linux/io.h>
|
|
#include <linux/of.h>
|
|
#include <linux/platform_device.h>
|
|
#include <linux/regulator/consumer.h>
|
|
#include <linux/usb/phy.h>
|
|
#include <linux/usb/msm_hsusb.h>
|
|
|
|
static int override_phy_init;
|
|
module_param(override_phy_init, int, S_IRUGO|S_IWUSR);
|
|
MODULE_PARM_DESC(override_phy_init, "Override HSPHY Init Seq");
|
|
|
|
|
|
#define PORT_OFFSET(i) ((i == 0) ? 0x0 : ((i == 1) ? 0x6c : 0x88))
|
|
|
|
/* QSCRATCH register settings differ based on MSM core ver */
|
|
#define MSM_CORE_VER_120 0x10020061
|
|
#define MSM_CORE_VER_160 0x10060000
|
|
#define MSM_CORE_VER_161 0x10060001
|
|
|
|
/* QSCRATCH register offsets */
|
|
#define GENERAL_CFG_REG (0x08)
|
|
#define HS_PHY_CTRL_REG(i) (0x10 + PORT_OFFSET(i))
|
|
#define PARAMETER_OVERRIDE_X_REG(i) (0x14 + PORT_OFFSET(i))
|
|
#define ALT_INTERRUPT_EN_REG(i) (0x20 + PORT_OFFSET(i))
|
|
#define HS_PHY_IRQ_STAT_REG(i) (0x24 + PORT_OFFSET(i))
|
|
#define HS_PHY_CTRL_COMMON_REG (0xEC) /* ver >= MSM_CORE_VER_120 */
|
|
|
|
/* GENERAL_CFG_REG bits */
|
|
#define SEC_UTMI_FREE_CLK_GFM_SEL1 (0x80)
|
|
|
|
/* HS_PHY_CTRL_REG bits */
|
|
#define RETENABLEN BIT(1)
|
|
#define FSEL_MASK (0x7 << 4)
|
|
#define FSEL_DEFAULT (0x3 << 4)
|
|
#define CLAMP_EN_N BIT(7)
|
|
#define OTGSESSVLD_HV_CLAMP_EN_N BIT(8)
|
|
#define ID_HV_CLAMP_EN_N BIT(9)
|
|
#define COMMONONN BIT(11)
|
|
#define OTGDISABLE0 BIT(12)
|
|
#define VBUSVLDEXT0 BIT(13)
|
|
#define VBUSVLDEXTSEL0 BIT(14)
|
|
#define OTGSESSVLDHV_INTEN BIT(15)
|
|
#define IDHV_INTEN BIT(16)
|
|
#define DPSEHV_CLAMP_EN_N BIT(17)
|
|
#define UTMI_OTG_VBUS_VALID BIT(20)
|
|
#define USB2_UTMI_CLK_EN BIT(21)
|
|
#define USB2_SUSPEND_N BIT(22)
|
|
#define USB2_SUSPEND_N_SEL BIT(23)
|
|
#define DMSEHV_CLAMP_EN_N BIT(24)
|
|
#define CLAMP_MPM_DPSE_DMSE_EN_N BIT(26)
|
|
/* Following exist only when core_ver >= MSM_CORE_VER_120 */
|
|
#define FREECLK_DIS_WHEN_SUSP BIT(27)
|
|
#define SW_SESSVLD_SEL BIT(28)
|
|
#define FREECLOCK_SEL BIT(29)
|
|
|
|
/* HS_PHY_CTRL_COMMON_REG bits used when core_ver >= MSM_CORE_VER_120 */
|
|
#define COMMON_PLLITUNE_1 BIT(18)
|
|
#define COMMON_PLLBTUNE BIT(15)
|
|
#define COMMON_CLKCORE BIT(14)
|
|
#define COMMON_VBUSVLDEXTSEL0 BIT(12)
|
|
#define COMMON_OTGDISABLE0 BIT(11)
|
|
#define COMMON_OTGTUNE0_MASK (0x7 << 8)
|
|
#define COMMON_OTGTUNE0_DEFAULT (0x4 << 8)
|
|
#define COMMON_COMMONONN BIT(7)
|
|
#define COMMON_FSEL (0x7 << 4)
|
|
#define COMMON_RETENABLEN BIT(3)
|
|
|
|
/* ALT_INTERRUPT_EN/HS_PHY_IRQ_STAT bits */
|
|
#define ACAINTEN BIT(0)
|
|
#define DMINTEN BIT(1)
|
|
#define DCDINTEN BIT(1)
|
|
#define DPINTEN BIT(3)
|
|
#define CHGDETINTEN BIT(4)
|
|
#define RIDFLOATNINTEN BIT(5)
|
|
#define DPSEHV_INTEN BIT(6)
|
|
#define DMSEHV_INTEN BIT(7)
|
|
#define DPSEHV_HI_INTEN BIT(8)
|
|
#define DPSEHV_LO_INTEN BIT(9)
|
|
#define DMSEHV_HI_INTEN BIT(10)
|
|
#define DMSEHV_LO_INTEN BIT(11)
|
|
#define LINESTATE_INTEN BIT(12)
|
|
#define DPDMHV_INT_MASK (0xFC0)
|
|
#define ALT_INTERRUPT_MASK (0x1FFF)
|
|
|
|
#define TCSR_USB30_CONTROL BIT(8)
|
|
#define TCSR_HSPHY_ARES BIT(11)
|
|
|
|
#define USB_HSPHY_3P3_VOL_MIN 3050000 /* uV */
|
|
#define USB_HSPHY_3P3_VOL_MAX 3300000 /* uV */
|
|
#define USB_HSPHY_3P3_HPM_LOAD 16000 /* uA */
|
|
#define USB_HSPHY_3P3_VOL_FSHOST 3150000 /* uV */
|
|
|
|
#define USB_HSPHY_1P8_VOL_MIN 1800000 /* uV */
|
|
#define USB_HSPHY_1P8_VOL_MAX 1800000 /* uV */
|
|
#define USB_HSPHY_1P8_HPM_LOAD 19000 /* uA */
|
|
|
|
struct msm_hsphy {
|
|
struct usb_phy phy;
|
|
void __iomem *base;
|
|
void __iomem *tcsr;
|
|
int hsphy_init_seq;
|
|
bool set_pllbtune;
|
|
u32 core_ver;
|
|
|
|
struct clk *sleep_clk;
|
|
bool sleep_clk_reset;
|
|
|
|
struct regulator *vdd;
|
|
struct regulator *vdda33;
|
|
struct regulator *vdda18;
|
|
int vdd_levels[3]; /* none, low, high */
|
|
u32 lpm_flags;
|
|
bool suspended;
|
|
bool vdda_force_on;
|
|
|
|
/* Using external VBUS/ID notification */
|
|
bool ext_vbus_id;
|
|
int num_ports;
|
|
bool cable_connected;
|
|
};
|
|
|
|
/* global reference counter between all HSPHY instances */
|
|
static atomic_t hsphy_active_count;
|
|
|
|
static int msm_hsusb_config_vdd(struct msm_hsphy *phy, int high)
|
|
{
|
|
int min, ret;
|
|
|
|
min = high ? 1 : 0; /* low or none? */
|
|
ret = regulator_set_voltage(phy->vdd, phy->vdd_levels[min],
|
|
phy->vdd_levels[2]);
|
|
if (ret) {
|
|
dev_err(phy->phy.dev, "unable to set voltage for hsusb vdd\n");
|
|
return ret;
|
|
}
|
|
|
|
dev_dbg(phy->phy.dev, "%s: min_vol:%d max_vol:%d\n", __func__,
|
|
phy->vdd_levels[min], phy->vdd_levels[2]);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static int msm_hsusb_ldo_enable(struct msm_hsphy *phy, int on)
|
|
{
|
|
int rc = 0;
|
|
|
|
dev_dbg(phy->phy.dev, "reg (%s)\n", on ? "HPM" : "LPM");
|
|
|
|
if (!on)
|
|
goto disable_regulators;
|
|
|
|
|
|
rc = regulator_set_optimum_mode(phy->vdda18, USB_HSPHY_1P8_HPM_LOAD);
|
|
if (rc < 0) {
|
|
dev_err(phy->phy.dev, "Unable to set HPM of vdda18\n");
|
|
return rc;
|
|
}
|
|
|
|
rc = regulator_set_voltage(phy->vdda18, USB_HSPHY_1P8_VOL_MIN,
|
|
USB_HSPHY_1P8_VOL_MAX);
|
|
if (rc) {
|
|
dev_err(phy->phy.dev, "unable to set voltage for vdda18\n");
|
|
goto put_vdda18_lpm;
|
|
}
|
|
|
|
rc = regulator_enable(phy->vdda18);
|
|
if (rc) {
|
|
dev_err(phy->phy.dev, "Unable to enable vdda18\n");
|
|
goto unset_vdda18;
|
|
}
|
|
|
|
rc = regulator_set_optimum_mode(phy->vdda33, USB_HSPHY_3P3_HPM_LOAD);
|
|
if (rc < 0) {
|
|
dev_err(phy->phy.dev, "Unable to set HPM of vdda33\n");
|
|
goto disable_vdda18;
|
|
}
|
|
|
|
rc = regulator_set_voltage(phy->vdda33, USB_HSPHY_3P3_VOL_MIN,
|
|
USB_HSPHY_3P3_VOL_MAX);
|
|
if (rc) {
|
|
dev_err(phy->phy.dev, "unable to set voltage for vdda33\n");
|
|
goto put_vdda33_lpm;
|
|
}
|
|
|
|
rc = regulator_enable(phy->vdda33);
|
|
if (rc) {
|
|
dev_err(phy->phy.dev, "Unable to enable vdda33\n");
|
|
goto unset_vdda33;
|
|
}
|
|
|
|
return 0;
|
|
|
|
disable_regulators:
|
|
rc = regulator_disable(phy->vdda33);
|
|
if (rc)
|
|
dev_err(phy->phy.dev, "Unable to disable vdda33\n");
|
|
|
|
unset_vdda33:
|
|
rc = regulator_set_voltage(phy->vdda33, 0, USB_HSPHY_3P3_VOL_MAX);
|
|
if (rc)
|
|
dev_err(phy->phy.dev, "unable to set voltage for vdda33\n");
|
|
|
|
put_vdda33_lpm:
|
|
rc = regulator_set_optimum_mode(phy->vdda33, 0);
|
|
if (rc < 0)
|
|
dev_err(phy->phy.dev, "Unable to set LPM of vdda33\n");
|
|
|
|
disable_vdda18:
|
|
rc = regulator_disable(phy->vdda18);
|
|
if (rc)
|
|
dev_err(phy->phy.dev, "Unable to disable vdda18\n");
|
|
|
|
unset_vdda18:
|
|
rc = regulator_set_voltage(phy->vdda18, 0, USB_HSPHY_1P8_VOL_MAX);
|
|
if (rc)
|
|
dev_err(phy->phy.dev, "unable to set voltage for vdda18\n");
|
|
|
|
put_vdda18_lpm:
|
|
rc = regulator_set_optimum_mode(phy->vdda18, 0);
|
|
if (rc < 0)
|
|
dev_err(phy->phy.dev, "Unable to set LPM of vdda18\n");
|
|
|
|
return rc < 0 ? rc : 0;
|
|
}
|
|
|
|
static void msm_usb_write_readback(void *base, u32 offset,
|
|
const u32 mask, u32 val)
|
|
{
|
|
u32 write_val, tmp = readl_relaxed(base + offset);
|
|
|
|
tmp &= ~mask; /* retain other bits */
|
|
write_val = tmp | val;
|
|
|
|
writel_relaxed(write_val, base + offset);
|
|
|
|
/* Read back to see if val was written */
|
|
tmp = readl_relaxed(base + offset);
|
|
tmp &= mask; /* clear other bits */
|
|
|
|
if (tmp != val)
|
|
pr_err("%s: write: %x to QSCRATCH: %x FAILED\n",
|
|
__func__, val, offset);
|
|
}
|
|
|
|
static int msm_hsphy_reset(struct usb_phy *uphy)
|
|
{
|
|
struct msm_hsphy *phy = container_of(uphy, struct msm_hsphy, phy);
|
|
u32 val;
|
|
int ret;
|
|
|
|
/* skip reset if there are other active PHY instances */
|
|
ret = atomic_read(&hsphy_active_count);
|
|
if (ret > 1) {
|
|
dev_dbg(uphy->dev, "skipping reset, inuse count=%d\n", ret);
|
|
return 0;
|
|
}
|
|
|
|
if (phy->tcsr) {
|
|
val = readl_relaxed(phy->tcsr);
|
|
|
|
/* Assert/deassert TCSR Reset */
|
|
writel_relaxed((val | TCSR_HSPHY_ARES), phy->tcsr);
|
|
usleep_range(1000, 1200);
|
|
writel_relaxed((val & ~TCSR_HSPHY_ARES), phy->tcsr);
|
|
} else if (phy->sleep_clk_reset) {
|
|
/* Reset PHY using sleep clock */
|
|
ret = clk_reset(phy->sleep_clk, CLK_RESET_ASSERT);
|
|
if (ret) {
|
|
dev_err(uphy->dev, "hsphy_sleep_clk assert failed\n");
|
|
return ret;
|
|
}
|
|
|
|
usleep_range(1000, 1200);
|
|
ret = clk_reset(phy->sleep_clk, CLK_RESET_DEASSERT);
|
|
if (ret) {
|
|
dev_err(uphy->dev, "hsphy_sleep_clk reset deassert failed\n");
|
|
return ret;
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int msm_hsphy_init(struct usb_phy *uphy)
|
|
{
|
|
struct msm_hsphy *phy = container_of(uphy, struct msm_hsphy, phy);
|
|
u32 val;
|
|
|
|
msm_hsphy_reset(uphy);
|
|
|
|
/* different sequences based on core version */
|
|
phy->core_ver = readl_relaxed(phy->base);
|
|
|
|
/*
|
|
* HSPHY Initialization: Enable UTMI clock and clamp enable HVINTs,
|
|
* and disable RETENTION (power-on default is ENABLED)
|
|
*/
|
|
val = readl_relaxed(phy->base + HS_PHY_CTRL_REG(0));
|
|
val |= (USB2_UTMI_CLK_EN | CLAMP_MPM_DPSE_DMSE_EN_N | RETENABLEN);
|
|
|
|
if (uphy->flags & ENABLE_SECONDARY_PHY) {
|
|
val &= ~(USB2_UTMI_CLK_EN | FREECLOCK_SEL);
|
|
val |= FREECLK_DIS_WHEN_SUSP;
|
|
}
|
|
|
|
writel_relaxed(val, phy->base + HS_PHY_CTRL_REG(0));
|
|
usleep_range(2000, 2200);
|
|
|
|
if (uphy->flags & ENABLE_SECONDARY_PHY)
|
|
msm_usb_write_readback(phy->base, GENERAL_CFG_REG,
|
|
SEC_UTMI_FREE_CLK_GFM_SEL1,
|
|
SEC_UTMI_FREE_CLK_GFM_SEL1);
|
|
|
|
if (phy->core_ver >= MSM_CORE_VER_120) {
|
|
if (phy->set_pllbtune) {
|
|
val = readl_relaxed(phy->base + HS_PHY_CTRL_COMMON_REG);
|
|
val |= COMMON_PLLBTUNE | COMMON_CLKCORE;
|
|
val &= ~COMMON_FSEL;
|
|
writel_relaxed(val, phy->base + HS_PHY_CTRL_COMMON_REG);
|
|
} else {
|
|
writel_relaxed(COMMON_OTGDISABLE0 |
|
|
COMMON_OTGTUNE0_DEFAULT |
|
|
COMMON_COMMONONN | FSEL_DEFAULT |
|
|
COMMON_RETENABLEN,
|
|
phy->base + HS_PHY_CTRL_COMMON_REG);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* write HSPHY init value to QSCRATCH reg to set HSPHY parameters like
|
|
* VBUS valid threshold, disconnect valid threshold, DC voltage level,
|
|
* preempasis and rise/fall time.
|
|
*/
|
|
if (override_phy_init)
|
|
phy->hsphy_init_seq = override_phy_init;
|
|
if (phy->hsphy_init_seq)
|
|
msm_usb_write_readback(phy->base,
|
|
PARAMETER_OVERRIDE_X_REG(0), 0x03FFFFFF,
|
|
phy->hsphy_init_seq & 0x03FFFFFF);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int msm_hsphy_set_suspend(struct usb_phy *uphy, int suspend)
|
|
{
|
|
struct msm_hsphy *phy = container_of(uphy, struct msm_hsphy, phy);
|
|
bool host = uphy->flags & PHY_HOST_MODE;
|
|
bool chg_connected = uphy->flags & PHY_CHARGER_CONNECTED;
|
|
int i, count;
|
|
|
|
if (!!suspend == phy->suspended) {
|
|
dev_dbg(uphy->dev, "%s\n", suspend ? "already suspended"
|
|
: "already resumed");
|
|
return 0;
|
|
}
|
|
|
|
if (suspend) {
|
|
for (i = 0; i < phy->num_ports; i++) {
|
|
/* Clear interrupt latch register */
|
|
writel_relaxed(ALT_INTERRUPT_MASK,
|
|
phy->base + HS_PHY_IRQ_STAT_REG(i));
|
|
|
|
if (host) {
|
|
/* Enable DP and DM HV interrupts */
|
|
if (phy->core_ver >= MSM_CORE_VER_120)
|
|
msm_usb_write_readback(phy->base,
|
|
ALT_INTERRUPT_EN_REG(i),
|
|
(LINESTATE_INTEN |
|
|
DPINTEN | DMINTEN),
|
|
(LINESTATE_INTEN |
|
|
DPINTEN | DMINTEN));
|
|
else
|
|
msm_usb_write_readback(phy->base,
|
|
ALT_INTERRUPT_EN_REG(i),
|
|
DPDMHV_INT_MASK,
|
|
DPDMHV_INT_MASK);
|
|
|
|
udelay(5);
|
|
} else {
|
|
/* set the following:
|
|
* OTGDISABLE0=1
|
|
* USB2_SUSPEND_N_SEL=1, USB2_SUSPEND_N=0
|
|
*/
|
|
if (phy->core_ver >= MSM_CORE_VER_120)
|
|
msm_usb_write_readback(phy->base,
|
|
HS_PHY_CTRL_COMMON_REG,
|
|
COMMON_OTGDISABLE0,
|
|
COMMON_OTGDISABLE0);
|
|
else
|
|
msm_usb_write_readback(phy->base,
|
|
HS_PHY_CTRL_REG(i),
|
|
OTGDISABLE0, OTGDISABLE0);
|
|
|
|
msm_usb_write_readback(phy->base,
|
|
HS_PHY_CTRL_REG(i),
|
|
(USB2_SUSPEND_N_SEL | USB2_SUSPEND_N),
|
|
USB2_SUSPEND_N_SEL);
|
|
/*
|
|
* Enable PHY retention
|
|
* RETENABLEN bit is not available on few platforms.
|
|
*/
|
|
if (!chg_connected) {
|
|
if (phy->set_pllbtune)
|
|
msm_usb_write_readback(phy->base,
|
|
HS_PHY_CTRL_COMMON_REG,
|
|
COMMON_PLLITUNE_1,
|
|
COMMON_PLLITUNE_1);
|
|
else
|
|
msm_usb_write_readback(phy->base,
|
|
HS_PHY_CTRL_REG(i),
|
|
RETENABLEN, 0);
|
|
phy->lpm_flags |= PHY_RETENTIONED;
|
|
}
|
|
}
|
|
|
|
if (!phy->ext_vbus_id)
|
|
/* Enable PHY-based IDHV and
|
|
*OTGSESSVLD HV interrupts
|
|
*/
|
|
msm_usb_write_readback(phy->base,
|
|
HS_PHY_CTRL_REG(i),
|
|
(OTGSESSVLDHV_INTEN | IDHV_INTEN),
|
|
(OTGSESSVLDHV_INTEN | IDHV_INTEN));
|
|
}
|
|
/* can turn off regulators if disconnected in device mode */
|
|
if (phy->lpm_flags & PHY_RETENTIONED && !phy->cable_connected) {
|
|
if (phy->ext_vbus_id) {
|
|
msm_hsusb_ldo_enable(phy, 0);
|
|
phy->lpm_flags |= PHY_PWR_COLLAPSED;
|
|
}
|
|
msm_hsusb_config_vdd(phy, 0);
|
|
}
|
|
|
|
count = atomic_dec_return(&hsphy_active_count);
|
|
if (count < 0) {
|
|
dev_WARN(uphy->dev, "hsphy_active_count=%d, something wrong?\n",
|
|
count);
|
|
atomic_set(&hsphy_active_count, 0);
|
|
}
|
|
} else {
|
|
atomic_inc(&hsphy_active_count);
|
|
if (phy->lpm_flags & PHY_RETENTIONED && !phy->cable_connected) {
|
|
msm_hsusb_config_vdd(phy, 1);
|
|
if (phy->ext_vbus_id) {
|
|
msm_hsusb_ldo_enable(phy, 1);
|
|
phy->lpm_flags &= ~PHY_PWR_COLLAPSED;
|
|
}
|
|
phy->lpm_flags &= ~PHY_RETENTIONED;
|
|
}
|
|
|
|
if (phy->core_ver >= MSM_CORE_VER_120) {
|
|
if (phy->set_pllbtune) {
|
|
msm_usb_write_readback(phy->base,
|
|
HS_PHY_CTRL_COMMON_REG,
|
|
FSEL_MASK, 0);
|
|
} else {
|
|
msm_usb_write_readback(phy->base,
|
|
HS_PHY_CTRL_COMMON_REG,
|
|
FSEL_MASK, FSEL_DEFAULT);
|
|
}
|
|
}
|
|
for (i = 0; i < phy->num_ports; i++) {
|
|
if (!phy->ext_vbus_id)
|
|
/* Disable HV interrupts */
|
|
msm_usb_write_readback(phy->base,
|
|
HS_PHY_CTRL_REG(i),
|
|
(OTGSESSVLDHV_INTEN | IDHV_INTEN),
|
|
0);
|
|
if (host) {
|
|
/* Clear interrupt latch register */
|
|
writel_relaxed(ALT_INTERRUPT_MASK,
|
|
phy->base + HS_PHY_IRQ_STAT_REG(i));
|
|
/* Disable DP and DM HV interrupt */
|
|
if (phy->core_ver >= MSM_CORE_VER_120)
|
|
msm_usb_write_readback(phy->base,
|
|
ALT_INTERRUPT_EN_REG(i),
|
|
LINESTATE_INTEN, 0);
|
|
else
|
|
msm_usb_write_readback(phy->base,
|
|
ALT_INTERRUPT_EN_REG(i),
|
|
DPDMHV_INT_MASK, 0);
|
|
} else {
|
|
/* Disable PHY retention */
|
|
if (phy->set_pllbtune)
|
|
msm_usb_write_readback(phy->base,
|
|
HS_PHY_CTRL_COMMON_REG,
|
|
COMMON_PLLITUNE_1, 0);
|
|
else
|
|
msm_usb_write_readback(phy->base,
|
|
HS_PHY_CTRL_REG(i),
|
|
RETENABLEN, RETENABLEN);
|
|
|
|
/* Bring PHY out of suspend */
|
|
msm_usb_write_readback(phy->base,
|
|
HS_PHY_CTRL_REG(i),
|
|
USB2_SUSPEND_N_SEL, 0);
|
|
|
|
if (phy->core_ver >= MSM_CORE_VER_120)
|
|
msm_usb_write_readback(phy->base,
|
|
HS_PHY_CTRL_COMMON_REG,
|
|
COMMON_OTGDISABLE0,
|
|
0);
|
|
else
|
|
msm_usb_write_readback(phy->base,
|
|
HS_PHY_CTRL_REG(i),
|
|
OTGDISABLE0, 0);
|
|
}
|
|
}
|
|
/*
|
|
* write HSPHY init value to QSCRATCH reg to set HSPHY
|
|
* parameters like VBUS valid threshold, disconnect valid
|
|
* threshold, DC voltage level,preempasis and rise/fall time
|
|
*/
|
|
if (override_phy_init)
|
|
phy->hsphy_init_seq = override_phy_init;
|
|
if (phy->hsphy_init_seq)
|
|
msm_usb_write_readback(phy->base,
|
|
PARAMETER_OVERRIDE_X_REG(0),
|
|
0x03FFFFFF,
|
|
phy->hsphy_init_seq & 0x03FFFFFF);
|
|
}
|
|
|
|
phy->suspended = !!suspend; /* double-NOT coerces to bool value */
|
|
return 0;
|
|
}
|
|
|
|
static int msm_hsphy_notify_connect(struct usb_phy *uphy,
|
|
enum usb_device_speed speed)
|
|
{
|
|
int rc = 0;
|
|
struct msm_hsphy *phy = container_of(uphy, struct msm_hsphy, phy);
|
|
|
|
phy->cable_connected = true;
|
|
|
|
if (uphy->flags & PHY_HOST_MODE) {
|
|
if (phy->core_ver == MSM_CORE_VER_160 ||
|
|
phy->core_ver == MSM_CORE_VER_161) {
|
|
/* Some snps usb2 picophy revisions require 3.15 V to
|
|
* operate correctly during full speed host mode at
|
|
* sub zero temperature.
|
|
*/
|
|
rc = regulator_set_voltage(phy->vdda33,
|
|
USB_HSPHY_3P3_VOL_FSHOST,
|
|
USB_HSPHY_3P3_VOL_MAX);
|
|
if (rc)
|
|
dev_err(phy->phy.dev,
|
|
"unable to set voltage for vdda33\n");
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
if (!(uphy->flags & PHY_VBUS_VALID_OVERRIDE))
|
|
return 0;
|
|
|
|
/* Set External VBUS Valid Select. Set once, can be left on */
|
|
if (phy->core_ver >= MSM_CORE_VER_120) {
|
|
msm_usb_write_readback(phy->base, HS_PHY_CTRL_COMMON_REG,
|
|
COMMON_VBUSVLDEXTSEL0,
|
|
COMMON_VBUSVLDEXTSEL0);
|
|
} else {
|
|
msm_usb_write_readback(phy->base,
|
|
HS_PHY_CTRL_REG(0),
|
|
VBUSVLDEXTSEL0, VBUSVLDEXTSEL0);
|
|
}
|
|
|
|
/* Enable D+ pull-up resistor */
|
|
msm_usb_write_readback(phy->base,
|
|
HS_PHY_CTRL_REG(0),
|
|
VBUSVLDEXT0, VBUSVLDEXT0);
|
|
|
|
/* Set OTG VBUS Valid from HSPHY to controller */
|
|
msm_usb_write_readback(phy->base, HS_PHY_CTRL_REG(0),
|
|
UTMI_OTG_VBUS_VALID,
|
|
UTMI_OTG_VBUS_VALID);
|
|
|
|
/* Indicate value is driven by UTMI_OTG_VBUS_VALID bit */
|
|
if (phy->core_ver >= MSM_CORE_VER_120)
|
|
msm_usb_write_readback(phy->base, HS_PHY_CTRL_REG(0),
|
|
SW_SESSVLD_SEL, SW_SESSVLD_SEL);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int msm_hsphy_notify_disconnect(struct usb_phy *uphy,
|
|
enum usb_device_speed speed)
|
|
{
|
|
int rc = 0;
|
|
struct msm_hsphy *phy = container_of(uphy, struct msm_hsphy, phy);
|
|
|
|
phy->cable_connected = false;
|
|
|
|
if (uphy->flags & PHY_HOST_MODE) {
|
|
if (phy->core_ver == MSM_CORE_VER_160 ||
|
|
phy->core_ver == MSM_CORE_VER_161) {
|
|
rc = regulator_set_voltage(phy->vdda33,
|
|
USB_HSPHY_3P3_VOL_MIN,
|
|
USB_HSPHY_3P3_VOL_MAX);
|
|
if (rc)
|
|
dev_err(phy->phy.dev,
|
|
"unable to set voltage for vdda33\n");
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
if (!(uphy->flags & PHY_VBUS_VALID_OVERRIDE))
|
|
return 0;
|
|
|
|
/* Clear OTG VBUS Valid to Controller */
|
|
msm_usb_write_readback(phy->base, HS_PHY_CTRL_REG(0),
|
|
UTMI_OTG_VBUS_VALID, 0);
|
|
|
|
/* Disable D+ pull-up resistor */
|
|
msm_usb_write_readback(phy->base,
|
|
HS_PHY_CTRL_REG(0), VBUSVLDEXT0, 0);
|
|
|
|
/* Indicate value is no longer driven by UTMI_OTG_VBUS_VALID bit */
|
|
if (phy->core_ver >= MSM_CORE_VER_120)
|
|
msm_usb_write_readback(phy->base, HS_PHY_CTRL_REG(0),
|
|
SW_SESSVLD_SEL, 0);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int msm_hsphy_probe(struct platform_device *pdev)
|
|
{
|
|
struct msm_hsphy *phy;
|
|
struct device *dev = &pdev->dev;
|
|
struct resource *res;
|
|
int ret = 0;
|
|
|
|
phy = devm_kzalloc(dev, sizeof(*phy), GFP_KERNEL);
|
|
if (!phy) {
|
|
ret = -ENOMEM;
|
|
goto err_ret;
|
|
}
|
|
|
|
res = platform_get_resource_byname(pdev, IORESOURCE_MEM, "core");
|
|
if (!res) {
|
|
dev_err(dev, "missing memory base resource\n");
|
|
ret = -ENODEV;
|
|
goto err_ret;
|
|
}
|
|
|
|
phy->base = devm_ioremap_nocache(dev, res->start, resource_size(res));
|
|
if (!phy->base) {
|
|
dev_err(dev, "ioremap failed\n");
|
|
ret = -ENODEV;
|
|
goto err_ret;
|
|
}
|
|
|
|
res = platform_get_resource_byname(pdev, IORESOURCE_MEM, "tcsr");
|
|
if (res) {
|
|
phy->tcsr = devm_ioremap_nocache(dev, res->start,
|
|
resource_size(res));
|
|
if (!phy->tcsr) {
|
|
dev_err(dev, "tcsr ioremap failed\n");
|
|
return -ENODEV;
|
|
}
|
|
|
|
/* switch MUX to let SNPS controller use the primary HSPHY */
|
|
writel_relaxed(readl_relaxed(phy->tcsr) | TCSR_USB30_CONTROL,
|
|
phy->tcsr);
|
|
}
|
|
|
|
if (of_get_property(dev->of_node, "qcom,primary-phy", NULL)) {
|
|
dev_dbg(dev, "secondary HSPHY\n");
|
|
phy->phy.flags |= ENABLE_SECONDARY_PHY;
|
|
}
|
|
|
|
ret = of_property_read_u32_array(dev->of_node, "qcom,vdd-voltage-level",
|
|
(u32 *) phy->vdd_levels,
|
|
ARRAY_SIZE(phy->vdd_levels));
|
|
if (ret) {
|
|
dev_err(dev, "error reading qcom,vdd-voltage-level property\n");
|
|
goto err_ret;
|
|
}
|
|
|
|
phy->ext_vbus_id = of_property_read_bool(dev->of_node,
|
|
"qcom,ext-vbus-id");
|
|
phy->phy.dev = dev;
|
|
|
|
phy->vdd = devm_regulator_get(dev, "vdd");
|
|
if (IS_ERR(phy->vdd)) {
|
|
dev_err(dev, "unable to get vdd supply\n");
|
|
ret = PTR_ERR(phy->vdd);
|
|
goto err_ret;
|
|
}
|
|
|
|
phy->vdda33 = devm_regulator_get(dev, "vdda33");
|
|
if (IS_ERR(phy->vdda33)) {
|
|
dev_err(dev, "unable to get vdda33 supply\n");
|
|
ret = PTR_ERR(phy->vdda33);
|
|
goto err_ret;
|
|
}
|
|
|
|
phy->vdda18 = devm_regulator_get(dev, "vdda18");
|
|
if (IS_ERR(phy->vdda18)) {
|
|
dev_err(dev, "unable to get vdda18 supply\n");
|
|
ret = PTR_ERR(phy->vdda18);
|
|
goto err_ret;
|
|
}
|
|
|
|
ret = msm_hsusb_config_vdd(phy, 1);
|
|
if (ret) {
|
|
dev_err(dev, "hsusb vdd_dig configuration failed\n");
|
|
goto err_ret;
|
|
}
|
|
|
|
ret = regulator_enable(phy->vdd);
|
|
if (ret) {
|
|
dev_err(dev, "unable to enable the hsusb vdd_dig\n");
|
|
goto unconfig_hs_vdd;
|
|
}
|
|
|
|
ret = msm_hsusb_ldo_enable(phy, 1);
|
|
if (ret) {
|
|
dev_err(dev, "hsusb vreg enable failed\n");
|
|
goto disable_hs_vdd;
|
|
}
|
|
|
|
phy->sleep_clk = devm_clk_get(&pdev->dev, "phy_sleep_clk");
|
|
if (IS_ERR(phy->sleep_clk)) {
|
|
dev_err(&pdev->dev, "failed to get phy_sleep_clk\n");
|
|
ret = PTR_ERR(phy->sleep_clk);
|
|
goto disable_hs_ldo;
|
|
}
|
|
clk_prepare_enable(phy->sleep_clk);
|
|
phy->sleep_clk_reset = of_property_read_bool(dev->of_node,
|
|
"qcom,sleep-clk-reset");
|
|
|
|
if (of_property_read_u32(dev->of_node, "qcom,hsphy-init",
|
|
&phy->hsphy_init_seq))
|
|
dev_dbg(dev, "unable to read hsphy init seq\n");
|
|
else if (!phy->hsphy_init_seq)
|
|
dev_warn(dev, "hsphy init seq cannot be 0. Using POR value\n");
|
|
|
|
if (of_property_read_u32(dev->of_node, "qcom,num-ports",
|
|
&phy->num_ports))
|
|
phy->num_ports = 1;
|
|
else if (phy->num_ports > 3) {
|
|
dev_err(dev, " number of ports more that 3 is not supported\n");
|
|
goto disable_clk;
|
|
}
|
|
|
|
phy->set_pllbtune = of_property_read_bool(dev->of_node,
|
|
"qcom,set-pllbtune");
|
|
|
|
/*
|
|
* If this workaround flag is enabled, the HW requires the 1.8 and 3.x
|
|
* regulators to be kept ON when entering suspend. The easiest way to
|
|
* do that is to call regulator_enable() an additional time here,
|
|
* since it will keep the regulators' reference counts nonzero.
|
|
*/
|
|
phy->vdda_force_on = of_property_read_bool(dev->of_node,
|
|
"qcom,vdda-force-on");
|
|
if (phy->vdda_force_on) {
|
|
ret = msm_hsusb_ldo_enable(phy, 1);
|
|
if (ret)
|
|
goto disable_clk;
|
|
}
|
|
|
|
platform_set_drvdata(pdev, phy);
|
|
|
|
if (of_property_read_bool(dev->of_node, "qcom,vbus-valid-override"))
|
|
phy->phy.flags |= PHY_VBUS_VALID_OVERRIDE;
|
|
|
|
phy->phy.init = msm_hsphy_init;
|
|
phy->phy.set_suspend = msm_hsphy_set_suspend;
|
|
phy->phy.notify_connect = msm_hsphy_notify_connect;
|
|
phy->phy.notify_disconnect = msm_hsphy_notify_disconnect;
|
|
phy->phy.reset = msm_hsphy_reset;
|
|
/*FIXME: this conflicts with dwc3_otg */
|
|
/*phy->phy.type = USB_PHY_TYPE_USB2; */
|
|
|
|
ret = usb_add_phy_dev(&phy->phy);
|
|
if (ret)
|
|
goto disable_clk;
|
|
|
|
atomic_inc(&hsphy_active_count);
|
|
return 0;
|
|
|
|
disable_clk:
|
|
clk_disable_unprepare(phy->sleep_clk);
|
|
disable_hs_ldo:
|
|
msm_hsusb_ldo_enable(phy, 0);
|
|
disable_hs_vdd:
|
|
regulator_disable(phy->vdd);
|
|
unconfig_hs_vdd:
|
|
msm_hsusb_config_vdd(phy, 0);
|
|
err_ret:
|
|
return ret;
|
|
}
|
|
|
|
static int msm_hsphy_remove(struct platform_device *pdev)
|
|
{
|
|
struct msm_hsphy *phy = platform_get_drvdata(pdev);
|
|
|
|
if (!phy)
|
|
return 0;
|
|
|
|
usb_remove_phy(&phy->phy);
|
|
clk_disable_unprepare(phy->sleep_clk);
|
|
|
|
/* Undo the additional regulator enable */
|
|
if (phy->vdda_force_on)
|
|
msm_hsusb_ldo_enable(phy, 0);
|
|
msm_hsusb_ldo_enable(phy, 0);
|
|
regulator_disable(phy->vdd);
|
|
msm_hsusb_config_vdd(phy, 0);
|
|
if (!phy->suspended)
|
|
atomic_dec(&hsphy_active_count);
|
|
kfree(phy);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static const struct of_device_id msm_usb_id_table[] = {
|
|
{
|
|
.compatible = "qcom,usb-hsphy",
|
|
},
|
|
{ },
|
|
};
|
|
MODULE_DEVICE_TABLE(of, msm_usb_id_table);
|
|
|
|
static struct platform_driver msm_hsphy_driver = {
|
|
.probe = msm_hsphy_probe,
|
|
.remove = msm_hsphy_remove,
|
|
.driver = {
|
|
.name = "msm-usb-hsphy",
|
|
.of_match_table = of_match_ptr(msm_usb_id_table),
|
|
},
|
|
};
|
|
|
|
module_platform_driver(msm_hsphy_driver);
|
|
|
|
MODULE_DESCRIPTION("MSM USB HS PHY driver");
|
|
MODULE_LICENSE("GPL v2");
|