diff options
author | Jonathan Gray <jsg@cvs.openbsd.org> | 2007-10-10 12:14:27 +0000 |
---|---|---|
committer | Jonathan Gray <jsg@cvs.openbsd.org> | 2007-10-10 12:14:27 +0000 |
commit | 5cd59df5b113b55740a4c17d728457d2f4f3b850 (patch) | |
tree | 0b5e6692a141ec11d2f0f73883cb51b9369c5c9b /sys | |
parent | ae21673ba1ff8f55993332f0e1bb893237f042d5 (diff) |
Add support for the RTL8211B PHY based on a patch by
Michael Eisele for FreeBSD.
This fixes nfe/rgephy for ckuethe & couderc.
Tested on re/rgephy by pyr.
Diffstat (limited to 'sys')
-rw-r--r-- | sys/dev/mii/rgephy.c | 81 | ||||
-rw-r--r-- | sys/dev/mii/rgephyreg.h | 13 |
2 files changed, 71 insertions, 23 deletions
diff --git a/sys/dev/mii/rgephy.c b/sys/dev/mii/rgephy.c index 2f04ebcd870..48b49014b6d 100644 --- a/sys/dev/mii/rgephy.c +++ b/sys/dev/mii/rgephy.c @@ -1,4 +1,4 @@ -/* $OpenBSD: rgephy.c,v 1.24 2007/01/21 22:37:56 kettenis Exp $ */ +/* $OpenBSD: rgephy.c,v 1.25 2007/10/10 12:14:26 jsg Exp $ */ /* * Copyright (c) 2003 * Bill Paul <wpaul@windriver.com>. All rights reserved. @@ -148,6 +148,7 @@ rgephy_service(struct mii_softc *sc, struct mii_data *mii, int cmd) { struct ifmedia_entry *ife = mii->mii_media.ifm_cur; int anar, reg, speed, gig = 0; + uint16_t id2; switch (cmd) { case MII_POLLSTAT: @@ -253,9 +254,16 @@ setit: * need to restart the autonegotiation process. Read * the BMSR twice in case it's latched. */ - reg = PHY_READ(sc, RL_GMEDIASTAT); - if (reg & RL_GMEDIASTAT_LINK) - break; + id2 = PHY_READ(sc, MII_PHYIDR2); + if (MII_REV(id2) < 2) { + reg = PHY_READ(sc, RL_GMEDIASTAT); + if (reg & RL_GMEDIASTAT_LINK) + break; + } else { + reg = PHY_READ(sc, RGEPHY_SR); + if (reg & RGEPHY_SR_LINK) + break; + } /* * Only retry autonegotiation every mii_anegticks seconds. @@ -292,14 +300,24 @@ rgephy_status(struct mii_softc *sc) { struct mii_data *mii = sc->mii_pdata; int bmsr, bmcr, gtsr; + uint16_t id2; mii->mii_media_status = IFM_AVALID; mii->mii_media_active = IFM_ETHER; - bmsr = PHY_READ(sc, RL_GMEDIASTAT); + id2 = PHY_READ(sc, MII_PHYIDR2); + + if (MII_REV(id2) < 2) { + bmsr = PHY_READ(sc, RL_GMEDIASTAT); + + if (bmsr & RL_GMEDIASTAT_LINK) + mii->mii_media_status |= IFM_ACTIVE; + } else { + bmsr = PHY_READ(sc, RGEPHY_SR); + if (bmsr & RGEPHY_SR_LINK) + mii->mii_media_status |= IFM_ACTIVE; + } - if (bmsr & RL_GMEDIASTAT_LINK) - mii->mii_media_status |= IFM_ACTIVE; bmsr = PHY_READ(sc, RGEPHY_MII_BMSR); bmcr = PHY_READ(sc, RGEPHY_MII_BMCR); @@ -315,18 +333,35 @@ rgephy_status(struct mii_softc *sc) } } - bmsr = PHY_READ(sc, RL_GMEDIASTAT); - if (bmsr & RL_GMEDIASTAT_1000MBPS) - mii->mii_media_active |= IFM_1000_T; - else if (bmsr & RL_GMEDIASTAT_100MBPS) - mii->mii_media_active |= IFM_100_TX; - else if (bmsr & RL_GMEDIASTAT_10MBPS) - mii->mii_media_active |= IFM_10_T; - - if (bmsr & RL_GMEDIASTAT_FDX) - mii->mii_media_active |= mii_phy_flowstatus(sc) | IFM_FDX; - else - mii->mii_media_active |= IFM_HDX; + if (MII_REV(id2) < 2) { + bmsr = PHY_READ(sc, RL_GMEDIASTAT); + if (bmsr & RL_GMEDIASTAT_1000MBPS) + mii->mii_media_active |= IFM_1000_T; + else if (bmsr & RL_GMEDIASTAT_100MBPS) + mii->mii_media_active |= IFM_100_TX; + else if (bmsr & RL_GMEDIASTAT_10MBPS) + mii->mii_media_active |= IFM_10_T; + + if (bmsr & RL_GMEDIASTAT_FDX) + mii->mii_media_active |= mii_phy_flowstatus(sc) | + IFM_FDX; + else + mii->mii_media_active |= IFM_HDX; + } else { + bmsr = PHY_READ(sc, RGEPHY_SR); + if (RGEPHY_SR_SPEED(bmsr) == RGEPHY_SR_SPEED_1000MBPS) + mii->mii_media_active |= IFM_1000_T; + else if (RGEPHY_SR_SPEED(bmsr) == RGEPHY_SR_SPEED_100MBPS) + mii->mii_media_active |= IFM_100_TX; + else if (bmsr & RL_GMEDIASTAT_10MBPS) + mii->mii_media_active |= IFM_10_T; + + if (bmsr & RGEPHY_SR_FDX) + mii->mii_media_active |= mii_phy_flowstatus(sc) | + IFM_FDX; + else + mii->mii_media_active |= IFM_HDX; + } gtsr = PHY_READ(sc, RGEPHY_MII_1000STS); if ((IFM_SUBTYPE(mii->mii_media_active) == IFM_1000_T) && @@ -363,10 +398,14 @@ void rgephy_loop(struct mii_softc *sc) { u_int32_t bmsr; + uint16_t id2; int i; - PHY_WRITE(sc, RGEPHY_MII_BMCR, RGEPHY_BMCR_PDOWN); - DELAY(1000); + id2 = PHY_READ(sc, MII_PHYIDR2); + if (MII_REV(id2) < 2) { + PHY_WRITE(sc, RGEPHY_MII_BMCR, RGEPHY_BMCR_PDOWN); + DELAY(1000); + } for (i = 0; i < 15000; i++) { bmsr = PHY_READ(sc, RGEPHY_MII_BMSR); diff --git a/sys/dev/mii/rgephyreg.h b/sys/dev/mii/rgephyreg.h index cf22daae07c..d4a66f3b6d0 100644 --- a/sys/dev/mii/rgephyreg.h +++ b/sys/dev/mii/rgephyreg.h @@ -1,4 +1,4 @@ -/* $OpenBSD: rgephyreg.h,v 1.2 2004/10/28 23:37:11 pvalchev Exp $ */ +/* $OpenBSD: rgephyreg.h,v 1.3 2007/10/10 12:14:26 jsg Exp $ */ /* * Copyright (c) 2003 * Bill Paul <wpaul@windriver.com>. All rights reserved. @@ -138,6 +138,15 @@ #define RGEPHY_EXTSTS_T_FD_CAP 0x2000 /* 1000base-T FD capable */ #define RGEPHY_EXTSTS_T_HD_CAP 0x1000 /* 1000base-T HD capable */ - +/* RTL8211B */ +#define RGEPHY_SR 0x11 /* PHY Specific Status */ +#define RGEPHY_SR_CROSSOVER (1 << 6) +#define RGEPHY_SR_LINK (1 << 10) +#define RGEPHY_SR_FDX (1 << 13) +#define RGEPHY_SR_SPEED_MASK (3 << 14) +#define RGEPHY_SR_SPEED_1000MBPS (2 << 14) +#define RGEPHY_SR_SPEED_100MBPS (1 << 14) +#define RGEPHY_SR_SPEED_10MBPS (0 << 14) +#define RGEPHY_SR_SPEED(X) ((X) & RGEPHY_SR_SPEED_MASK) #endif /* _DEV_RGEPHY_MIIREG_H_ */ |