Commit dc1b5d9a authored by Enric Balletbo i Serra's avatar Enric Balletbo i Serra Committed by Felipe Balbi

usb: dwc3: Fix core validation in probe, move after clocks are enabled

The required clocks needs to be enabled before the first register
access. After commit fe8abf33 ("usb: dwc3: support clocks and resets
for DWC3 core"), this happens when the dwc3_core_is_valid function is
called, but the mentioned commit adds that call in the wrong place,
before the clocks are enabled. So, move that call after the
clk_bulk_enable() to ensure the clocks are enabled and the reset
deasserted.

I detected this while, as experiment, I tried to move the clocks and resets
from the glue layer to the DWC3 core on a Samsung Chromebook Plus.

That was not detected before because, in most cases, the glue layer
initializes SoC-specific things and then populates the child "snps,dwc3"
with those clocks already enabled.

Fixes: b873e2d0 ("usb: dwc3: Do core validation early on probe")
Signed-off-by: default avatarEnric Balletbo i Serra <enric.balletbo@collabora.com>
Signed-off-by: default avatarFelipe Balbi <felipe.balbi@linux.intel.com>
parent 7f5d6a46
...@@ -1423,11 +1423,6 @@ static int dwc3_probe(struct platform_device *pdev) ...@@ -1423,11 +1423,6 @@ static int dwc3_probe(struct platform_device *pdev)
dwc->regs = regs; dwc->regs = regs;
dwc->regs_size = resource_size(&dwc_res); dwc->regs_size = resource_size(&dwc_res);
if (!dwc3_core_is_valid(dwc)) {
dev_err(dwc->dev, "this is not a DesignWare USB3 DRD Core\n");
return -ENODEV;
}
dwc3_get_properties(dwc); dwc3_get_properties(dwc);
dwc->reset = devm_reset_control_get_optional_shared(dev, NULL); dwc->reset = devm_reset_control_get_optional_shared(dev, NULL);
...@@ -1460,6 +1455,12 @@ static int dwc3_probe(struct platform_device *pdev) ...@@ -1460,6 +1455,12 @@ static int dwc3_probe(struct platform_device *pdev)
if (ret) if (ret)
goto unprepare_clks; goto unprepare_clks;
if (!dwc3_core_is_valid(dwc)) {
dev_err(dwc->dev, "this is not a DesignWare USB3 DRD Core\n");
ret = -ENODEV;
goto disable_clks;
}
platform_set_drvdata(pdev, dwc); platform_set_drvdata(pdev, dwc);
dwc3_cache_hwparams(dwc); dwc3_cache_hwparams(dwc);
...@@ -1525,6 +1526,7 @@ static int dwc3_probe(struct platform_device *pdev) ...@@ -1525,6 +1526,7 @@ static int dwc3_probe(struct platform_device *pdev)
pm_runtime_put_sync(&pdev->dev); pm_runtime_put_sync(&pdev->dev);
pm_runtime_disable(&pdev->dev); pm_runtime_disable(&pdev->dev);
disable_clks:
clk_bulk_disable(dwc->num_clks, dwc->clks); clk_bulk_disable(dwc->num_clks, dwc->clks);
unprepare_clks: unprepare_clks:
clk_bulk_unprepare(dwc->num_clks, dwc->clks); clk_bulk_unprepare(dwc->num_clks, dwc->clks);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment