root/dev/mii/urlphy.c

/* [<][>][^][v][top][bottom][index][help] */

DEFINITIONS

This source file includes following definitions.
  1. urlphy_match
  2. urlphy_attach
  3. urlphy_service
  4. urlphy_status

    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 }

/* [<][>][^][v][top][bottom][index][help] */