Message ID | E1qgoNf-007a4O-47@rmk-PC.armlinux.org.uk (mailing list archive) |
---|---|
State | Accepted |
Commit | 6e19b3502c59ad21aa405383dfd0bdf4e841de57 |
Delegated to: | Netdev Maintainers |
Headers | show |
Series | net: phy: avoid race when erroring stopping PHY | expand |
On 9/14/23 08:35, Russell King (Oracle) wrote: > Move the call to phy_suspend() to the end of phy_state_machine() after > we release the lock so that we can combine the locked areas. > phy_suspend() can not be called while holding phydev->lock as it has > caused deadlocks in the past. > > Tested-by: Jijie Shao <shaojijie@huawei.com> > Signed-off-by: Russell King (Oracle) <rmk+kernel@armlinux.org.uk> Reviewed-by: Florian Fainelli <florian.fainelli@broadcom.com>
diff --git a/drivers/net/phy/phy.c b/drivers/net/phy/phy.c index 5bb33af2a4cb..756326f38b14 100644 --- a/drivers/net/phy/phy.c +++ b/drivers/net/phy/phy.c @@ -1494,15 +1494,11 @@ void phy_state_machine(struct work_struct *work) func = &_phy_start_aneg; } - mutex_unlock(&phydev->lock); - - if (do_suspend) - phy_suspend(phydev); - - if (err == -ENODEV) + if (err == -ENODEV) { + mutex_unlock(&phydev->lock); return; + } - mutex_lock(&phydev->lock); if (err < 0) phy_error_precise(phydev, func, err); @@ -1519,6 +1515,9 @@ void phy_state_machine(struct work_struct *work) if (phy_polling_mode(phydev) && phy_is_started(phydev)) phy_queue_state_machine(phydev, PHY_STATE_TIME); mutex_unlock(&phydev->lock); + + if (do_suspend) + phy_suspend(phydev); } /**