Commit 66bdede4 authored by Geert Uytterhoeven's avatar Geert Uytterhoeven Committed by David S. Miller

of_mdio: Fix broken PHY IRQ in case of probe deferral

If an Ethernet PHY is initialized before the interrupt controller it is
connected to, a message like the following is printed:

    irq: no irq domain found for /interrupt-controller@e61c0000 !

However, the actual error is ignored, leading to a non-functional (POLL)
PHY interrupt later:

    Micrel KSZ8041RNLI ee700000.ethernet-ffffffff:01: attached PHY driver [Micrel KSZ8041RNLI] (mii_bus:phy_addr=ee700000.ethernet-ffffffff:01, irq=POLL)

Depending on whether the PHY driver will fall back to polling, Ethernet
may or may not work.

To fix this:
  1. Switch of_mdiobus_register_phy() from irq_of_parse_and_map() to
     of_irq_get().
     Unlike the former, the latter returns -EPROBE_DEFER if the
     interrupt controller is not yet available, so this condition can be
     detected.
     Other errors are handled the same as before, i.e. use the passed
     mdio->irq[addr] as interrupt.
  2. Propagate and handle errors from of_mdiobus_register_phy() and
     of_mdiobus_register_device().
Signed-off-by: default avatarGeert Uytterhoeven <geert+renesas@glider.be>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent 7433a8d6
...@@ -44,7 +44,7 @@ static int of_get_phy_id(struct device_node *device, u32 *phy_id) ...@@ -44,7 +44,7 @@ static int of_get_phy_id(struct device_node *device, u32 *phy_id)
return -EINVAL; return -EINVAL;
} }
static void of_mdiobus_register_phy(struct mii_bus *mdio, static int of_mdiobus_register_phy(struct mii_bus *mdio,
struct device_node *child, u32 addr) struct device_node *child, u32 addr)
{ {
struct phy_device *phy; struct phy_device *phy;
...@@ -60,9 +60,13 @@ static void of_mdiobus_register_phy(struct mii_bus *mdio, ...@@ -60,9 +60,13 @@ static void of_mdiobus_register_phy(struct mii_bus *mdio,
else else
phy = get_phy_device(mdio, addr, is_c45); phy = get_phy_device(mdio, addr, is_c45);
if (IS_ERR(phy)) if (IS_ERR(phy))
return; return PTR_ERR(phy);
rc = irq_of_parse_and_map(child, 0); rc = of_irq_get(child, 0);
if (rc == -EPROBE_DEFER) {
phy_device_free(phy);
return rc;
}
if (rc > 0) { if (rc > 0) {
phy->irq = rc; phy->irq = rc;
mdio->irq[addr] = rc; mdio->irq[addr] = rc;
...@@ -84,22 +88,23 @@ static void of_mdiobus_register_phy(struct mii_bus *mdio, ...@@ -84,22 +88,23 @@ static void of_mdiobus_register_phy(struct mii_bus *mdio,
if (rc) { if (rc) {
phy_device_free(phy); phy_device_free(phy);
of_node_put(child); of_node_put(child);
return; return rc;
} }
dev_dbg(&mdio->dev, "registered phy %s at address %i\n", dev_dbg(&mdio->dev, "registered phy %s at address %i\n",
child->name, addr); child->name, addr);
return 0;
} }
static void of_mdiobus_register_device(struct mii_bus *mdio, static int of_mdiobus_register_device(struct mii_bus *mdio,
struct device_node *child, u32 addr) struct device_node *child, u32 addr)
{ {
struct mdio_device *mdiodev; struct mdio_device *mdiodev;
int rc; int rc;
mdiodev = mdio_device_create(mdio, addr); mdiodev = mdio_device_create(mdio, addr);
if (IS_ERR(mdiodev)) if (IS_ERR(mdiodev))
return; return PTR_ERR(mdiodev);
/* Associate the OF node with the device structure so it /* Associate the OF node with the device structure so it
* can be looked up later. * can be looked up later.
...@@ -112,11 +117,12 @@ static void of_mdiobus_register_device(struct mii_bus *mdio, ...@@ -112,11 +117,12 @@ static void of_mdiobus_register_device(struct mii_bus *mdio,
if (rc) { if (rc) {
mdio_device_free(mdiodev); mdio_device_free(mdiodev);
of_node_put(child); of_node_put(child);
return; return rc;
} }
dev_dbg(&mdio->dev, "registered mdio device %s at address %i\n", dev_dbg(&mdio->dev, "registered mdio device %s at address %i\n",
child->name, addr); child->name, addr);
return 0;
} }
/* The following is a list of PHY compatible strings which appear in /* The following is a list of PHY compatible strings which appear in
...@@ -219,9 +225,11 @@ int of_mdiobus_register(struct mii_bus *mdio, struct device_node *np) ...@@ -219,9 +225,11 @@ int of_mdiobus_register(struct mii_bus *mdio, struct device_node *np)
} }
if (of_mdiobus_child_is_phy(child)) if (of_mdiobus_child_is_phy(child))
of_mdiobus_register_phy(mdio, child, addr); rc = of_mdiobus_register_phy(mdio, child, addr);
else else
of_mdiobus_register_device(mdio, child, addr); rc = of_mdiobus_register_device(mdio, child, addr);
if (rc)
goto unregister;
} }
if (!scanphys) if (!scanphys)
...@@ -242,12 +250,19 @@ int of_mdiobus_register(struct mii_bus *mdio, struct device_node *np) ...@@ -242,12 +250,19 @@ int of_mdiobus_register(struct mii_bus *mdio, struct device_node *np)
dev_info(&mdio->dev, "scan phy %s at address %i\n", dev_info(&mdio->dev, "scan phy %s at address %i\n",
child->name, addr); child->name, addr);
if (of_mdiobus_child_is_phy(child)) if (of_mdiobus_child_is_phy(child)) {
of_mdiobus_register_phy(mdio, child, addr); rc = of_mdiobus_register_phy(mdio, child, addr);
if (rc)
goto unregister;
}
} }
} }
return 0; return 0;
unregister:
mdiobus_unregister(mdio);
return rc;
} }
EXPORT_SYMBOL(of_mdiobus_register); EXPORT_SYMBOL(of_mdiobus_register);
......
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