Commit 54e19bc7 authored by Timur Tabi's avatar Timur Tabi Committed by David S. Miller

net: qcom/emac: do not use devm on internal phy pdev

The platform_device returned by of_find_device_by_node() is not
automatically released when the driver unprobes.  Therefore,
managed calls like devm_ioremap_resource() should not be used.
Instead, we manually allocate the resources and then free them
on driver release.
Signed-off-by: default avatarTimur Tabi <timur@codeaurora.org>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent 48461135
...@@ -681,6 +681,7 @@ int emac_sgmii_config(struct platform_device *pdev, struct emac_adapter *adpt) ...@@ -681,6 +681,7 @@ int emac_sgmii_config(struct platform_device *pdev, struct emac_adapter *adpt)
struct resource *res; struct resource *res;
const struct of_device_id *match; const struct of_device_id *match;
struct device_node *np; struct device_node *np;
int ret;
np = of_parse_phandle(pdev->dev.of_node, "internal-phy", 0); np = of_parse_phandle(pdev->dev.of_node, "internal-phy", 0);
if (!np) { if (!np) {
...@@ -697,25 +698,48 @@ int emac_sgmii_config(struct platform_device *pdev, struct emac_adapter *adpt) ...@@ -697,25 +698,48 @@ int emac_sgmii_config(struct platform_device *pdev, struct emac_adapter *adpt)
match = of_match_device(emac_sgmii_dt_match, &sgmii_pdev->dev); match = of_match_device(emac_sgmii_dt_match, &sgmii_pdev->dev);
if (!match) { if (!match) {
dev_err(&pdev->dev, "unrecognized internal phy node\n"); dev_err(&pdev->dev, "unrecognized internal phy node\n");
return -ENODEV; ret = -ENODEV;
goto error_put_device;
} }
phy->initialize = (emac_sgmii_initialize)match->data; phy->initialize = (emac_sgmii_initialize)match->data;
/* Base address is the first address */ /* Base address is the first address */
res = platform_get_resource(sgmii_pdev, IORESOURCE_MEM, 0); res = platform_get_resource(sgmii_pdev, IORESOURCE_MEM, 0);
phy->base = devm_ioremap_resource(&sgmii_pdev->dev, res); phy->base = ioremap(res->start, resource_size(res));
if (IS_ERR(phy->base)) if (IS_ERR(phy->base)) {
return PTR_ERR(phy->base); ret = PTR_ERR(phy->base);
goto error_put_device;
}
/* v2 SGMII has a per-lane digital digital, so parse it if it exists */ /* v2 SGMII has a per-lane digital digital, so parse it if it exists */
res = platform_get_resource(sgmii_pdev, IORESOURCE_MEM, 1); res = platform_get_resource(sgmii_pdev, IORESOURCE_MEM, 1);
if (res) { if (res) {
phy->digital = devm_ioremap_resource(&sgmii_pdev->dev, res); phy->digital = ioremap(res->start, resource_size(res));
if (IS_ERR(phy->base)) if (IS_ERR(phy->digital)) {
return PTR_ERR(phy->base); ret = PTR_ERR(phy->digital);
goto error_unmap_base;
}
} }
return phy->initialize(adpt); ret = phy->initialize(adpt);
if (ret)
goto error;
/* We've remapped the addresses, so we don't need the device any
* more. of_find_device_by_node() says we should release it.
*/
put_device(&sgmii_pdev->dev);
return 0;
error:
if (phy->digital)
iounmap(phy->digital);
error_unmap_base:
iounmap(phy->base);
error_put_device:
put_device(&sgmii_pdev->dev);
return ret;
} }
...@@ -723,6 +723,10 @@ static int emac_remove(struct platform_device *pdev) ...@@ -723,6 +723,10 @@ static int emac_remove(struct platform_device *pdev)
mdiobus_unregister(adpt->mii_bus); mdiobus_unregister(adpt->mii_bus);
free_netdev(netdev); free_netdev(netdev);
if (adpt->phy.digital)
iounmap(adpt->phy.digital);
iounmap(adpt->phy.base);
return 0; return 0;
} }
......
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