---- a/drivers/net/wireless/ath/ath9k/ar9001_initvals.h
-+++ b/drivers/net/wireless/ath/ath9k/ar9001_initvals.h
-@@ -459,97 +459,6 @@ static const u32 ar5416Common_9100[][2]
- {0x0000a3e0, 0x000001ce},
- };
+--- a/drivers/net/wireless/ath/ath5k/base.c
++++ b/drivers/net/wireless/ath/ath5k/base.c
+@@ -2416,6 +2416,22 @@ ath5k_tx_complete_poll_work(struct work_
+ * Initialization routines *
+ \*************************/
+
++static const struct ieee80211_iface_limit if_limits[] = {
++ { .max = 2048, .types = BIT(NL80211_IFTYPE_STATION) },
++ { .max = 4, .types =
++#ifdef CONFIG_MAC80211_MESH
++ BIT(NL80211_IFTYPE_MESH_POINT) |
++#endif
++ BIT(NL80211_IFTYPE_AP) },
++};
++
++static const struct ieee80211_iface_combination if_comb = {
++ .limits = if_limits,
++ .n_limits = ARRAY_SIZE(if_limits),
++ .max_interfaces = 2048,
++ .num_different_channels = 1,
++};
++
+ int __devinit
+ ath5k_init_ah(struct ath5k_hw *ah, const struct ath_bus_ops *bus_ops)
+ {
+@@ -2437,6 +2453,9 @@ ath5k_init_ah(struct ath5k_hw *ah, const
+ BIT(NL80211_IFTYPE_ADHOC) |
+ BIT(NL80211_IFTYPE_MESH_POINT);
--static const u32 ar5416Bank0_9100[][2] = {
-- /* Addr allmodes */
-- {0x000098b0, 0x1e5795e5},
-- {0x000098e0, 0x02008020},
--};
--
--static const u32 ar5416BB_RfGain_9100[][3] = {
-- /* Addr 5G_HT20 5G_HT40 */
-- {0x00009a00, 0x00000000, 0x00000000},
-- {0x00009a04, 0x00000040, 0x00000040},
-- {0x00009a08, 0x00000080, 0x00000080},
-- {0x00009a0c, 0x000001a1, 0x00000141},
-- {0x00009a10, 0x000001e1, 0x00000181},
-- {0x00009a14, 0x00000021, 0x000001c1},
-- {0x00009a18, 0x00000061, 0x00000001},
-- {0x00009a1c, 0x00000168, 0x00000041},
-- {0x00009a20, 0x000001a8, 0x000001a8},
-- {0x00009a24, 0x000001e8, 0x000001e8},
-- {0x00009a28, 0x00000028, 0x00000028},
-- {0x00009a2c, 0x00000068, 0x00000068},
-- {0x00009a30, 0x00000189, 0x000000a8},
-- {0x00009a34, 0x000001c9, 0x00000169},
-- {0x00009a38, 0x00000009, 0x000001a9},
-- {0x00009a3c, 0x00000049, 0x000001e9},
-- {0x00009a40, 0x00000089, 0x00000029},
-- {0x00009a44, 0x00000170, 0x00000069},
-- {0x00009a48, 0x000001b0, 0x00000190},
-- {0x00009a4c, 0x000001f0, 0x000001d0},
-- {0x00009a50, 0x00000030, 0x00000010},
-- {0x00009a54, 0x00000070, 0x00000050},
-- {0x00009a58, 0x00000191, 0x00000090},
-- {0x00009a5c, 0x000001d1, 0x00000151},
-- {0x00009a60, 0x00000011, 0x00000191},
-- {0x00009a64, 0x00000051, 0x000001d1},
-- {0x00009a68, 0x00000091, 0x00000011},
-- {0x00009a6c, 0x000001b8, 0x00000051},
-- {0x00009a70, 0x000001f8, 0x00000198},
-- {0x00009a74, 0x00000038, 0x000001d8},
-- {0x00009a78, 0x00000078, 0x00000018},
-- {0x00009a7c, 0x00000199, 0x00000058},
-- {0x00009a80, 0x000001d9, 0x00000098},
-- {0x00009a84, 0x00000019, 0x00000159},
-- {0x00009a88, 0x00000059, 0x00000199},
-- {0x00009a8c, 0x00000099, 0x000001d9},
-- {0x00009a90, 0x000000d9, 0x00000019},
-- {0x00009a94, 0x000000f9, 0x00000059},
-- {0x00009a98, 0x000000f9, 0x00000099},
-- {0x00009a9c, 0x000000f9, 0x000000d9},
-- {0x00009aa0, 0x000000f9, 0x000000f9},
-- {0x00009aa4, 0x000000f9, 0x000000f9},
-- {0x00009aa8, 0x000000f9, 0x000000f9},
-- {0x00009aac, 0x000000f9, 0x000000f9},
-- {0x00009ab0, 0x000000f9, 0x000000f9},
-- {0x00009ab4, 0x000000f9, 0x000000f9},
-- {0x00009ab8, 0x000000f9, 0x000000f9},
-- {0x00009abc, 0x000000f9, 0x000000f9},
-- {0x00009ac0, 0x000000f9, 0x000000f9},
-- {0x00009ac4, 0x000000f9, 0x000000f9},
-- {0x00009ac8, 0x000000f9, 0x000000f9},
-- {0x00009acc, 0x000000f9, 0x000000f9},
-- {0x00009ad0, 0x000000f9, 0x000000f9},
-- {0x00009ad4, 0x000000f9, 0x000000f9},
-- {0x00009ad8, 0x000000f9, 0x000000f9},
-- {0x00009adc, 0x000000f9, 0x000000f9},
-- {0x00009ae0, 0x000000f9, 0x000000f9},
-- {0x00009ae4, 0x000000f9, 0x000000f9},
-- {0x00009ae8, 0x000000f9, 0x000000f9},
-- {0x00009aec, 0x000000f9, 0x000000f9},
-- {0x00009af0, 0x000000f9, 0x000000f9},
-- {0x00009af4, 0x000000f9, 0x000000f9},
-- {0x00009af8, 0x000000f9, 0x000000f9},
-- {0x00009afc, 0x000000f9, 0x000000f9},
--};
--
--static const u32 ar5416Bank1_9100[][2] = {
-- /* Addr allmodes */
-- {0x000098b0, 0x02108421},
-- {0x000098ec, 0x00000008},
--};
--
--static const u32 ar5416Bank2_9100[][2] = {
-- /* Addr allmodes */
-- {0x000098b0, 0x0e73ff17},
-- {0x000098e0, 0x00000420},
--};
--
--static const u32 ar5416Bank3_9100[][3] = {
-- /* Addr 5G_HT20 5G_HT40 */
-- {0x000098f0, 0x01400018, 0x01c00018},
--};
++ hw->wiphy->iface_combinations = &if_comb;
++ hw->wiphy->n_iface_combinations = 1;
++
+ /* SW support for IBSS_RSN is provided by mac80211 */
+ hw->wiphy->flags |= WIPHY_FLAG_IBSS_RSN;
+
+--- a/drivers/net/wireless/ath/ath9k/ar5008_phy.c
++++ b/drivers/net/wireless/ath/ath9k/ar5008_phy.c
+@@ -618,19 +618,10 @@ static void ar5008_hw_init_bb(struct ath
+ u32 synthDelay;
+
+ synthDelay = REG_READ(ah, AR_PHY_RX_DELAY) & AR_PHY_RX_DELAY_DELAY;
+- if (IS_CHAN_B(chan))
+- synthDelay = (4 * synthDelay) / 22;
+- else
+- synthDelay /= 10;
-
- static const u32 ar5416Bank6_9100[][3] = {
- /* Addr 5G_HT20 5G_HT40 */
- {0x0000989c, 0x00000000, 0x00000000},
-@@ -624,13 +533,6 @@ static const u32 ar5416Bank6TPC_9100[][3
- {0x000098d0, 0x0000000f, 0x0010000f},
- };
+- if (IS_CHAN_HALF_RATE(chan))
+- synthDelay *= 2;
+- else if (IS_CHAN_QUARTER_RATE(chan))
+- synthDelay *= 4;
--static const u32 ar5416Bank7_9100[][2] = {
-- /* Addr allmodes */
-- {0x0000989c, 0x00000500},
-- {0x0000989c, 0x00000800},
-- {0x000098cc, 0x0000000e},
--};
--
- static const u32 ar5416Addac_9100[][2] = {
- /* Addr allmodes */
- {0x0000989c, 0x00000000},
-@@ -1113,178 +1015,6 @@ static const u32 ar5416Common_9160[][2]
- {0x0000a3e0, 0x000001ce},
- };
+ REG_WRITE(ah, AR_PHY_ACTIVE, AR_PHY_ACTIVE_EN);
--static const u32 ar5416Bank0_9160[][2] = {
-- /* Addr allmodes */
-- {0x000098b0, 0x1e5795e5},
-- {0x000098e0, 0x02008020},
--};
--
--static const u32 ar5416BB_RfGain_9160[][3] = {
-- /* Addr 5G_HT20 5G_HT40 */
-- {0x00009a00, 0x00000000, 0x00000000},
-- {0x00009a04, 0x00000040, 0x00000040},
-- {0x00009a08, 0x00000080, 0x00000080},
-- {0x00009a0c, 0x000001a1, 0x00000141},
-- {0x00009a10, 0x000001e1, 0x00000181},
-- {0x00009a14, 0x00000021, 0x000001c1},
-- {0x00009a18, 0x00000061, 0x00000001},
-- {0x00009a1c, 0x00000168, 0x00000041},
-- {0x00009a20, 0x000001a8, 0x000001a8},
-- {0x00009a24, 0x000001e8, 0x000001e8},
-- {0x00009a28, 0x00000028, 0x00000028},
-- {0x00009a2c, 0x00000068, 0x00000068},
-- {0x00009a30, 0x00000189, 0x000000a8},
-- {0x00009a34, 0x000001c9, 0x00000169},
-- {0x00009a38, 0x00000009, 0x000001a9},
-- {0x00009a3c, 0x00000049, 0x000001e9},
-- {0x00009a40, 0x00000089, 0x00000029},
-- {0x00009a44, 0x00000170, 0x00000069},
-- {0x00009a48, 0x000001b0, 0x00000190},
-- {0x00009a4c, 0x000001f0, 0x000001d0},
-- {0x00009a50, 0x00000030, 0x00000010},
-- {0x00009a54, 0x00000070, 0x00000050},
-- {0x00009a58, 0x00000191, 0x00000090},
-- {0x00009a5c, 0x000001d1, 0x00000151},
-- {0x00009a60, 0x00000011, 0x00000191},
-- {0x00009a64, 0x00000051, 0x000001d1},
-- {0x00009a68, 0x00000091, 0x00000011},
-- {0x00009a6c, 0x000001b8, 0x00000051},
-- {0x00009a70, 0x000001f8, 0x00000198},
-- {0x00009a74, 0x00000038, 0x000001d8},
-- {0x00009a78, 0x00000078, 0x00000018},
-- {0x00009a7c, 0x00000199, 0x00000058},
-- {0x00009a80, 0x000001d9, 0x00000098},
-- {0x00009a84, 0x00000019, 0x00000159},
-- {0x00009a88, 0x00000059, 0x00000199},
-- {0x00009a8c, 0x00000099, 0x000001d9},
-- {0x00009a90, 0x000000d9, 0x00000019},
-- {0x00009a94, 0x000000f9, 0x00000059},
-- {0x00009a98, 0x000000f9, 0x00000099},
-- {0x00009a9c, 0x000000f9, 0x000000d9},
-- {0x00009aa0, 0x000000f9, 0x000000f9},
-- {0x00009aa4, 0x000000f9, 0x000000f9},
-- {0x00009aa8, 0x000000f9, 0x000000f9},
-- {0x00009aac, 0x000000f9, 0x000000f9},
-- {0x00009ab0, 0x000000f9, 0x000000f9},
-- {0x00009ab4, 0x000000f9, 0x000000f9},
-- {0x00009ab8, 0x000000f9, 0x000000f9},
-- {0x00009abc, 0x000000f9, 0x000000f9},
-- {0x00009ac0, 0x000000f9, 0x000000f9},
-- {0x00009ac4, 0x000000f9, 0x000000f9},
-- {0x00009ac8, 0x000000f9, 0x000000f9},
-- {0x00009acc, 0x000000f9, 0x000000f9},
-- {0x00009ad0, 0x000000f9, 0x000000f9},
-- {0x00009ad4, 0x000000f9, 0x000000f9},
-- {0x00009ad8, 0x000000f9, 0x000000f9},
-- {0x00009adc, 0x000000f9, 0x000000f9},
-- {0x00009ae0, 0x000000f9, 0x000000f9},
-- {0x00009ae4, 0x000000f9, 0x000000f9},
-- {0x00009ae8, 0x000000f9, 0x000000f9},
-- {0x00009aec, 0x000000f9, 0x000000f9},
-- {0x00009af0, 0x000000f9, 0x000000f9},
-- {0x00009af4, 0x000000f9, 0x000000f9},
-- {0x00009af8, 0x000000f9, 0x000000f9},
-- {0x00009afc, 0x000000f9, 0x000000f9},
--};
--
--static const u32 ar5416Bank1_9160[][2] = {
-- /* Addr allmodes */
-- {0x000098b0, 0x02108421},
-- {0x000098ec, 0x00000008},
--};
--
--static const u32 ar5416Bank2_9160[][2] = {
-- /* Addr allmodes */
-- {0x000098b0, 0x0e73ff17},
-- {0x000098e0, 0x00000420},
--};
--
--static const u32 ar5416Bank3_9160[][3] = {
-- /* Addr 5G_HT20 5G_HT40 */
-- {0x000098f0, 0x01400018, 0x01c00018},
--};
--
--static const u32 ar5416Bank6_9160[][3] = {
-- /* Addr 5G_HT20 5G_HT40 */
-- {0x0000989c, 0x00000000, 0x00000000},
-- {0x0000989c, 0x00000000, 0x00000000},
-- {0x0000989c, 0x00000000, 0x00000000},
-- {0x0000989c, 0x00e00000, 0x00e00000},
-- {0x0000989c, 0x005e0000, 0x005e0000},
-- {0x0000989c, 0x00120000, 0x00120000},
-- {0x0000989c, 0x00620000, 0x00620000},
-- {0x0000989c, 0x00020000, 0x00020000},
-- {0x0000989c, 0x00ff0000, 0x00ff0000},
-- {0x0000989c, 0x00ff0000, 0x00ff0000},
-- {0x0000989c, 0x00ff0000, 0x00ff0000},
-- {0x0000989c, 0x40ff0000, 0x40ff0000},
-- {0x0000989c, 0x005f0000, 0x005f0000},
-- {0x0000989c, 0x00870000, 0x00870000},
-- {0x0000989c, 0x00f90000, 0x00f90000},
-- {0x0000989c, 0x007b0000, 0x007b0000},
-- {0x0000989c, 0x00ff0000, 0x00ff0000},
-- {0x0000989c, 0x00f50000, 0x00f50000},
-- {0x0000989c, 0x00dc0000, 0x00dc0000},
-- {0x0000989c, 0x00110000, 0x00110000},
-- {0x0000989c, 0x006100a8, 0x006100a8},
-- {0x0000989c, 0x004210a2, 0x004210a2},
-- {0x0000989c, 0x0014008f, 0x0014008f},
-- {0x0000989c, 0x00c40003, 0x00c40003},
-- {0x0000989c, 0x003000f2, 0x003000f2},
-- {0x0000989c, 0x00440016, 0x00440016},
-- {0x0000989c, 0x00410040, 0x00410040},
-- {0x0000989c, 0x0001805e, 0x0001805e},
-- {0x0000989c, 0x0000c0ab, 0x0000c0ab},
-- {0x0000989c, 0x000000f1, 0x000000f1},
-- {0x0000989c, 0x00002081, 0x00002081},
-- {0x0000989c, 0x000000d4, 0x000000d4},
-- {0x000098d0, 0x0000000f, 0x0010000f},
--};
+- udelay(synthDelay + BASE_ACTIVATE_DELAY);
++ ath9k_hw_synth_delay(ah, chan, synthDelay);
+ }
+
+ static void ar5008_hw_init_chain_masks(struct ath_hw *ah)
+@@ -868,7 +859,7 @@ static int ar5008_hw_process_ini(struct
+ ar5008_hw_set_channel_regs(ah, chan);
+ ar5008_hw_init_chain_masks(ah);
+ ath9k_olc_init(ah);
+- ath9k_hw_apply_txpower(ah, chan);
++ ath9k_hw_apply_txpower(ah, chan, false);
+
+ /* Write analog registers */
+ if (!ath9k_hw_set_rf_regs(ah, chan, freqIndex)) {
+@@ -948,12 +939,8 @@ static bool ar5008_hw_rfbus_req(struct a
+ static void ar5008_hw_rfbus_done(struct ath_hw *ah)
+ {
+ u32 synthDelay = REG_READ(ah, AR_PHY_RX_DELAY) & AR_PHY_RX_DELAY_DELAY;
+- if (IS_CHAN_B(ah->curchan))
+- synthDelay = (4 * synthDelay) / 22;
+- else
+- synthDelay /= 10;
+
+- udelay(synthDelay + BASE_ACTIVATE_DELAY);
++ ath9k_hw_synth_delay(ah, ah->curchan, synthDelay);
+
+ REG_WRITE(ah, AR_PHY_RFBUS_REQ, 0);
+ }
+--- a/drivers/net/wireless/ath/ath9k/ar9003_calib.c
++++ b/drivers/net/wireless/ath/ath9k/ar9003_calib.c
+@@ -1000,10 +1000,12 @@ static bool ar9003_hw_init_cal(struct at
+ if (mci && IS_CHAN_2GHZ(chan) && run_agc_cal)
+ ar9003_mci_init_cal_req(ah, &is_reusable);
+
+- txiqcal_done = ar9003_hw_tx_iq_cal_run(ah);
+- REG_WRITE(ah, AR_PHY_ACTIVE, AR_PHY_ACTIVE_DIS);
+- udelay(5);
+- REG_WRITE(ah, AR_PHY_ACTIVE, AR_PHY_ACTIVE_EN);
++ if (!(IS_CHAN_HALF_RATE(chan) || IS_CHAN_QUARTER_RATE(chan))) {
++ txiqcal_done = ar9003_hw_tx_iq_cal_run(ah);
++ REG_WRITE(ah, AR_PHY_ACTIVE, AR_PHY_ACTIVE_DIS);
++ udelay(5);
++ REG_WRITE(ah, AR_PHY_ACTIVE, AR_PHY_ACTIVE_EN);
++ }
+
+ skip_tx_iqcal:
+ if (run_agc_cal || !(ah->ah_flags & AH_FASTCC)) {
+--- a/drivers/net/wireless/ath/ath9k/ar9003_eeprom.c
++++ b/drivers/net/wireless/ath/ath9k/ar9003_eeprom.c
+@@ -4281,18 +4281,10 @@ static int ar9003_hw_tx_power_regwrite(s
+ #undef POW_SM
+ }
+
+-static void ar9003_hw_set_target_power_eeprom(struct ath_hw *ah, u16 freq,
+- u8 *targetPowerValT2)
++static void ar9003_hw_get_legacy_target_powers(struct ath_hw *ah, u16 freq,
++ u8 *targetPowerValT2,
++ bool is2GHz)
+ {
+- /* XXX: hard code for now, need to get from eeprom struct */
+- u8 ht40PowerIncForPdadc = 0;
+- bool is2GHz = false;
+- unsigned int i = 0;
+- struct ath_common *common = ath9k_hw_common(ah);
-
--static const u32 ar5416Bank6TPC_9160[][3] = {
-- /* Addr 5G_HT20 5G_HT40 */
-- {0x0000989c, 0x00000000, 0x00000000},
-- {0x0000989c, 0x00000000, 0x00000000},
-- {0x0000989c, 0x00000000, 0x00000000},
-- {0x0000989c, 0x00e00000, 0x00e00000},
-- {0x0000989c, 0x005e0000, 0x005e0000},
-- {0x0000989c, 0x00120000, 0x00120000},
-- {0x0000989c, 0x00620000, 0x00620000},
-- {0x0000989c, 0x00020000, 0x00020000},
-- {0x0000989c, 0x00ff0000, 0x00ff0000},
-- {0x0000989c, 0x00ff0000, 0x00ff0000},
-- {0x0000989c, 0x00ff0000, 0x00ff0000},
-- {0x0000989c, 0x40ff0000, 0x40ff0000},
-- {0x0000989c, 0x005f0000, 0x005f0000},
-- {0x0000989c, 0x00870000, 0x00870000},
-- {0x0000989c, 0x00f90000, 0x00f90000},
-- {0x0000989c, 0x007b0000, 0x007b0000},
-- {0x0000989c, 0x00ff0000, 0x00ff0000},
-- {0x0000989c, 0x00f50000, 0x00f50000},
-- {0x0000989c, 0x00dc0000, 0x00dc0000},
-- {0x0000989c, 0x00110000, 0x00110000},
-- {0x0000989c, 0x006100a8, 0x006100a8},
-- {0x0000989c, 0x00423022, 0x00423022},
-- {0x0000989c, 0x2014008f, 0x2014008f},
-- {0x0000989c, 0x00c40002, 0x00c40002},
-- {0x0000989c, 0x003000f2, 0x003000f2},
-- {0x0000989c, 0x00440016, 0x00440016},
-- {0x0000989c, 0x00410040, 0x00410040},
-- {0x0000989c, 0x0001805e, 0x0001805e},
-- {0x0000989c, 0x0000c0ab, 0x0000c0ab},
-- {0x0000989c, 0x000000e1, 0x000000e1},
-- {0x0000989c, 0x00007080, 0x00007080},
-- {0x0000989c, 0x000000d4, 0x000000d4},
-- {0x000098d0, 0x0000000f, 0x0010000f},
--};
+- if (freq < 4000)
+- is2GHz = true;
-
--static const u32 ar5416Bank7_9160[][2] = {
-- /* Addr allmodes */
-- {0x0000989c, 0x00000500},
-- {0x0000989c, 0x00000800},
-- {0x000098cc, 0x0000000e},
--};
+ targetPowerValT2[ALL_TARGET_LEGACY_6_24] =
+ ar9003_hw_eeprom_get_tgt_pwr(ah, LEGACY_TARGET_RATE_6_24, freq,
+ is2GHz);
+@@ -4305,6 +4297,11 @@ static void ar9003_hw_set_target_power_e
+ targetPowerValT2[ALL_TARGET_LEGACY_54] =
+ ar9003_hw_eeprom_get_tgt_pwr(ah, LEGACY_TARGET_RATE_54, freq,
+ is2GHz);
++}
++
++static void ar9003_hw_get_cck_target_powers(struct ath_hw *ah, u16 freq,
++ u8 *targetPowerValT2)
++{
+ targetPowerValT2[ALL_TARGET_LEGACY_1L_5L] =
+ ar9003_hw_eeprom_get_cck_tgt_pwr(ah, LEGACY_TARGET_RATE_1L_5L,
+ freq);
+@@ -4314,6 +4311,11 @@ static void ar9003_hw_set_target_power_e
+ ar9003_hw_eeprom_get_cck_tgt_pwr(ah, LEGACY_TARGET_RATE_11L, freq);
+ targetPowerValT2[ALL_TARGET_LEGACY_11S] =
+ ar9003_hw_eeprom_get_cck_tgt_pwr(ah, LEGACY_TARGET_RATE_11S, freq);
++}
++
++static void ar9003_hw_get_ht20_target_powers(struct ath_hw *ah, u16 freq,
++ u8 *targetPowerValT2, bool is2GHz)
++{
+ targetPowerValT2[ALL_TARGET_HT20_0_8_16] =
+ ar9003_hw_eeprom_get_ht20_tgt_pwr(ah, HT_TARGET_RATE_0_8_16, freq,
+ is2GHz);
+@@ -4356,6 +4358,16 @@ static void ar9003_hw_set_target_power_e
+ targetPowerValT2[ALL_TARGET_HT20_23] =
+ ar9003_hw_eeprom_get_ht20_tgt_pwr(ah, HT_TARGET_RATE_23, freq,
+ is2GHz);
++}
++
++static void ar9003_hw_get_ht40_target_powers(struct ath_hw *ah,
++ u16 freq,
++ u8 *targetPowerValT2,
++ bool is2GHz)
++{
++ /* XXX: hard code for now, need to get from eeprom struct */
++ u8 ht40PowerIncForPdadc = 0;
++
+ targetPowerValT2[ALL_TARGET_HT40_0_8_16] =
+ ar9003_hw_eeprom_get_ht40_tgt_pwr(ah, HT_TARGET_RATE_0_8_16, freq,
+ is2GHz) + ht40PowerIncForPdadc;
+@@ -4399,6 +4411,26 @@ static void ar9003_hw_set_target_power_e
+ targetPowerValT2[ALL_TARGET_HT40_23] =
+ ar9003_hw_eeprom_get_ht40_tgt_pwr(ah, HT_TARGET_RATE_23, freq,
+ is2GHz) + ht40PowerIncForPdadc;
++}
++
++static void ar9003_hw_get_target_power_eeprom(struct ath_hw *ah,
++ struct ath9k_channel *chan,
++ u8 *targetPowerValT2)
++{
++ bool is2GHz = IS_CHAN_2GHZ(chan);
++ unsigned int i = 0;
++ struct ath_common *common = ath9k_hw_common(ah);
++ u16 freq = chan->channel;
++
++ if (is2GHz)
++ ar9003_hw_get_cck_target_powers(ah, freq, targetPowerValT2);
++
++ ar9003_hw_get_legacy_target_powers(ah, freq, targetPowerValT2, is2GHz);
++ ar9003_hw_get_ht20_target_powers(ah, freq, targetPowerValT2, is2GHz);
++
++ if (IS_CHAN_HT40(chan))
++ ar9003_hw_get_ht40_target_powers(ah, freq, targetPowerValT2,
++ is2GHz);
+
+ for (i = 0; i < ar9300RateSize; i++) {
+ ath_dbg(common, EEPROM, "TPC[%02d] 0x%08x\n",
+@@ -4778,9 +4810,6 @@ static void ar9003_hw_set_power_per_rate
+ scaledPower = ath9k_hw_get_scaled_power(ah, powerLimit,
+ antenna_reduction);
+
+- /*
+- * Get target powers from EEPROM - our baseline for TX Power
+- */
+ if (is2ghz) {
+ /* Setup for CTL modes */
+ /* CTL_11B, CTL_11G, CTL_2GHT20 */
+@@ -4952,7 +4981,12 @@ static void ath9k_hw_ar9300_set_txpower(
+ unsigned int i = 0, paprd_scale_factor = 0;
+ u8 pwr_idx, min_pwridx = 0;
+
+- ar9003_hw_set_target_power_eeprom(ah, chan->channel, targetPowerValT2);
++ memset(targetPowerValT2, 0 , sizeof(targetPowerValT2));
++
++ /*
++ * Get target powers from EEPROM - our baseline for TX Power
++ */
++ ar9003_hw_get_target_power_eeprom(ah, chan, targetPowerValT2);
+
+ if (ah->eep_ops->get_eeprom(ah, EEP_PAPRD)) {
+ if (IS_CHAN_2GHZ(chan))
+--- a/drivers/net/wireless/ath/ath9k/ar9003_paprd.c
++++ b/drivers/net/wireless/ath/ath9k/ar9003_paprd.c
+@@ -54,7 +54,7 @@ void ar9003_paprd_enable(struct ath_hw *
+
+ if (val) {
+ ah->paprd_table_write_done = true;
+- ath9k_hw_apply_txpower(ah, chan);
++ ath9k_hw_apply_txpower(ah, chan, false);
+ }
+
+ REG_RMW_FIELD(ah, AR_PHY_PAPRD_CTRL0_B0,
+--- a/drivers/net/wireless/ath/ath9k/ar9003_phy.c
++++ b/drivers/net/wireless/ath/ath9k/ar9003_phy.c
+@@ -373,7 +373,7 @@ static void ar9003_hw_spur_ofdm_work(str
+ else
+ spur_subchannel_sd = 0;
+
+- spur_freq_sd = (freq_offset << 9) / 11;
++ spur_freq_sd = ((freq_offset + 10) << 9) / 11;
+
+ } else {
+ if (REG_READ_FIELD(ah, AR_PHY_GEN_CTRL,
+@@ -382,7 +382,7 @@ static void ar9003_hw_spur_ofdm_work(str
+ else
+ spur_subchannel_sd = 1;
+
+- spur_freq_sd = (freq_offset << 9) / 11;
++ spur_freq_sd = ((freq_offset - 10) << 9) / 11;
+
+ }
+
+@@ -526,22 +526,10 @@ static void ar9003_hw_init_bb(struct ath
+ * Value is in 100ns increments.
+ */
+ synthDelay = REG_READ(ah, AR_PHY_RX_DELAY) & AR_PHY_RX_DELAY_DELAY;
+- if (IS_CHAN_B(chan))
+- synthDelay = (4 * synthDelay) / 22;
+- else
+- synthDelay /= 10;
+
+ /* Activate the PHY (includes baseband activate + synthesizer on) */
+ REG_WRITE(ah, AR_PHY_ACTIVE, AR_PHY_ACTIVE_EN);
-
- static const u32 ar5416Addac_9160[][2] = {
- /* Addr allmodes */
- {0x0000989c, 0x00000000},
---- a/drivers/net/wireless/ath/ath9k/ar9002_hw.c
-+++ b/drivers/net/wireless/ath/ath9k/ar9002_hw.c
-@@ -35,11 +35,11 @@ static void ar9002_hw_init_mode_regs(str
- INIT_INI_ARRAY(&ah->iniCommon, ar9271Common_9271,
- ARRAY_SIZE(ar9271Common_9271), 2);
- INIT_INI_ARRAY(&ah->iniCommon_normal_cck_fir_coeff_9271,
-- ar9271Common_normal_cck_fir_coeff_9271,
-- ARRAY_SIZE(ar9271Common_normal_cck_fir_coeff_9271), 2);
-+ ar9287Common_normal_cck_fir_coeff_9287_1_1,
-+ ARRAY_SIZE(ar9287Common_normal_cck_fir_coeff_9287_1_1), 2);
- INIT_INI_ARRAY(&ah->iniCommon_japan_2484_cck_fir_coeff_9271,
-- ar9271Common_japan_2484_cck_fir_coeff_9271,
-- ARRAY_SIZE(ar9271Common_japan_2484_cck_fir_coeff_9271), 2);
-+ ar9287Common_japan_2484_cck_fir_coeff_9287_1_1,
-+ ARRAY_SIZE(ar9287Common_japan_2484_cck_fir_coeff_9287_1_1), 2);
- INIT_INI_ARRAY(&ah->iniModes_9271_1_0_only,
- ar9271Modes_9271_1_0_only,
- ARRAY_SIZE(ar9271Modes_9271_1_0_only), 5);
-@@ -54,53 +54,31 @@ static void ar9002_hw_init_mode_regs(str
- return;
+- /*
+- * There is an issue if the AP starts the calibration before
+- * the base band timeout completes. This could result in the
+- * rx_clear false triggering. As a workaround we add delay an
+- * extra BASE_ACTIVATE_DELAY usecs to ensure this condition
+- * does not happen.
+- */
+- udelay(synthDelay + BASE_ACTIVATE_DELAY);
++ ath9k_hw_synth_delay(ah, chan, synthDelay);
+ }
+
+ static void ar9003_hw_set_chain_masks(struct ath_hw *ah, u8 rx, u8 tx)
+@@ -692,7 +680,7 @@ static int ar9003_hw_process_ini(struct
+ ar9003_hw_override_ini(ah);
+ ar9003_hw_set_channel_regs(ah, chan);
+ ar9003_hw_set_chain_masks(ah, ah->rxchainmask, ah->txchainmask);
+- ath9k_hw_apply_txpower(ah, chan);
++ ath9k_hw_apply_txpower(ah, chan, false);
+
+ if (AR_SREV_9462(ah)) {
+ if (REG_READ_FIELD(ah, AR_PHY_TX_IQCAL_CONTROL_0,
+@@ -723,6 +711,14 @@ static void ar9003_hw_set_rfmode(struct
+
+ if (IS_CHAN_A_FAST_CLOCK(ah, chan))
+ rfMode |= (AR_PHY_MODE_DYNAMIC | AR_PHY_MODE_DYN_CCK_DISABLE);
++ if (IS_CHAN_QUARTER_RATE(chan))
++ rfMode |= AR_PHY_MODE_QUARTER;
++ if (IS_CHAN_HALF_RATE(chan))
++ rfMode |= AR_PHY_MODE_HALF;
++
++ if (rfMode & (AR_PHY_MODE_QUARTER | AR_PHY_MODE_HALF))
++ REG_RMW_FIELD(ah, AR_PHY_FRAME_CTL,
++ AR_PHY_FRAME_CTL_CF_OVERLAP_WINDOW, 3);
+
+ REG_WRITE(ah, AR_PHY_MODE, rfMode);
+ }
+@@ -793,12 +789,8 @@ static bool ar9003_hw_rfbus_req(struct a
+ static void ar9003_hw_rfbus_done(struct ath_hw *ah)
+ {
+ u32 synthDelay = REG_READ(ah, AR_PHY_RX_DELAY) & AR_PHY_RX_DELAY_DELAY;
+- if (IS_CHAN_B(ah->curchan))
+- synthDelay = (4 * synthDelay) / 22;
+- else
+- synthDelay /= 10;
+
+- udelay(synthDelay + BASE_ACTIVATE_DELAY);
++ ath9k_hw_synth_delay(ah, ah->curchan, synthDelay);
+
+ REG_WRITE(ah, AR_PHY_RFBUS_REQ, 0);
+ }
+--- a/drivers/net/wireless/ath/ath9k/ar9003_phy.h
++++ b/drivers/net/wireless/ath/ath9k/ar9003_phy.h
+@@ -468,6 +468,9 @@
+ #define AR_PHY_ADDAC_PARA_CTL (AR_SM_BASE + 0x150)
+ #define AR_PHY_XPA_CFG (AR_SM_BASE + 0x158)
+
++#define AR_PHY_FRAME_CTL_CF_OVERLAP_WINDOW 3
++#define AR_PHY_FRAME_CTL_CF_OVERLAP_WINDOW_S 0
++
+ #define AR_PHY_SPUR_MASK_A_CF_PUNC_MASK_IDX_A 0x0001FC00
+ #define AR_PHY_SPUR_MASK_A_CF_PUNC_MASK_IDX_A_S 10
+ #define AR_PHY_SPUR_MASK_A_CF_PUNC_MASK_A 0x3FF
+--- a/drivers/net/wireless/ath/ath9k/eeprom_9287.c
++++ b/drivers/net/wireless/ath/ath9k/eeprom_9287.c
+@@ -798,6 +798,8 @@ static void ath9k_hw_ar9287_set_txpower(
+ regulatory->max_power_level = ratesArray[i];
}
-+ if (ah->config.pcie_clock_req)
-+ INIT_INI_ARRAY(&ah->iniPcieSerdes,
-+ ar9280PciePhy_clkreq_off_L1_9280,
-+ ARRAY_SIZE(ar9280PciePhy_clkreq_off_L1_9280), 2);
++ ath9k_hw_update_regulatory_maxpower(ah);
++
+ if (test)
+ return;
+
+--- a/drivers/net/wireless/ath/ath9k/hw.c
++++ b/drivers/net/wireless/ath/ath9k/hw.c
+@@ -191,6 +191,22 @@ bool ath9k_hw_wait(struct ath_hw *ah, u3
+ }
+ EXPORT_SYMBOL(ath9k_hw_wait);
+
++void ath9k_hw_synth_delay(struct ath_hw *ah, struct ath9k_channel *chan,
++ int hw_delay)
++{
++ if (IS_CHAN_B(chan))
++ hw_delay = (4 * hw_delay) / 22;
+ else
-+ INIT_INI_ARRAY(&ah->iniPcieSerdes,
-+ ar9280PciePhy_clkreq_always_on_L1_9280,
-+ ARRAY_SIZE(ar9280PciePhy_clkreq_always_on_L1_9280), 2);
++ hw_delay /= 10;
+
- if (AR_SREV_9287_11_OR_LATER(ah)) {
- INIT_INI_ARRAY(&ah->iniModes, ar9287Modes_9287_1_1,
- ARRAY_SIZE(ar9287Modes_9287_1_1), 5);
- INIT_INI_ARRAY(&ah->iniCommon, ar9287Common_9287_1_1,
- ARRAY_SIZE(ar9287Common_9287_1_1), 2);
-- if (ah->config.pcie_clock_req)
-- INIT_INI_ARRAY(&ah->iniPcieSerdes,
-- ar9287PciePhy_clkreq_off_L1_9287_1_1,
-- ARRAY_SIZE(ar9287PciePhy_clkreq_off_L1_9287_1_1), 2);
-- else
-- INIT_INI_ARRAY(&ah->iniPcieSerdes,
-- ar9287PciePhy_clkreq_always_on_L1_9287_1_1,
-- ARRAY_SIZE(ar9287PciePhy_clkreq_always_on_L1_9287_1_1),
-- 2);
- } else if (AR_SREV_9285_12_OR_LATER(ah)) {
--
--
- INIT_INI_ARRAY(&ah->iniModes, ar9285Modes_9285_1_2,
- ARRAY_SIZE(ar9285Modes_9285_1_2), 5);
- INIT_INI_ARRAY(&ah->iniCommon, ar9285Common_9285_1_2,
- ARRAY_SIZE(ar9285Common_9285_1_2), 2);
--
-- if (ah->config.pcie_clock_req) {
-- INIT_INI_ARRAY(&ah->iniPcieSerdes,
-- ar9285PciePhy_clkreq_off_L1_9285_1_2,
-- ARRAY_SIZE(ar9285PciePhy_clkreq_off_L1_9285_1_2), 2);
-- } else {
-- INIT_INI_ARRAY(&ah->iniPcieSerdes,
-- ar9285PciePhy_clkreq_always_on_L1_9285_1_2,
-- ARRAY_SIZE(ar9285PciePhy_clkreq_always_on_L1_9285_1_2),
-- 2);
-- }
- } else if (AR_SREV_9280_20_OR_LATER(ah)) {
- INIT_INI_ARRAY(&ah->iniModes, ar9280Modes_9280_2,
- ARRAY_SIZE(ar9280Modes_9280_2), 5);
- INIT_INI_ARRAY(&ah->iniCommon, ar9280Common_9280_2,
- ARRAY_SIZE(ar9280Common_9280_2), 2);
-
-- if (ah->config.pcie_clock_req) {
-- INIT_INI_ARRAY(&ah->iniPcieSerdes,
-- ar9280PciePhy_clkreq_off_L1_9280,
-- ARRAY_SIZE(ar9280PciePhy_clkreq_off_L1_9280), 2);
-- } else {
-- INIT_INI_ARRAY(&ah->iniPcieSerdes,
-- ar9280PciePhy_clkreq_always_on_L1_9280,
-- ARRAY_SIZE(ar9280PciePhy_clkreq_always_on_L1_9280), 2);
-- }
- INIT_INI_ARRAY(&ah->iniModesAdditional,
- ar9280Modes_fast_clock_9280_2,
- ARRAY_SIZE(ar9280Modes_fast_clock_9280_2), 3);
-@@ -109,22 +87,6 @@ static void ar9002_hw_init_mode_regs(str
- ARRAY_SIZE(ar5416Modes_9160), 5);
- INIT_INI_ARRAY(&ah->iniCommon, ar5416Common_9160,
- ARRAY_SIZE(ar5416Common_9160), 2);
-- INIT_INI_ARRAY(&ah->iniBank0, ar5416Bank0_9160,
-- ARRAY_SIZE(ar5416Bank0_9160), 2);
-- INIT_INI_ARRAY(&ah->iniBB_RfGain, ar5416BB_RfGain_9160,
-- ARRAY_SIZE(ar5416BB_RfGain_9160), 3);
-- INIT_INI_ARRAY(&ah->iniBank1, ar5416Bank1_9160,
-- ARRAY_SIZE(ar5416Bank1_9160), 2);
-- INIT_INI_ARRAY(&ah->iniBank2, ar5416Bank2_9160,
-- ARRAY_SIZE(ar5416Bank2_9160), 2);
-- INIT_INI_ARRAY(&ah->iniBank3, ar5416Bank3_9160,
-- ARRAY_SIZE(ar5416Bank3_9160), 3);
-- INIT_INI_ARRAY(&ah->iniBank6, ar5416Bank6_9160,
-- ARRAY_SIZE(ar5416Bank6_9160), 3);
-- INIT_INI_ARRAY(&ah->iniBank6TPC, ar5416Bank6TPC_9160,
-- ARRAY_SIZE(ar5416Bank6TPC_9160), 3);
-- INIT_INI_ARRAY(&ah->iniBank7, ar5416Bank7_9160,
-- ARRAY_SIZE(ar5416Bank7_9160), 2);
- if (AR_SREV_9160_11(ah)) {
- INIT_INI_ARRAY(&ah->iniAddac,
- ar5416Addac_9160_1_1,
-@@ -138,22 +100,8 @@ static void ar9002_hw_init_mode_regs(str
- ARRAY_SIZE(ar5416Modes_9100), 5);
- INIT_INI_ARRAY(&ah->iniCommon, ar5416Common_9100,
- ARRAY_SIZE(ar5416Common_9100), 2);
-- INIT_INI_ARRAY(&ah->iniBank0, ar5416Bank0_9100,
-- ARRAY_SIZE(ar5416Bank0_9100), 2);
-- INIT_INI_ARRAY(&ah->iniBB_RfGain, ar5416BB_RfGain_9100,
-- ARRAY_SIZE(ar5416BB_RfGain_9100), 3);
-- INIT_INI_ARRAY(&ah->iniBank1, ar5416Bank1_9100,
-- ARRAY_SIZE(ar5416Bank1_9100), 2);
-- INIT_INI_ARRAY(&ah->iniBank2, ar5416Bank2_9100,
-- ARRAY_SIZE(ar5416Bank2_9100), 2);
-- INIT_INI_ARRAY(&ah->iniBank3, ar5416Bank3_9100,
-- ARRAY_SIZE(ar5416Bank3_9100), 3);
- INIT_INI_ARRAY(&ah->iniBank6, ar5416Bank6_9100,
- ARRAY_SIZE(ar5416Bank6_9100), 3);
-- INIT_INI_ARRAY(&ah->iniBank6TPC, ar5416Bank6TPC_9100,
-- ARRAY_SIZE(ar5416Bank6TPC_9100), 3);
-- INIT_INI_ARRAY(&ah->iniBank7, ar5416Bank7_9100,
-- ARRAY_SIZE(ar5416Bank7_9100), 2);
- INIT_INI_ARRAY(&ah->iniAddac, ar5416Addac_9100,
- ARRAY_SIZE(ar5416Addac_9100), 2);
- } else {
-@@ -161,24 +109,37 @@ static void ar9002_hw_init_mode_regs(str
- ARRAY_SIZE(ar5416Modes), 5);
- INIT_INI_ARRAY(&ah->iniCommon, ar5416Common,
- ARRAY_SIZE(ar5416Common), 2);
-- INIT_INI_ARRAY(&ah->iniBank0, ar5416Bank0,
-- ARRAY_SIZE(ar5416Bank0), 2);
-+ INIT_INI_ARRAY(&ah->iniBank6TPC, ar5416Bank6TPC,
-+ ARRAY_SIZE(ar5416Bank6TPC), 3);
-+ INIT_INI_ARRAY(&ah->iniAddac, ar5416Addac,
-+ ARRAY_SIZE(ar5416Addac), 2);
-+ }
++ if (IS_CHAN_HALF_RATE(chan))
++ hw_delay *= 2;
++ else if (IS_CHAN_QUARTER_RATE(chan))
++ hw_delay *= 4;
+
-+ if (!AR_SREV_9280_20_OR_LATER(ah)) {
-+ /* Common for AR5416, AR913x, AR9160 */
- INIT_INI_ARRAY(&ah->iniBB_RfGain, ar5416BB_RfGain,
- ARRAY_SIZE(ar5416BB_RfGain), 3);
++ udelay(hw_delay + BASE_ACTIVATE_DELAY);
++}
+
-+ INIT_INI_ARRAY(&ah->iniBank0, ar5416Bank0,
-+ ARRAY_SIZE(ar5416Bank0), 2);
- INIT_INI_ARRAY(&ah->iniBank1, ar5416Bank1,
- ARRAY_SIZE(ar5416Bank1), 2);
- INIT_INI_ARRAY(&ah->iniBank2, ar5416Bank2,
- ARRAY_SIZE(ar5416Bank2), 2);
- INIT_INI_ARRAY(&ah->iniBank3, ar5416Bank3,
- ARRAY_SIZE(ar5416Bank3), 3);
-- INIT_INI_ARRAY(&ah->iniBank6, ar5416Bank6,
-- ARRAY_SIZE(ar5416Bank6), 3);
-- INIT_INI_ARRAY(&ah->iniBank6TPC, ar5416Bank6TPC,
-- ARRAY_SIZE(ar5416Bank6TPC), 3);
- INIT_INI_ARRAY(&ah->iniBank7, ar5416Bank7,
- ARRAY_SIZE(ar5416Bank7), 2);
-- INIT_INI_ARRAY(&ah->iniAddac, ar5416Addac,
-- ARRAY_SIZE(ar5416Addac), 2);
+ void ath9k_hw_write_array(struct ath_hw *ah, struct ar5416IniArray *array,
+ int column, unsigned int *writecnt)
+ {
+@@ -1020,7 +1036,7 @@ void ath9k_hw_init_global_settings(struc
+ struct ath_common *common = ath9k_hw_common(ah);
+ struct ieee80211_conf *conf = &common->hw->conf;
+ const struct ath9k_channel *chan = ah->curchan;
+- int acktimeout, ctstimeout;
++ int acktimeout, ctstimeout, ack_offset = 0;
+ int slottime;
+ int sifstime;
+ int rx_lat = 0, tx_lat = 0, eifs = 0;
+@@ -1041,6 +1057,11 @@ void ath9k_hw_init_global_settings(struc
+ rx_lat = 37;
+ tx_lat = 54;
+
++ if (IS_CHAN_5GHZ(chan))
++ sifstime = 16;
++ else
++ sifstime = 10;
+
-+ /* Common for AR5416, AR9160 */
-+ if (!AR_SREV_9100(ah))
-+ INIT_INI_ARRAY(&ah->iniBank6, ar5416Bank6,
-+ ARRAY_SIZE(ar5416Bank6), 3);
+ if (IS_CHAN_HALF_RATE(chan)) {
+ eifs = 175;
+ rx_lat *= 2;
+@@ -1048,8 +1069,9 @@ void ath9k_hw_init_global_settings(struc
+ if (IS_CHAN_A_FAST_CLOCK(ah, chan))
+ tx_lat += 11;
+
++ sifstime *= 2;
++ ack_offset = 16;
+ slottime = 13;
+- sifstime = 32;
+ } else if (IS_CHAN_QUARTER_RATE(chan)) {
+ eifs = 340;
+ rx_lat = (rx_lat * 4) - 1;
+@@ -1057,8 +1079,9 @@ void ath9k_hw_init_global_settings(struc
+ if (IS_CHAN_A_FAST_CLOCK(ah, chan))
+ tx_lat += 22;
+
++ sifstime *= 4;
++ ack_offset = 32;
+ slottime = 21;
+- sifstime = 64;
+ } else {
+ if (AR_SREV_9287(ah) && AR_SREV_9287_13_OR_LATER(ah)) {
+ eifs = AR_D_GBL_IFS_EIFS_ASYNC_FIFO;
+@@ -1072,14 +1095,10 @@ void ath9k_hw_init_global_settings(struc
+ tx_lat = MS(reg, AR_USEC_TX_LAT);
+
+ slottime = ah->slottime;
+- if (IS_CHAN_5GHZ(chan))
+- sifstime = 16;
+- else
+- sifstime = 10;
+ }
+
+ /* As defined by IEEE 802.11-2007 17.3.8.6 */
+- acktimeout = slottime + sifstime + 3 * ah->coverage_class;
++ acktimeout = slottime + sifstime + 3 * ah->coverage_class + ack_offset;
+ ctstimeout = acktimeout;
+
+ /*
+@@ -1089,7 +1108,8 @@ void ath9k_hw_init_global_settings(struc
+ * BA frames in some implementations, but it has been found to fix ACK
+ * timeout issues in other cases as well.
+ */
+- if (conf->channel && conf->channel->band == IEEE80211_BAND_2GHZ) {
++ if (conf->channel && conf->channel->band == IEEE80211_BAND_2GHZ &&
++ !IS_CHAN_HALF_RATE(chan) && !IS_CHAN_QUARTER_RATE(chan)) {
+ acktimeout += 64 - sifstime - ah->slottime;
+ ctstimeout += 48 - sifstime - ah->slottime;
+ }
+@@ -1469,6 +1489,10 @@ static bool ath9k_hw_channel_change(stru
+ CHANNEL_5GHZ));
+ mode_diff = (chan->chanmode != ah->curchan->chanmode);
+
++ if ((ah->curchan->channelFlags | chan->channelFlags) &
++ (CHANNEL_HALF | CHANNEL_QUARTER))
++ return false;
+
-+ /* Common for AR913x, AR9160 */
-+ if (!AR_SREV_5416(ah))
-+ INIT_INI_ARRAY(&ah->iniBank6TPC, ar5416Bank6TPC_9100,
-+ ARRAY_SIZE(ar5416Bank6TPC_9100), 3);
+ for (qnum = 0; qnum < AR_NUM_QCU; qnum++) {
+ if (ath9k_hw_numtxpending(ah, qnum)) {
+ ath_dbg(common, QUEUE,
+@@ -1502,7 +1526,7 @@ static bool ath9k_hw_channel_change(stru
+ return false;
}
+ ath9k_hw_set_clockrate(ah);
+- ath9k_hw_apply_txpower(ah, chan);
++ ath9k_hw_apply_txpower(ah, chan, false);
+ ath9k_hw_rfbus_done(ah);
+
+ if (IS_CHAN_OFDM(chan) || IS_CHAN_HT(chan))
+@@ -2773,7 +2797,8 @@ static int get_antenna_gain(struct ath_h
+ return ah->eep_ops->get_eeprom(ah, gain_param);
+ }
- /* iniAddac needs to be modified for these chips */
---- a/drivers/net/wireless/ath/ath9k/ar9002_initvals.h
-+++ b/drivers/net/wireless/ath/ath9k/ar9002_initvals.h
-@@ -925,34 +925,6 @@ static const u32 ar9280PciePhy_clkreq_al
- {0x00004044, 0x00000000},
- };
+-void ath9k_hw_apply_txpower(struct ath_hw *ah, struct ath9k_channel *chan)
++void ath9k_hw_apply_txpower(struct ath_hw *ah, struct ath9k_channel *chan,
++ bool test)
+ {
+ struct ath_regulatory *reg = ath9k_hw_regulatory(ah);
+ struct ieee80211_channel *channel;
+@@ -2794,7 +2819,7 @@ void ath9k_hw_apply_txpower(struct ath_h
+
+ ah->eep_ops->set_txpower(ah, chan,
+ ath9k_regd_get_ctl(reg, chan),
+- ant_reduction, new_pwr, false);
++ ant_reduction, new_pwr, test);
+ }
--static const u32 ar9285PciePhy_clkreq_always_on_L1_9285[][2] = {
-- /* Addr allmodes */
-- {0x00004040, 0x9248fd00},
-- {0x00004040, 0x24924924},
-- {0x00004040, 0xa8000019},
-- {0x00004040, 0x13160820},
-- {0x00004040, 0xe5980560},
-- {0x00004040, 0xc01dcffd},
-- {0x00004040, 0x1aaabe41},
-- {0x00004040, 0xbe105554},
-- {0x00004040, 0x00043007},
-- {0x00004044, 0x00000000},
--};
--
--static const u32 ar9285PciePhy_clkreq_off_L1_9285[][2] = {
-- /* Addr allmodes */
-- {0x00004040, 0x9248fd00},
-- {0x00004040, 0x24924924},
-- {0x00004040, 0xa8000019},
-- {0x00004040, 0x13160820},
-- {0x00004040, 0xe5980560},
-- {0x00004040, 0xc01dcffc},
-- {0x00004040, 0x1aaabe41},
-- {0x00004040, 0xbe105554},
-- {0x00004040, 0x00043007},
-- {0x00004044, 0x00000000},
--};
--
- static const u32 ar9285Modes_9285_1_2[][5] = {
- /* Addr 5G_HT20 5G_HT40 2G_HT40 2G_HT20 */
- {0x00001030, 0x00000230, 0x00000460, 0x000002c0, 0x00000160},
-@@ -1743,34 +1715,6 @@ static const u32 ar9285Modes_XE2_0_high_
- {0x0000a3e0, 0x000000e7, 0x000000e7, 0x000000e7, 0x000000e7},
- };
+ void ath9k_hw_set_txpowerlimit(struct ath_hw *ah, u32 limit, bool test)
+@@ -2807,7 +2832,7 @@ void ath9k_hw_set_txpowerlimit(struct at
+ if (test)
+ channel->max_power = MAX_RATE_POWER / 2;
+
+- ath9k_hw_apply_txpower(ah, chan);
++ ath9k_hw_apply_txpower(ah, chan, test);
+
+ if (test)
+ channel->max_power = DIV_ROUND_UP(reg->max_power_level, 2);
+--- a/drivers/net/wireless/ath/ath9k/hw.h
++++ b/drivers/net/wireless/ath/ath9k/hw.h
+@@ -923,6 +923,8 @@ void ath9k_hw_set_gpio(struct ath_hw *ah
+ void ath9k_hw_setantenna(struct ath_hw *ah, u32 antenna);
+
+ /* General Operation */
++void ath9k_hw_synth_delay(struct ath_hw *ah, struct ath9k_channel *chan,
++ int hw_delay);
+ bool ath9k_hw_wait(struct ath_hw *ah, u32 reg, u32 mask, u32 val, u32 timeout);
+ void ath9k_hw_write_array(struct ath_hw *ah, struct ar5416IniArray *array,
+ int column, unsigned int *writecnt);
+@@ -982,7 +984,8 @@ void ath9k_hw_name(struct ath_hw *ah, ch
+ /* PHY */
+ void ath9k_hw_get_delta_slope_vals(struct ath_hw *ah, u32 coef_scaled,
+ u32 *coef_mantissa, u32 *coef_exponent);
+-void ath9k_hw_apply_txpower(struct ath_hw *ah, struct ath9k_channel *chan);
++void ath9k_hw_apply_txpower(struct ath_hw *ah, struct ath9k_channel *chan,
++ bool test);
+
+ /*
+ * Code Specific to AR5008, AR9001 or AR9002,
+--- a/drivers/net/wireless/ath/ath9k/init.c
++++ b/drivers/net/wireless/ath/ath9k/init.c
+@@ -647,6 +647,24 @@ void ath9k_reload_chainmask_settings(str
+ setup_ht_cap(sc, &sc->sbands[IEEE80211_BAND_5GHZ].ht_cap);
+ }
--static const u32 ar9285PciePhy_clkreq_always_on_L1_9285_1_2[][2] = {
-- /* Addr allmodes */
-- {0x00004040, 0x9248fd00},
-- {0x00004040, 0x24924924},
-- {0x00004040, 0xa8000019},
-- {0x00004040, 0x13160820},
-- {0x00004040, 0xe5980560},
-- {0x00004040, 0xc01dcffd},
-- {0x00004040, 0x1aaabe41},
-- {0x00004040, 0xbe105554},
-- {0x00004040, 0x00043007},
-- {0x00004044, 0x00000000},
--};
--
--static const u32 ar9285PciePhy_clkreq_off_L1_9285_1_2[][2] = {
-- /* Addr allmodes */
-- {0x00004040, 0x9248fd00},
-- {0x00004040, 0x24924924},
-- {0x00004040, 0xa8000019},
-- {0x00004040, 0x13160820},
-- {0x00004040, 0xe5980560},
-- {0x00004040, 0xc01dcffc},
-- {0x00004040, 0x1aaabe41},
-- {0x00004040, 0xbe105554},
-- {0x00004040, 0x00043007},
-- {0x00004044, 0x00000000},
--};
--
- static const u32 ar9287Modes_9287_1_1[][5] = {
- /* Addr 5G_HT20 5G_HT40 2G_HT40 2G_HT20 */
- {0x00001030, 0x00000000, 0x00000000, 0x000002c0, 0x00000160},
-@@ -2512,34 +2456,6 @@ static const u32 ar9287Modes_rx_gain_928
- {0x0000a848, 0x00000000, 0x00000000, 0x00001067, 0x00001067},
- };
++static const struct ieee80211_iface_limit if_limits[] = {
++ { .max = 2048, .types = BIT(NL80211_IFTYPE_STATION) |
++ BIT(NL80211_IFTYPE_P2P_CLIENT) |
++ BIT(NL80211_IFTYPE_WDS) },
++ { .max = 8, .types =
++#ifdef CONFIG_MAC80211_MESH
++ BIT(NL80211_IFTYPE_MESH_POINT) |
++#endif
++ BIT(NL80211_IFTYPE_AP) |
++ BIT(NL80211_IFTYPE_P2P_GO) },
++};
++
++static const struct ieee80211_iface_combination if_comb = {
++ .limits = if_limits,
++ .n_limits = ARRAY_SIZE(if_limits),
++ .max_interfaces = 2048,
++ .num_different_channels = 1,
++};
+
+ void ath9k_set_hw_capab(struct ath_softc *sc, struct ieee80211_hw *hw)
+ {
+@@ -676,6 +694,9 @@ void ath9k_set_hw_capab(struct ath_softc
+ BIT(NL80211_IFTYPE_ADHOC) |
+ BIT(NL80211_IFTYPE_MESH_POINT);
--static const u32 ar9287PciePhy_clkreq_always_on_L1_9287_1_1[][2] = {
-- /* Addr allmodes */
-- {0x00004040, 0x9248fd00},
-- {0x00004040, 0x24924924},
-- {0x00004040, 0xa8000019},
-- {0x00004040, 0x13160820},
-- {0x00004040, 0xe5980560},
-- {0x00004040, 0xc01dcffd},
-- {0x00004040, 0x1aaabe41},
-- {0x00004040, 0xbe105554},
-- {0x00004040, 0x00043007},
-- {0x00004044, 0x00000000},
--};
--
--static const u32 ar9287PciePhy_clkreq_off_L1_9287_1_1[][2] = {
-- /* Addr allmodes */
-- {0x00004040, 0x9248fd00},
-- {0x00004040, 0x24924924},
-- {0x00004040, 0xa8000019},
-- {0x00004040, 0x13160820},
-- {0x00004040, 0xe5980560},
-- {0x00004040, 0xc01dcffc},
-- {0x00004040, 0x1aaabe41},
-- {0x00004040, 0xbe105554},
-- {0x00004040, 0x00043007},
-- {0x00004044, 0x00000000},
--};
--
- static const u32 ar9271Modes_9271[][5] = {
- /* Addr 5G_HT20 5G_HT40 2G_HT40 2G_HT20 */
- {0x00001030, 0x00000230, 0x00000460, 0x000002c0, 0x00000160},
-@@ -3176,20 +3092,6 @@ static const u32 ar9271Common_9271[][2]
- {0x0000d384, 0xf3307ff0},
- };
++ hw->wiphy->iface_combinations = &if_comb;
++ hw->wiphy->n_iface_combinations = 1;
++
+ if (AR_SREV_5416(sc->sc_ah))
+ hw->wiphy->flags &= ~WIPHY_FLAG_PS_ON_BY_DEFAULT;
+
+--- a/drivers/net/wireless/ath/ath9k/mac.c
++++ b/drivers/net/wireless/ath/ath9k/mac.c
+@@ -133,8 +133,16 @@ EXPORT_SYMBOL(ath9k_hw_updatetxtriglevel
+
+ void ath9k_hw_abort_tx_dma(struct ath_hw *ah)
+ {
++ int maxdelay = 1000;
+ int i, q;
+
++ if (ah->curchan) {
++ if (IS_CHAN_HALF_RATE(ah->curchan))
++ maxdelay *= 2;
++ else if (IS_CHAN_QUARTER_RATE(ah->curchan))
++ maxdelay *= 4;
++ }
++
+ REG_WRITE(ah, AR_Q_TXD, AR_Q_TXD_M);
+
+ REG_SET_BIT(ah, AR_PCU_MISC, AR_PCU_FORCE_QUIET_COLL | AR_PCU_CLEAR_VMF);
+@@ -142,7 +150,7 @@ void ath9k_hw_abort_tx_dma(struct ath_hw
+ REG_SET_BIT(ah, AR_D_GBL_IFS_MISC, AR_D_GBL_IFS_MISC_IGNORE_BACKOFF);
+
+ for (q = 0; q < AR_NUM_QCU; q++) {
+- for (i = 0; i < 1000; i++) {
++ for (i = 0; i < maxdelay; i++) {
+ if (i)
+ udelay(5);
--static const u32 ar9271Common_normal_cck_fir_coeff_9271[][2] = {
-- /* Addr allmodes */
-- {0x0000a1f4, 0x00fffeff},
-- {0x0000a1f8, 0x00f5f9ff},
-- {0x0000a1fc, 0xb79f6427},
--};
--
--static const u32 ar9271Common_japan_2484_cck_fir_coeff_9271[][2] = {
-- /* Addr allmodes */
-- {0x0000a1f4, 0x00000000},
-- {0x0000a1f8, 0xefff0301},
-- {0x0000a1fc, 0xca9228ee},
--};
--
- static const u32 ar9271Modes_9271_1_0_only[][5] = {
- /* Addr 5G_HT20 5G_HT40 2G_HT40 2G_HT20 */
- {0x00009910, 0x30002311, 0x30002311, 0x30002311, 0x30002311},
--- a/net/mac80211/agg-rx.c
+++ b/net/mac80211/agg-rx.c
-@@ -187,6 +187,8 @@ static void ieee80211_send_addba_resp(st
+@@ -200,6 +200,8 @@ static void ieee80211_send_addba_resp(st
memcpy(mgmt->bssid, sdata->u.mgd.bssid, ETH_ALEN);
else if (sdata->vif.type == NL80211_IFTYPE_ADHOC)
memcpy(mgmt->bssid, sdata->u.ibss.bssid, ETH_ALEN);
memcpy(mgmt->bssid, sdata->vif.addr, ETH_ALEN);
else if (sdata->vif.type == NL80211_IFTYPE_STATION)
memcpy(mgmt->bssid, sdata->u.mgd.bssid, ETH_ALEN);
-@@ -471,6 +472,7 @@ int ieee80211_start_tx_ba_session(struct
+@@ -484,6 +485,7 @@ int ieee80211_start_tx_ba_session(struct
sdata->vif.type != NL80211_IFTYPE_MESH_POINT &&
sdata->vif.type != NL80211_IFTYPE_AP_VLAN &&
sdata->vif.type != NL80211_IFTYPE_AP &&
test_sta_flag(sta, WLAN_STA_##flg) ? #flg "\n" : ""
int res = scnprintf(buf, sizeof(buf),
-- "%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s",
-+ "%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s",
+- "%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s",
++ "%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s",
TEST(AUTH), TEST(ASSOC), TEST(PS_STA),
TEST(PS_DRIVER), TEST(AUTHORIZED),
TEST(SHORT_PREAMBLE),
TEST(TDLS_PEER_AUTH), TEST(4ADDR_EVENT),
--- a/net/mac80211/iface.c
+++ b/net/mac80211/iface.c
-@@ -178,7 +178,6 @@ static int ieee80211_do_open(struct net_
+@@ -282,7 +282,6 @@ static int ieee80211_do_open(struct net_
{
struct ieee80211_sub_if_data *sdata = IEEE80211_DEV_TO_SUB_IF(dev);
struct ieee80211_local *local = sdata->local;
u32 changed = 0;
int res;
u32 hw_reconf_flags = 0;
-@@ -309,28 +308,6 @@ static int ieee80211_do_open(struct net_
+@@ -428,28 +427,6 @@ static int ieee80211_do_open(struct net_
set_bit(SDATA_STATE_RUNNING, &sdata->state);
/*
* set_multicast_list will be invoked by the networking core
* which will check whether any increments here were done in
-@@ -357,8 +334,7 @@ static int ieee80211_do_open(struct net_
- netif_tx_start_all_queues(dev);
-
- return 0;
-- err_del_interface:
-- drv_remove_interface(local, sdata);
-+
- err_stop:
- if (!local->open_count)
- drv_stop(local);
-@@ -654,6 +630,8 @@ static void ieee80211_teardown_sdata(str
-
- if (ieee80211_vif_is_mesh(&sdata->vif))
- mesh_rmc_free(sdata);
-+ else if (sdata->vif.type == NL80211_IFTYPE_STATION)
-+ ieee80211_mgd_teardown(sdata);
-
- flushed = sta_info_flush(local, sdata);
- WARN_ON(flushed);
-@@ -720,6 +698,70 @@ static void ieee80211_if_setup(struct ne
+@@ -846,6 +823,72 @@ static void ieee80211_if_setup(struct ne
dev->destructor = free_netdev;
}
+ ieee802_11_parse_elems(mgmt->u.probe_resp.variable,
+ skb->len - baselen, &elems);
+
-+ rates = ieee80211_sta_get_rates(local, &elems, band);
++ rates = ieee80211_sta_get_rates(local, &elems, band, NULL);
+
+ rcu_read_lock();
+
+ set_sta_flag(sta, WLAN_STA_WME);
+
+ if (new) {
-+ set_sta_flag(sta, WLAN_STA_AUTHORIZED);
++ sta_info_pre_move_state(sta, IEEE80211_STA_AUTH);
++ sta_info_pre_move_state(sta, IEEE80211_STA_ASSOC);
++ sta_info_pre_move_state(sta, IEEE80211_STA_AUTHORIZED);
+ rate_control_rate_init(sta);
+ sta_info_insert_rcu(sta);
+ }
static void ieee80211_iface_work(struct work_struct *work)
{
struct ieee80211_sub_if_data *sdata =
-@@ -824,6 +866,9 @@ static void ieee80211_iface_work(struct
+@@ -950,6 +993,9 @@ static void ieee80211_iface_work(struct
break;
ieee80211_mesh_rx_queued_mgmt(sdata, skb);
break;
break;
--- a/net/mac80211/rx.c
+++ b/net/mac80211/rx.c
-@@ -2281,6 +2281,7 @@ ieee80211_rx_h_action(struct ieee80211_r
+@@ -2283,6 +2283,7 @@ ieee80211_rx_h_action(struct ieee80211_r
sdata->vif.type != NL80211_IFTYPE_MESH_POINT &&
sdata->vif.type != NL80211_IFTYPE_AP_VLAN &&
sdata->vif.type != NL80211_IFTYPE_AP &&
sdata->vif.type != NL80211_IFTYPE_ADHOC)
break;
-@@ -2491,14 +2492,15 @@ ieee80211_rx_h_mgmt(struct ieee80211_rx_
+@@ -2497,14 +2498,15 @@ ieee80211_rx_h_mgmt(struct ieee80211_rx_
if (!ieee80211_vif_is_mesh(&sdata->vif) &&
sdata->vif.type != NL80211_IFTYPE_ADHOC &&
break;
case cpu_to_le16(IEEE80211_STYPE_ASSOC_RESP):
case cpu_to_le16(IEEE80211_STYPE_REASSOC_RESP):
-@@ -2852,10 +2854,16 @@ static int prepare_for_handlers(struct i
+@@ -2838,10 +2840,16 @@ static int prepare_for_handlers(struct i
}
break;
case NL80211_IFTYPE_WDS:
/* should never get here */
--- a/net/mac80211/sta_info.h
+++ b/net/mac80211/sta_info.h
-@@ -31,7 +31,6 @@
+@@ -32,7 +32,6 @@
* @WLAN_STA_SHORT_PREAMBLE: Station is capable of receiving short-preamble
* frames.
* @WLAN_STA_WME: Station is a QoS-STA.
* @WLAN_STA_CLEAR_PS_FILT: Clear PS filter in hardware (using the
* IEEE80211_TX_CTL_CLEAR_PS_FILT control flag) when the next
* frame to this station is transmitted.
-@@ -62,7 +61,6 @@ enum ieee80211_sta_info_flags {
+@@ -64,7 +63,6 @@ enum ieee80211_sta_info_flags {
WLAN_STA_AUTHORIZED,
WLAN_STA_SHORT_PREAMBLE,
WLAN_STA_WME,
WLAN_STA_CLEAR_PS_FILT,
WLAN_STA_MFP,
WLAN_STA_BLOCK_BA,
---- a/drivers/net/wireless/ath/ath9k/ar9003_mac.c
-+++ b/drivers/net/wireless/ath/ath9k/ar9003_mac.c
-@@ -530,7 +530,11 @@ int ath9k_hw_process_rxdesc_edma(struct
- */
- if (rxsp->status11 & AR_CRCErr)
- rxs->rs_status |= ATH9K_RXERR_CRC;
-- else if (rxsp->status11 & AR_PHYErr) {
-+ else if (rxsp->status11 & AR_DecryptCRCErr)
-+ rxs->rs_status |= ATH9K_RXERR_DECRYPT;
-+ else if (rxsp->status11 & AR_MichaelErr)
-+ rxs->rs_status |= ATH9K_RXERR_MIC;
-+ if (rxsp->status11 & AR_PHYErr) {
- phyerr = MS(rxsp->status11, AR_PHYErrCode);
- /*
- * If we reach a point here where AR_PostDelimCRCErr is
-@@ -552,11 +556,7 @@ int ath9k_hw_process_rxdesc_edma(struct
- rxs->rs_status |= ATH9K_RXERR_PHY;
- rxs->rs_phyerr = phyerr;
- }
--
-- } else if (rxsp->status11 & AR_DecryptCRCErr)
-- rxs->rs_status |= ATH9K_RXERR_DECRYPT;
-- else if (rxsp->status11 & AR_MichaelErr)
-- rxs->rs_status |= ATH9K_RXERR_MIC;
-+ };
+--- a/drivers/net/wireless/iwlwifi/iwl-agn.h
++++ b/drivers/net/wireless/iwlwifi/iwl-agn.h
+@@ -425,6 +425,7 @@ void iwl_testmode_cleanup(struct iwl_pri
+ #ifdef CONFIG_IWLWIFI_DEBUG
+ void iwl_print_rx_config_cmd(struct iwl_priv *priv,
+ enum iwl_rxon_context_id ctxid);
++int iwl_alloc_traffic_mem(struct iwl_priv *priv);
+ #else
+ static inline void iwl_print_rx_config_cmd(struct iwl_priv *priv,
+ enum iwl_rxon_context_id ctxid)
+@@ -510,7 +511,6 @@ void iwl_setup_deferred_work(struct iwl_
+ int iwl_send_wimax_coex(struct iwl_priv *priv);
+ int iwl_send_bt_env(struct iwl_priv *priv, u8 action, u8 type);
+ void iwl_debug_config(struct iwl_priv *priv);
+-int iwl_alloc_traffic_mem(struct iwl_priv *priv);
+ void iwl_set_hw_params(struct iwl_priv *priv);
+ void iwl_init_context(struct iwl_priv *priv, u32 ucode_flags);
+ int iwl_init_drv(struct iwl_priv *priv);
+--- a/drivers/net/wireless/libertas/firmware.c
++++ b/drivers/net/wireless/libertas/firmware.c
+@@ -5,6 +5,7 @@
+ #include <linux/firmware.h>
+ #include <linux/firmware.h>
+ #include <linux/module.h>
++#include <linux/sched.h>
+
+ #include "dev.h"
+ #include "decl.h"
+--- a/drivers/net/wireless/ath/ath9k/recv.c
++++ b/drivers/net/wireless/ath/ath9k/recv.c
+@@ -812,6 +812,7 @@ static bool ath9k_rx_accept(struct ath_c
+ is_valid_tkip = rx_stats->rs_keyix != ATH9K_RXKEYIX_INVALID &&
+ test_bit(rx_stats->rs_keyix, common->tkip_keymap);
+ strip_mic = is_valid_tkip && ieee80211_is_data(fc) &&
++ ieee80211_has_protected(fc) &&
+ !(rx_stats->rs_status &
+ (ATH9K_RXERR_DECRYPT | ATH9K_RXERR_CRC | ATH9K_RXERR_MIC |
+ ATH9K_RXERR_KEYMISS));
+--- a/net/mac80211/cfg.c
++++ b/net/mac80211/cfg.c
+@@ -1005,6 +1005,9 @@ static int ieee80211_change_station(stru
}
- if (rxsp->status11 & AR_KeyMiss)
---- a/drivers/net/wireless/ath/carl9170/tx.c
-+++ b/drivers/net/wireless/ath/carl9170/tx.c
-@@ -1236,6 +1236,7 @@ static bool carl9170_tx_ps_drop(struct a
- {
- struct ieee80211_sta *sta;
- struct carl9170_sta_info *sta_info;
-+ struct ieee80211_tx_info *tx_info;
-
- rcu_read_lock();
- sta = __carl9170_get_tx_sta(ar, skb);
-@@ -1243,12 +1244,13 @@ static bool carl9170_tx_ps_drop(struct a
- goto out_rcu;
-
- sta_info = (void *) sta->drv_priv;
-- if (unlikely(sta_info->sleeping)) {
-- struct ieee80211_tx_info *tx_info;
-+ tx_info = IEEE80211_SKB_CB(skb);
-
-+ if (unlikely(sta_info->sleeping) &&
-+ !(tx_info->flags & (IEEE80211_TX_CTL_NO_PS_BUFFER |
-+ IEEE80211_TX_CTL_CLEAR_PS_FILT))) {
- rcu_read_unlock();
-
-- tx_info = IEEE80211_SKB_CB(skb);
- if (tx_info->flags & IEEE80211_TX_CTL_AMPDU)
- atomic_dec(&ar->tx_ampdu_upload);
-
---- a/drivers/net/wireless/iwlegacy/4965-mac.c
-+++ b/drivers/net/wireless/iwlegacy/4965-mac.c
-@@ -1694,7 +1694,7 @@ il4965_tx_skb(struct il_priv *il, struct
- sta_priv = (void *)sta->drv_priv;
-
- if (sta_priv && sta_priv->asleep &&
-- (info->flags & IEEE80211_TX_CTL_POLL_RESPONSE)) {
-+ (info->flags & IEEE80211_TX_CTL_NO_PS_BUFFER)) {
- /*
- * This sends an asynchronous command to the device,
- * but we can rely on it being processed before the
---- a/drivers/net/wireless/iwlwifi/iwl-agn-tx.c
-+++ b/drivers/net/wireless/iwlwifi/iwl-agn-tx.c
-@@ -322,7 +322,7 @@ int iwlagn_tx_skb(struct iwl_priv *priv,
- sta_priv = (void *)info->control.sta->drv_priv;
-
- if (sta_priv && sta_priv->asleep &&
-- (info->flags & IEEE80211_TX_CTL_POLL_RESPONSE)) {
-+ (info->flags & IEEE80211_TX_CTL_NO_PS_BUFFER)) {
- /*
- * This sends an asynchronous command to the device,
- * but we can rely on it being processed before the
-@@ -331,6 +331,10 @@ int iwlagn_tx_skb(struct iwl_priv *priv,
- * counter.
- * For now set the counter to just 1 since we do not
- * support uAPSD yet.
-+ *
-+ * FIXME: If we get two non-bufferable frames one
-+ * after the other, we might only send out one of
-+ * them because this is racy.
- */
- iwl_sta_modify_sleep_tx_count(priv, sta_id, 1);
- }
---- a/drivers/net/wireless/p54/txrx.c
-+++ b/drivers/net/wireless/p54/txrx.c
-@@ -690,7 +690,7 @@ static void p54_tx_80211_header(struct p
- if (!(info->flags & IEEE80211_TX_CTL_ASSIGN_SEQ))
- *flags |= P54_HDR_FLAG_DATA_OUT_SEQNR;
-
-- if (info->flags & IEEE80211_TX_CTL_POLL_RESPONSE)
-+ if (info->flags & IEEE80211_TX_CTL_NO_PS_BUFFER)
- *flags |= P54_HDR_FLAG_DATA_OUT_NOCANCEL;
-
- if (info->flags & IEEE80211_TX_CTL_CLEAR_PS_FILT)
---- a/include/net/mac80211.h
-+++ b/include/net/mac80211.h
-@@ -341,9 +341,9 @@ struct ieee80211_bss_conf {
- * used to indicate that a frame was already retried due to PS
- * @IEEE80211_TX_INTFL_DONT_ENCRYPT: completely internal to mac80211,
- * used to indicate frame should not be encrypted
-- * @IEEE80211_TX_CTL_POLL_RESPONSE: This frame is a response to a poll
-- * frame (PS-Poll or uAPSD) and should be sent although the station
-- * is in powersave mode.
-+ * @IEEE80211_TX_CTL_NO_PS_BUFFER: This frame is a response to a poll
-+ * frame (PS-Poll or uAPSD) or a non-bufferable MMPDU and must
-+ * be sent although the station is in powersave mode.
- * @IEEE80211_TX_CTL_MORE_FRAMES: More frames will be passed to the
- * transmit function after the current frame, this can be used
- * by drivers to kick the DMA queue only if unset or when the
-@@ -399,7 +399,7 @@ enum mac80211_tx_control_flags {
- IEEE80211_TX_INTFL_NEED_TXPROCESSING = BIT(14),
- IEEE80211_TX_INTFL_RETRIED = BIT(15),
- IEEE80211_TX_INTFL_DONT_ENCRYPT = BIT(16),
-- IEEE80211_TX_CTL_POLL_RESPONSE = BIT(17),
-+ IEEE80211_TX_CTL_NO_PS_BUFFER = BIT(17),
- IEEE80211_TX_CTL_MORE_FRAMES = BIT(18),
- IEEE80211_TX_INTFL_RETRANSMISSION = BIT(19),
- /* hole at 20, use later */
-@@ -425,7 +425,7 @@ enum mac80211_tx_control_flags {
- IEEE80211_TX_CTL_SEND_AFTER_DTIM | IEEE80211_TX_CTL_AMPDU | \
- IEEE80211_TX_STAT_TX_FILTERED | IEEE80211_TX_STAT_ACK | \
- IEEE80211_TX_STAT_AMPDU | IEEE80211_TX_STAT_AMPDU_NO_BACK | \
-- IEEE80211_TX_CTL_RATE_CTRL_PROBE | IEEE80211_TX_CTL_POLL_RESPONSE | \
-+ IEEE80211_TX_CTL_RATE_CTRL_PROBE | IEEE80211_TX_CTL_NO_PS_BUFFER | \
- IEEE80211_TX_CTL_MORE_FRAMES | IEEE80211_TX_CTL_LDPC | \
- IEEE80211_TX_CTL_STBC | IEEE80211_TX_STATUS_EOSP)
-
-@@ -1634,7 +1634,7 @@ void ieee80211_free_txskb(struct ieee802
- * the station sends a PS-Poll or a uAPSD trigger frame, mac80211
- * will inform the driver of this with the @allow_buffered_frames
- * callback; this callback is optional. mac80211 will then transmit
-- * the frames as usual and set the %IEEE80211_TX_CTL_POLL_RESPONSE
-+ * the frames as usual and set the %IEEE80211_TX_CTL_NO_PS_BUFFER
- * on each frame. The last frame in the service period (or the only
- * response to a PS-Poll) also has %IEEE80211_TX_STATUS_EOSP set to
- * indicate that it ends the service period; as this frame must have
-@@ -1642,6 +1642,9 @@ void ieee80211_free_txskb(struct ieee802
- * When TX status is reported for this frame, the service period is
- * marked has having ended and a new one can be started by the peer.
- *
-+ * Additionally, non-bufferable MMPDUs can also be transmitted by
-+ * mac80211 with the %IEEE80211_TX_CTL_NO_PS_BUFFER set in them.
-+ *
- * Another race condition can happen on some devices like iwlwifi
- * when there are frames queued for the station and it wakes up
- * or polls; the frames that are already queued could end up being
-@@ -2140,7 +2143,7 @@ enum ieee80211_frame_release_type {
- * @allow_buffered_frames: Prepare device to allow the given number of frames
- * to go out to the given station. The frames will be sent by mac80211
- * via the usual TX path after this call. The TX information for frames
-- * released will also have the %IEEE80211_TX_CTL_POLL_RESPONSE flag set
-+ * released will also have the %IEEE80211_TX_CTL_NO_PS_BUFFER flag set
- * and the last one will also have %IEEE80211_TX_STATUS_EOSP set. In case
- * frames from multiple TIDs are released and the driver might reorder
- * them between the TIDs, it must set the %IEEE80211_TX_STATUS_EOSP flag
---- a/net/mac80211/ieee80211_i.h
-+++ b/net/mac80211/ieee80211_i.h
-@@ -1183,6 +1183,7 @@ void ieee80211_sta_rx_queued_mgmt(struct
- struct sk_buff *skb);
- void ieee80211_sta_reset_beacon_monitor(struct ieee80211_sub_if_data *sdata);
- void ieee80211_sta_reset_conn_monitor(struct ieee80211_sub_if_data *sdata);
-+void ieee80211_mgd_teardown(struct ieee80211_sub_if_data *sdata);
-
- /* IBSS code */
- void ieee80211_ibss_notify_scan_completed(struct ieee80211_local *local);
---- a/net/mac80211/mlme.c
-+++ b/net/mac80211/mlme.c
-@@ -3496,6 +3496,19 @@ int ieee80211_mgd_disassoc(struct ieee80
- return 0;
- }
+ if (params->vlan && params->vlan != sta->sdata->dev) {
++ bool prev_4addr = false;
++ bool new_4addr = false;
++
+ vlansdata = IEEE80211_DEV_TO_SUB_IF(params->vlan);
-+void ieee80211_mgd_teardown(struct ieee80211_sub_if_data *sdata)
-+{
-+ struct ieee80211_if_managed *ifmgd = &sdata->u.mgd;
+ if (vlansdata->vif.type != NL80211_IFTYPE_AP_VLAN &&
+@@ -1020,9 +1023,25 @@ static int ieee80211_change_station(stru
+ }
+
+ rcu_assign_pointer(vlansdata->u.vlan.sta, sta);
++ new_4addr = true;
++ }
+
-+ mutex_lock(&ifmgd->mtx);
-+ if (ifmgd->assoc_data)
-+ ieee80211_destroy_assoc_data(sdata, false);
-+ if (ifmgd->auth_data)
-+ ieee80211_destroy_auth_data(sdata, false);
-+ del_timer_sync(&ifmgd->timer);
-+ mutex_unlock(&ifmgd->mtx);
-+}
++ if (sta->sdata->vif.type == NL80211_IFTYPE_AP_VLAN &&
++ sta->sdata->u.vlan.sta) {
++ rcu_assign_pointer(sta->sdata->u.vlan.sta, NULL);
++ prev_4addr = true;
+ }
+
+ sta->sdata = vlansdata;
+
- void ieee80211_cqm_rssi_notify(struct ieee80211_vif *vif,
- enum nl80211_cqm_rssi_threshold_event rssi_event,
- gfp_t gfp)
---- a/net/mac80211/sta_info.c
-+++ b/net/mac80211/sta_info.c
-@@ -865,8 +865,10 @@ int sta_info_flush(struct ieee80211_loca
-
- mutex_lock(&local->sta_mtx);
- list_for_each_entry_safe(sta, tmp, &local->sta_list, list) {
-- if (!sdata || sdata == sta->sdata)
-+ if (!sdata || sdata == sta->sdata) {
- WARN_ON(__sta_info_destroy(sta));
-+ ret++;
++ if (sta->sta_state == IEEE80211_STA_AUTHORIZED &&
++ prev_4addr != new_4addr) {
++ if (new_4addr)
++ atomic_dec(&sta->sdata->bss->num_mcast_sta);
++ else
++ atomic_inc(&sta->sdata->bss->num_mcast_sta);
+ }
++
+ ieee80211_send_layer2_update(sta);
}
- mutex_unlock(&local->sta_mtx);
-@@ -1048,7 +1050,7 @@ static void ieee80211_send_null_response
- * exchange. Also set EOSP to indicate this packet
- * ends the poll/service period.
- */
-- info->flags |= IEEE80211_TX_CTL_POLL_RESPONSE |
-+ info->flags |= IEEE80211_TX_CTL_NO_PS_BUFFER |
- IEEE80211_TX_STATUS_EOSP |
- IEEE80211_TX_CTL_REQ_TX_STATUS;
-
-@@ -1175,7 +1177,7 @@ ieee80211_sta_ps_deliver_response(struct
- * STA may still remain is PS mode after this frame
- * exchange.
- */
-- info->flags |= IEEE80211_TX_CTL_POLL_RESPONSE;
-+ info->flags |= IEEE80211_TX_CTL_NO_PS_BUFFER;
-
- /*
- * Use MoreData flag to indicate whether there are
+--- a/net/mac80211/debugfs_netdev.c
++++ b/net/mac80211/debugfs_netdev.c
+@@ -394,7 +394,7 @@ static ssize_t ieee80211_if_parse_uapsd_
+ __IEEE80211_IF_FILE_W(uapsd_max_sp_len);
+
+ /* AP attributes */
+-IEEE80211_IF_FILE(num_sta_authorized, u.ap.num_sta_authorized, ATOMIC);
++IEEE80211_IF_FILE(num_mcast_sta, u.ap.num_mcast_sta, ATOMIC);
+ IEEE80211_IF_FILE(num_sta_ps, u.ap.num_sta_ps, ATOMIC);
+ IEEE80211_IF_FILE(dtim_count, u.ap.dtim_count, DEC);
+
+@@ -540,7 +540,7 @@ static void add_sta_files(struct ieee802
+
+ static void add_ap_files(struct ieee80211_sub_if_data *sdata)
+ {
+- DEBUGFS_ADD(num_sta_authorized);
++ DEBUGFS_ADD(num_mcast_sta);
+ DEBUGFS_ADD(num_sta_ps);
+ DEBUGFS_ADD(dtim_count);
+ DEBUGFS_ADD(num_buffered_multicast);
+--- a/net/mac80211/ieee80211_i.h
++++ b/net/mac80211/ieee80211_i.h
+@@ -282,7 +282,7 @@ struct ieee80211_if_ap {
+ u8 tim[sizeof(unsigned long) * BITS_TO_LONGS(IEEE80211_MAX_AID + 1)];
+ struct sk_buff_head ps_bc_buf;
+ atomic_t num_sta_ps; /* number of stations in PS mode */
+- atomic_t num_sta_authorized; /* number of authorized stations */
++ atomic_t num_mcast_sta; /* number of stations receiving multicast */
+ int dtim_count;
+ bool dtim_bc_mc;
+ };
+--- a/net/mac80211/sta_info.c
++++ b/net/mac80211/sta_info.c
+@@ -1417,15 +1417,19 @@ int sta_info_move_state(struct sta_info
+ if (sta->sta_state == IEEE80211_STA_AUTH) {
+ set_bit(WLAN_STA_ASSOC, &sta->_flags);
+ } else if (sta->sta_state == IEEE80211_STA_AUTHORIZED) {
+- if (sta->sdata->vif.type == NL80211_IFTYPE_AP)
+- atomic_dec(&sta->sdata->u.ap.num_sta_authorized);
++ if (sta->sdata->vif.type == NL80211_IFTYPE_AP ||
++ (sta->sdata->vif.type == NL80211_IFTYPE_AP_VLAN &&
++ !sta->sdata->u.vlan.sta))
++ atomic_dec(&sta->sdata->bss->num_mcast_sta);
+ clear_bit(WLAN_STA_AUTHORIZED, &sta->_flags);
+ }
+ break;
+ case IEEE80211_STA_AUTHORIZED:
+ if (sta->sta_state == IEEE80211_STA_ASSOC) {
+- if (sta->sdata->vif.type == NL80211_IFTYPE_AP)
+- atomic_inc(&sta->sdata->u.ap.num_sta_authorized);
++ if (sta->sdata->vif.type == NL80211_IFTYPE_AP ||
++ (sta->sdata->vif.type == NL80211_IFTYPE_AP_VLAN &&
++ !sta->sdata->u.vlan.sta))
++ atomic_inc(&sta->sdata->bss->num_mcast_sta);
+ set_bit(WLAN_STA_AUTHORIZED, &sta->_flags);
+ }
+ break;
--- a/net/mac80211/tx.c
+++ b/net/mac80211/tx.c
-@@ -448,18 +448,23 @@ ieee80211_tx_h_unicast_ps_buf(struct iee
- struct ieee80211_hdr *hdr = (struct ieee80211_hdr *)tx->skb->data;
- struct ieee80211_local *local = tx->local;
-
-- if (unlikely(!sta ||
-- ieee80211_is_probe_resp(hdr->frame_control) ||
-- ieee80211_is_auth(hdr->frame_control) ||
-- ieee80211_is_assoc_resp(hdr->frame_control) ||
-- ieee80211_is_reassoc_resp(hdr->frame_control)))
-+ if (unlikely(!sta))
- return TX_CONTINUE;
-
- if (unlikely((test_sta_flag(sta, WLAN_STA_PS_STA) ||
- test_sta_flag(sta, WLAN_STA_PS_DRIVER)) &&
-- !(info->flags & IEEE80211_TX_CTL_POLL_RESPONSE))) {
-+ !(info->flags & IEEE80211_TX_CTL_NO_PS_BUFFER))) {
- int ac = skb_get_queue_mapping(tx->skb);
-
-+ /* only deauth, disassoc and action are bufferable MMPDUs */
-+ if (ieee80211_is_mgmt(hdr->frame_control) &&
-+ !ieee80211_is_deauth(hdr->frame_control) &&
-+ !ieee80211_is_disassoc(hdr->frame_control) &&
-+ !ieee80211_is_action(hdr->frame_control)) {
-+ info->flags |= IEEE80211_TX_CTL_NO_PS_BUFFER;
-+ return TX_CONTINUE;
-+ }
-+
- #ifdef CONFIG_MAC80211_VERBOSE_PS_DEBUG
- printk(KERN_DEBUG "STA %pM aid %d: PS buffer for AC %d\n",
- sta->sta.addr, sta->sta.aid, ac);
+@@ -306,7 +306,7 @@ ieee80211_tx_h_check_assoc(struct ieee80
+ }
+ } else if (unlikely(tx->sdata->vif.type == NL80211_IFTYPE_AP &&
+ ieee80211_is_data(hdr->frame_control) &&
+- !atomic_read(&tx->sdata->u.ap.num_sta_authorized))) {
++ !atomic_read(&tx->sdata->u.ap.num_mcast_sta))) {
+ /*
+ * No associated STAs - no need to send multicast
+ * frames.
+@@ -1159,7 +1159,8 @@ ieee80211_tx_prepare(struct ieee80211_su
+ tx->sta = rcu_dereference(sdata->u.vlan.sta);
+ if (!tx->sta && sdata->dev->ieee80211_ptr->use_4addr)
+ return TX_DROP;
+- } else if (info->flags & IEEE80211_TX_CTL_INJECTED) {
++ } else if (info->flags & IEEE80211_TX_CTL_INJECTED ||
++ tx->sdata->control_port_protocol == tx->skb->protocol) {
+ tx->sta = sta_info_get_bss(sdata, hdr->addr1);
+ }
+ if (!tx->sta)
+--- a/net/wireless/scan.c
++++ b/net/wireless/scan.c
+@@ -378,7 +378,11 @@ static int cmp_bss_core(struct cfg80211_
+ b->len_information_elements);
+ }
+
+- return compare_ether_addr(a->bssid, b->bssid);
++ /*
++ * we can't use compare_ether_addr here since we need a < > operator.
++ * The binary return value of compare_ether_addr isn't enough
++ */
++ return memcmp(a->bssid, b->bssid, sizeof(a->bssid));
+ }
+
+ static int cmp_bss(struct cfg80211_bss *a,