summaryrefslogtreecommitdiff
path: root/sys/dev
diff options
context:
space:
mode:
Diffstat (limited to 'sys/dev')
-rw-r--r--sys/dev/pci/if_sis.c107
1 files changed, 72 insertions, 35 deletions
diff --git a/sys/dev/pci/if_sis.c b/sys/dev/pci/if_sis.c
index 2bb43878c30..8aff9fc5c93 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.52 2005/10/12 21:14:37 brad Exp $ */
+/* $OpenBSD: if_sis.c,v 1.53 2005/10/13 01:44:33 brad Exp $ */
/*
* Copyright (c) 1997, 1998, 1999
* Bill Paul <wpaul@ctr.columbia.edu>. All rights reserved.
@@ -142,6 +142,7 @@ int sis_miibus_readreg(struct device *, int, int);
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_sis(struct sis_softc *);
void sis_setmulti_ns(struct sis_softc *);
void sis_reset(struct sis_softc *);
@@ -696,8 +697,31 @@ sis_miibus_statchg(self)
return;
}
-void sis_setmulti_ns(sc)
- struct sis_softc *sc;
+u_int32_t
+sis_mchash(struct sis_softc *sc, const uint8_t *addr)
+{
+ uint32_t crc;
+
+ /* Compute CRC for the address value. */
+ crc = ether_crc32_be(addr, ETHER_ADDR_LEN);
+
+ /*
+ * return the filter bit position
+ *
+ * The NatSemi chip has a 512-bit filter, which is
+ * different than the SiS, so we special-case it.
+ */
+ if (sc->sis_type == SIS_TYPE_83815)
+ return (crc >> 23);
+ else if (sc->sis_rev >= SIS_REV_635 ||
+ sc->sis_rev == SIS_REV_900B)
+ return (crc >> 24);
+ else
+ return (crc >> 25);
+}
+
+void
+sis_setmulti_ns(struct sis_softc *sc)
{
struct ifnet *ifp;
struct arpcom *ac = &sc->arpcom;
@@ -737,7 +761,7 @@ allmulti:
goto allmulti;
}
- h = ether_crc32_be(enm->enm_addrlo, ETHER_ADDR_LEN) >> 23;
+ h = sis_mchash(sc, enm->enm_addrlo);
index = h >> 3;
bit = h & 0x1F;
CSR_WRITE_4(sc, SIS_RXFILT_CTL, NS_FILTADDR_FMEM_LO + index);
@@ -748,55 +772,68 @@ allmulti:
}
CSR_WRITE_4(sc, SIS_RXFILT_CTL, filtsave);
-
- return;
}
-void sis_setmulti_sis(sc)
- struct sis_softc *sc;
+void
+sis_setmulti_sis(struct sis_softc *sc)
{
struct ifnet *ifp;
struct arpcom *ac = &sc->arpcom;
struct ether_multi *enm;
struct ether_multistep step;
- u_int32_t h = 0, i, filtsave;
+ u_int32_t h, i, n, ctl;
+ u_int16_t hashes[16];
ifp = &sc->arpcom.ac_if;
-allmulti:
- if (ifp->if_flags & IFF_ALLMULTI || ifp->if_flags & IFF_PROMISC) {
- SIS_SETBIT(sc, SIS_RXFILT_CTL, SIS_RXFILTCTL_ALLMULTI);
- return;
- }
+ /* hash table size */
+ if (sc->sis_rev >= SIS_REV_635 ||
+ sc->sis_rev == SIS_REV_900B)
+ n = 16;
+ else
+ n = 8;
- SIS_CLRBIT(sc, SIS_RXFILT_CTL, SIS_RXFILTCTL_ALLMULTI);
+ ctl = CSR_READ_4(sc, SIS_RXFILT_CTL) & SIS_RXFILTCTL_ENABLE;
- filtsave = CSR_READ_4(sc, SIS_RXFILT_CTL);
+ if (ifp->if_flags & IFF_BROADCAST)
+ ctl |= SIS_RXFILTCTL_BROAD;
- /* first, zot all the existing hash bits */
- for (i = 0; i < 8; i++) {
- CSR_WRITE_4(sc, SIS_RXFILT_CTL, (4 + ((i * 16) >> 4)) << 16);
- CSR_WRITE_4(sc, SIS_RXFILT_DATA, 0);
- }
+allmulti:
+ if (ifp->if_flags & IFF_ALLMULTI || ifp->if_flags & IFF_PROMISC) {
+ ctl |= SIS_RXFILTCTL_ALLMULTI;
+ if (ifp->if_flags & IFF_PROMISC)
+ ctl |= SIS_RXFILTCTL_BROAD|SIS_RXFILTCTL_ALLPHYS;
+ for (i = 0; i < n; i++)
+ hashes[i] = ~0;
+ } else {
+ for (i = 0; i < n; i++)
+ hashes[i] = 0;
+ i = 0;
+ ETHER_FIRST_MULTI(step, ac, enm);
+ while (enm != NULL) {
+ if (bcmp(enm->enm_addrlo, enm->enm_addrhi, ETHER_ADDR_LEN)) {
+ ifp->if_flags |= IFF_ALLMULTI;
+ goto allmulti;
+ }
- /* now program new ones */
- ETHER_FIRST_MULTI(step, ac, enm);
- while (enm != NULL) {
- if (bcmp(enm->enm_addrlo, enm->enm_addrhi, ETHER_ADDR_LEN)) {
- ifp->if_flags |= IFF_ALLMULTI;
- goto allmulti;
+ h = sis_mchash(sc, enm->enm_addrlo);
+ hashes[h >> 4] |= 1 << (h & 0xf);
+ i++;
+ ETHER_NEXT_MULTI(step, enm);
+ }
+ if (i > n) {
+ ctl |= SIS_RXFILTCTL_ALLMULTI;
+ for (i = 0; i < n; i++)
+ hashes[i] = ~0;
}
-
- h = (ether_crc32_be(enm->enm_addrlo,
- ETHER_ADDR_LEN) >> 25) & 0x0000007F;
- CSR_WRITE_4(sc, SIS_RXFILT_CTL, (4 + (h >> 4)) << 16);
- SIS_SETBIT(sc, SIS_RXFILT_DATA, (1 << (h & 0xF)));
- ETHER_NEXT_MULTI(step, enm);
}
- CSR_WRITE_4(sc, SIS_RXFILT_CTL, filtsave);
+ for (i = 0; i < n; i++) {
+ CSR_WRITE_4(sc, SIS_RXFILT_CTL, (4 + i) << 16);
+ CSR_WRITE_4(sc, SIS_RXFILT_DATA, hashes[i]);
+ }
- return;
+ CSR_WRITE_4(sc, SIS_RXFILT_CTL, ctl);
}
void sis_reset(sc)