1*60b9567dSKevin Lo /*- 2*60b9567dSKevin Lo * Copyright (c) 2017 Kevin Lo <kevlo@FreeBSD.org> 3*60b9567dSKevin Lo * All rights reserved. 4*60b9567dSKevin Lo * 5*60b9567dSKevin Lo * Redistribution and use in source and binary forms, with or without 6*60b9567dSKevin Lo * modification, are permitted provided that the following conditions 7*60b9567dSKevin Lo * are met: 8*60b9567dSKevin Lo * 1. Redistributions of source code must retain the above copyright 9*60b9567dSKevin Lo * notice, this list of conditions and the following disclaimer. 10*60b9567dSKevin Lo * 2. Redistributions in binary form must reproduce the above copyright 11*60b9567dSKevin Lo * notice, this list of conditions and the following disclaimer in the 12*60b9567dSKevin Lo * documentation and/or other materials provided with the distribution. 13*60b9567dSKevin Lo * 14*60b9567dSKevin Lo * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND 15*60b9567dSKevin Lo * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 16*60b9567dSKevin Lo * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 17*60b9567dSKevin Lo * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE 18*60b9567dSKevin Lo * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 19*60b9567dSKevin Lo * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 20*60b9567dSKevin Lo * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 21*60b9567dSKevin Lo * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 22*60b9567dSKevin Lo * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 23*60b9567dSKevin Lo * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 24*60b9567dSKevin Lo * SUCH DAMAGE. 25*60b9567dSKevin Lo */ 26*60b9567dSKevin Lo 27*60b9567dSKevin Lo #include <sys/cdefs.h> 28*60b9567dSKevin Lo __FBSDID("$FreeBSD$"); 29*60b9567dSKevin Lo 30*60b9567dSKevin Lo #include "opt_wlan.h" 31*60b9567dSKevin Lo 32*60b9567dSKevin Lo #include <sys/param.h> 33*60b9567dSKevin Lo #include <sys/lock.h> 34*60b9567dSKevin Lo #include <sys/mutex.h> 35*60b9567dSKevin Lo #include <sys/mbuf.h> 36*60b9567dSKevin Lo #include <sys/kernel.h> 37*60b9567dSKevin Lo #include <sys/socket.h> 38*60b9567dSKevin Lo #include <sys/systm.h> 39*60b9567dSKevin Lo #include <sys/malloc.h> 40*60b9567dSKevin Lo #include <sys/queue.h> 41*60b9567dSKevin Lo #include <sys/taskqueue.h> 42*60b9567dSKevin Lo #include <sys/bus.h> 43*60b9567dSKevin Lo #include <sys/endian.h> 44*60b9567dSKevin Lo #include <sys/linker.h> 45*60b9567dSKevin Lo 46*60b9567dSKevin Lo #include <net/if.h> 47*60b9567dSKevin Lo #include <net/ethernet.h> 48*60b9567dSKevin Lo #include <net/if_media.h> 49*60b9567dSKevin Lo 50*60b9567dSKevin Lo #include <net80211/ieee80211_var.h> 51*60b9567dSKevin Lo #include <net80211/ieee80211_radiotap.h> 52*60b9567dSKevin Lo 53*60b9567dSKevin Lo #include <dev/rtwn/if_rtwnreg.h> 54*60b9567dSKevin Lo #include <dev/rtwn/if_rtwnvar.h> 55*60b9567dSKevin Lo 56*60b9567dSKevin Lo #include <dev/rtwn/if_rtwn_debug.h> 57*60b9567dSKevin Lo 58*60b9567dSKevin Lo #include <dev/rtwn/rtl8188e/r88e.h> 59*60b9567dSKevin Lo #include <dev/rtwn/rtl8188e/r88e_reg.h> 60*60b9567dSKevin Lo 61*60b9567dSKevin Lo #include <dev/rtwn/rtl8812a/r12a_fw_cmd.h> 62*60b9567dSKevin Lo 63*60b9567dSKevin Lo #include <dev/rtwn/rtl8192e/r92e.h> 64*60b9567dSKevin Lo 65*60b9567dSKevin Lo #ifndef RTWN_WITHOUT_UCODE 66*60b9567dSKevin Lo void 67*60b9567dSKevin Lo r92e_fw_reset(struct rtwn_softc *sc, int reason) 68*60b9567dSKevin Lo { 69*60b9567dSKevin Lo /* Reset MCU IO wrapper. */ 70*60b9567dSKevin Lo rtwn_setbits_1(sc, R92C_RSV_CTRL + 1, 0x01, 0); 71*60b9567dSKevin Lo 72*60b9567dSKevin Lo rtwn_setbits_1_shift(sc, R92C_SYS_FUNC_EN, 73*60b9567dSKevin Lo R92C_SYS_FUNC_EN_CPUEN, 0, 1); 74*60b9567dSKevin Lo 75*60b9567dSKevin Lo /* Enable MCU IO wrapper. */ 76*60b9567dSKevin Lo rtwn_setbits_1(sc, R92C_RSV_CTRL + 1, 0, 0x01); 77*60b9567dSKevin Lo 78*60b9567dSKevin Lo rtwn_setbits_1_shift(sc, R92C_SYS_FUNC_EN, 79*60b9567dSKevin Lo 0, R92C_SYS_FUNC_EN_CPUEN, 1); 80*60b9567dSKevin Lo } 81*60b9567dSKevin Lo 82*60b9567dSKevin Lo void 83*60b9567dSKevin Lo r92e_set_media_status(struct rtwn_softc *sc, int macid) 84*60b9567dSKevin Lo { 85*60b9567dSKevin Lo struct r88e_fw_cmd_msrrpt status; 86*60b9567dSKevin Lo 87*60b9567dSKevin Lo if (macid & RTWN_MACID_VALID) 88*60b9567dSKevin Lo status.msrb0 = R88E_MSRRPT_B0_ASSOC; 89*60b9567dSKevin Lo else 90*60b9567dSKevin Lo status.msrb0 = R88E_MSRRPT_B0_DISASSOC; 91*60b9567dSKevin Lo status.macid = (macid & ~RTWN_MACID_VALID); 92*60b9567dSKevin Lo 93*60b9567dSKevin Lo if (r88e_fw_cmd(sc, R88E_CMD_MSR_RPT, &status, sizeof(status)) != 0) { 94*60b9567dSKevin Lo device_printf(sc->sc_dev, "%s: cannot change media status!\n", 95*60b9567dSKevin Lo __func__); 96*60b9567dSKevin Lo } 97*60b9567dSKevin Lo } 98*60b9567dSKevin Lo 99*60b9567dSKevin Lo int 100*60b9567dSKevin Lo r92e_set_pwrmode(struct rtwn_softc *sc, struct ieee80211vap *vap, int off) 101*60b9567dSKevin Lo { 102*60b9567dSKevin Lo struct r12a_fw_cmd_pwrmode mode; 103*60b9567dSKevin Lo int error; 104*60b9567dSKevin Lo 105*60b9567dSKevin Lo if (off && vap->iv_state == IEEE80211_S_RUN && 106*60b9567dSKevin Lo (vap->iv_flags & IEEE80211_F_PMGTON)) { 107*60b9567dSKevin Lo mode.mode = R88E_PWRMODE_LEG; 108*60b9567dSKevin Lo /* 109*60b9567dSKevin Lo * TODO: switch to RFOFF state 110*60b9567dSKevin Lo * (something is missing here - Rx stops with it). 111*60b9567dSKevin Lo */ 112*60b9567dSKevin Lo #ifdef RTWN_TODO 113*60b9567dSKevin Lo mode.pwr_state = R88E_PWRMODE_STATE_RFOFF; 114*60b9567dSKevin Lo #else 115*60b9567dSKevin Lo mode.pwr_state = R88E_PWRMODE_STATE_RFON; 116*60b9567dSKevin Lo #endif 117*60b9567dSKevin Lo } else { 118*60b9567dSKevin Lo mode.mode = R88E_PWRMODE_CAM; 119*60b9567dSKevin Lo mode.pwr_state = R88E_PWRMODE_STATE_ALLON; 120*60b9567dSKevin Lo } 121*60b9567dSKevin Lo mode.pwrb1 = 122*60b9567dSKevin Lo SM(R88E_PWRMODE_B1_SMART_PS, R88E_PWRMODE_B1_LEG_NULLDATA) | 123*60b9567dSKevin Lo SM(R88E_PWRMODE_B1_RLBM, R88E_PWRMODE_B1_MODE_MIN); 124*60b9567dSKevin Lo /* XXX ignored */ 125*60b9567dSKevin Lo mode.bcn_pass = 0; 126*60b9567dSKevin Lo mode.queue_uapsd = 0; 127*60b9567dSKevin Lo mode.pwrb5 = 0; 128*60b9567dSKevin Lo error = r88e_fw_cmd(sc, R88E_CMD_SET_PWRMODE, &mode, sizeof(mode)); 129*60b9567dSKevin Lo if (error != 0) { 130*60b9567dSKevin Lo device_printf(sc->sc_dev, 131*60b9567dSKevin Lo "%s: CMD_SET_PWRMODE was not sent, error %d\n", 132*60b9567dSKevin Lo __func__, error); 133*60b9567dSKevin Lo } 134*60b9567dSKevin Lo 135*60b9567dSKevin Lo return (error); 136*60b9567dSKevin Lo } 137*60b9567dSKevin Lo #endif 138