2
0
mirror of https://github.com/edk2-porting/linux-next.git synced 2025-01-20 19:43:58 +08:00

usb: chipidea: msm: Handle phy power states

The ULPI phy on qcom platforms needs to be initialized and
powered on after a USB reset and before we toggle the run/stop
bit. Otherwise, the phy locks up and doesn't work properly. Hook
the phy initialization into the RESET event and the phy power off
into the STOPPED event.

Acked-by: Peter Chen <peter.chen@nxp.com>
Cc: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
Signed-off-by: Stephen Boyd <stephen.boyd@linaro.org>
Signed-off-by: Peter Chen <peter.chen@nxp.com>
This commit is contained in:
Stephen Boyd 2016-12-28 14:57:06 -08:00 committed by Peter Chen
parent 1b8fc5a525
commit 11893dae63
4 changed files with 33 additions and 25 deletions

View File

@ -80,20 +80,33 @@ static const struct reset_control_ops ci_hdrc_msm_reset_ops = {
.reset = ci_hdrc_msm_por_reset, .reset = ci_hdrc_msm_por_reset,
}; };
static void ci_hdrc_msm_notify_event(struct ci_hdrc *ci, unsigned event) static int ci_hdrc_msm_notify_event(struct ci_hdrc *ci, unsigned event)
{ {
struct device *dev = ci->dev->parent; struct device *dev = ci->dev->parent;
struct ci_hdrc_msm *msm_ci = dev_get_drvdata(dev); struct ci_hdrc_msm *msm_ci = dev_get_drvdata(dev);
int ret;
switch (event) { switch (event) {
case CI_HDRC_CONTROLLER_RESET_EVENT: case CI_HDRC_CONTROLLER_RESET_EVENT:
dev_dbg(dev, "CI_HDRC_CONTROLLER_RESET_EVENT received\n"); dev_dbg(dev, "CI_HDRC_CONTROLLER_RESET_EVENT received\n");
hw_phymode_configure(ci);
if (msm_ci->secondary_phy) { if (msm_ci->secondary_phy) {
u32 val = readl_relaxed(msm_ci->base + HS_PHY_SEC_CTRL); u32 val = readl_relaxed(msm_ci->base + HS_PHY_SEC_CTRL);
val |= HS_PHY_DIG_CLAMP_N; val |= HS_PHY_DIG_CLAMP_N;
writel_relaxed(val, msm_ci->base + HS_PHY_SEC_CTRL); writel_relaxed(val, msm_ci->base + HS_PHY_SEC_CTRL);
} }
ret = phy_init(ci->phy);
if (ret)
return ret;
ret = phy_power_on(ci->phy);
if (ret) {
phy_exit(ci->phy);
return ret;
}
/* use AHB transactor, allow posted data writes */ /* use AHB transactor, allow posted data writes */
hw_write_id_reg(ci, HS_PHY_AHB_MODE, 0xffffffff, 0x8); hw_write_id_reg(ci, HS_PHY_AHB_MODE, 0xffffffff, 0x8);
@ -113,21 +126,18 @@ static void ci_hdrc_msm_notify_event(struct ci_hdrc *ci, unsigned event)
HSPHY_SESS_VLD_CTRL); HSPHY_SESS_VLD_CTRL);
} }
usb_phy_init(ci->usb_phy);
break; break;
case CI_HDRC_CONTROLLER_STOPPED_EVENT: case CI_HDRC_CONTROLLER_STOPPED_EVENT:
dev_dbg(dev, "CI_HDRC_CONTROLLER_STOPPED_EVENT received\n"); dev_dbg(dev, "CI_HDRC_CONTROLLER_STOPPED_EVENT received\n");
/* phy_power_off(ci->phy);
* Put the phy in non-driving mode. Otherwise host phy_exit(ci->phy);
* may not detect soft-disconnection.
*/
usb_phy_notify_disconnect(ci->usb_phy, USB_SPEED_UNKNOWN);
break; break;
default: default:
dev_dbg(dev, "unknown ci_hdrc event\n"); dev_dbg(dev, "unknown ci_hdrc event\n");
break; break;
} }
return 0;
} }
static int ci_hdrc_msm_mux_phy(struct ci_hdrc_msm *ci, static int ci_hdrc_msm_mux_phy(struct ci_hdrc_msm *ci,
@ -167,7 +177,6 @@ static int ci_hdrc_msm_probe(struct platform_device *pdev)
{ {
struct ci_hdrc_msm *ci; struct ci_hdrc_msm *ci;
struct platform_device *plat_ci; struct platform_device *plat_ci;
struct usb_phy *phy;
struct clk *clk; struct clk *clk;
struct reset_control *reset; struct reset_control *reset;
struct resource *res; struct resource *res;
@ -181,21 +190,12 @@ static int ci_hdrc_msm_probe(struct platform_device *pdev)
return -ENOMEM; return -ENOMEM;
platform_set_drvdata(pdev, ci); platform_set_drvdata(pdev, ci);
/*
* OTG(PHY) driver takes care of PHY initialization, clock management,
* powering up VBUS, mapping of registers address space and power
* management.
*/
phy = devm_usb_get_phy_by_phandle(&pdev->dev, "usb-phy", 0);
if (IS_ERR(phy))
return PTR_ERR(phy);
ci->pdata.name = "ci_hdrc_msm"; ci->pdata.name = "ci_hdrc_msm";
ci->pdata.capoffset = DEF_CAPOFFSET; ci->pdata.capoffset = DEF_CAPOFFSET;
ci->pdata.flags = CI_HDRC_REGS_SHARED | CI_HDRC_DISABLE_STREAMING | ci->pdata.flags = CI_HDRC_REGS_SHARED | CI_HDRC_DISABLE_STREAMING |
CI_HDRC_OVERRIDE_AHB_BURST; CI_HDRC_OVERRIDE_AHB_BURST |
CI_HDRC_OVERRIDE_PHY_CONTROL;
ci->pdata.notify_event = ci_hdrc_msm_notify_event; ci->pdata.notify_event = ci_hdrc_msm_notify_event;
ci->pdata.usb_phy = phy;
reset = devm_reset_control_get(&pdev->dev, "core"); reset = devm_reset_control_get(&pdev->dev, "core");
if (IS_ERR(reset)) if (IS_ERR(reset))

View File

@ -327,6 +327,7 @@ void hw_phymode_configure(struct ci_hdrc *ci)
hw_write(ci, OP_PORTSC, PORTSC_STS, PORTSC_STS); hw_write(ci, OP_PORTSC, PORTSC_STS, PORTSC_STS);
} }
} }
EXPORT_SYMBOL_GPL(hw_phymode_configure);
/** /**
* _ci_usb_phy_init: initialize phy taking in account both phy and usb_phy * _ci_usb_phy_init: initialize phy taking in account both phy and usb_phy
@ -503,9 +504,12 @@ int hw_device_reset(struct ci_hdrc *ci)
return ret; return ret;
} }
if (ci->platdata->notify_event) if (ci->platdata->notify_event) {
ci->platdata->notify_event(ci, ret = ci->platdata->notify_event(ci,
CI_HDRC_CONTROLLER_RESET_EVENT); CI_HDRC_CONTROLLER_RESET_EVENT);
if (ret)
return ret;
}
/* USBMODE should be configured step by step */ /* USBMODE should be configured step by step */
hw_write(ci, OP_USBMODE, USBMODE_CM, USBMODE_CM_IDLE); hw_write(ci, OP_USBMODE, USBMODE_CM, USBMODE_CM_IDLE);

