summaryrefslogtreecommitdiff
path: root/sys/dev
diff options
context:
space:
mode:
Diffstat (limited to 'sys/dev')
-rw-r--r--sys/dev/pci/if_tht.c75
1 files changed, 74 insertions, 1 deletions
diff --git a/sys/dev/pci/if_tht.c b/sys/dev/pci/if_tht.c
index eb65634fade..331bb309667 100644
--- a/sys/dev/pci/if_tht.c
+++ b/sys/dev/pci/if_tht.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: if_tht.c,v 1.99 2007/05/08 14:07:14 dlg Exp $ */
+/* $OpenBSD: if_tht.c,v 1.100 2007/05/16 09:27:44 dlg Exp $ */
/*
* Copyright (c) 2007 David Gwynne <dlg@openbsd.org>
@@ -616,6 +616,8 @@ void tht_dmamem_free(struct tht_softc *,
/* bus space operations */
u_int32_t tht_read(struct tht_softc *, bus_size_t);
void tht_write(struct tht_softc *, bus_size_t, u_int32_t);
+void tht_write_region(struct tht_softc *, bus_size_t,
+ void *, size_t);
int tht_wait_eq(struct tht_softc *, bus_size_t, u_int32_t,
u_int32_t, int);
int tht_wait_ne(struct tht_softc *, bus_size_t, u_int32_t,
@@ -893,15 +895,29 @@ tht_ioctl(struct ifnet *ifp, u_long cmd, caddr_t addr)
}
break;
+ case SIOCADDMULTI:
+ error = ether_addmulti(ifr, &sc->sc_ac);
+ break;
+ case SIOCDELMULTI:
+ error = ether_delmulti(ifr, &sc->sc_ac);
+ break;
+
case SIOCGIFMEDIA:
case SIOCSIFMEDIA:
error = ifmedia_ioctl(ifp, ifr, &sc->sc_media, cmd);
break;
+
default:
error = ENOTTY;
break;
}
+ if (error == ENETRESET) {
+ if (ifp->if_flags & IFF_RUNNING)
+ tht_iff(sc);
+ error = 0;
+ }
+
err:
splx(s);
rw_exit_write(&sc->sc_lock);
@@ -992,13 +1008,61 @@ void
tht_iff(struct tht_softc *sc)
{
struct ifnet *ifp = &sc->sc_ac.ac_if;
+ struct ether_multi *enm;
+ struct ether_multistep step;
u_int32_t rxf;
+ u_int8_t imf[THT_REG_RX_MCST_HASH_SIZE];
+ u_int8_t hash;
+ int i;
+
+ ifp->if_flags &= ~IFF_ALLMULTI;
rxf = THT_REG_RX_FLT_OSEN | THT_REG_RX_FLT_AM | THT_REG_RX_FLT_AB;
+ for (i = 0; i < THT_REG_RX_MAC_MCST_CNT; i++) {
+ tht_write(sc, THT_REG_RX_MAC_MCST0(i), 0);
+ tht_write(sc, THT_REG_RX_MAC_MCST1(i), 0);
+ }
+ memset(imf, 0x00, sizeof(imf));
if (ifp->if_flags & IFF_PROMISC)
rxf |= THT_REG_RX_FLT_PRM_ALL;
+ else if (sc->sc_ac.ac_multirangecnt > 0) {
+ ifp->if_flags |= IFF_ALLMULTI;
+ memset(imf, 0xff, sizeof(imf));
+ } else {
+ ETHER_FIRST_MULTI(step, &sc->sc_ac, enm);
+
+#if 0
+ /* fill the perfect multicast filters */
+ for (i = 0; i < THT_REG_RX_MAC_MCST_CNT; i++) {
+ if (enm == NULL)
+ break;
+
+ tht_write(sc, THT_REG_RX_MAC_MCST0(i),
+ (enm->enm_addrlo[0] << 0) |
+ (enm->enm_addrlo[1] << 8) |
+ (enm->enm_addrlo[2] << 16) |
+ (enm->enm_addrlo[3] << 24));
+ tht_write(sc, THT_REG_RX_MAC_MCST1(i),
+ (enm->enm_addrlo[4] << 0) |
+ (enm->enm_addrlo[5] << 8));
+
+ ETHER_NEXT_MULTI(step, enm);
+ }
+#endif
+
+ /* fill the imperfect multicast filter with whats left */
+ while (enm != NULL) {
+ hash = 0x00;
+ for (i = 0; i < ETHER_ADDR_LEN; i++)
+ hash ^= enm->enm_addrlo[i];
+ setbit(imf, hash);
+
+ ETHER_NEXT_MULTI(step, enm);
+ }
+ }
+ tht_write_region(sc, THT_REG_RX_MCST_HASH, imf, sizeof(imf));
tht_write(sc, THT_REG_RX_FLT, rxf);
}
@@ -1718,6 +1782,15 @@ tht_write(struct tht_softc *sc, bus_size_t r, u_int32_t v)
BUS_SPACE_BARRIER_WRITE);
}
+void
+tht_write_region(struct tht_softc *sc, bus_size_t r, void *buf, size_t len)
+{
+ bus_space_write_raw_region_4(sc->sc_thtc->sc_memt, sc->sc_memh, r,
+ buf, len);
+ bus_space_barrier(sc->sc_thtc->sc_memt, sc->sc_memh, r, len,
+ BUS_SPACE_BARRIER_WRITE);
+}
+
int
tht_wait_eq(struct tht_softc *sp, bus_size_t r, u_int32_t m, u_int32_t v,
int timeout)