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/rtl8192e/r92e.h> 59*60b9567dSKevin Lo #include <dev/rtwn/rtl8192e/r92e_var.h> 60*60b9567dSKevin Lo #include <dev/rtwn/rtl8192e/r92e_rom_image.h> 61*60b9567dSKevin Lo 62*60b9567dSKevin Lo void 63*60b9567dSKevin Lo r92e_parse_rom(struct rtwn_softc *sc, uint8_t *buf) 64*60b9567dSKevin Lo { 65*60b9567dSKevin Lo struct r92e_softc *rs = sc->sc_priv; 66*60b9567dSKevin Lo struct r92e_rom *rom = (struct r92e_rom *)buf; 67*60b9567dSKevin Lo uint8_t pwr_diff; 68*60b9567dSKevin Lo int i, j, k; 69*60b9567dSKevin Lo 70*60b9567dSKevin Lo sc->thermal_meter = rom->thermal_meter; 71*60b9567dSKevin Lo rs->crystalcap = RTWN_GET_ROM_VAR(rom->crystalcap, 72*60b9567dSKevin Lo R92E_ROM_CRYSTALCAP_DEF); 73*60b9567dSKevin Lo 74*60b9567dSKevin Lo for (i = 0; i < sc->ntxchains; i++) { 75*60b9567dSKevin Lo struct r92e_tx_pwr_2g *pwr_2g = &rom->tx_pwr[i].pwr_2g; 76*60b9567dSKevin Lo struct r92e_tx_pwr_diff_2g *pwr_diff_2g = 77*60b9567dSKevin Lo &rom->tx_pwr[i].pwr_diff_2g; 78*60b9567dSKevin Lo 79*60b9567dSKevin Lo for (j = 0; j < R92E_GROUP_2G - 1; j++) { 80*60b9567dSKevin Lo rs->cck_tx_pwr[i][j] = 81*60b9567dSKevin Lo RTWN_GET_ROM_VAR(pwr_2g->cck[j], 82*60b9567dSKevin Lo R92E_DEF_TX_PWR_2G); 83*60b9567dSKevin Lo rs->ht40_tx_pwr_2g[i][j] = 84*60b9567dSKevin Lo RTWN_GET_ROM_VAR(pwr_2g->ht40[j], 85*60b9567dSKevin Lo R92E_DEF_TX_PWR_2G); 86*60b9567dSKevin Lo } 87*60b9567dSKevin Lo rs->cck_tx_pwr[i][j] = RTWN_GET_ROM_VAR(pwr_2g->cck[j], 88*60b9567dSKevin Lo R92E_DEF_TX_PWR_2G); 89*60b9567dSKevin Lo 90*60b9567dSKevin Lo rs->cck_tx_pwr_diff_2g[i][0] = 0; 91*60b9567dSKevin Lo rs->ofdm_tx_pwr_diff_2g[i][0] = RTWN_SIGN4TO8( 92*60b9567dSKevin Lo MS(pwr_diff_2g->ht20_ofdm, LOW_PART)); 93*60b9567dSKevin Lo rs->bw20_tx_pwr_diff_2g[i][0] = RTWN_SIGN4TO8( 94*60b9567dSKevin Lo MS(pwr_diff_2g->ht20_ofdm, HIGH_PART)); 95*60b9567dSKevin Lo rs->bw40_tx_pwr_diff_2g[i][0] = 0; 96*60b9567dSKevin Lo pwr_diff = RTWN_GET_ROM_VAR(pwr_diff_2g->ht20_ofdm, 97*60b9567dSKevin Lo R92E_DEF_TX_PWR_HT20_DIFF); 98*60b9567dSKevin Lo if (pwr_diff != R92E_DEF_TX_PWR_HT20_DIFF) { 99*60b9567dSKevin Lo rs->ofdm_tx_pwr_diff_2g[i][0] = RTWN_SIGN4TO8( 100*60b9567dSKevin Lo MS(pwr_diff_2g->ht20_ofdm, LOW_PART)); 101*60b9567dSKevin Lo rs->bw20_tx_pwr_diff_2g[i][0] = RTWN_SIGN4TO8( 102*60b9567dSKevin Lo MS(pwr_diff_2g->ht20_ofdm, HIGH_PART)); 103*60b9567dSKevin Lo } else { 104*60b9567dSKevin Lo rs->ofdm_tx_pwr_diff_2g[i][0] = 105*60b9567dSKevin Lo rs->bw20_tx_pwr_diff_2g[i][0] = pwr_diff; 106*60b9567dSKevin Lo } 107*60b9567dSKevin Lo 108*60b9567dSKevin Lo for (j = 1, k = 0; k < nitems(pwr_diff_2g->diff123); j++, k++) { 109*60b9567dSKevin Lo pwr_diff = RTWN_GET_ROM_VAR( 110*60b9567dSKevin Lo pwr_diff_2g->diff123[k].ofdm_cck, 111*60b9567dSKevin Lo R92E_DEF_TX_PWR_DIFF); 112*60b9567dSKevin Lo if (pwr_diff != R92E_DEF_TX_PWR_DIFF) { 113*60b9567dSKevin Lo rs->cck_tx_pwr_diff_2g[i][j] = RTWN_SIGN4TO8( 114*60b9567dSKevin Lo MS(pwr_diff_2g->diff123[k].ofdm_cck, 115*60b9567dSKevin Lo LOW_PART)); 116*60b9567dSKevin Lo rs->ofdm_tx_pwr_diff_2g[i][j] = RTWN_SIGN4TO8( 117*60b9567dSKevin Lo MS(pwr_diff_2g->diff123[k].ofdm_cck, 118*60b9567dSKevin Lo HIGH_PART)); 119*60b9567dSKevin Lo } else { 120*60b9567dSKevin Lo rs->cck_tx_pwr_diff_2g[i][j] = 121*60b9567dSKevin Lo rs->ofdm_tx_pwr_diff_2g[i][j] = pwr_diff; 122*60b9567dSKevin Lo } 123*60b9567dSKevin Lo pwr_diff = RTWN_GET_ROM_VAR( 124*60b9567dSKevin Lo pwr_diff_2g->diff123[k].ht40_ht20, 125*60b9567dSKevin Lo R92E_DEF_TX_PWR_DIFF); 126*60b9567dSKevin Lo if (pwr_diff != R92E_DEF_TX_PWR_DIFF) { 127*60b9567dSKevin Lo rs->bw20_tx_pwr_diff_2g[i][j] = RTWN_SIGN4TO8( 128*60b9567dSKevin Lo MS(pwr_diff_2g->diff123[k].ht40_ht20, 129*60b9567dSKevin Lo LOW_PART)); 130*60b9567dSKevin Lo rs->bw40_tx_pwr_diff_2g[i][j] = RTWN_SIGN4TO8( 131*60b9567dSKevin Lo MS(pwr_diff_2g->diff123[k].ht40_ht20, 132*60b9567dSKevin Lo HIGH_PART)); 133*60b9567dSKevin Lo } else { 134*60b9567dSKevin Lo rs->bw20_tx_pwr_diff_2g[i][j] = 135*60b9567dSKevin Lo rs->bw40_tx_pwr_diff_2g[i][j] = pwr_diff; 136*60b9567dSKevin Lo } 137*60b9567dSKevin Lo } 138*60b9567dSKevin Lo } 139*60b9567dSKevin Lo 140*60b9567dSKevin Lo rs->regulatory = MS(rom->rf_board_opt, R92C_ROM_RF1_REGULATORY); 141*60b9567dSKevin Lo 142*60b9567dSKevin Lo /* Read MAC address. */ 143*60b9567dSKevin Lo IEEE80211_ADDR_COPY(sc->sc_ic.ic_macaddr, rom->macaddr); 144*60b9567dSKevin Lo } 145