Commit 1075a474 authored by David S. Miller's avatar David S. Miller

Merge branch 'Clause-45-PHY-probing-improvements'

Russell King says:

====================
Clause 45 PHY probing improvements

Last time this series was posted back in May, Florian reviewed the
patches, which was the only feedback I received.  I'm now posting
them without the RFC tag.

This series aims to improve the probing for Clause 45 PHYs.

The first four patches clean up get_phy_device() and called functions,
updating the kernel doc, adding information about the various error
return values.

We then provide better kerneldoc for get_phy_device(), describing what
is going on, and more importantly what the various return codes mean.

Patch 6 adds support for probing MMDs >= 8 to check for their presence.

Patch 7 changes get_phy_c45_ids() to only set the returned
devices_in_package if we successfully find a PHY.

Patch 8 splits the use of "devices in package" from the "mmds present".

Patch 9 expands our ID reading to cover the other MMDs.
====================
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parents e034c6d2 389a3389
...@@ -219,7 +219,7 @@ int genphy_c45_read_link(struct phy_device *phydev) ...@@ -219,7 +219,7 @@ int genphy_c45_read_link(struct phy_device *phydev)
int val, devad; int val, devad;
bool link = true; bool link = true;
if (phydev->c45_ids.devices_in_package & MDIO_DEVS_AN) { if (phydev->c45_ids.mmds_present & MDIO_DEVS_AN) {
val = phy_read_mmd(phydev, MDIO_MMD_AN, MDIO_CTRL1); val = phy_read_mmd(phydev, MDIO_MMD_AN, MDIO_CTRL1);
if (val < 0) if (val < 0)
return val; return val;
...@@ -409,7 +409,7 @@ int genphy_c45_pma_read_abilities(struct phy_device *phydev) ...@@ -409,7 +409,7 @@ int genphy_c45_pma_read_abilities(struct phy_device *phydev)
int val; int val;
linkmode_clear_bit(ETHTOOL_LINK_MODE_Autoneg_BIT, phydev->supported); linkmode_clear_bit(ETHTOOL_LINK_MODE_Autoneg_BIT, phydev->supported);
if (phydev->c45_ids.devices_in_package & MDIO_DEVS_AN) { if (phydev->c45_ids.mmds_present & MDIO_DEVS_AN) {
val = phy_read_mmd(phydev, MDIO_MMD_AN, MDIO_STAT1); val = phy_read_mmd(phydev, MDIO_MMD_AN, MDIO_STAT1);
if (val < 0) if (val < 0)
return val; return val;
......
...@@ -661,6 +661,28 @@ struct phy_device *phy_device_create(struct mii_bus *bus, int addr, u32 phy_id, ...@@ -661,6 +661,28 @@ struct phy_device *phy_device_create(struct mii_bus *bus, int addr, u32 phy_id,
} }
EXPORT_SYMBOL(phy_device_create); EXPORT_SYMBOL(phy_device_create);
/* phy_c45_probe_present - checks to see if a MMD is present in the package
* @bus: the target MII bus
* @prtad: PHY package address on the MII bus
* @devad: PHY device (MMD) address
*
* Read the MDIO_STAT2 register, and check whether a device is responding
* at this address.
*
* Returns: negative error number on bus access error, zero if no device
* is responding, or positive if a device is present.
*/
static int phy_c45_probe_present(struct mii_bus *bus, int prtad, int devad)
{
int stat2;
stat2 = mdiobus_c45_read(bus, prtad, devad, MDIO_STAT2);
if (stat2 < 0)
return stat2;
return (stat2 & MDIO_STAT2_DEVPRST) == MDIO_STAT2_DEVPRST_VAL;
}
/* get_phy_c45_devs_in_pkg - reads a MMD's devices in package registers. /* get_phy_c45_devs_in_pkg - reads a MMD's devices in package registers.
* @bus: the target MII bus * @bus: the target MII bus
* @addr: PHY address on the MII bus * @addr: PHY address on the MII bus
...@@ -687,9 +709,6 @@ static int get_phy_c45_devs_in_pkg(struct mii_bus *bus, int addr, int dev_addr, ...@@ -687,9 +709,6 @@ static int get_phy_c45_devs_in_pkg(struct mii_bus *bus, int addr, int dev_addr,
return -EIO; return -EIO;
*devices_in_package |= phy_reg; *devices_in_package |= phy_reg;
/* Bit 0 doesn't represent a device, it indicates c22 regs presence */
*devices_in_package &= ~BIT(0);
return 0; return 0;
} }
...@@ -697,53 +716,76 @@ static int get_phy_c45_devs_in_pkg(struct mii_bus *bus, int addr, int dev_addr, ...@@ -697,53 +716,76 @@ static int get_phy_c45_devs_in_pkg(struct mii_bus *bus, int addr, int dev_addr,
* get_phy_c45_ids - reads the specified addr for its 802.3-c45 IDs. * get_phy_c45_ids - reads the specified addr for its 802.3-c45 IDs.
* @bus: the target MII bus * @bus: the target MII bus
* @addr: PHY address on the MII bus * @addr: PHY address on the MII bus
* @phy_id: where to store the ID retrieved.
* @c45_ids: where to store the c45 ID information. * @c45_ids: where to store the c45 ID information.
* *
* If the PHY devices-in-package appears to be valid, it and the * Read the PHY "devices in package". If this appears to be valid, read
* corresponding identifiers are stored in @c45_ids, zero is stored * the PHY identifiers for each device. Return the "devices in package"
* in @phy_id. Otherwise 0xffffffff is stored in @phy_id. Returns * and identifiers in @c45_ids.
* zero on success.
* *
* Returns zero on success, %-EIO on bus access error, or %-ENODEV if
* the "devices in package" is invalid.
*/ */
static int get_phy_c45_ids(struct mii_bus *bus, int addr, u32 *phy_id, static int get_phy_c45_ids(struct mii_bus *bus, int addr,
struct phy_c45_device_ids *c45_ids) struct phy_c45_device_ids *c45_ids)
{ {
const int num_ids = ARRAY_SIZE(c45_ids->device_ids); const int num_ids = ARRAY_SIZE(c45_ids->device_ids);
u32 *devs = &c45_ids->devices_in_package; u32 devs_in_pkg = 0;
int i, phy_reg; int i, ret, phy_reg;
/* Find first non-zero Devices In package. Device zero is reserved /* Find first non-zero Devices In package. Device zero is reserved
* for 802.3 c45 complied PHYs, so don't probe it at first. * for 802.3 c45 complied PHYs, so don't probe it at first.
*/ */
for (i = 1; i < num_ids && *devs == 0; i++) { for (i = 1; i < MDIO_MMD_NUM && devs_in_pkg == 0; i++) {
phy_reg = get_phy_c45_devs_in_pkg(bus, addr, i, devs); if (i == MDIO_MMD_VEND1 || i == MDIO_MMD_VEND2) {
/* Check that there is a device present at this
* address before reading the devices-in-package
* register to avoid reading garbage from the PHY.
* Some PHYs (88x3310) vendor space is not IEEE802.3
* compliant.
*/
ret = phy_c45_probe_present(bus, addr, i);
if (ret < 0)
return -EIO;
if (!ret)
continue;
}
phy_reg = get_phy_c45_devs_in_pkg(bus, addr, i, &devs_in_pkg);
if (phy_reg < 0) if (phy_reg < 0)
return -EIO; return -EIO;
}
if ((*devs & 0x1fffffff) == 0x1fffffff) { if ((devs_in_pkg & 0x1fffffff) == 0x1fffffff) {
/* If mostly Fs, there is no device there, /* If mostly Fs, there is no device there, then let's probe
* then let's continue to probe more, as some * MMD 0, as some 10G PHYs have zero Devices In package,
* 10G PHYs have zero Devices In package,
* e.g. Cortina CS4315/CS4340 PHY. * e.g. Cortina CS4315/CS4340 PHY.
*/ */
phy_reg = get_phy_c45_devs_in_pkg(bus, addr, 0, devs); phy_reg = get_phy_c45_devs_in_pkg(bus, addr, 0, &devs_in_pkg);
if (phy_reg < 0) if (phy_reg < 0)
return -EIO; return -EIO;
/* no device there, let's get out of here */ /* no device there, let's get out of here */
if ((*devs & 0x1fffffff) == 0x1fffffff) { if ((devs_in_pkg & 0x1fffffff) == 0x1fffffff)
*phy_id = 0xffffffff; return -ENODEV;
return 0;
} else {
break;
}
}
} }
/* Now probe Device Identifiers for each device present. */ /* Now probe Device Identifiers for each device present. */
for (i = 1; i < num_ids; i++) { for (i = 1; i < num_ids; i++) {
if (!(c45_ids->devices_in_package & (1 << i))) if (!(devs_in_pkg & (1 << i)))
continue;
if (i == MDIO_MMD_VEND1 || i == MDIO_MMD_VEND2) {
/* Probe the "Device Present" bits for the vendor MMDs
* to ignore these if they do not contain IEEE 802.3
* registers.
*/
ret = phy_c45_probe_present(bus, addr, i);
if (ret < 0)
return ret;
if (!ret)
continue; continue;
}
phy_reg = mdiobus_c45_read(bus, addr, i, MII_PHYSID1); phy_reg = mdiobus_c45_read(bus, addr, i, MII_PHYSID1);
if (phy_reg < 0) if (phy_reg < 0)
...@@ -755,34 +797,29 @@ static int get_phy_c45_ids(struct mii_bus *bus, int addr, u32 *phy_id, ...@@ -755,34 +797,29 @@ static int get_phy_c45_ids(struct mii_bus *bus, int addr, u32 *phy_id,
return -EIO; return -EIO;
c45_ids->device_ids[i] |= phy_reg; c45_ids->device_ids[i] |= phy_reg;
} }
*phy_id = 0;
c45_ids->devices_in_package = devs_in_pkg;
/* Bit 0 doesn't represent a device, it indicates c22 regs presence */
c45_ids->mmds_present = devs_in_pkg & ~BIT(0);
return 0; return 0;
} }
/** /**
* get_phy_id - reads the specified addr for its ID. * get_phy_c22_id - reads the specified addr for its clause 22 ID.
* @bus: the target MII bus * @bus: the target MII bus
* @addr: PHY address on the MII bus * @addr: PHY address on the MII bus
* @phy_id: where to store the ID retrieved. * @phy_id: where to store the ID retrieved.
* @is_c45: If true the PHY uses the 802.3 clause 45 protocol
* @c45_ids: where to store the c45 ID information.
*
* Description: In the case of a 802.3-c22 PHY, reads the ID registers
* of the PHY at @addr on the @bus, stores it in @phy_id and returns
* zero on success.
*
* In the case of a 802.3-c45 PHY, get_phy_c45_ids() is invoked, and
* its return value is in turn returned.
* *
* Read the 802.3 clause 22 PHY ID from the PHY at @addr on the @bus,
* placing it in @phy_id. Return zero on successful read and the ID is
* valid, %-EIO on bus access error, or %-ENODEV if no device responds
* or invalid ID.
*/ */
static int get_phy_id(struct mii_bus *bus, int addr, u32 *phy_id, static int get_phy_c22_id(struct mii_bus *bus, int addr, u32 *phy_id)
bool is_c45, struct phy_c45_device_ids *c45_ids)
{ {
int phy_reg; int phy_reg;
if (is_c45)
return get_phy_c45_ids(bus, addr, phy_id, c45_ids);
/* Grab the bits from PHYIR1, and put them in the upper half */ /* Grab the bits from PHYIR1, and put them in the upper half */
phy_reg = mdiobus_read(bus, addr, MII_PHYSID1); phy_reg = mdiobus_read(bus, addr, MII_PHYSID1);
if (phy_reg < 0) { if (phy_reg < 0) {
...@@ -799,6 +836,10 @@ static int get_phy_id(struct mii_bus *bus, int addr, u32 *phy_id, ...@@ -799,6 +836,10 @@ static int get_phy_id(struct mii_bus *bus, int addr, u32 *phy_id,
*phy_id |= phy_reg; *phy_id |= phy_reg;
/* If the phy_id is mostly Fs, there is no device there */
if ((*phy_id & 0x1fffffff) == 0x1fffffff)
return -ENODEV;
return 0; return 0;
} }
...@@ -809,8 +850,17 @@ static int get_phy_id(struct mii_bus *bus, int addr, u32 *phy_id, ...@@ -809,8 +850,17 @@ static int get_phy_id(struct mii_bus *bus, int addr, u32 *phy_id,
* @addr: PHY address on the MII bus * @addr: PHY address on the MII bus
* @is_c45: If true the PHY uses the 802.3 clause 45 protocol * @is_c45: If true the PHY uses the 802.3 clause 45 protocol
* *
* Description: Reads the ID registers of the PHY at @addr on the * Probe for a PHY at @addr on @bus.
* @bus, then allocates and returns the phy_device to represent it. *
* When probing for a clause 22 PHY, then read the ID registers. If we find
* a valid ID, allocate and return a &struct phy_device.
*
* When probing for a clause 45 PHY, read the "devices in package" registers.
* If the "devices in package" appears valid, read the ID registers for each
* MMD, allocate and return a &struct phy_device.
*
* Returns an allocated &struct phy_device on success, %-ENODEV if there is
* no PHY present, or %-EIO on bus access error.
*/ */
struct phy_device *get_phy_device(struct mii_bus *bus, int addr, bool is_c45) struct phy_device *get_phy_device(struct mii_bus *bus, int addr, bool is_c45)
{ {
...@@ -819,16 +869,17 @@ struct phy_device *get_phy_device(struct mii_bus *bus, int addr, bool is_c45) ...@@ -819,16 +869,17 @@ struct phy_device *get_phy_device(struct mii_bus *bus, int addr, bool is_c45)
int r; int r;
c45_ids.devices_in_package = 0; c45_ids.devices_in_package = 0;
c45_ids.mmds_present = 0;
memset(c45_ids.device_ids, 0xff, sizeof(c45_ids.device_ids)); memset(c45_ids.device_ids, 0xff, sizeof(c45_ids.device_ids));
r = get_phy_id(bus, addr, &phy_id, is_c45, &c45_ids); if (is_c45)
r = get_phy_c45_ids(bus, addr, &c45_ids);
else
r = get_phy_c22_id(bus, addr, &phy_id);
if (r) if (r)
return ERR_PTR(r); return ERR_PTR(r);
/* If the phy_id is mostly Fs, there is no device there */
if ((phy_id & 0x1fffffff) == 0x1fffffff)
return ERR_PTR(-ENODEV);
return phy_device_create(bus, addr, phy_id, is_c45, &c45_ids); return phy_device_create(bus, addr, phy_id, is_c45, &c45_ids);
} }
EXPORT_SYMBOL(get_phy_device); EXPORT_SYMBOL(get_phy_device);
......
...@@ -1638,11 +1638,11 @@ static int phylink_phy_read(struct phylink *pl, unsigned int phy_id, ...@@ -1638,11 +1638,11 @@ static int phylink_phy_read(struct phylink *pl, unsigned int phy_id,
case MII_BMSR: case MII_BMSR:
case MII_PHYSID1: case MII_PHYSID1:
case MII_PHYSID2: case MII_PHYSID2:
devad = __ffs(phydev->c45_ids.devices_in_package); devad = __ffs(phydev->c45_ids.mmds_present);
break; break;
case MII_ADVERTISE: case MII_ADVERTISE:
case MII_LPA: case MII_LPA:
if (!(phydev->c45_ids.devices_in_package & MDIO_DEVS_AN)) if (!(phydev->c45_ids.mmds_present & MDIO_DEVS_AN))
return -EINVAL; return -EINVAL;
devad = MDIO_MMD_AN; devad = MDIO_MMD_AN;
if (reg == MII_ADVERTISE) if (reg == MII_ADVERTISE)
...@@ -1678,11 +1678,11 @@ static int phylink_phy_write(struct phylink *pl, unsigned int phy_id, ...@@ -1678,11 +1678,11 @@ static int phylink_phy_write(struct phylink *pl, unsigned int phy_id,
case MII_BMSR: case MII_BMSR:
case MII_PHYSID1: case MII_PHYSID1:
case MII_PHYSID2: case MII_PHYSID2:
devad = __ffs(phydev->c45_ids.devices_in_package); devad = __ffs(phydev->c45_ids.mmds_present);
break; break;
case MII_ADVERTISE: case MII_ADVERTISE:
case MII_LPA: case MII_LPA:
if (!(phydev->c45_ids.devices_in_package & MDIO_DEVS_AN)) if (!(phydev->c45_ids.mmds_present & MDIO_DEVS_AN))
return -EINVAL; return -EINVAL;
devad = MDIO_MMD_AN; devad = MDIO_MMD_AN;
if (reg == MII_ADVERTISE) if (reg == MII_ADVERTISE)
......
...@@ -388,14 +388,18 @@ enum phy_state { ...@@ -388,14 +388,18 @@ enum phy_state {
PHY_CABLETEST, PHY_CABLETEST,
}; };
#define MDIO_MMD_NUM 32
/** /**
* struct phy_c45_device_ids - 802.3-c45 Device Identifiers * struct phy_c45_device_ids - 802.3-c45 Device Identifiers
* @devices_in_package: Bit vector of devices present. * @devices_in_package: IEEE 802.3 devices in package register value.
* @mmds_present: bit vector of MMDs present.
* @device_ids: The device identifer for each present device. * @device_ids: The device identifer for each present device.
*/ */
struct phy_c45_device_ids { struct phy_c45_device_ids {
u32 devices_in_package; u32 devices_in_package;
u32 device_ids[8]; u32 mmds_present;
u32 device_ids[MDIO_MMD_NUM];
}; };
struct macsec_context; struct macsec_context;
......
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