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:
parent
1b8fc5a525
commit
11893dae63
@ -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))
|
||||||
|
@ -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);
|
||||||
|
@ -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);
|
||||||
|
|
||||||
|
@ -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;
|
||||||
|
Loading…
Reference in New Issue
Block a user