[<prev] [next>] [thread-next>] [day] [month] [year] [list]
Message-ID: <20250709132425.48631-1-buday.csaba@prolan.hu>
Date: Wed, 9 Jul 2025 15:24:24 +0200
From: Buday Csaba <buday.csaba@...lan.hu>
To: <netdev@...r.kernel.org>, <linux-kernel@...r.kernel.org>
CC: Buday Csaba <buday.csaba@...lan.hu>, Andrew Lunn <andrew@...n.ch>, "Heiner
Kallweit" <hkallweit1@...il.com>, Russell King <linux@...linux.org.uk>,
"David S. Miller" <davem@...emloft.net>, Eric Dumazet <edumazet@...gle.com>,
"Jakub Kicinski" <kuba@...nel.org>, Paolo Abeni <pabeni@...hat.com>
Subject: [PATCH] net: mdio: reset PHY before attempting to access registers in fwnode_mdiobus_register_phy
Some PHYs (e.g. LAN8710A) require a reset after power-on,even for
MDIO register access.
The current implementation of fwnode_mdiobus_register_phy() and
get_phy_device() attempt to read the id registers without ensuring
that the PHY had a reset before, which can fail on these devices.
This patch addresses that shortcoming, by always resetting the PHY
(when such property is given in the device tree). To keep the code
impact minimal, a change was also needed in phy_device_remove() to
prevent asserting the reset on device removal.
According to the documentation of phy_device_remove(), it should
reverse the effect of phy_device_register(). Since the reset GPIO
is in undefined state before that, it should be acceptable to leave
it unchanged during removal.
Signed-off-by: Buday Csaba <buday.csaba@...lan.hu>
---
drivers/net/mdio/fwnode_mdio.c | 20 ++++++++++++++++++--
drivers/net/phy/phy_device.c | 3 ---
2 files changed, 18 insertions(+), 5 deletions(-)
diff --git a/drivers/net/mdio/fwnode_mdio.c b/drivers/net/mdio/fwnode_mdio.c
index aea0f03575689..36b60544327b6 100644
--- a/drivers/net/mdio/fwnode_mdio.c
+++ b/drivers/net/mdio/fwnode_mdio.c
@@ -139,8 +139,24 @@ int fwnode_mdiobus_register_phy(struct mii_bus *bus,
}
is_c45 = fwnode_device_is_compatible(child, "ethernet-phy-ieee802.3-c45");
- if (is_c45 || fwnode_get_phy_id(child, &phy_id))
- phy = get_phy_device(bus, addr, is_c45);
+ if (is_c45 || fwnode_get_phy_id(child, &phy_id)) {
+ /* get_phy_device is NOT SAFE HERE, since the PHY may need a HW RESET.
+ * First create a dummy PHY device, reset the PHY, then call
+ * get_phy_device.
+ */
+ phy = phy_device_create(bus, addr, 0, 0, NULL);
+ if (!IS_ERR(phy)) {
+ if (is_of_node(child)) {
+ /* fwnode_mdiobus_phy_device_register performs the reset */
+ rc = fwnode_mdiobus_phy_device_register(bus, phy, child, addr);
+ if (!rc)
+ phy_device_remove(phy);
+ /* PHY has been reset at this point. */
+ }
+ phy_device_free(phy);
+ phy = get_phy_device(bus, addr, is_c45);
+ }
+ }
else
phy = phy_device_create(bus, addr, phy_id, 0, NULL);
if (IS_ERR(phy)) {
diff --git a/drivers/net/phy/phy_device.c b/drivers/net/phy/phy_device.c
index 13dea33d86ffa..da4ddce04e5fb 100644
--- a/drivers/net/phy/phy_device.c
+++ b/drivers/net/phy/phy_device.c
@@ -1102,9 +1102,6 @@ void phy_device_remove(struct phy_device *phydev)
device_del(&phydev->mdio.dev);
- /* Assert the reset signal */
- phy_device_reset(phydev, 1);
-
mdiobus_unregister_device(&phydev->mdio);
}
EXPORT_SYMBOL(phy_device_remove);
--
2.39.5
Powered by blists - more mailing lists