Commit 3e3aaf64 authored by Russell King's avatar Russell King Committed by David S. Miller

phy: fix mdiobus module safety

Re-implement the mdiobus module refcounting to ensure that we actually
ensure that the mdiobus module code does not go away while we might call
into it.

The old scheme using bus->dev.driver was buggy, because bus->dev is a
class device which never has a struct device_driver associated with it,
and hence the associated code trying to obtain a refcount did nothing
useful.

Instead, take the approach that other subsystems do: pass the module
when calling mdiobus_register(), and record that in the mii_bus struct.
When we need to increment the module use count in the phy code, use
this stored pointer.  When the phy is deteched, drop the module
refcount, remembering that the phy device might go away at that point.

This doesn't stop the mii_bus going away while there are in-use phys -
it merely stops the underlying code vanishing.
Signed-off-by: default avatarRussell King <rmk+kernel@arm.linux.org.uk>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent e496ae69
...@@ -244,7 +244,7 @@ static inline void of_mdiobus_link_phydev(struct mii_bus *mdio, ...@@ -244,7 +244,7 @@ static inline void of_mdiobus_link_phydev(struct mii_bus *mdio,
* *
* Returns 0 on success or < 0 on error. * Returns 0 on success or < 0 on error.
*/ */
int mdiobus_register(struct mii_bus *bus) int __mdiobus_register(struct mii_bus *bus, struct module *owner)
{ {
int i, err; int i, err;
...@@ -255,6 +255,7 @@ int mdiobus_register(struct mii_bus *bus) ...@@ -255,6 +255,7 @@ int mdiobus_register(struct mii_bus *bus)
BUG_ON(bus->state != MDIOBUS_ALLOCATED && BUG_ON(bus->state != MDIOBUS_ALLOCATED &&
bus->state != MDIOBUS_UNREGISTERED); bus->state != MDIOBUS_UNREGISTERED);
bus->owner = owner;
bus->dev.parent = bus->parent; bus->dev.parent = bus->parent;
bus->dev.class = &mdio_bus_class; bus->dev.class = &mdio_bus_class;
bus->dev.groups = NULL; bus->dev.groups = NULL;
...@@ -296,7 +297,7 @@ int mdiobus_register(struct mii_bus *bus) ...@@ -296,7 +297,7 @@ int mdiobus_register(struct mii_bus *bus)
device_del(&bus->dev); device_del(&bus->dev);
return err; return err;
} }
EXPORT_SYMBOL(mdiobus_register); EXPORT_SYMBOL(__mdiobus_register);
void mdiobus_unregister(struct mii_bus *bus) void mdiobus_unregister(struct mii_bus *bus)
{ {
......
...@@ -582,10 +582,15 @@ EXPORT_SYMBOL(phy_init_hw); ...@@ -582,10 +582,15 @@ EXPORT_SYMBOL(phy_init_hw);
int phy_attach_direct(struct net_device *dev, struct phy_device *phydev, int phy_attach_direct(struct net_device *dev, struct phy_device *phydev,
u32 flags, phy_interface_t interface) u32 flags, phy_interface_t interface)
{ {
struct mii_bus *bus = phydev->bus;
struct device *d = &phydev->dev; struct device *d = &phydev->dev;
struct module *bus_module;
int err; int err;
if (!try_module_get(bus->owner)) {
dev_err(&dev->dev, "failed to get the bus module\n");
return -EIO;
}
/* Assume that if there is no driver, that it doesn't /* Assume that if there is no driver, that it doesn't
* exist, and we should use the genphy driver. * exist, and we should use the genphy driver.
*/ */
...@@ -600,20 +605,13 @@ int phy_attach_direct(struct net_device *dev, struct phy_device *phydev, ...@@ -600,20 +605,13 @@ int phy_attach_direct(struct net_device *dev, struct phy_device *phydev,
err = device_bind_driver(d); err = device_bind_driver(d);
if (err) if (err)
return err; goto error;
} }
if (phydev->attached_dev) { if (phydev->attached_dev) {
dev_err(&dev->dev, "PHY already attached\n"); dev_err(&dev->dev, "PHY already attached\n");
return -EBUSY; err = -EBUSY;
} goto error;
/* Increment the bus module reference count */
bus_module = phydev->bus->dev.driver ?
phydev->bus->dev.driver->owner : NULL;
if (!try_module_get(bus_module)) {
dev_err(&dev->dev, "failed to get the bus module\n");
return -EIO;
} }
phydev->attached_dev = dev; phydev->attached_dev = dev;
...@@ -636,6 +634,10 @@ int phy_attach_direct(struct net_device *dev, struct phy_device *phydev, ...@@ -636,6 +634,10 @@ int phy_attach_direct(struct net_device *dev, struct phy_device *phydev,
phy_resume(phydev); phy_resume(phydev);
return err; return err;
error:
module_put(bus->owner);
return err;
} }
EXPORT_SYMBOL(phy_attach_direct); EXPORT_SYMBOL(phy_attach_direct);
...@@ -680,11 +682,9 @@ EXPORT_SYMBOL(phy_attach); ...@@ -680,11 +682,9 @@ EXPORT_SYMBOL(phy_attach);
*/ */
void phy_detach(struct phy_device *phydev) void phy_detach(struct phy_device *phydev)
{ {
struct mii_bus *bus;
int i; int i;
if (phydev->bus->dev.driver)
module_put(phydev->bus->dev.driver->owner);
phydev->attached_dev->phydev = NULL; phydev->attached_dev->phydev = NULL;
phydev->attached_dev = NULL; phydev->attached_dev = NULL;
phy_suspend(phydev); phy_suspend(phydev);
...@@ -700,6 +700,10 @@ void phy_detach(struct phy_device *phydev) ...@@ -700,6 +700,10 @@ void phy_detach(struct phy_device *phydev)
break; break;
} }
} }
bus = phydev->bus;
module_put(bus->owner);
} }
EXPORT_SYMBOL(phy_detach); EXPORT_SYMBOL(phy_detach);
......
...@@ -19,6 +19,7 @@ ...@@ -19,6 +19,7 @@
#include <linux/spinlock.h> #include <linux/spinlock.h>
#include <linux/ethtool.h> #include <linux/ethtool.h>
#include <linux/mii.h> #include <linux/mii.h>
#include <linux/module.h>
#include <linux/timer.h> #include <linux/timer.h>
#include <linux/workqueue.h> #include <linux/workqueue.h>
#include <linux/mod_devicetable.h> #include <linux/mod_devicetable.h>
...@@ -153,6 +154,7 @@ struct sk_buff; ...@@ -153,6 +154,7 @@ struct sk_buff;
* PHYs should register using this structure * PHYs should register using this structure
*/ */
struct mii_bus { struct mii_bus {
struct module *owner;
const char *name; const char *name;
char id[MII_BUS_ID_SIZE]; char id[MII_BUS_ID_SIZE];
void *priv; void *priv;
...@@ -198,7 +200,8 @@ static inline struct mii_bus *mdiobus_alloc(void) ...@@ -198,7 +200,8 @@ static inline struct mii_bus *mdiobus_alloc(void)
return mdiobus_alloc_size(0); return mdiobus_alloc_size(0);
} }
int mdiobus_register(struct mii_bus *bus); int __mdiobus_register(struct mii_bus *bus, struct module *owner);
#define mdiobus_register(bus) __mdiobus_register(bus, THIS_MODULE)
void mdiobus_unregister(struct mii_bus *bus); void mdiobus_unregister(struct mii_bus *bus);
void mdiobus_free(struct mii_bus *bus); void mdiobus_free(struct mii_bus *bus);
struct mii_bus *devm_mdiobus_alloc_size(struct device *dev, int sizeof_priv); struct mii_bus *devm_mdiobus_alloc_size(struct device *dev, int sizeof_priv);
......
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