17453645fSAndriy Voskoboinyk /* $OpenBSD: if_urtwn.c,v 1.16 2011/02/10 17:26:40 jakemsr Exp $ */ 27453645fSAndriy Voskoboinyk 37453645fSAndriy Voskoboinyk /*- 47453645fSAndriy Voskoboinyk * Copyright (c) 2010 Damien Bergamini <damien.bergamini@free.fr> 57453645fSAndriy Voskoboinyk * Copyright (c) 2014 Kevin Lo <kevlo@FreeBSD.org> 67453645fSAndriy Voskoboinyk * Copyright (c) 2015-2016 Andriy Voskoboinyk <avos@FreeBSD.org> 77453645fSAndriy Voskoboinyk * 87453645fSAndriy Voskoboinyk * Permission to use, copy, modify, and distribute this software for any 97453645fSAndriy Voskoboinyk * purpose with or without fee is hereby granted, provided that the above 107453645fSAndriy Voskoboinyk * copyright notice and this permission notice appear in all copies. 117453645fSAndriy Voskoboinyk * 127453645fSAndriy Voskoboinyk * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES 137453645fSAndriy Voskoboinyk * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF 147453645fSAndriy Voskoboinyk * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR 157453645fSAndriy Voskoboinyk * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES 167453645fSAndriy Voskoboinyk * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN 177453645fSAndriy Voskoboinyk * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF 187453645fSAndriy Voskoboinyk * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. 197453645fSAndriy Voskoboinyk */ 207453645fSAndriy Voskoboinyk 217453645fSAndriy Voskoboinyk #include <sys/cdefs.h> 227453645fSAndriy Voskoboinyk __FBSDID("$FreeBSD$"); 237453645fSAndriy Voskoboinyk 247453645fSAndriy Voskoboinyk #include "opt_wlan.h" 257453645fSAndriy Voskoboinyk 267453645fSAndriy Voskoboinyk #include <sys/param.h> 277453645fSAndriy Voskoboinyk #include <sys/lock.h> 287453645fSAndriy Voskoboinyk #include <sys/mutex.h> 297453645fSAndriy Voskoboinyk #include <sys/mbuf.h> 307453645fSAndriy Voskoboinyk #include <sys/kernel.h> 317453645fSAndriy Voskoboinyk #include <sys/socket.h> 327453645fSAndriy Voskoboinyk #include <sys/systm.h> 337453645fSAndriy Voskoboinyk #include <sys/malloc.h> 347453645fSAndriy Voskoboinyk #include <sys/queue.h> 357453645fSAndriy Voskoboinyk #include <sys/taskqueue.h> 367453645fSAndriy Voskoboinyk #include <sys/bus.h> 377453645fSAndriy Voskoboinyk #include <sys/endian.h> 387453645fSAndriy Voskoboinyk #include <sys/linker.h> 397453645fSAndriy Voskoboinyk 407453645fSAndriy Voskoboinyk #include <net/if.h> 417453645fSAndriy Voskoboinyk #include <net/ethernet.h> 427453645fSAndriy Voskoboinyk #include <net/if_media.h> 437453645fSAndriy Voskoboinyk 447453645fSAndriy Voskoboinyk #include <net80211/ieee80211_var.h> 457453645fSAndriy Voskoboinyk #include <net80211/ieee80211_radiotap.h> 467453645fSAndriy Voskoboinyk 477453645fSAndriy Voskoboinyk #include <dev/rtwn/if_rtwnreg.h> 487453645fSAndriy Voskoboinyk #include <dev/rtwn/if_rtwnvar.h> 497453645fSAndriy Voskoboinyk 507453645fSAndriy Voskoboinyk #include <dev/rtwn/if_rtwn_debug.h> 517453645fSAndriy Voskoboinyk #include <dev/rtwn/if_rtwn_ridx.h> 527453645fSAndriy Voskoboinyk 537453645fSAndriy Voskoboinyk #include <dev/rtwn/rtl8192c/r92c_var.h> 547453645fSAndriy Voskoboinyk 557453645fSAndriy Voskoboinyk #include <dev/rtwn/rtl8188e/r88e.h> 567453645fSAndriy Voskoboinyk #include <dev/rtwn/rtl8188e/r88e_priv.h> 577453645fSAndriy Voskoboinyk #include <dev/rtwn/rtl8188e/r88e_reg.h> 587453645fSAndriy Voskoboinyk 597453645fSAndriy Voskoboinyk 607453645fSAndriy Voskoboinyk static int 617453645fSAndriy Voskoboinyk r88e_get_power_group(struct rtwn_softc *sc, struct ieee80211_channel *c) 627453645fSAndriy Voskoboinyk { 637453645fSAndriy Voskoboinyk uint8_t chan; 647453645fSAndriy Voskoboinyk int group; 657453645fSAndriy Voskoboinyk 667453645fSAndriy Voskoboinyk chan = rtwn_chan2centieee(c); 677453645fSAndriy Voskoboinyk if (IEEE80211_IS_CHAN_2GHZ(c)) { 687453645fSAndriy Voskoboinyk if (chan <= 2) group = 0; 697453645fSAndriy Voskoboinyk else if (chan <= 5) group = 1; 707453645fSAndriy Voskoboinyk else if (chan <= 8) group = 2; 717453645fSAndriy Voskoboinyk else if (chan <= 11) group = 3; 727453645fSAndriy Voskoboinyk else if (chan <= 13) group = 4; 737453645fSAndriy Voskoboinyk else if (chan <= 14) group = 5; 747453645fSAndriy Voskoboinyk else { 757453645fSAndriy Voskoboinyk KASSERT(0, ("wrong 2GHz channel %d!\n", chan)); 767453645fSAndriy Voskoboinyk return (-1); 777453645fSAndriy Voskoboinyk } 787453645fSAndriy Voskoboinyk } else { 797453645fSAndriy Voskoboinyk KASSERT(0, ("wrong channel band (flags %08X)\n", c->ic_flags)); 807453645fSAndriy Voskoboinyk return (-1); 817453645fSAndriy Voskoboinyk } 827453645fSAndriy Voskoboinyk 837453645fSAndriy Voskoboinyk return (group); 847453645fSAndriy Voskoboinyk } 857453645fSAndriy Voskoboinyk 867453645fSAndriy Voskoboinyk void 877453645fSAndriy Voskoboinyk r88e_get_txpower(struct rtwn_softc *sc, int chain, 885c7083ceSAndriy Voskoboinyk struct ieee80211_channel *c, uint8_t power[RTWN_RIDX_COUNT]) 897453645fSAndriy Voskoboinyk { 907453645fSAndriy Voskoboinyk struct r92c_softc *rs = sc->sc_priv; 917453645fSAndriy Voskoboinyk const struct rtwn_r88e_txpwr *rt = rs->rs_txpwr; 922cd32557SKevin Lo uint8_t cckpow, ofdmpow, bw20pow, htpow = 0; 937453645fSAndriy Voskoboinyk int max_mcs, ridx, group; 947453645fSAndriy Voskoboinyk 957453645fSAndriy Voskoboinyk /* Determine channel group. */ 967453645fSAndriy Voskoboinyk group = r88e_get_power_group(sc, c); 977453645fSAndriy Voskoboinyk if (group == -1) { /* shouldn't happen */ 987453645fSAndriy Voskoboinyk device_printf(sc->sc_dev, "%s: incorrect channel\n", __func__); 997453645fSAndriy Voskoboinyk return; 1007453645fSAndriy Voskoboinyk } 1017453645fSAndriy Voskoboinyk 1027453645fSAndriy Voskoboinyk /* XXX net80211 regulatory */ 1037453645fSAndriy Voskoboinyk 104*0cc18edfSAndriy Voskoboinyk max_mcs = RTWN_RIDX_HT_MCS(sc->ntxchains * 8 - 1); 1057453645fSAndriy Voskoboinyk KASSERT(max_mcs <= RTWN_RIDX_COUNT, ("increase ridx limit\n")); 1067453645fSAndriy Voskoboinyk 1077453645fSAndriy Voskoboinyk memset(power, 0, max_mcs * sizeof(power[0])); 1087453645fSAndriy Voskoboinyk 1097453645fSAndriy Voskoboinyk /* Compute per-CCK rate Tx power. */ 1107453645fSAndriy Voskoboinyk cckpow = rt->cck_tx_pwr[group]; 1112cd32557SKevin Lo for (ridx = RTWN_RIDX_CCK1; ridx <= RTWN_RIDX_CCK11; ridx++) { 1122cd32557SKevin Lo power[ridx] = (ridx == RTWN_RIDX_CCK2) ? cckpow - 9 : cckpow; 1132cd32557SKevin Lo } 1147453645fSAndriy Voskoboinyk 1152cd32557SKevin Lo if (group < 5) 1167453645fSAndriy Voskoboinyk htpow = rt->ht40_tx_pwr[group]; 1177453645fSAndriy Voskoboinyk 1187453645fSAndriy Voskoboinyk /* Compute per-OFDM rate Tx power. */ 1197453645fSAndriy Voskoboinyk ofdmpow = htpow + rt->ofdm_tx_pwr_diff; 1207453645fSAndriy Voskoboinyk for (ridx = RTWN_RIDX_OFDM6; ridx <= RTWN_RIDX_OFDM54; ridx++) 1212cd32557SKevin Lo power[ridx] = ofdmpow; 1227453645fSAndriy Voskoboinyk 1237453645fSAndriy Voskoboinyk bw20pow = htpow + rt->bw20_tx_pwr_diff; 124*0cc18edfSAndriy Voskoboinyk for (ridx = RTWN_RIDX_HT_MCS(0); ridx <= max_mcs; ridx++) 1252cd32557SKevin Lo power[ridx] = bw20pow; 1267453645fSAndriy Voskoboinyk 1277453645fSAndriy Voskoboinyk /* Apply max limit. */ 1287453645fSAndriy Voskoboinyk for (ridx = RTWN_RIDX_CCK1; ridx <= max_mcs; ridx++) { 1297453645fSAndriy Voskoboinyk if (power[ridx] > R92C_MAX_TX_PWR) 1307453645fSAndriy Voskoboinyk power[ridx] = R92C_MAX_TX_PWR; 1317453645fSAndriy Voskoboinyk } 1327453645fSAndriy Voskoboinyk } 1337453645fSAndriy Voskoboinyk 1347453645fSAndriy Voskoboinyk void 1357453645fSAndriy Voskoboinyk r88e_set_bw20(struct rtwn_softc *sc, uint8_t chan) 1367453645fSAndriy Voskoboinyk { 1377453645fSAndriy Voskoboinyk struct r92c_softc *rs = sc->sc_priv; 1387453645fSAndriy Voskoboinyk 1397453645fSAndriy Voskoboinyk rtwn_setbits_1(sc, R92C_BWOPMODE, 0, R92C_BWOPMODE_20MHZ); 1407453645fSAndriy Voskoboinyk 1417453645fSAndriy Voskoboinyk rtwn_bb_setbits(sc, R92C_FPGA0_RFMOD, R92C_RFMOD_40MHZ, 0); 1427453645fSAndriy Voskoboinyk rtwn_bb_setbits(sc, R92C_FPGA1_RFMOD, R92C_RFMOD_40MHZ, 0); 1437453645fSAndriy Voskoboinyk 1447453645fSAndriy Voskoboinyk /* Select 20MHz bandwidth. */ 1457453645fSAndriy Voskoboinyk rtwn_rf_write(sc, 0, R92C_RF_CHNLBW, 1467453645fSAndriy Voskoboinyk (rs->rf_chnlbw[0] & ~0xfff) | chan | R88E_RF_CHNLBW_BW20); 1477453645fSAndriy Voskoboinyk } 1487453645fSAndriy Voskoboinyk 1497453645fSAndriy Voskoboinyk void 1507453645fSAndriy Voskoboinyk r88e_set_gain(struct rtwn_softc *sc, uint8_t gain) 1517453645fSAndriy Voskoboinyk { 1527453645fSAndriy Voskoboinyk 1537453645fSAndriy Voskoboinyk rtwn_bb_setbits(sc, R92C_OFDM0_AGCCORE1(0), 1547453645fSAndriy Voskoboinyk R92C_OFDM0_AGCCORE1_GAIN_M, gain); 1557453645fSAndriy Voskoboinyk } 156