diff options
Diffstat (limited to 'sys/dev/ic/re.c')
-rw-r--r-- | sys/dev/ic/re.c | 63 |
1 files changed, 36 insertions, 27 deletions
diff --git a/sys/dev/ic/re.c b/sys/dev/ic/re.c index 8845422cdd6..5faa23d6f79 100644 --- a/sys/dev/ic/re.c +++ b/sys/dev/ic/re.c @@ -1,4 +1,4 @@ -/* $OpenBSD: re.c,v 1.38 2006/08/05 21:03:22 brad Exp $ */ +/* $OpenBSD: re.c,v 1.39 2006/08/05 21:38:20 brad Exp $ */ /* $FreeBSD: if_re.c,v 1.31 2004/09/04 07:54:05 ru Exp $ */ /* * Copyright (c) 1997, 1998-2003 @@ -187,6 +187,7 @@ void re_miibus_writereg(struct device *, int, int, int); void re_miibus_statchg(struct device *); void re_setmulti(struct rl_softc *); +void re_setpromisc(struct rl_softc *); void re_reset(struct rl_softc *); #ifdef RE_DIAG @@ -520,6 +521,22 @@ re_setmulti(struct rl_softc *sc) } void +re_setpromisc(struct rl_softc *sc) +{ + struct ifnet *ifp; + u_int32_t rxcfg = 0; + + ifp = &sc->sc_arpcom.ac_if; + + rxcfg = CSR_READ_4(sc, RL_RXCFG); + if (ifp->if_flags & IFF_PROMISC) + rxcfg |= RL_RXCFG_RX_ALLPHYS; + else + rxcfg &= ~RL_RXCFG_RX_ALLPHYS; + CSR_WRITE_4(sc, RL_RXCFG, rxcfg); +} + +void re_reset(struct rl_softc *sc) { int i; @@ -886,11 +903,8 @@ re_attach(struct rl_softc *sc) ifp->if_start = re_start; ifp->if_watchdog = re_watchdog; ifp->if_init = re_init; - if (sc->rl_type == RL_8169) { - ifp->if_baudrate = 1000000000; + if (sc->rl_type == RL_8169) ifp->if_hardmtu = RL_JUMBO_MTU; - } else - ifp->if_baudrate = 100000000; IFQ_SET_MAXLEN(&ifp->if_snd, RL_TX_QLEN); IFQ_SET_READY(&ifp->if_snd); @@ -1754,13 +1768,6 @@ re_init(struct ifnet *ifp) rxcfg = CSR_READ_4(sc, RL_RXCFG); rxcfg |= RL_RXCFG_RX_INDIV; - /* If we want promiscuous mode, set the allframes bit. */ - if (ifp->if_flags & IFF_PROMISC) - rxcfg |= RL_RXCFG_RX_ALLPHYS; - else - rxcfg &= ~RL_RXCFG_RX_ALLPHYS; - CSR_WRITE_4(sc, RL_RXCFG, rxcfg); - /* * Set capture broadcast bit to capture broadcast frames. */ @@ -1768,8 +1775,12 @@ re_init(struct ifnet *ifp) rxcfg |= RL_RXCFG_RX_BROAD; else rxcfg &= ~RL_RXCFG_RX_BROAD; + CSR_WRITE_4(sc, RL_RXCFG, rxcfg); + /* Set promiscuous mode. */ + re_setpromisc(sc); + /* * Program the multicast filter, if necessary. */ @@ -1888,36 +1899,34 @@ re_ioctl(struct ifnet *ifp, u_long command, caddr_t data) switch(command) { case SIOCSIFADDR: ifp->if_flags |= IFF_UP; - switch (ifa->ifa_addr->sa_family) { -#ifdef INET - case AF_INET: + if (!(ifp->if_flags & IFF_RUNNING)) re_init(ifp); +#ifdef INET + if (ifa->ifa_addr->sa_family == AF_INET) arp_ifinit(&sc->sc_arpcom, ifa); - break; #endif /* INET */ - default: - re_init(ifp); - break; - } break; case SIOCSIFMTU: - if (ifr->ifr_mtu < ETHERMIN || - ((sc->rl_type == RL_8169 && - ifr->ifr_mtu > RL_JUMBO_MTU) || - (sc->rl_type == RL_8139 && - ifr->ifr_mtu > ETHERMTU))) + if (ifr->ifr_mtu < ETHERMIN || ifr->ifr_mtu > ifp->if_hardmtu) error = EINVAL; else if (ifp->if_mtu != ifr->ifr_mtu) ifp->if_mtu = ifr->ifr_mtu; break; case SIOCSIFFLAGS: if (ifp->if_flags & IFF_UP) { - re_init(ifp); + if (ifp->if_flags & IFF_RUNNING && + ((ifp->if_flags ^ sc->if_flags) & + IFF_PROMISC)) { + re_setpromisc(sc); + } else { + if (!(ifp->if_flags & IFF_RUNNING)) + re_init(ifp); + } } else { if (ifp->if_flags & IFF_RUNNING) re_stop(ifp, 1); } - error = 0; + sc->if_flags = ifp->if_flags; break; case SIOCADDMULTI: case SIOCDELMULTI: |