summaryrefslogtreecommitdiffstats
path: root/sys/dev/urtwn/if_urtwn.c
diff options
context:
space:
mode:
authorRenato Botelho <renato@netgate.com>2016-05-19 10:45:17 -0300
committerRenato Botelho <renato@netgate.com>2016-05-19 10:45:17 -0300
commit57a447a9c3cb69fb2e4f874104327cdc3203313a (patch)
tree5f882a4db4469e125c743f84b4f8998df4f015c3 /sys/dev/urtwn/if_urtwn.c
parent5cf4da15d7c435594db5cc482d961f98986d0808 (diff)
parent0412106b46a9cffa2790ddc17339d96c0dc607c9 (diff)
downloadFreeBSD-src-57a447a9c3cb69fb2e4f874104327cdc3203313a.zip
FreeBSD-src-57a447a9c3cb69fb2e4f874104327cdc3203313a.tar.gz
Merge remote-tracking branch 'origin/master' into devel-11
Diffstat (limited to 'sys/dev/urtwn/if_urtwn.c')
-rw-r--r--sys/dev/urtwn/if_urtwn.c75
1 files changed, 71 insertions, 4 deletions
diff --git a/sys/dev/urtwn/if_urtwn.c b/sys/dev/urtwn/if_urtwn.c
index dce9d19..efbdaba 100644
--- a/sys/dev/urtwn/if_urtwn.c
+++ b/sys/dev/urtwn/if_urtwn.c
@@ -356,6 +356,8 @@ static void urtwn_update_slot(struct ieee80211com *);
static void urtwn_update_slot_cb(struct urtwn_softc *,
union sec_param *);
static void urtwn_update_aifs(struct urtwn_softc *, uint8_t);
+static uint8_t urtwn_get_multi_pos(const uint8_t[]);
+static void urtwn_set_multi(struct urtwn_softc *);
static void urtwn_set_promisc(struct urtwn_softc *);
static void urtwn_update_promisc(struct ieee80211com *);
static void urtwn_update_mcast(struct ieee80211com *);
@@ -4359,9 +4361,8 @@ urtwn_rxfilter_init(struct urtwn_softc *sc)
URTWN_ASSERT_LOCKED(sc);
- /* Accept all multicast frames. */
- urtwn_write_4(sc, R92C_MAR + 0, 0xffffffff);
- urtwn_write_4(sc, R92C_MAR + 4, 0xffffffff);
+ /* Setup multicast filter. */
+ urtwn_set_multi(sc);
/* Filter for management frames. */
filter = 0x7f3f;
@@ -4822,6 +4823,67 @@ urtwn_update_aifs(struct urtwn_softc *sc, uint8_t slottime)
}
}
+static uint8_t
+urtwn_get_multi_pos(const uint8_t maddr[])
+{
+ uint64_t mask = 0x00004d101df481b4;
+ uint8_t pos = 0x27; /* initial value */
+ int i, j;
+
+ for (i = 0; i < IEEE80211_ADDR_LEN; i++)
+ for (j = (i == 0) ? 1 : 0; j < 8; j++)
+ if ((maddr[i] >> j) & 1)
+ pos ^= (mask >> (i * 8 + j - 1));
+
+ pos &= 0x3f;
+
+ return (pos);
+}
+
+static void
+urtwn_set_multi(struct urtwn_softc *sc)
+{
+ struct ieee80211com *ic = &sc->sc_ic;
+ uint32_t mfilt[2];
+
+ URTWN_ASSERT_LOCKED(sc);
+
+ /* general structure was copied from ath(4). */
+ if (ic->ic_allmulti == 0) {
+ struct ieee80211vap *vap;
+ struct ifnet *ifp;
+ struct ifmultiaddr *ifma;
+
+ /*
+ * Merge multicast addresses to form the hardware filter.
+ */
+ mfilt[0] = mfilt[1] = 0;
+ TAILQ_FOREACH(vap, &ic->ic_vaps, iv_next) {
+ ifp = vap->iv_ifp;
+ if_maddr_rlock(ifp);
+ TAILQ_FOREACH(ifma, &ifp->if_multiaddrs, ifma_link) {
+ caddr_t dl;
+ uint8_t pos;
+
+ dl = LLADDR((struct sockaddr_dl *)
+ ifma->ifma_addr);
+ pos = urtwn_get_multi_pos(dl);
+
+ mfilt[pos / 32] |= (1 << (pos % 32));
+ }
+ if_maddr_runlock(ifp);
+ }
+ } else
+ mfilt[0] = mfilt[1] = ~0;
+
+
+ urtwn_write_4(sc, R92C_MAR + 0, mfilt[0]);
+ urtwn_write_4(sc, R92C_MAR + 4, mfilt[1]);
+
+ URTWN_DPRINTF(sc, URTWN_DEBUG_STATE, "%s: MC filter %08x:%08x\n",
+ __func__, mfilt[0], mfilt[1]);
+}
+
static void
urtwn_set_promisc(struct urtwn_softc *sc)
{
@@ -4877,7 +4939,12 @@ urtwn_update_promisc(struct ieee80211com *ic)
static void
urtwn_update_mcast(struct ieee80211com *ic)
{
- /* XXX do nothing? */
+ struct urtwn_softc *sc = ic->ic_softc;
+
+ URTWN_LOCK(sc);
+ if (sc->sc_flags & URTWN_RUNNING)
+ urtwn_set_multi(sc);
+ URTWN_UNLOCK(sc);
}
static struct ieee80211_node *
OpenPOWER on IntegriCloud