Annotation of sys/dev/mii/urlphy.c, Revision 1.1
1.1 ! nbrk 1: /* $OpenBSD: urlphy.c,v 1.12 2006/12/27 19:11:09 kettenis Exp $ */
! 2: /* $NetBSD: urlphy.c,v 1.1 2002/03/28 21:07:53 ichiro Exp $ */
! 3: /*
! 4: * Copyright (c) 2001, 2002
! 5: * Shingo WATANABE <nabe@nabechan.org>. All rights reserved.
! 6: *
! 7: * Redistribution and use in source and binary forms, with or without
! 8: * modification, are permitted provided that the following conditions
! 9: * are met:
! 10: * 1. Redistributions of source code must retain the above copyright
! 11: * notice, this list of conditions and the following disclaimer.
! 12: * 2. Redistributions in binary form must reproduce the above copyright
! 13: * notice, this list of conditions and the following disclaimer in the
! 14: * documentation and/or other materials provided with the distribution.
! 15: * 3. Neither the name of the author nor the names of any co-contributors
! 16: * may be used to endorse or promote products derived from this software
! 17: * without specific prior written permission.
! 18: *
! 19: * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
! 20: * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
! 21: * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
! 22: * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
! 23: * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
! 24: * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
! 25: * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
! 26: * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
! 27: * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
! 28: * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
! 29: * SUCH DAMAGE.
! 30: *
! 31: */
! 32:
! 33: /*
! 34: * driver for Realtek RL8150L internal phy
! 35: */
! 36:
! 37: #include <sys/param.h>
! 38: #include <sys/systm.h>
! 39: #include <sys/kernel.h>
! 40: #include <sys/device.h>
! 41: #include <sys/socket.h>
! 42:
! 43: #include <net/if.h>
! 44: #include <net/if_media.h>
! 45:
! 46: #include <dev/mii/mii.h>
! 47: #include <dev/mii/miivar.h>
! 48: #include <dev/mii/miidevs.h>
! 49: #include <dev/mii/urlphyreg.h>
! 50:
! 51: #define URLPHY_DEBUG 0
! 52: #ifdef URLPHY_DEBUG
! 53: #define DPRINTF(x) if (urlphydebug) printf x
! 54: #define DPRINTFN(n,x) if (urlphydebug>(n)) printf x
! 55: int urlphydebug = URLPHY_DEBUG;
! 56: #else
! 57: #define DPRINTF(x)
! 58: #define DPRINTFN(n,x)
! 59: #endif
! 60:
! 61: int urlphy_match(struct device *, void *, void *);
! 62: void urlphy_attach(struct device *, struct device *, void *);
! 63:
! 64: struct cfattach urlphy_ca = {
! 65: sizeof(struct mii_softc), urlphy_match, urlphy_attach, mii_phy_detach,
! 66: mii_phy_activate
! 67: };
! 68:
! 69: struct cfdriver urlphy_cd = {
! 70: NULL, "urlphy", DV_DULL
! 71: };
! 72:
! 73: int urlphy_service(struct mii_softc *, struct mii_data *, int);
! 74: void urlphy_status(struct mii_softc *);
! 75:
! 76: const struct mii_phy_funcs urlphy_funcs = {
! 77: urlphy_service, urlphy_status, mii_phy_reset,
! 78: };
! 79:
! 80: int
! 81: urlphy_match(struct device *parent, void *match, void *aux)
! 82: {
! 83: struct mii_attach_args *ma = aux;
! 84:
! 85: /*
! 86: * RTL8150 reports OUT == 0, MODEL == 0
! 87: */
! 88: if (MII_OUI(ma->mii_id1, ma->mii_id2) != 0 &&
! 89: MII_MODEL(ma->mii_id2) != 0)
! 90: return (0);
! 91:
! 92: /*
! 93: * Make sure the parent is an 'url' device.
! 94: */
! 95: if (strcmp(parent->dv_cfdata->cf_driver->cd_name, "url") != 0)
! 96: return (0);
! 97:
! 98: return (10);
! 99: }
! 100:
! 101: void
! 102: urlphy_attach(struct device *parent, struct device *self, void *aux)
! 103: {
! 104: struct mii_softc *sc = (struct mii_softc *)self;
! 105: struct mii_attach_args *ma = aux;
! 106: struct mii_data *mii = ma->mii_data;
! 107:
! 108: printf(": RTL internal phy\n");
! 109:
! 110: DPRINTF(("%s: %s: enter\n", sc->mii_dev.dv_xname, __func__));
! 111:
! 112: sc->mii_inst = mii->mii_instance;
! 113: sc->mii_phy = ma->mii_phyno;
! 114: sc->mii_funcs = &urlphy_funcs;
! 115: sc->mii_pdata = mii;
! 116: sc->mii_flags = ma->mii_flags;
! 117: sc->mii_anegticks = MII_ANEGTICKS_GIGE;
! 118:
! 119: /* Don't do loopback on this PHY. */
! 120: sc->mii_flags |= MIIF_NOLOOP;
! 121: /* Don't do isolate on this PHY. */
! 122: sc->mii_flags |= MIIF_NOISOLATE;
! 123:
! 124: if (mii->mii_instance != 0) {
! 125: printf("%s: ignoring this PHY, non-zero instance\n",
! 126: sc->mii_dev.dv_xname);
! 127: return;
! 128: }
! 129: PHY_RESET(sc);
! 130:
! 131: sc->mii_capabilities = PHY_READ(sc, MII_BMSR) & ma->mii_capmask;
! 132: if (sc->mii_capabilities & BMSR_MEDIAMASK)
! 133: mii_phy_add_media(sc);
! 134: }
! 135:
! 136: int
! 137: urlphy_service(struct mii_softc *sc, struct mii_data *mii, int cmd)
! 138: {
! 139: struct ifmedia_entry *ife = mii->mii_media.ifm_cur;
! 140: int reg;
! 141:
! 142: DPRINTF(("%s: %s: enter\n", sc->mii_dev.dv_xname, __func__));
! 143:
! 144: if ((sc->mii_dev.dv_flags & DVF_ACTIVE) == 0)
! 145: return (ENXIO);
! 146:
! 147: switch (cmd) {
! 148: case MII_POLLSTAT:
! 149: /*
! 150: * If we're not polling our PHY instance, just return.
! 151: */
! 152: if (IFM_INST(ife->ifm_media) != sc->mii_inst)
! 153: return (0);
! 154: break;
! 155:
! 156: case MII_MEDIACHG:
! 157: /*
! 158: * If we're not currently selected, just return.
! 159: */
! 160: if (IFM_INST(ife->ifm_media) != sc->mii_inst)
! 161: return (0);
! 162:
! 163: /* If the interface is not up, don't do anything. */
! 164: if ((mii->mii_ifp->if_flags & IFF_UP) == 0)
! 165: break;
! 166:
! 167: mii_phy_setmedia(sc);
! 168: break;
! 169:
! 170: case MII_TICK:
! 171: /*
! 172: * If we're not currently selected, just return.
! 173: */
! 174: if (IFM_INST(ife->ifm_media) != sc->mii_inst)
! 175: return (0);
! 176:
! 177: /* Just bail now if the interface is down. */
! 178: if ((mii->mii_ifp->if_flags & IFF_UP) == 0)
! 179: return (0);
! 180:
! 181: /*
! 182: * If we're not doing autonegotiation, we don't need to do
! 183: * any extra work here. However, we need to check the link
! 184: * status so we can generate an announcement if the status
! 185: * changes.
! 186: */
! 187: if (IFM_SUBTYPE(ife->ifm_media) != IFM_AUTO)
! 188: return (0);
! 189:
! 190: /* Read the status register twice; MSR_LINK is latch-low. */
! 191: reg = PHY_READ(sc, URLPHY_MSR) | PHY_READ(sc, URLPHY_MSR);
! 192: if (reg & URLPHY_MSR_LINK)
! 193: return (0);
! 194:
! 195: /*
! 196: * Only retry autonegotiation every mii_anegticks seconds.
! 197: */
! 198: if (++sc->mii_ticks <= sc->mii_anegticks)
! 199: return (0);
! 200:
! 201: sc->mii_ticks = 0;
! 202: PHY_RESET(sc);
! 203:
! 204: if (mii_phy_auto(sc, 0) == EJUSTRETURN)
! 205: return (0);
! 206:
! 207: break;
! 208:
! 209: case MII_DOWN:
! 210: mii_phy_down(sc);
! 211: return (0);
! 212: }
! 213:
! 214: /* Update the media status. */
! 215: mii_phy_status(sc);
! 216:
! 217: /* Callback if something changed. */
! 218: mii_phy_update(sc, cmd);
! 219:
! 220: return (0);
! 221: }
! 222:
! 223: void
! 224: urlphy_status(struct mii_softc *sc)
! 225: {
! 226: struct mii_data *mii = sc->mii_pdata;
! 227: struct ifmedia_entry *ife = mii->mii_media.ifm_cur;
! 228: int msr, bmsr, bmcr;
! 229:
! 230: DPRINTF(("%s: %s: enter\n", sc->mii_dev.dv_xname, __func__));
! 231:
! 232: mii->mii_media_status = IFM_AVALID;
! 233: mii->mii_media_active = IFM_ETHER;
! 234:
! 235: /*
! 236: * The link status bit is not exist in the BMSR register,
! 237: * so we need to read the MSR register to get link status.
! 238: */
! 239: msr = PHY_READ(sc, URLPHY_MSR) | PHY_READ(sc, URLPHY_MSR);
! 240: if (msr & URLPHY_MSR_LINK)
! 241: mii->mii_media_status |= IFM_ACTIVE;
! 242:
! 243: DPRINTF(("%s: %s: link %s\n", sc->mii_dev.dv_xname, __func__,
! 244: mii->mii_media_status & IFM_ACTIVE ? "up" : "down"));
! 245:
! 246: bmcr = PHY_READ(sc, MII_BMCR);
! 247: if (bmcr & BMCR_AUTOEN) {
! 248: bmsr = PHY_READ(sc, MII_BMSR) | PHY_READ(sc, MII_BMSR);
! 249: if ((bmsr & BMSR_ACOMP) == 0) {
! 250: /* Erg, still trying, I guess... */
! 251: mii->mii_media_active |= IFM_NONE;
! 252: return;
! 253: }
! 254:
! 255: if (msr & URLPHY_MSR_SPEED_100)
! 256: mii->mii_media_active |= IFM_100_TX;
! 257: else
! 258: mii->mii_media_active |= IFM_10_T;
! 259:
! 260: if (msr & URLPHY_MSR_DUPLEX)
! 261: mii->mii_media_active |= IFM_FDX;
! 262: else
! 263: mii->mii_media_active |= IFM_HDX;
! 264: } else
! 265: mii->mii_media_active = ife->ifm_media;
! 266: }
CVSweb