summaryrefslogtreecommitdiff
path: root/sys/dev
diff options
context:
space:
mode:
Diffstat (limited to 'sys/dev')
-rw-r--r--sys/dev/pci/if_sis.c61
-rw-r--r--sys/dev/pci/if_sisreg.h3
2 files changed, 41 insertions, 23 deletions
diff --git a/sys/dev/pci/if_sis.c b/sys/dev/pci/if_sis.c
index f4842437414..7ba76e81c97 100644
--- a/sys/dev/pci/if_sis.c
+++ b/sys/dev/pci/if_sis.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: if_sis.c,v 1.62 2006/03/20 16:15:03 brad Exp $ */
+/* $OpenBSD: if_sis.c,v 1.63 2006/03/25 03:21:56 brad Exp $ */
/*
* Copyright (c) 1997, 1998, 1999
* Bill Paul <wpaul@ctr.columbia.edu>. All rights reserved.
@@ -143,6 +143,7 @@ void sis_miibus_writereg(struct device *, int, int, int);
void sis_miibus_statchg(struct device *);
u_int32_t sis_mchash(struct sis_softc *, const uint8_t *);
+void sis_setmulti(struct sis_softc *);
void sis_setmulti_sis(struct sis_softc *);
void sis_setmulti_ns(struct sis_softc *);
void sis_reset(struct sis_softc *);
@@ -724,6 +725,15 @@ sis_mchash(struct sis_softc *sc, const uint8_t *addr)
}
void
+sis_setmulti(struct sis_softc *sc)
+{
+ if (sc->sis_type == SIS_TYPE_83815)
+ sis_setmulti_ns(sc);
+ else
+ sis_setmulti_sis(sc);
+}
+
+void
sis_setmulti_ns(struct sis_softc *sc)
{
struct ifnet *ifp;
@@ -735,8 +745,8 @@ sis_setmulti_ns(struct sis_softc *sc)
ifp = &sc->arpcom.ac_if;
-allmulti:
if (ifp->if_flags & IFF_ALLMULTI || ifp->if_flags & IFF_PROMISC) {
+allmulti:
SIS_CLRBIT(sc, SIS_RXFILT_CTL, NS_RXFILTCTL_MCHASH);
SIS_SETBIT(sc, SIS_RXFILT_CTL, SIS_RXFILTCTL_ALLMULTI);
return;
@@ -801,8 +811,8 @@ sis_setmulti_sis(struct sis_softc *sc)
if (ifp->if_flags & IFF_BROADCAST)
ctl |= SIS_RXFILTCTL_BROAD;
-allmulti:
if (ifp->if_flags & IFF_ALLMULTI || ifp->if_flags & IFF_PROMISC) {
+allmulti:
ctl |= SIS_RXFILTCTL_ALLMULTI;
if (ifp->if_flags & IFF_PROMISC)
ctl |= SIS_RXFILTCTL_BROAD|SIS_RXFILTCTL_ALLPHYS;
@@ -1764,10 +1774,7 @@ void sis_init(xsc)
/*
* Load the multicast filter.
*/
- if (sc->sis_type == SIS_TYPE_83815)
- sis_setmulti_ns(sc);
- else
- sis_setmulti_sis(sc);
+ sis_setmulti(sc);
/* Turn the receive filter on */
SIS_SETBIT(sc, SIS_RXFILT_CTL, SIS_RXFILTCTL_ENABLE);
@@ -1931,24 +1938,38 @@ int sis_ioctl(ifp, command, data)
switch(command) {
case SIOCSIFADDR:
ifp->if_flags |= IFF_UP;
- switch (ifa->ifa_addr->sa_family) {
- case AF_INET:
+ if ((ifp->if_flags & IFF_RUNNING) == 0)
sis_init(sc);
+#ifdef INET
+ if (ifa->ifa_addr->sa_family == AF_INET)
arp_ifinit(&sc->arpcom, ifa);
- break;
- default:
- sis_init(sc);
- break;
- }
+#endif
break;
case SIOCSIFFLAGS:
if (ifp->if_flags & IFF_UP) {
- sis_init(sc);
+ if (ifp->if_flags & IFF_RUNNING &&
+ ifp->if_flags & IFF_PROMISC &&
+ !(sc->sc_if_flags & IFF_PROMISC)) {
+ SIS_SETBIT(sc, SIS_RXFILT_CTL,
+ SIS_RXFILTCTL_ALLPHYS);
+ sis_setmulti(sc);
+ } else if (ifp->if_flags & IFF_RUNNING &&
+ !(ifp->if_flags & IFF_PROMISC) &&
+ sc->sc_if_flags & IFF_PROMISC) {
+ SIS_CLRBIT(sc, SIS_RXFILT_CTL,
+ SIS_RXFILTCTL_ALLPHYS);
+ sis_setmulti(sc);
+ } else if (ifp->if_flags & IFF_RUNNING &&
+ (ifp->if_flags ^ sc->sc_if_flags) & IFF_ALLMULTI) {
+ sis_setmulti(sc);
+ } else {
+ if ((ifp->if_flags & IFF_RUNNING) == 0)
+ sis_init(sc);
+ }
} else {
if (ifp->if_flags & IFF_RUNNING)
sis_stop(sc);
}
- error = 0;
break;
case SIOCSIFMTU:
if (ifr->ifr_mtu < ETHERMIN || ifr->ifr_mtu > ETHERMTU)
@@ -1967,12 +1988,8 @@ int sis_ioctl(ifp, command, data)
* Multicast list has changed; set the hardware
* filter accordingly.
*/
- if (ifp->if_flags & IFF_RUNNING) {
- if (sc->sis_type == SIS_TYPE_83815)
- sis_setmulti_ns(sc);
- else
- sis_setmulti_sis(sc);
- }
+ if (ifp->if_flags & IFF_RUNNING)
+ sis_setmulti(sc);
error = 0;
}
break;
diff --git a/sys/dev/pci/if_sisreg.h b/sys/dev/pci/if_sisreg.h
index 11992a286e6..c6ff985ace6 100644
--- a/sys/dev/pci/if_sisreg.h
+++ b/sys/dev/pci/if_sisreg.h
@@ -1,4 +1,4 @@
-/* $OpenBSD: if_sisreg.h,v 1.24 2005/10/20 21:47:56 brad Exp $ */
+/* $OpenBSD: if_sisreg.h,v 1.25 2006/03/25 03:21:56 brad Exp $ */
/*
* Copyright (c) 1997, 1998, 1999
* Bill Paul <wpaul@ee.columbia.edu>. All rights reserved.
@@ -449,6 +449,7 @@ struct sis_softc {
bus_dmamap_t sc_tx_sparemap;
int sis_stopped;
int sc_rxbufs;
+ int sc_if_flags;
};
/*