/*- * Copyright (c) 2016 Andriy Voskoboinyk * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF * SUCH DAMAGE. */ #include __FBSDID("$FreeBSD$"); #include "opt_wlan.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include int r12a_ioctl_net(struct ieee80211com *ic, u_long cmd, void *data) { struct rtwn_softc *sc = ic->ic_softc; struct r12a_softc *rs = sc->sc_priv; struct ifreq *ifr = (struct ifreq *)data; int error; error = 0; switch (cmd) { case SIOCSIFCAP: { struct ieee80211vap *vap; int changed, rxmask; rxmask = ifr->ifr_reqcap & (IFCAP_RXCSUM | IFCAP_RXCSUM_IPV6); RTWN_LOCK(sc); changed = 0; if (!(rs->rs_flags & R12A_RXCKSUM_EN) ^ !(ifr->ifr_reqcap & IFCAP_RXCSUM)) { rs->rs_flags ^= R12A_RXCKSUM_EN; changed = 1; } if (!(rs->rs_flags & R12A_RXCKSUM6_EN) ^ !(ifr->ifr_reqcap & IFCAP_RXCSUM_IPV6)) { rs->rs_flags ^= R12A_RXCKSUM6_EN; changed = 1; } if (changed) { if (rxmask == 0) { sc->rcr &= ~R12A_RCR_TCP_OFFLD_EN; if (sc->sc_flags & RTWN_RUNNING) { rtwn_setbits_4(sc, R92C_RCR, R12A_RCR_TCP_OFFLD_EN, 0); } } else { sc->rcr |= R12A_RCR_TCP_OFFLD_EN; if (sc->sc_flags & RTWN_RUNNING) { rtwn_setbits_4(sc, R92C_RCR, 0, R12A_RCR_TCP_OFFLD_EN); } } } RTWN_UNLOCK(sc); IEEE80211_LOCK(ic); /* XXX */ TAILQ_FOREACH(vap, &ic->ic_vaps, iv_next) { struct ifnet *ifp = vap->iv_ifp; ifp->if_capenable &= ~(IFCAP_RXCSUM | IFCAP_RXCSUM_IPV6); ifp->if_capenable |= rxmask; } IEEE80211_UNLOCK(ic); break; } default: error = ENOTTY; /* for net80211 */ break; } return (error); }