Skip to content

Commit

Permalink
---
Browse files Browse the repository at this point in the history
yaml
---
r: 150012
b: refs/heads/master
c: fa94f6d
h: refs/heads/master
v: v3
  • Loading branch information
Grant Likely authored and David S. Miller committed Apr 27, 2009
1 parent d9a9308 commit 9531c11
Show file tree
Hide file tree
Showing 3 changed files with 91 additions and 34 deletions.
2 changes: 1 addition & 1 deletion [refs]
Original file line number Diff line number Diff line change
@@ -1,2 +1,2 @@
---
refs/heads/master: 4dea547fef1ba23f9d23f5e7f5218187a7dcf1b3
refs/heads/master: fa94f6d93c5382810ff41f010f12ca8698fc775e
118 changes: 85 additions & 33 deletions trunk/drivers/net/phy/phy_device.c
Original file line number Diff line number Diff line change
Expand Up @@ -291,6 +291,33 @@ void phy_prepare_link(struct phy_device *phydev,
phydev->adjust_link = handler;
}

/**
* phy_connect_direct - connect an ethernet device to a specific phy_device
* @dev: the network device to connect
* @phydev: the pointer to the phy device
* @handler: callback function for state change notifications
* @flags: PHY device's dev_flags
* @interface: PHY device's interface
*/
int phy_connect_direct(struct net_device *dev, struct phy_device *phydev,
void (*handler)(struct net_device *), u32 flags,
phy_interface_t interface)
{
int rc;

rc = phy_attach_direct(dev, phydev, flags, interface);
if (rc)
return rc;

phy_prepare_link(phydev, handler);
phy_start_machine(phydev, NULL);
if (phydev->irq > 0)
phy_start_interrupts(phydev);

return 0;
}
EXPORT_SYMBOL(phy_connect_direct);

/**
* phy_connect - connect an ethernet device to a PHY device
* @dev: the network device to connect
Expand All @@ -312,18 +339,21 @@ struct phy_device * phy_connect(struct net_device *dev, const char *bus_id,
phy_interface_t interface)
{
struct phy_device *phydev;
struct device *d;
int rc;

phydev = phy_attach(dev, bus_id, flags, interface);

if (IS_ERR(phydev))
return phydev;

phy_prepare_link(phydev, handler);

phy_start_machine(phydev, NULL);
/* Search the list of PHY devices on the mdio bus for the
* PHY with the requested name */
d = bus_find_device_by_name(&mdio_bus_type, NULL, bus_id);
if (!d) {
pr_err("PHY %s not found\n", bus_id);
return ERR_PTR(-ENODEV);
}
phydev = to_phy_device(d);

if (phydev->irq > 0)
phy_start_interrupts(phydev);
rc = phy_connect_direct(dev, phydev, handler, flags, interface);
if (rc)
return ERR_PTR(rc);

return phydev;
}
Expand All @@ -347,9 +377,9 @@ void phy_disconnect(struct phy_device *phydev)
EXPORT_SYMBOL(phy_disconnect);

/**
* phy_attach - attach a network device to a particular PHY device
* phy_attach_direct - attach a network device to a given PHY device pointer
* @dev: network device to attach
* @bus_id: PHY device to attach
* @phydev: Pointer to phy_device to attach
* @flags: PHY device's dev_flags
* @interface: PHY device's interface
*
Expand All @@ -360,22 +390,10 @@ EXPORT_SYMBOL(phy_disconnect);
* the attaching device, and given a callback for link status
* change. The phy_device is returned to the attaching driver.
*/
struct phy_device *phy_attach(struct net_device *dev,
const char *bus_id, u32 flags, phy_interface_t interface)
int phy_attach_direct(struct net_device *dev, struct phy_device *phydev,
u32 flags, phy_interface_t interface)
{
struct bus_type *bus = &mdio_bus_type;
struct phy_device *phydev;
struct device *d;

/* Search the list of PHY devices on the mdio bus for the
* PHY with the requested name */
d = bus_find_device_by_name(bus, NULL, bus_id);
if (d) {
phydev = to_phy_device(d);
} else {
printk(KERN_ERR "%s not found\n", bus_id);
return ERR_PTR(-ENODEV);
}
struct device *d = &phydev->dev;

/* Assume that if there is no driver, that it doesn't
* exist, and we should use the genphy driver. */
Expand All @@ -388,13 +406,12 @@ struct phy_device *phy_attach(struct net_device *dev,
err = device_bind_driver(d);

if (err)
return ERR_PTR(err);
return err;
}

if (phydev->attached_dev) {
printk(KERN_ERR "%s: %s already attached\n",
dev->name, bus_id);
return ERR_PTR(-EBUSY);
dev_err(&dev->dev, "PHY already attached\n");
return -EBUSY;
}

phydev->attached_dev = dev;
Expand All @@ -412,13 +429,48 @@ struct phy_device *phy_attach(struct net_device *dev,
err = phy_scan_fixups(phydev);

if (err < 0)
return ERR_PTR(err);
return err;

err = phydev->drv->config_init(phydev);

if (err < 0)
return ERR_PTR(err);
return err;
}

return 0;
}
EXPORT_SYMBOL(phy_attach_direct);

/**
* phy_attach - attach a network device to a particular PHY device
* @dev: network device to attach
* @bus_id: Bus ID of PHY device to attach
* @flags: PHY device's dev_flags
* @interface: PHY device's interface
*
* Description: Same as phy_attach_direct() except that a PHY bus_id
* string is passed instead of a pointer to a struct phy_device.
*/
struct phy_device *phy_attach(struct net_device *dev,
const char *bus_id, u32 flags, phy_interface_t interface)
{
struct bus_type *bus = &mdio_bus_type;
struct phy_device *phydev;
struct device *d;
int rc;

/* Search the list of PHY devices on the mdio bus for the
* PHY with the requested name */
d = bus_find_device_by_name(bus, NULL, bus_id);
if (!d) {
pr_err("PHY %s not found\n", bus_id);
return ERR_PTR(-ENODEV);
}
phydev = to_phy_device(d);

rc = phy_attach_direct(dev, phydev, flags, interface);
if (rc)
return ERR_PTR(rc);

return phydev;
}
Expand Down
5 changes: 5 additions & 0 deletions trunk/include/linux/phy.h
Original file line number Diff line number Diff line change
Expand Up @@ -447,8 +447,13 @@ struct phy_device* get_phy_device(struct mii_bus *bus, int addr);
int phy_device_register(struct phy_device *phy);
int phy_clear_interrupt(struct phy_device *phydev);
int phy_config_interrupt(struct phy_device *phydev, u32 interrupts);
int phy_attach_direct(struct net_device *dev, struct phy_device *phydev,
u32 flags, phy_interface_t interface);
struct phy_device * phy_attach(struct net_device *dev,
const char *bus_id, u32 flags, phy_interface_t interface);
int phy_connect_direct(struct net_device *dev, struct phy_device *phydev,
void (*handler)(struct net_device *), u32 flags,
phy_interface_t interface);
struct phy_device * phy_connect(struct net_device *dev, const char *bus_id,
void (*handler)(struct net_device *), u32 flags,
phy_interface_t interface);
Expand Down

0 comments on commit 9531c11

Please sign in to comment.