summaryrefslogtreecommitdiff
path: root/sys
diff options
context:
space:
mode:
authorJonathan Gray <jsg@cvs.openbsd.org>2007-10-10 12:14:27 +0000
committerJonathan Gray <jsg@cvs.openbsd.org>2007-10-10 12:14:27 +0000
commit5cd59df5b113b55740a4c17d728457d2f4f3b850 (patch)
tree0b5e6692a141ec11d2f0f73883cb51b9369c5c9b /sys
parentae21673ba1ff8f55993332f0e1bb893237f042d5 (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.c81
-rw-r--r--sys/dev/mii/rgephyreg.h13
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_ */