View File

@ -90,8 +90,12 @@ static int ehci_ci_reset(struct usb_hcd *hcd)
ehci->need_io_watchdog = 0; ehci->need_io_watchdog = 0;
if (ci->platdata->notify_event) if (ci->platdata->notify_event) {
ci->platdata->notify_event(ci, CI_HDRC_CONTROLLER_RESET_EVENT); ret = ci->platdata->notify_event(ci,
CI_HDRC_CONTROLLER_RESET_EVENT);
if (ret)
return ret;
}
ci_platform_configure(ci); ci_platform_configure(ci);

View File

@ -61,7 +61,7 @@ struct ci_hdrc_platform_data {
enum usb_dr_mode dr_mode; enum usb_dr_mode dr_mode;
#define CI_HDRC_CONTROLLER_RESET_EVENT 0 #define CI_HDRC_CONTROLLER_RESET_EVENT 0
#define CI_HDRC_CONTROLLER_STOPPED_EVENT 1 #define CI_HDRC_CONTROLLER_STOPPED_EVENT 1
void (*notify_event) (struct ci_hdrc *ci, unsigned event); int (*notify_event) (struct ci_hdrc *ci, unsigned event);
struct regulator *reg_vbus; struct regulator *reg_vbus;
struct usb_otg_caps ci_otg_caps; struct usb_otg_caps ci_otg_caps;
bool tpl_support; bool tpl_support;