blob: e4f7d0ea5f20bce76274e08e8a3882e5816ce049 [file] [log] [blame]
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07001/*
Sujith Manoharan5b681382011-05-17 13:36:18 +05302 * Copyright (c) 2008-2011 Atheros Communications Inc.
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07003 *
4 * Permission to use, copy, modify, and/or distribute this software for any
5 * purpose with or without fee is hereby granted, provided that the above
6 * copyright notice and this permission notice appear in all copies.
7 *
8 * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
9 * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
10 * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
11 * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
12 * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
13 * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
14 * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
15 */
16
Luis R. Rodriguezf078f202008-08-04 00:16:41 -070017#include <linux/nl80211.h>
Vasanthakumar Thiagarajan69081622011-02-19 01:13:42 -080018#include <linux/delay.h>
Sujith394cf0a2009-02-09 13:26:54 +053019#include "ath9k.h"
Luis R. Rodriguezaf03abe2009-09-09 02:33:11 -070020#include "btcoex.h"
Luis R. Rodriguezf078f202008-08-04 00:16:41 -070021
Sujithff37e332008-11-24 12:07:55 +053022static u8 parse_mpdudensity(u8 mpdudensity)
23{
24 /*
25 * 802.11n D2.0 defined values for "Minimum MPDU Start Spacing":
26 * 0 for no restriction
27 * 1 for 1/4 us
28 * 2 for 1/2 us
29 * 3 for 1 us
30 * 4 for 2 us
31 * 5 for 4 us
32 * 6 for 8 us
33 * 7 for 16 us
34 */
35 switch (mpdudensity) {
36 case 0:
37 return 0;
38 case 1:
39 case 2:
40 case 3:
41 /* Our lower layer calculations limit our precision to
42 1 microsecond */
43 return 1;
44 case 4:
45 return 2;
46 case 5:
47 return 4;
48 case 6:
49 return 8;
50 case 7:
51 return 16;
52 default:
53 return 0;
54 }
55}
56
Vasanthakumar Thiagarajan69081622011-02-19 01:13:42 -080057static bool ath9k_has_pending_frames(struct ath_softc *sc, struct ath_txq *txq)
58{
59 bool pending = false;
60
61 spin_lock_bh(&txq->axq_lock);
62
63 if (txq->axq_depth || !list_empty(&txq->axq_acq))
64 pending = true;
Vasanthakumar Thiagarajan69081622011-02-19 01:13:42 -080065
66 spin_unlock_bh(&txq->axq_lock);
67 return pending;
68}
69
Mohammed Shafi Shajakhan6d79cb42011-05-19 17:40:46 +053070static bool ath9k_setpower(struct ath_softc *sc, enum ath9k_power_mode mode)
Luis R. Rodriguez8c77a562009-09-09 21:02:34 -070071{
72 unsigned long flags;
73 bool ret;
74
Luis R. Rodriguez9ecdef42009-09-09 21:10:09 -070075 spin_lock_irqsave(&sc->sc_pm_lock, flags);
76 ret = ath9k_hw_setpower(sc->sc_ah, mode);
77 spin_unlock_irqrestore(&sc->sc_pm_lock, flags);
Luis R. Rodriguez8c77a562009-09-09 21:02:34 -070078
79 return ret;
80}
81
Luis R. Rodrigueza91d75ae2009-09-09 20:29:18 -070082void ath9k_ps_wakeup(struct ath_softc *sc)
83{
Felix Fietkau898c9142010-10-12 14:02:53 +020084 struct ath_common *common = ath9k_hw_common(sc->sc_ah);
Luis R. Rodrigueza91d75ae2009-09-09 20:29:18 -070085 unsigned long flags;
Felix Fietkaufbb078f2010-11-03 01:36:51 +010086 enum ath9k_power_mode power_mode;
Luis R. Rodrigueza91d75ae2009-09-09 20:29:18 -070087
88 spin_lock_irqsave(&sc->sc_pm_lock, flags);
89 if (++sc->ps_usecount != 1)
90 goto unlock;
91
Felix Fietkaufbb078f2010-11-03 01:36:51 +010092 power_mode = sc->sc_ah->power_mode;
Luis R. Rodriguez9ecdef42009-09-09 21:10:09 -070093 ath9k_hw_setpower(sc->sc_ah, ATH9K_PM_AWAKE);
Luis R. Rodrigueza91d75ae2009-09-09 20:29:18 -070094
Felix Fietkau898c9142010-10-12 14:02:53 +020095 /*
96 * While the hardware is asleep, the cycle counters contain no
97 * useful data. Better clear them now so that they don't mess up
98 * survey data results.
99 */
Felix Fietkaufbb078f2010-11-03 01:36:51 +0100100 if (power_mode != ATH9K_PM_AWAKE) {
101 spin_lock(&common->cc_lock);
102 ath_hw_cycle_counters_update(common);
103 memset(&common->cc_survey, 0, sizeof(common->cc_survey));
Rajkumar Manoharanc9ae6ab2012-06-04 16:28:41 +0530104 memset(&common->cc_ani, 0, sizeof(common->cc_ani));
Felix Fietkaufbb078f2010-11-03 01:36:51 +0100105 spin_unlock(&common->cc_lock);
106 }
Felix Fietkau898c9142010-10-12 14:02:53 +0200107
Luis R. Rodrigueza91d75ae2009-09-09 20:29:18 -0700108 unlock:
109 spin_unlock_irqrestore(&sc->sc_pm_lock, flags);
110}
111
112void ath9k_ps_restore(struct ath_softc *sc)
113{
Felix Fietkau898c9142010-10-12 14:02:53 +0200114 struct ath_common *common = ath9k_hw_common(sc->sc_ah);
Felix Fietkauc6c539f2011-09-14 21:24:24 +0200115 enum ath9k_power_mode mode;
Luis R. Rodrigueza91d75ae2009-09-09 20:29:18 -0700116 unsigned long flags;
Sujith Manoharanad128862012-04-24 10:23:20 +0530117 bool reset;
Luis R. Rodrigueza91d75ae2009-09-09 20:29:18 -0700118
119 spin_lock_irqsave(&sc->sc_pm_lock, flags);
120 if (--sc->ps_usecount != 0)
121 goto unlock;
122
Sujith Manoharanad128862012-04-24 10:23:20 +0530123 if (sc->ps_idle) {
124 ath9k_hw_setrxabort(sc->sc_ah, 1);
125 ath9k_hw_stopdmarecv(sc->sc_ah, &reset);
Felix Fietkauc6c539f2011-09-14 21:24:24 +0200126 mode = ATH9K_PM_FULL_SLEEP;
Sujith Manoharanad128862012-04-24 10:23:20 +0530127 } else if (sc->ps_enabled &&
128 !(sc->ps_flags & (PS_WAIT_FOR_BEACON |
129 PS_WAIT_FOR_CAB |
130 PS_WAIT_FOR_PSPOLL_DATA |
131 PS_WAIT_FOR_TX_ACK))) {
Felix Fietkauc6c539f2011-09-14 21:24:24 +0200132 mode = ATH9K_PM_NETWORK_SLEEP;
Sujith Manoharanad128862012-04-24 10:23:20 +0530133 } else {
Felix Fietkauc6c539f2011-09-14 21:24:24 +0200134 goto unlock;
Sujith Manoharanad128862012-04-24 10:23:20 +0530135 }
Felix Fietkauc6c539f2011-09-14 21:24:24 +0200136
137 spin_lock(&common->cc_lock);
138 ath_hw_cycle_counters_update(common);
139 spin_unlock(&common->cc_lock);
140
Felix Fietkau1a8f0d392011-09-22 08:04:32 -0600141 ath9k_hw_setpower(sc->sc_ah, mode);
Luis R. Rodrigueza91d75ae2009-09-09 20:29:18 -0700142
143 unlock:
144 spin_unlock_irqrestore(&sc->sc_pm_lock, flags);
145}
146
Felix Fietkau9adcf442011-09-03 01:40:26 +0200147static void __ath_cancel_work(struct ath_softc *sc)
148{
149 cancel_work_sync(&sc->paprd_work);
150 cancel_work_sync(&sc->hw_check_work);
151 cancel_delayed_work_sync(&sc->tx_complete_work);
152 cancel_delayed_work_sync(&sc->hw_pll_work);
153}
154
155static void ath_cancel_work(struct ath_softc *sc)
156{
157 __ath_cancel_work(sc);
158 cancel_work_sync(&sc->hw_reset_work);
159}
160
Sujith Manoharanaf68aba2012-06-04 20:23:43 +0530161static void ath_restart_work(struct ath_softc *sc)
162{
163 struct ath_common *common = ath9k_hw_common(sc->sc_ah);
164
165 ieee80211_queue_delayed_work(sc->hw, &sc->tx_complete_work, 0);
166
167 if (AR_SREV_9485(sc->sc_ah) || AR_SREV_9340(sc->sc_ah))
168 ieee80211_queue_delayed_work(sc->hw, &sc->hw_pll_work,
169 msecs_to_jiffies(ATH_PLL_WORK_INTERVAL));
170
171 ath_start_rx_poll(sc, 3);
172
173 if (!common->disable_ani)
174 ath_start_ani(common);
175}
176
Felix Fietkau9adcf442011-09-03 01:40:26 +0200177static bool ath_prepare_reset(struct ath_softc *sc, bool retry_tx, bool flush)
178{
179 struct ath_hw *ah = sc->sc_ah;
180 struct ath_common *common = ath9k_hw_common(ah);
Felix Fietkauceea2a52012-05-24 14:32:19 +0200181 bool ret = true;
Felix Fietkau9adcf442011-09-03 01:40:26 +0200182
183 ieee80211_stop_queues(sc->hw);
184
185 sc->hw_busy_count = 0;
186 del_timer_sync(&common->ani.timer);
Rajkumar Manoharan01e18912012-03-15 05:34:27 +0530187 del_timer_sync(&sc->rx_poll_timer);
Felix Fietkau9adcf442011-09-03 01:40:26 +0200188
189 ath9k_debug_samp_bb_mac(sc);
190 ath9k_hw_disable_interrupts(ah);
191
Felix Fietkau9adcf442011-09-03 01:40:26 +0200192 if (!ath_stoprecv(sc))
193 ret = false;
194
Felix Fietkauceea2a52012-05-24 14:32:19 +0200195 if (!ath_drain_all_txq(sc, retry_tx))
196 ret = false;
197
Felix Fietkau9adcf442011-09-03 01:40:26 +0200198 if (!flush) {
199 if (ah->caps.hw_caps & ATH9K_HW_CAP_EDMA)
Felix Fietkau34832882011-09-14 21:23:03 +0200200 ath_rx_tasklet(sc, 1, true);
201 ath_rx_tasklet(sc, 1, false);
Felix Fietkau9adcf442011-09-03 01:40:26 +0200202 } else {
203 ath_flushrecv(sc);
204 }
205
206 return ret;
207}
208
209static bool ath_complete_reset(struct ath_softc *sc, bool start)
210{
211 struct ath_hw *ah = sc->sc_ah;
212 struct ath_common *common = ath9k_hw_common(ah);
213
214 if (ath_startrecv(sc) != 0) {
215 ath_err(common, "Unable to restart recv logic\n");
216 return false;
217 }
218
219 ath9k_cmn_update_txpow(ah, sc->curtxpow,
220 sc->config.txpowlimit, &sc->curtxpow);
Felix Fietkau72d874c2011-10-08 20:06:19 +0200221 ath9k_hw_set_interrupts(ah);
Felix Fietkau9adcf442011-09-03 01:40:26 +0200222 ath9k_hw_enable_interrupts(ah);
223
Sujith Manoharan4cb54fa2012-06-04 16:27:52 +0530224 if (!(sc->hw->conf.flags & IEEE80211_CONF_OFFCHANNEL) && start) {
Sujith Manoharan781b14a2012-06-04 20:23:55 +0530225 if (test_bit(SC_OP_BEACONS, &sc->sc_flags))
Felix Fietkau9adcf442011-09-03 01:40:26 +0200226 ath_set_beacon(sc);
227
Sujith Manoharanaf68aba2012-06-04 20:23:43 +0530228 ath_restart_work(sc);
Felix Fietkau9adcf442011-09-03 01:40:26 +0200229 }
230
Sujith Manoharan8da07832012-06-04 20:23:49 +0530231 if ((ah->caps.hw_caps & ATH9K_HW_CAP_ANT_DIV_COMB) && sc->ant_rx != 3)
232 ath_ant_comb_update(sc);
Felix Fietkau43c35282011-09-03 01:40:27 +0200233
Felix Fietkau9adcf442011-09-03 01:40:26 +0200234 ieee80211_wake_queues(sc->hw);
235
236 return true;
237}
238
239static int ath_reset_internal(struct ath_softc *sc, struct ath9k_channel *hchan,
240 bool retry_tx)
241{
242 struct ath_hw *ah = sc->sc_ah;
243 struct ath_common *common = ath9k_hw_common(ah);
244 struct ath9k_hw_cal_data *caldata = NULL;
245 bool fastcc = true;
246 bool flush = false;
247 int r;
248
249 __ath_cancel_work(sc);
250
251 spin_lock_bh(&sc->sc_pcu_lock);
252
Sujith Manoharan4cb54fa2012-06-04 16:27:52 +0530253 if (!(sc->hw->conf.flags & IEEE80211_CONF_OFFCHANNEL)) {
Felix Fietkau9adcf442011-09-03 01:40:26 +0200254 fastcc = false;
255 caldata = &sc->caldata;
256 }
257
258 if (!hchan) {
259 fastcc = false;
260 flush = true;
261 hchan = ah->curchan;
262 }
263
Felix Fietkau9adcf442011-09-03 01:40:26 +0200264 if (!ath_prepare_reset(sc, retry_tx, flush))
265 fastcc = false;
266
Joe Perchesd2182b62011-12-15 14:55:53 -0800267 ath_dbg(common, CONFIG, "Reset to %u MHz, HT40: %d fastcc: %d\n",
Sujith Manoharanfeced202012-01-30 14:21:42 +0530268 hchan->channel, IS_CHAN_HT40(hchan), fastcc);
Felix Fietkau9adcf442011-09-03 01:40:26 +0200269
270 r = ath9k_hw_reset(ah, hchan, caldata, fastcc);
271 if (r) {
272 ath_err(common,
273 "Unable to reset channel, reset status %d\n", r);
274 goto out;
275 }
276
277 if (!ath_complete_reset(sc, true))
278 r = -EIO;
279
280out:
281 spin_unlock_bh(&sc->sc_pcu_lock);
282 return r;
283}
284
285
Sujithff37e332008-11-24 12:07:55 +0530286/*
287 * Set/change channels. If the channel is really being changed, it's done
288 * by reseting the chip. To accomplish this we must first cleanup any pending
289 * DMA, then restart stuff.
290*/
Mohammed Shafi Shajakhan5595f112011-05-19 18:08:57 +0530291static int ath_set_channel(struct ath_softc *sc, struct ieee80211_hw *hw,
Jouni Malinen0e2dedf2009-03-03 19:23:32 +0200292 struct ath9k_channel *hchan)
Sujithff37e332008-11-24 12:07:55 +0530293{
Luis R. Rodriguezae8d2852008-12-23 15:58:40 -0800294 int r;
Sujithff37e332008-11-24 12:07:55 +0530295
Sujith Manoharan781b14a2012-06-04 20:23:55 +0530296 if (test_bit(SC_OP_INVALID, &sc->sc_flags))
Sujithff37e332008-11-24 12:07:55 +0530297 return -EIO;
298
Felix Fietkau9adcf442011-09-03 01:40:26 +0200299 r = ath_reset_internal(sc, hchan, false);
Luis R. Rodriguez6a6733f2010-10-26 15:27:25 -0700300
Gabor Juhos39892792009-06-15 17:49:09 +0200301 return r;
Sujithff37e332008-11-24 12:07:55 +0530302}
303
Ben Greear7e1e3862011-11-03 11:33:13 -0700304static void ath_node_attach(struct ath_softc *sc, struct ieee80211_sta *sta,
305 struct ieee80211_vif *vif)
Sujithff37e332008-11-24 12:07:55 +0530306{
307 struct ath_node *an;
Sujithff37e332008-11-24 12:07:55 +0530308 an = (struct ath_node *)sta->drv_priv;
309
Ben Greear7f010c92011-01-09 23:11:49 -0800310#ifdef CONFIG_ATH9K_DEBUGFS
311 spin_lock(&sc->nodes_lock);
312 list_add(&an->list, &sc->nodes);
313 spin_unlock(&sc->nodes_lock);
Felix Fietkau156369f2011-12-14 22:08:04 +0100314#endif
Ben Greear7f010c92011-01-09 23:11:49 -0800315 an->sta = sta;
Ben Greear7e1e3862011-11-03 11:33:13 -0700316 an->vif = vif;
Sujith Manoharan3d4e20f2012-03-14 14:40:58 +0530317
Sujith Manoharana4d63672012-03-27 10:08:55 +0530318 if (sc->sc_ah->caps.hw_caps & ATH9K_HW_CAP_HT) {
Sujithff37e332008-11-24 12:07:55 +0530319 ath_tx_node_init(sc, an);
Sujith9e98ac62009-07-23 15:32:34 +0530320 an->maxampdu = 1 << (IEEE80211_HT_MAX_AMPDU_FACTOR +
Sujith87792ef2009-03-30 15:28:48 +0530321 sta->ht_cap.ampdu_factor);
322 an->mpdudensity = parse_mpdudensity(sta->ht_cap.ampdu_density);
323 }
Sujithff37e332008-11-24 12:07:55 +0530324}
325
326static void ath_node_detach(struct ath_softc *sc, struct ieee80211_sta *sta)
327{
328 struct ath_node *an = (struct ath_node *)sta->drv_priv;
329
Ben Greear7f010c92011-01-09 23:11:49 -0800330#ifdef CONFIG_ATH9K_DEBUGFS
331 spin_lock(&sc->nodes_lock);
332 list_del(&an->list);
333 spin_unlock(&sc->nodes_lock);
334 an->sta = NULL;
335#endif
336
Sujith Manoharana4d63672012-03-27 10:08:55 +0530337 if (sc->sc_ah->caps.hw_caps & ATH9K_HW_CAP_HT)
Sujithff37e332008-11-24 12:07:55 +0530338 ath_tx_node_cleanup(sc, an);
339}
340
Sujith55624202010-01-08 10:36:02 +0530341void ath9k_tasklet(unsigned long data)
Sujithff37e332008-11-24 12:07:55 +0530342{
343 struct ath_softc *sc = (struct ath_softc *)data;
Luis R. Rodriguezaf03abe2009-09-09 02:33:11 -0700344 struct ath_hw *ah = sc->sc_ah;
Luis R. Rodriguezc46917b2009-09-13 02:42:02 -0700345 struct ath_common *common = ath9k_hw_common(ah);
Luis R. Rodriguezaf03abe2009-09-09 02:33:11 -0700346
Sujith17d79042009-02-09 13:27:03 +0530347 u32 status = sc->intrstatus;
Felix Fietkaub5c804752010-04-15 17:38:48 -0400348 u32 rxmask;
Sujithff37e332008-11-24 12:07:55 +0530349
Felix Fietkaue3927002011-09-14 21:23:01 +0200350 ath9k_ps_wakeup(sc);
351 spin_lock(&sc->sc_pcu_lock);
352
Rajkumar Manoharana4d86d92011-05-20 17:52:10 +0530353 if ((status & ATH9K_INT_FATAL) ||
354 (status & ATH9K_INT_BB_WATCHDOG)) {
Felix Fietkau030d6292011-10-07 02:28:13 +0200355#ifdef CONFIG_ATH9K_DEBUGFS
356 enum ath_reset_type type;
357
358 if (status & ATH9K_INT_FATAL)
359 type = RESET_TYPE_FATAL_INT;
360 else
361 type = RESET_TYPE_BB_WATCHDOG;
362
363 RESET_STAT_INC(sc, type);
364#endif
Felix Fietkau236de512011-09-03 01:40:25 +0200365 ieee80211_queue_work(sc->hw, &sc->hw_reset_work);
Felix Fietkaue3927002011-09-14 21:23:01 +0200366 goto out;
Sujithff37e332008-11-24 12:07:55 +0530367 }
368
Rajkumar Manoharan4105f802011-05-06 18:27:47 +0530369 if ((status & ATH9K_INT_TSFOOR) && sc->ps_enabled) {
370 /*
371 * TSF sync does not look correct; remain awake to sync with
372 * the next Beacon.
373 */
Joe Perchesd2182b62011-12-15 14:55:53 -0800374 ath_dbg(common, PS, "TSFOOR - Sync with next Beacon\n");
Rajkumar Manoharane8fe7332011-08-05 18:59:41 +0530375 sc->ps_flags |= PS_WAIT_FOR_BEACON | PS_BEACON_SYNC;
Rajkumar Manoharan4105f802011-05-06 18:27:47 +0530376 }
377
Felix Fietkaub5c804752010-04-15 17:38:48 -0400378 if (ah->caps.hw_caps & ATH9K_HW_CAP_EDMA)
379 rxmask = (ATH9K_INT_RXHP | ATH9K_INT_RXLP | ATH9K_INT_RXEOL |
380 ATH9K_INT_RXORN);
381 else
382 rxmask = (ATH9K_INT_RX | ATH9K_INT_RXEOL | ATH9K_INT_RXORN);
383
384 if (status & rxmask) {
Felix Fietkaub5c804752010-04-15 17:38:48 -0400385 /* Check for high priority Rx first */
386 if ((ah->caps.hw_caps & ATH9K_HW_CAP_EDMA) &&
387 (status & ATH9K_INT_RXHP))
388 ath_rx_tasklet(sc, 0, true);
389
390 ath_rx_tasklet(sc, 0, false);
Sujith063d8be2009-03-30 15:28:49 +0530391 }
392
Vasanthakumar Thiagarajane5003242010-04-15 17:39:36 -0400393 if (status & ATH9K_INT_TX) {
394 if (ah->caps.hw_caps & ATH9K_HW_CAP_EDMA)
395 ath_tx_edma_tasklet(sc);
396 else
397 ath_tx_tasklet(sc);
398 }
Sujith063d8be2009-03-30 15:28:49 +0530399
Sujith Manoharan56ca0db2012-02-22 12:40:32 +0530400 ath9k_btcoex_handle_interrupt(sc, status);
Mohammed Shafi Shajakhan19686dd2011-11-30 10:41:28 +0530401
Felix Fietkaue3927002011-09-14 21:23:01 +0200402out:
Sujithff37e332008-11-24 12:07:55 +0530403 /* re-enable hardware interrupt */
Felix Fietkau4df30712010-11-08 20:54:47 +0100404 ath9k_hw_enable_interrupts(ah);
Luis R. Rodriguez6a6733f2010-10-26 15:27:25 -0700405
Senthil Balasubramanian52671e42010-12-23 21:06:57 +0530406 spin_unlock(&sc->sc_pcu_lock);
Vasanthakumar Thiagarajan153e0802009-05-15 02:47:16 -0400407 ath9k_ps_restore(sc);
Sujithff37e332008-11-24 12:07:55 +0530408}
409
Gabor Juhos6baff7f2009-01-14 20:17:06 +0100410irqreturn_t ath_isr(int irq, void *dev)
Sujithff37e332008-11-24 12:07:55 +0530411{
Sujith063d8be2009-03-30 15:28:49 +0530412#define SCHED_INTR ( \
413 ATH9K_INT_FATAL | \
Rajkumar Manoharana4d86d92011-05-20 17:52:10 +0530414 ATH9K_INT_BB_WATCHDOG | \
Sujith063d8be2009-03-30 15:28:49 +0530415 ATH9K_INT_RXORN | \
416 ATH9K_INT_RXEOL | \
417 ATH9K_INT_RX | \
Felix Fietkaub5c804752010-04-15 17:38:48 -0400418 ATH9K_INT_RXLP | \
419 ATH9K_INT_RXHP | \
Sujith063d8be2009-03-30 15:28:49 +0530420 ATH9K_INT_TX | \
421 ATH9K_INT_BMISS | \
422 ATH9K_INT_CST | \
Vasanthakumar Thiagarajanebb8e1d2009-09-01 17:46:32 +0530423 ATH9K_INT_TSFOOR | \
Mohammed Shafi Shajakhan40dc5392011-11-30 10:41:18 +0530424 ATH9K_INT_GENTIMER | \
425 ATH9K_INT_MCI)
Sujith063d8be2009-03-30 15:28:49 +0530426
Sujithff37e332008-11-24 12:07:55 +0530427 struct ath_softc *sc = dev;
Sujithcbe61d82009-02-09 13:27:12 +0530428 struct ath_hw *ah = sc->sc_ah;
Felix Fietkaub5bfc562010-10-08 22:13:53 +0200429 struct ath_common *common = ath9k_hw_common(ah);
Sujithff37e332008-11-24 12:07:55 +0530430 enum ath9k_int status;
431 bool sched = false;
432
Sujith063d8be2009-03-30 15:28:49 +0530433 /*
434 * The hardware is not ready/present, don't
435 * touch anything. Note this can happen early
436 * on if the IRQ is shared.
437 */
Sujith Manoharan781b14a2012-06-04 20:23:55 +0530438 if (test_bit(SC_OP_INVALID, &sc->sc_flags))
Sujith063d8be2009-03-30 15:28:49 +0530439 return IRQ_NONE;
Sujithff37e332008-11-24 12:07:55 +0530440
Sujithff37e332008-11-24 12:07:55 +0530441
Sujith063d8be2009-03-30 15:28:49 +0530442 /* shared irq, not for us */
Sujithff37e332008-11-24 12:07:55 +0530443
Vasanthakumar Thiagarajan153e0802009-05-15 02:47:16 -0400444 if (!ath9k_hw_intrpend(ah))
Sujith063d8be2009-03-30 15:28:49 +0530445 return IRQ_NONE;
Sujithff37e332008-11-24 12:07:55 +0530446
Sujith063d8be2009-03-30 15:28:49 +0530447 /*
448 * Figure out the reason(s) for the interrupt. Note
449 * that the hal returns a pseudo-ISR that may include
450 * bits we haven't explicitly enabled so we mask the
451 * value to insure we only process bits we requested.
452 */
453 ath9k_hw_getisr(ah, &status); /* NB: clears ISR too */
Pavel Roskin30691682010-03-31 18:05:31 -0400454 status &= ah->imask; /* discard unasked-for bits */
Sujith063d8be2009-03-30 15:28:49 +0530455
456 /*
457 * If there are no status bits set, then this interrupt was not
458 * for me (should have been caught above).
459 */
Vasanthakumar Thiagarajan153e0802009-05-15 02:47:16 -0400460 if (!status)
Sujith063d8be2009-03-30 15:28:49 +0530461 return IRQ_NONE;
Sujith063d8be2009-03-30 15:28:49 +0530462
463 /* Cache the status */
464 sc->intrstatus = status;
465
466 if (status & SCHED_INTR)
467 sched = true;
468
469 /*
470 * If a FATAL or RXORN interrupt is received, we have to reset the
471 * chip immediately.
472 */
Felix Fietkaub5c804752010-04-15 17:38:48 -0400473 if ((status & ATH9K_INT_FATAL) || ((status & ATH9K_INT_RXORN) &&
474 !(ah->caps.hw_caps & ATH9K_HW_CAP_EDMA)))
Sujith063d8be2009-03-30 15:28:49 +0530475 goto chip_reset;
476
Luis R. Rodriguez08578b82010-05-13 13:33:44 -0400477 if ((ah->caps.hw_caps & ATH9K_HW_CAP_EDMA) &&
478 (status & ATH9K_INT_BB_WATCHDOG)) {
Felix Fietkaub5bfc562010-10-08 22:13:53 +0200479
480 spin_lock(&common->cc_lock);
481 ath_hw_cycle_counters_update(common);
Luis R. Rodriguez08578b82010-05-13 13:33:44 -0400482 ar9003_hw_bb_watchdog_dbg_info(ah);
Felix Fietkaub5bfc562010-10-08 22:13:53 +0200483 spin_unlock(&common->cc_lock);
484
Luis R. Rodriguez08578b82010-05-13 13:33:44 -0400485 goto chip_reset;
486 }
487
Sujith063d8be2009-03-30 15:28:49 +0530488 if (status & ATH9K_INT_SWBA)
489 tasklet_schedule(&sc->bcon_tasklet);
490
491 if (status & ATH9K_INT_TXURN)
492 ath9k_hw_updatetxtriglevel(ah, true);
493
Rajkumar Manoharan0682c9b2011-08-13 10:28:09 +0530494 if (status & ATH9K_INT_RXEOL) {
495 ah->imask &= ~(ATH9K_INT_RXEOL | ATH9K_INT_RXORN);
Felix Fietkau72d874c2011-10-08 20:06:19 +0200496 ath9k_hw_set_interrupts(ah);
Felix Fietkaub5c804752010-04-15 17:38:48 -0400497 }
498
Sujith063d8be2009-03-30 15:28:49 +0530499 if (status & ATH9K_INT_MIB) {
500 /*
501 * Disable interrupts until we service the MIB
502 * interrupt; otherwise it will continue to
503 * fire.
504 */
Felix Fietkau4df30712010-11-08 20:54:47 +0100505 ath9k_hw_disable_interrupts(ah);
Sujith063d8be2009-03-30 15:28:49 +0530506 /*
507 * Let the hal handle the event. We assume
508 * it will clear whatever condition caused
509 * the interrupt.
510 */
Felix Fietkau88eac2d2010-10-12 16:08:03 +0200511 spin_lock(&common->cc_lock);
Felix Fietkaubfc472b2010-10-04 20:09:48 +0200512 ath9k_hw_proc_mib_event(ah);
Felix Fietkau88eac2d2010-10-12 16:08:03 +0200513 spin_unlock(&common->cc_lock);
Felix Fietkau4df30712010-11-08 20:54:47 +0100514 ath9k_hw_enable_interrupts(ah);
Sujith063d8be2009-03-30 15:28:49 +0530515 }
516
Vasanthakumar Thiagarajan153e0802009-05-15 02:47:16 -0400517 if (!(ah->caps.hw_caps & ATH9K_HW_CAP_AUTOSLEEP))
518 if (status & ATH9K_INT_TIM_TIMER) {
Luis R. Rodriguezff9f0b62010-12-07 15:13:22 -0800519 if (ATH_DBG_WARN_ON_ONCE(sc->ps_idle))
520 goto chip_reset;
Sujith063d8be2009-03-30 15:28:49 +0530521 /* Clear RxAbort bit so that we can
522 * receive frames */
Luis R. Rodriguez9ecdef42009-09-09 21:10:09 -0700523 ath9k_setpower(sc, ATH9K_PM_AWAKE);
Vasanthakumar Thiagarajan153e0802009-05-15 02:47:16 -0400524 ath9k_hw_setrxabort(sc->sc_ah, 0);
Sujith1b04b932010-01-08 10:36:05 +0530525 sc->ps_flags |= PS_WAIT_FOR_BEACON;
Sujith063d8be2009-03-30 15:28:49 +0530526 }
Sujith063d8be2009-03-30 15:28:49 +0530527
528chip_reset:
529
Sujith817e11d2008-12-07 21:42:44 +0530530 ath_debug_stat_interrupt(sc, status);
531
Sujithff37e332008-11-24 12:07:55 +0530532 if (sched) {
Felix Fietkau4df30712010-11-08 20:54:47 +0100533 /* turn off every interrupt */
534 ath9k_hw_disable_interrupts(ah);
Sujithff37e332008-11-24 12:07:55 +0530535 tasklet_schedule(&sc->intr_tq);
536 }
537
538 return IRQ_HANDLED;
Sujith063d8be2009-03-30 15:28:49 +0530539
540#undef SCHED_INTR
Sujithff37e332008-11-24 12:07:55 +0530541}
542
Felix Fietkau236de512011-09-03 01:40:25 +0200543static int ath_reset(struct ath_softc *sc, bool retry_tx)
Sujithff37e332008-11-24 12:07:55 +0530544{
Luis R. Rodriguezae8d2852008-12-23 15:58:40 -0800545 int r;
Sujithff37e332008-11-24 12:07:55 +0530546
Felix Fietkau783cd012011-01-21 18:52:38 +0100547 ath9k_ps_wakeup(sc);
Luis R. Rodriguez6a6733f2010-10-26 15:27:25 -0700548
Felix Fietkau9adcf442011-09-03 01:40:26 +0200549 r = ath_reset_internal(sc, NULL, retry_tx);
Sujithff37e332008-11-24 12:07:55 +0530550
551 if (retry_tx) {
552 int i;
553 for (i = 0; i < ATH9K_NUM_TX_QUEUES; i++) {
554 if (ATH_TXQ_SETUP(sc, i)) {
Sujithb77f4832008-12-07 21:44:03 +0530555 spin_lock_bh(&sc->tx.txq[i].axq_lock);
556 ath_txq_schedule(sc, &sc->tx.txq[i]);
557 spin_unlock_bh(&sc->tx.txq[i].axq_lock);
Sujithff37e332008-11-24 12:07:55 +0530558 }
559 }
560 }
561
Felix Fietkau783cd012011-01-21 18:52:38 +0100562 ath9k_ps_restore(sc);
Sujith2ab81d42009-12-14 16:34:56 +0530563
Luis R. Rodriguezae8d2852008-12-23 15:58:40 -0800564 return r;
Sujithff37e332008-11-24 12:07:55 +0530565}
566
Felix Fietkau236de512011-09-03 01:40:25 +0200567void ath_reset_work(struct work_struct *work)
568{
569 struct ath_softc *sc = container_of(work, struct ath_softc, hw_reset_work);
570
Felix Fietkau236de512011-09-03 01:40:25 +0200571 ath_reset(sc, true);
Felix Fietkau236de512011-09-03 01:40:25 +0200572}
573
Sujithff37e332008-11-24 12:07:55 +0530574/**********************/
575/* mac80211 callbacks */
576/**********************/
577
Luis R. Rodriguezf078f202008-08-04 00:16:41 -0700578static int ath9k_start(struct ieee80211_hw *hw)
579{
Felix Fietkau9ac586152011-01-24 19:23:18 +0100580 struct ath_softc *sc = hw->priv;
Luis R. Rodriguezaf03abe2009-09-09 02:33:11 -0700581 struct ath_hw *ah = sc->sc_ah;
Luis R. Rodriguezc46917b2009-09-13 02:42:02 -0700582 struct ath_common *common = ath9k_hw_common(ah);
Luis R. Rodriguezf078f202008-08-04 00:16:41 -0700583 struct ieee80211_channel *curchan = hw->conf.channel;
Sujithff37e332008-11-24 12:07:55 +0530584 struct ath9k_channel *init_channel;
Vasanthakumar Thiagarajan82880a72009-06-13 14:50:24 +0530585 int r;
Luis R. Rodriguezf078f202008-08-04 00:16:41 -0700586
Joe Perchesd2182b62011-12-15 14:55:53 -0800587 ath_dbg(common, CONFIG,
Joe Perches226afe62010-12-02 19:12:37 -0800588 "Starting driver with initial channel: %d MHz\n",
589 curchan->center_freq);
Luis R. Rodriguezf078f202008-08-04 00:16:41 -0700590
Felix Fietkauf62d8162011-03-25 17:43:41 +0100591 ath9k_ps_wakeup(sc);
Sujith141b38b2009-02-04 08:10:07 +0530592 mutex_lock(&sc->mutex);
593
Rajkumar Manoharanc344c9c2011-01-31 23:47:43 +0530594 init_channel = ath9k_cmn_get_curchannel(hw, ah);
Luis R. Rodriguezf078f202008-08-04 00:16:41 -0700595
Sujithff37e332008-11-24 12:07:55 +0530596 /* Reset SERDES registers */
Stanislaw Gruszka84c87dc2011-08-05 13:10:32 +0200597 ath9k_hw_configpcipowersave(ah, false);
Sujithff37e332008-11-24 12:07:55 +0530598
599 /*
600 * The basic interface to setting the hardware in a good
601 * state is ``reset''. On return the hardware is known to
602 * be powered up and with interrupts disabled. This must
603 * be followed by initialization of the appropriate bits
604 * and then setup of the interrupt mask.
605 */
Luis R. Rodriguez4bdd1e92010-10-26 15:27:24 -0700606 spin_lock_bh(&sc->sc_pcu_lock);
Felix Fietkauc0c11742011-11-16 13:08:41 +0100607
608 atomic_set(&ah->intr_ref_cnt, -1);
609
Felix Fietkau20bd2a02010-07-31 00:12:00 +0200610 r = ath9k_hw_reset(ah, init_channel, ah->caldata, false);
Luis R. Rodriguezae8d2852008-12-23 15:58:40 -0800611 if (r) {
Joe Perches38002762010-12-02 19:12:36 -0800612 ath_err(common,
613 "Unable to reset hardware; reset status %d (freq %u MHz)\n",
614 r, curchan->center_freq);
Luis R. Rodriguez4bdd1e92010-10-26 15:27:24 -0700615 spin_unlock_bh(&sc->sc_pcu_lock);
Sujith141b38b2009-02-04 08:10:07 +0530616 goto mutex_unlock;
Luis R. Rodriguezf078f202008-08-04 00:16:41 -0700617 }
Sujithff37e332008-11-24 12:07:55 +0530618
Sujithff37e332008-11-24 12:07:55 +0530619 /* Setup our intr mask. */
Felix Fietkaub5c804752010-04-15 17:38:48 -0400620 ah->imask = ATH9K_INT_TX | ATH9K_INT_RXEOL |
621 ATH9K_INT_RXORN | ATH9K_INT_FATAL |
622 ATH9K_INT_GLOBAL;
623
624 if (ah->caps.hw_caps & ATH9K_HW_CAP_EDMA)
Luis R. Rodriguez08578b82010-05-13 13:33:44 -0400625 ah->imask |= ATH9K_INT_RXHP |
626 ATH9K_INT_RXLP |
627 ATH9K_INT_BB_WATCHDOG;
Felix Fietkaub5c804752010-04-15 17:38:48 -0400628 else
629 ah->imask |= ATH9K_INT_RX;
Sujithff37e332008-11-24 12:07:55 +0530630
Felix Fietkau364734f2010-09-14 20:22:44 +0200631 ah->imask |= ATH9K_INT_GTT;
Sujithff37e332008-11-24 12:07:55 +0530632
Luis R. Rodriguezaf03abe2009-09-09 02:33:11 -0700633 if (ah->caps.hw_caps & ATH9K_HW_CAP_HT)
Pavel Roskin30691682010-03-31 18:05:31 -0400634 ah->imask |= ATH9K_INT_CST;
Sujithff37e332008-11-24 12:07:55 +0530635
Sujith Manoharane270e772012-06-04 16:27:19 +0530636 ath_mci_enable(sc);
Mohammed Shafi Shajakhan40dc5392011-11-30 10:41:18 +0530637
Sujith Manoharan781b14a2012-06-04 20:23:55 +0530638 clear_bit(SC_OP_INVALID, &sc->sc_flags);
Rajkumar Manoharan5f841b42010-10-27 18:31:15 +0530639 sc->sc_ah->is_monitoring = false;
Sujithff37e332008-11-24 12:07:55 +0530640
Felix Fietkau9adcf442011-09-03 01:40:26 +0200641 if (!ath_complete_reset(sc, false)) {
642 r = -EIO;
643 spin_unlock_bh(&sc->sc_pcu_lock);
644 goto mutex_unlock;
645 }
Luis R. Rodriguezf078f202008-08-04 00:16:41 -0700646
Felix Fietkauc0c11742011-11-16 13:08:41 +0100647 if (ah->led_pin >= 0) {
648 ath9k_hw_cfg_output(ah, ah->led_pin,
649 AR_GPIO_OUTPUT_MUX_AS_OUTPUT);
650 ath9k_hw_set_gpio(ah, ah->led_pin, 0);
651 }
652
653 /*
654 * Reset key cache to sane defaults (all entries cleared) instead of
655 * semi-random values after suspend/resume.
656 */
657 ath9k_cmn_init_crypto(sc->sc_ah);
658
Felix Fietkau9adcf442011-09-03 01:40:26 +0200659 spin_unlock_bh(&sc->sc_pcu_lock);
Senthil Balasubramanian164ace32009-07-14 20:17:09 -0400660
Sujith Manoharandf198b12012-02-22 12:40:27 +0530661 ath9k_start_btcoex(sc);
Vasanthakumar Thiagarajan17739122009-08-26 21:08:50 +0530662
Vasanthakumar Thiagarajan8060e162010-12-06 04:27:42 -0800663 if (ah->caps.pcie_lcr_extsync_en && common->bus_ops->extn_synch_en)
664 common->bus_ops->extn_synch_en(common);
665
Sujith141b38b2009-02-04 08:10:07 +0530666mutex_unlock:
667 mutex_unlock(&sc->mutex);
668
Felix Fietkauf62d8162011-03-25 17:43:41 +0100669 ath9k_ps_restore(sc);
670
Luis R. Rodriguezae8d2852008-12-23 15:58:40 -0800671 return r;
Luis R. Rodriguezf078f202008-08-04 00:16:41 -0700672}
673
Johannes Berg7bb45682011-02-24 14:42:06 +0100674static void ath9k_tx(struct ieee80211_hw *hw, struct sk_buff *skb)
Luis R. Rodriguezf078f202008-08-04 00:16:41 -0700675{
Felix Fietkau9ac586152011-01-24 19:23:18 +0100676 struct ath_softc *sc = hw->priv;
Luis R. Rodriguezc46917b2009-09-13 02:42:02 -0700677 struct ath_common *common = ath9k_hw_common(sc->sc_ah);
Sujith528f0c62008-10-29 10:14:26 +0530678 struct ath_tx_control txctl;
Benoit Papillault1bc14882009-11-24 15:49:18 +0100679 struct ieee80211_hdr *hdr = (struct ieee80211_hdr *) skb->data;
Sujith528f0c62008-10-29 10:14:26 +0530680
Gabor Juhos96148322009-07-24 17:27:21 +0200681 if (sc->ps_enabled) {
Jouni Malinendc8c4582009-05-19 17:01:42 +0300682 /*
683 * mac80211 does not set PM field for normal data frames, so we
684 * need to update that based on the current PS mode.
685 */
686 if (ieee80211_is_data(hdr->frame_control) &&
687 !ieee80211_is_nullfunc(hdr->frame_control) &&
688 !ieee80211_has_pm(hdr->frame_control)) {
Joe Perchesd2182b62011-12-15 14:55:53 -0800689 ath_dbg(common, PS,
Joe Perches226afe62010-12-02 19:12:37 -0800690 "Add PM=1 for a TX frame while in PS mode\n");
Jouni Malinendc8c4582009-05-19 17:01:42 +0300691 hdr->frame_control |= cpu_to_le16(IEEE80211_FCTL_PM);
692 }
693 }
694
Sujith Manoharanad128862012-04-24 10:23:20 +0530695 if (unlikely(sc->sc_ah->power_mode == ATH9K_PM_NETWORK_SLEEP)) {
Jouni Malinen9a23f9c2009-05-19 17:01:38 +0300696 /*
697 * We are using PS-Poll and mac80211 can request TX while in
698 * power save mode. Need to wake up hardware for the TX to be
699 * completed and if needed, also for RX of buffered frames.
700 */
Jouni Malinen9a23f9c2009-05-19 17:01:38 +0300701 ath9k_ps_wakeup(sc);
Vasanthakumar Thiagarajanfdf76622010-05-17 18:57:55 -0700702 if (!(sc->sc_ah->caps.hw_caps & ATH9K_HW_CAP_AUTOSLEEP))
703 ath9k_hw_setrxabort(sc->sc_ah, 0);
Jouni Malinen9a23f9c2009-05-19 17:01:38 +0300704 if (ieee80211_is_pspoll(hdr->frame_control)) {
Joe Perchesd2182b62011-12-15 14:55:53 -0800705 ath_dbg(common, PS,
Joe Perches226afe62010-12-02 19:12:37 -0800706 "Sending PS-Poll to pick a buffered frame\n");
Sujith1b04b932010-01-08 10:36:05 +0530707 sc->ps_flags |= PS_WAIT_FOR_PSPOLL_DATA;
Jouni Malinen9a23f9c2009-05-19 17:01:38 +0300708 } else {
Joe Perchesd2182b62011-12-15 14:55:53 -0800709 ath_dbg(common, PS, "Wake up to complete TX\n");
Sujith1b04b932010-01-08 10:36:05 +0530710 sc->ps_flags |= PS_WAIT_FOR_TX_ACK;
Jouni Malinen9a23f9c2009-05-19 17:01:38 +0300711 }
712 /*
713 * The actual restore operation will happen only after
Sujith Manoharanad128862012-04-24 10:23:20 +0530714 * the ps_flags bit is cleared. We are just dropping
Jouni Malinen9a23f9c2009-05-19 17:01:38 +0300715 * the ps_usecount here.
716 */
717 ath9k_ps_restore(sc);
718 }
719
Sujith Manoharanad128862012-04-24 10:23:20 +0530720 /*
721 * Cannot tx while the hardware is in full sleep, it first needs a full
722 * chip reset to recover from that
723 */
724 if (unlikely(sc->sc_ah->power_mode == ATH9K_PM_FULL_SLEEP)) {
725 ath_err(common, "TX while HW is in FULL_SLEEP mode\n");
726 goto exit;
727 }
728
Sujith528f0c62008-10-29 10:14:26 +0530729 memset(&txctl, 0, sizeof(struct ath_tx_control));
Felix Fietkau066dae92010-11-07 14:59:39 +0100730 txctl.txq = sc->tx.txq_map[skb_get_queue_mapping(skb)];
Sujith528f0c62008-10-29 10:14:26 +0530731
Joe Perchesd2182b62011-12-15 14:55:53 -0800732 ath_dbg(common, XMIT, "transmitting packet, skb: %p\n", skb);
Luis R. Rodriguezf078f202008-08-04 00:16:41 -0700733
Jouni Malinenc52f33d2009-03-03 19:23:29 +0200734 if (ath_tx_start(hw, skb, &txctl) != 0) {
Joe Perchesd2182b62011-12-15 14:55:53 -0800735 ath_dbg(common, XMIT, "TX failed\n");
Ben Greeara5a0bca2012-04-03 09:16:55 -0700736 TX_STAT_INC(txctl.txq->axq_qnum, txfailed);
Sujith528f0c62008-10-29 10:14:26 +0530737 goto exit;
Luis R. Rodriguezf078f202008-08-04 00:16:41 -0700738 }
739
Johannes Berg7bb45682011-02-24 14:42:06 +0100740 return;
Sujith528f0c62008-10-29 10:14:26 +0530741exit:
742 dev_kfree_skb_any(skb);
Luis R. Rodriguezf078f202008-08-04 00:16:41 -0700743}
744
745static void ath9k_stop(struct ieee80211_hw *hw)
746{
Felix Fietkau9ac586152011-01-24 19:23:18 +0100747 struct ath_softc *sc = hw->priv;
Luis R. Rodriguezaf03abe2009-09-09 02:33:11 -0700748 struct ath_hw *ah = sc->sc_ah;
Luis R. Rodriguezc46917b2009-09-13 02:42:02 -0700749 struct ath_common *common = ath9k_hw_common(ah);
Felix Fietkauc0c11742011-11-16 13:08:41 +0100750 bool prev_idle;
Sujith9c84b792008-10-29 10:17:13 +0530751
Sujith4c483812009-08-18 10:51:52 +0530752 mutex_lock(&sc->mutex);
753
Felix Fietkau9adcf442011-09-03 01:40:26 +0200754 ath_cancel_work(sc);
Rajkumar Manoharan01e18912012-03-15 05:34:27 +0530755 del_timer_sync(&sc->rx_poll_timer);
Luis R. Rodriguezc94dbff2009-07-27 11:53:04 -0700756
Sujith Manoharan781b14a2012-06-04 20:23:55 +0530757 if (test_bit(SC_OP_INVALID, &sc->sc_flags)) {
Joe Perchesd2182b62011-12-15 14:55:53 -0800758 ath_dbg(common, ANY, "Device not present\n");
Sujith4c483812009-08-18 10:51:52 +0530759 mutex_unlock(&sc->mutex);
Sujith9c84b792008-10-29 10:17:13 +0530760 return;
761 }
762
Sujith3867cf62009-12-23 20:03:27 -0500763 /* Ensure HW is awake when we try to shut it down. */
764 ath9k_ps_wakeup(sc);
765
Sujith Manoharandf198b12012-02-22 12:40:27 +0530766 ath9k_stop_btcoex(sc);
Vasanthakumar Thiagarajan17739122009-08-26 21:08:50 +0530767
Luis R. Rodriguez6a6733f2010-10-26 15:27:25 -0700768 spin_lock_bh(&sc->sc_pcu_lock);
769
Stanislaw Gruszka203043f2011-01-25 14:08:40 +0100770 /* prevent tasklets to enable interrupts once we disable them */
771 ah->imask &= ~ATH9K_INT_GLOBAL;
772
Sujithff37e332008-11-24 12:07:55 +0530773 /* make sure h/w will not generate any interrupt
774 * before setting the invalid flag. */
Felix Fietkau4df30712010-11-08 20:54:47 +0100775 ath9k_hw_disable_interrupts(ah);
Sujithff37e332008-11-24 12:07:55 +0530776
Luis R. Rodriguez6a6733f2010-10-26 15:27:25 -0700777 spin_unlock_bh(&sc->sc_pcu_lock);
778
Stanislaw Gruszka203043f2011-01-25 14:08:40 +0100779 /* we can now sync irq and kill any running tasklets, since we already
780 * disabled interrupts and not holding a spin lock */
781 synchronize_irq(sc->irq);
782 tasklet_kill(&sc->intr_tq);
783 tasklet_kill(&sc->bcon_tasklet);
784
Felix Fietkauc0c11742011-11-16 13:08:41 +0100785 prev_idle = sc->ps_idle;
786 sc->ps_idle = true;
787
788 spin_lock_bh(&sc->sc_pcu_lock);
789
790 if (ah->led_pin >= 0) {
791 ath9k_hw_set_gpio(ah, ah->led_pin, 1);
792 ath9k_hw_cfg_gpio_input(ah, ah->led_pin);
793 }
794
795 ath_prepare_reset(sc, false, true);
796
797 if (sc->rx.frag) {
798 dev_kfree_skb_any(sc->rx.frag);
799 sc->rx.frag = NULL;
800 }
801
802 if (!ah->curchan)
803 ah->curchan = ath9k_cmn_get_curchannel(hw, ah);
804
805 ath9k_hw_reset(ah, ah->curchan, ah->caldata, false);
806 ath9k_hw_phy_disable(ah);
807
808 ath9k_hw_configpcipowersave(ah, true);
809
810 spin_unlock_bh(&sc->sc_pcu_lock);
811
Sujith3867cf62009-12-23 20:03:27 -0500812 ath9k_ps_restore(sc);
813
Sujith Manoharan781b14a2012-06-04 20:23:55 +0530814 set_bit(SC_OP_INVALID, &sc->sc_flags);
Felix Fietkauc0c11742011-11-16 13:08:41 +0100815 sc->ps_idle = prev_idle;
Luis R. Rodriguezf078f202008-08-04 00:16:41 -0700816
Sujith141b38b2009-02-04 08:10:07 +0530817 mutex_unlock(&sc->mutex);
818
Joe Perchesd2182b62011-12-15 14:55:53 -0800819 ath_dbg(common, CONFIG, "Driver halt\n");
Luis R. Rodriguezf078f202008-08-04 00:16:41 -0700820}
821
Ben Greear48014162011-01-15 19:13:48 +0000822bool ath9k_uses_beacons(int type)
Luis R. Rodriguezf078f202008-08-04 00:16:41 -0700823{
Ben Greear48014162011-01-15 19:13:48 +0000824 switch (type) {
Johannes Berg05c914f2008-09-11 00:01:58 +0200825 case NL80211_IFTYPE_AP:
Ben Greear48014162011-01-15 19:13:48 +0000826 case NL80211_IFTYPE_ADHOC:
Pat Erley9cb54122009-03-20 22:59:59 -0400827 case NL80211_IFTYPE_MESH_POINT:
Ben Greear48014162011-01-15 19:13:48 +0000828 return true;
Luis R. Rodriguezf078f202008-08-04 00:16:41 -0700829 default:
Ben Greear48014162011-01-15 19:13:48 +0000830 return false;
Luis R. Rodriguezf078f202008-08-04 00:16:41 -0700831 }
Luis R. Rodriguezf078f202008-08-04 00:16:41 -0700832}
833
Rajkumar Manoharan6b3b9912010-12-08 19:38:55 +0530834static void ath9k_reclaim_beacon(struct ath_softc *sc,
835 struct ieee80211_vif *vif)
836{
837 struct ath_vif *avp = (void *)vif->drv_priv;
838
Rajkumar Manoharan014cf3b2011-02-09 17:46:39 +0530839 ath9k_set_beaconing_status(sc, false);
Rajkumar Manoharan6b3b9912010-12-08 19:38:55 +0530840 ath_beacon_return(sc, avp);
Rajkumar Manoharan014cf3b2011-02-09 17:46:39 +0530841 ath9k_set_beaconing_status(sc, true);
Rajkumar Manoharan6b3b9912010-12-08 19:38:55 +0530842}
843
Ben Greear48014162011-01-15 19:13:48 +0000844static void ath9k_vif_iter(void *data, u8 *mac, struct ieee80211_vif *vif)
845{
846 struct ath9k_vif_iter_data *iter_data = data;
847 int i;
848
849 if (iter_data->hw_macaddr)
850 for (i = 0; i < ETH_ALEN; i++)
851 iter_data->mask[i] &=
852 ~(iter_data->hw_macaddr[i] ^ mac[i]);
853
854 switch (vif->type) {
855 case NL80211_IFTYPE_AP:
856 iter_data->naps++;
857 break;
858 case NL80211_IFTYPE_STATION:
859 iter_data->nstations++;
860 break;
861 case NL80211_IFTYPE_ADHOC:
862 iter_data->nadhocs++;
863 break;
864 case NL80211_IFTYPE_MESH_POINT:
865 iter_data->nmeshes++;
866 break;
867 case NL80211_IFTYPE_WDS:
868 iter_data->nwds++;
869 break;
870 default:
Ben Greear48014162011-01-15 19:13:48 +0000871 break;
872 }
873}
874
875/* Called with sc->mutex held. */
876void ath9k_calculate_iter_data(struct ieee80211_hw *hw,
877 struct ieee80211_vif *vif,
878 struct ath9k_vif_iter_data *iter_data)
879{
Felix Fietkau9ac586152011-01-24 19:23:18 +0100880 struct ath_softc *sc = hw->priv;
Ben Greear48014162011-01-15 19:13:48 +0000881 struct ath_hw *ah = sc->sc_ah;
882 struct ath_common *common = ath9k_hw_common(ah);
Ben Greear48014162011-01-15 19:13:48 +0000883
884 /*
885 * Use the hardware MAC address as reference, the hardware uses it
886 * together with the BSSID mask when matching addresses.
887 */
888 memset(iter_data, 0, sizeof(*iter_data));
889 iter_data->hw_macaddr = common->macaddr;
890 memset(&iter_data->mask, 0xff, ETH_ALEN);
891
892 if (vif)
893 ath9k_vif_iter(iter_data, vif->addr, vif);
894
895 /* Get list of all active MAC addresses */
Ben Greear48014162011-01-15 19:13:48 +0000896 ieee80211_iterate_active_interfaces_atomic(sc->hw, ath9k_vif_iter,
897 iter_data);
Ben Greear48014162011-01-15 19:13:48 +0000898}
899
900/* Called with sc->mutex held. */
901static void ath9k_calculate_summary_state(struct ieee80211_hw *hw,
902 struct ieee80211_vif *vif)
903{
Felix Fietkau9ac586152011-01-24 19:23:18 +0100904 struct ath_softc *sc = hw->priv;
Ben Greear48014162011-01-15 19:13:48 +0000905 struct ath_hw *ah = sc->sc_ah;
906 struct ath_common *common = ath9k_hw_common(ah);
907 struct ath9k_vif_iter_data iter_data;
908
909 ath9k_calculate_iter_data(hw, vif, &iter_data);
910
911 /* Set BSSID mask. */
912 memcpy(common->bssidmask, iter_data.mask, ETH_ALEN);
913 ath_hw_setbssidmask(common);
914
915 /* Set op-mode & TSF */
916 if (iter_data.naps > 0) {
917 ath9k_hw_set_tsfadjust(ah, 1);
Sujith Manoharan781b14a2012-06-04 20:23:55 +0530918 set_bit(SC_OP_TSF_RESET, &sc->sc_flags);
Ben Greear48014162011-01-15 19:13:48 +0000919 ah->opmode = NL80211_IFTYPE_AP;
920 } else {
921 ath9k_hw_set_tsfadjust(ah, 0);
Sujith Manoharan781b14a2012-06-04 20:23:55 +0530922 clear_bit(SC_OP_TSF_RESET, &sc->sc_flags);
Ben Greear48014162011-01-15 19:13:48 +0000923
Javier Cardonafd5999c2011-05-03 16:57:19 -0700924 if (iter_data.nmeshes)
925 ah->opmode = NL80211_IFTYPE_MESH_POINT;
926 else if (iter_data.nwds)
Ben Greear48014162011-01-15 19:13:48 +0000927 ah->opmode = NL80211_IFTYPE_AP;
928 else if (iter_data.nadhocs)
929 ah->opmode = NL80211_IFTYPE_ADHOC;
930 else
931 ah->opmode = NL80211_IFTYPE_STATION;
932 }
933
934 /*
935 * Enable MIB interrupts when there are hardware phy counters.
936 */
937 if ((iter_data.nstations + iter_data.nadhocs + iter_data.nmeshes) > 0) {
938 if (ah->config.enable_ani)
939 ah->imask |= ATH9K_INT_MIB;
940 ah->imask |= ATH9K_INT_TSFOOR;
941 } else {
942 ah->imask &= ~ATH9K_INT_MIB;
943 ah->imask &= ~ATH9K_INT_TSFOOR;
944 }
945
Felix Fietkau72d874c2011-10-08 20:06:19 +0200946 ath9k_hw_set_interrupts(ah);
Ben Greear48014162011-01-15 19:13:48 +0000947
948 /* Set up ANI */
Rajkumar Manoharan2e5ef452011-05-20 17:52:12 +0530949 if (iter_data.naps > 0) {
Rajkumar Manoharan729da3902011-05-09 19:11:29 +0530950 sc->sc_ah->stats.avgbrssi = ATH_RSSI_DUMMY_MARKER;
Mohammed Shafi Shajakhan05c0be22011-05-26 10:56:15 +0530951
952 if (!common->disable_ani) {
Sujith Manoharan781b14a2012-06-04 20:23:55 +0530953 set_bit(SC_OP_ANI_RUN, &sc->sc_flags);
Mohammed Shafi Shajakhan05c0be22011-05-26 10:56:15 +0530954 ath_start_ani(common);
955 }
956
Rajkumar Manoharanf60c49b2011-04-08 17:06:25 +0530957 } else {
Sujith Manoharan781b14a2012-06-04 20:23:55 +0530958 clear_bit(SC_OP_ANI_RUN, &sc->sc_flags);
Rajkumar Manoharanf60c49b2011-04-08 17:06:25 +0530959 del_timer_sync(&common->ani.timer);
Ben Greear48014162011-01-15 19:13:48 +0000960 }
961}
962
963/* Called with sc->mutex held, vif counts set up properly. */
964static void ath9k_do_vif_add_setup(struct ieee80211_hw *hw,
965 struct ieee80211_vif *vif)
966{
Felix Fietkau9ac586152011-01-24 19:23:18 +0100967 struct ath_softc *sc = hw->priv;
Ben Greear48014162011-01-15 19:13:48 +0000968
969 ath9k_calculate_summary_state(hw, vif);
970
971 if (ath9k_uses_beacons(vif->type)) {
Rajkumar Manoharaned2578c2012-04-19 19:13:51 +0530972 /* Reserve a beacon slot for the vif */
Rajkumar Manoharan014cf3b2011-02-09 17:46:39 +0530973 ath9k_set_beaconing_status(sc, false);
Rajkumar Manoharaned2578c2012-04-19 19:13:51 +0530974 ath_beacon_alloc(sc, vif);
Rajkumar Manoharan014cf3b2011-02-09 17:46:39 +0530975 ath9k_set_beaconing_status(sc, true);
Ben Greear48014162011-01-15 19:13:48 +0000976 }
977}
978
Ben Greear48014162011-01-15 19:13:48 +0000979static int ath9k_add_interface(struct ieee80211_hw *hw,
980 struct ieee80211_vif *vif)
981{
Felix Fietkau9ac586152011-01-24 19:23:18 +0100982 struct ath_softc *sc = hw->priv;
Ben Greear48014162011-01-15 19:13:48 +0000983 struct ath_hw *ah = sc->sc_ah;
984 struct ath_common *common = ath9k_hw_common(ah);
Ben Greear48014162011-01-15 19:13:48 +0000985 int ret = 0;
986
Felix Fietkau96f372c2011-04-07 19:07:17 +0200987 ath9k_ps_wakeup(sc);
Ben Greear48014162011-01-15 19:13:48 +0000988 mutex_lock(&sc->mutex);
989
990 switch (vif->type) {
991 case NL80211_IFTYPE_STATION:
992 case NL80211_IFTYPE_WDS:
993 case NL80211_IFTYPE_ADHOC:
994 case NL80211_IFTYPE_AP:
995 case NL80211_IFTYPE_MESH_POINT:
996 break;
997 default:
998 ath_err(common, "Interface type %d not yet supported\n",
999 vif->type);
1000 ret = -EOPNOTSUPP;
1001 goto out;
1002 }
1003
1004 if (ath9k_uses_beacons(vif->type)) {
1005 if (sc->nbcnvifs >= ATH_BCBUF) {
1006 ath_err(common, "Not enough beacon buffers when adding"
1007 " new interface of type: %i\n",
1008 vif->type);
1009 ret = -ENOBUFS;
1010 goto out;
1011 }
1012 }
1013
Rajkumar Manoharan59575d12011-04-04 22:56:16 +05301014 if ((ah->opmode == NL80211_IFTYPE_ADHOC) ||
1015 ((vif->type == NL80211_IFTYPE_ADHOC) &&
1016 sc->nvifs > 0)) {
Ben Greear48014162011-01-15 19:13:48 +00001017 ath_err(common, "Cannot create ADHOC interface when other"
1018 " interfaces already exist.\n");
1019 ret = -EINVAL;
1020 goto out;
1021 }
1022
Joe Perchesd2182b62011-12-15 14:55:53 -08001023 ath_dbg(common, CONFIG, "Attach a VIF of type: %d\n", vif->type);
Ben Greear48014162011-01-15 19:13:48 +00001024
Ben Greear48014162011-01-15 19:13:48 +00001025 sc->nvifs++;
1026
1027 ath9k_do_vif_add_setup(hw, vif);
1028out:
1029 mutex_unlock(&sc->mutex);
Felix Fietkau96f372c2011-04-07 19:07:17 +02001030 ath9k_ps_restore(sc);
Ben Greear48014162011-01-15 19:13:48 +00001031 return ret;
1032}
1033
Rajkumar Manoharan6b3b9912010-12-08 19:38:55 +05301034static int ath9k_change_interface(struct ieee80211_hw *hw,
1035 struct ieee80211_vif *vif,
1036 enum nl80211_iftype new_type,
1037 bool p2p)
1038{
Felix Fietkau9ac586152011-01-24 19:23:18 +01001039 struct ath_softc *sc = hw->priv;
Rajkumar Manoharan6b3b9912010-12-08 19:38:55 +05301040 struct ath_common *common = ath9k_hw_common(sc->sc_ah);
Dan Carpenter6dab55b2010-12-21 06:59:06 +03001041 int ret = 0;
Rajkumar Manoharan6b3b9912010-12-08 19:38:55 +05301042
Joe Perchesd2182b62011-12-15 14:55:53 -08001043 ath_dbg(common, CONFIG, "Change Interface\n");
Rajkumar Manoharan6b3b9912010-12-08 19:38:55 +05301044 mutex_lock(&sc->mutex);
Felix Fietkau96f372c2011-04-07 19:07:17 +02001045 ath9k_ps_wakeup(sc);
Rajkumar Manoharan6b3b9912010-12-08 19:38:55 +05301046
Ben Greear48014162011-01-15 19:13:48 +00001047 /* See if new interface type is valid. */
1048 if ((new_type == NL80211_IFTYPE_ADHOC) &&
1049 (sc->nvifs > 1)) {
1050 ath_err(common, "When using ADHOC, it must be the only"
1051 " interface.\n");
1052 ret = -EINVAL;
1053 goto out;
1054 }
1055
1056 if (ath9k_uses_beacons(new_type) &&
1057 !ath9k_uses_beacons(vif->type)) {
Rajkumar Manoharan6b3b9912010-12-08 19:38:55 +05301058 if (sc->nbcnvifs >= ATH_BCBUF) {
1059 ath_err(common, "No beacon slot available\n");
Dan Carpenter6dab55b2010-12-21 06:59:06 +03001060 ret = -ENOBUFS;
1061 goto out;
Rajkumar Manoharan6b3b9912010-12-08 19:38:55 +05301062 }
Rajkumar Manoharan6b3b9912010-12-08 19:38:55 +05301063 }
Ben Greear48014162011-01-15 19:13:48 +00001064
1065 /* Clean up old vif stuff */
1066 if (ath9k_uses_beacons(vif->type))
1067 ath9k_reclaim_beacon(sc, vif);
1068
1069 /* Add new settings */
Rajkumar Manoharan6b3b9912010-12-08 19:38:55 +05301070 vif->type = new_type;
1071 vif->p2p = p2p;
1072
Ben Greear48014162011-01-15 19:13:48 +00001073 ath9k_do_vif_add_setup(hw, vif);
Dan Carpenter6dab55b2010-12-21 06:59:06 +03001074out:
Felix Fietkau96f372c2011-04-07 19:07:17 +02001075 ath9k_ps_restore(sc);
Rajkumar Manoharan6b3b9912010-12-08 19:38:55 +05301076 mutex_unlock(&sc->mutex);
Dan Carpenter6dab55b2010-12-21 06:59:06 +03001077 return ret;
Rajkumar Manoharan6b3b9912010-12-08 19:38:55 +05301078}
1079
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07001080static void ath9k_remove_interface(struct ieee80211_hw *hw,
Johannes Berg1ed32e42009-12-23 13:15:45 +01001081 struct ieee80211_vif *vif)
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07001082{
Felix Fietkau9ac586152011-01-24 19:23:18 +01001083 struct ath_softc *sc = hw->priv;
Luis R. Rodriguezc46917b2009-09-13 02:42:02 -07001084 struct ath_common *common = ath9k_hw_common(sc->sc_ah);
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07001085
Joe Perchesd2182b62011-12-15 14:55:53 -08001086 ath_dbg(common, CONFIG, "Detach Interface\n");
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07001087
Felix Fietkau96f372c2011-04-07 19:07:17 +02001088 ath9k_ps_wakeup(sc);
Sujith141b38b2009-02-04 08:10:07 +05301089 mutex_lock(&sc->mutex);
1090
Ben Greear48014162011-01-15 19:13:48 +00001091 sc->nvifs--;
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07001092
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07001093 /* Reclaim beacon resources */
Ben Greear48014162011-01-15 19:13:48 +00001094 if (ath9k_uses_beacons(vif->type))
Rajkumar Manoharan6b3b9912010-12-08 19:38:55 +05301095 ath9k_reclaim_beacon(sc, vif);
Jouni Malinen2c3db3d2009-03-03 19:23:26 +02001096
Ben Greear48014162011-01-15 19:13:48 +00001097 ath9k_calculate_summary_state(hw, NULL);
Sujith141b38b2009-02-04 08:10:07 +05301098
1099 mutex_unlock(&sc->mutex);
Felix Fietkau96f372c2011-04-07 19:07:17 +02001100 ath9k_ps_restore(sc);
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07001101}
1102
Senthil Balasubramanianfbab7392010-10-05 20:36:40 +05301103static void ath9k_enable_ps(struct ath_softc *sc)
Senthil Balasubramanian3f7c5c12010-02-03 22:51:13 +05301104{
Pavel Roskin30691682010-03-31 18:05:31 -04001105 struct ath_hw *ah = sc->sc_ah;
Sujith Manoharanad128862012-04-24 10:23:20 +05301106 struct ath_common *common = ath9k_hw_common(ah);
Pavel Roskin30691682010-03-31 18:05:31 -04001107
Senthil Balasubramanian3f7c5c12010-02-03 22:51:13 +05301108 sc->ps_enabled = true;
Pavel Roskin30691682010-03-31 18:05:31 -04001109 if (!(ah->caps.hw_caps & ATH9K_HW_CAP_AUTOSLEEP)) {
1110 if ((ah->imask & ATH9K_INT_TIM_TIMER) == 0) {
1111 ah->imask |= ATH9K_INT_TIM_TIMER;
Felix Fietkau72d874c2011-10-08 20:06:19 +02001112 ath9k_hw_set_interrupts(ah);
Senthil Balasubramanian3f7c5c12010-02-03 22:51:13 +05301113 }
Vasanthakumar Thiagarajanfdf76622010-05-17 18:57:55 -07001114 ath9k_hw_setrxabort(ah, 1);
Senthil Balasubramanian3f7c5c12010-02-03 22:51:13 +05301115 }
Sujith Manoharanad128862012-04-24 10:23:20 +05301116 ath_dbg(common, PS, "PowerSave enabled\n");
Senthil Balasubramanian3f7c5c12010-02-03 22:51:13 +05301117}
1118
Senthil Balasubramanian845d7082010-10-05 20:36:41 +05301119static void ath9k_disable_ps(struct ath_softc *sc)
1120{
1121 struct ath_hw *ah = sc->sc_ah;
Sujith Manoharanad128862012-04-24 10:23:20 +05301122 struct ath_common *common = ath9k_hw_common(ah);
Senthil Balasubramanian845d7082010-10-05 20:36:41 +05301123
1124 sc->ps_enabled = false;
1125 ath9k_hw_setpower(ah, ATH9K_PM_AWAKE);
1126 if (!(ah->caps.hw_caps & ATH9K_HW_CAP_AUTOSLEEP)) {
1127 ath9k_hw_setrxabort(ah, 0);
1128 sc->ps_flags &= ~(PS_WAIT_FOR_BEACON |
1129 PS_WAIT_FOR_CAB |
1130 PS_WAIT_FOR_PSPOLL_DATA |
1131 PS_WAIT_FOR_TX_ACK);
1132 if (ah->imask & ATH9K_INT_TIM_TIMER) {
1133 ah->imask &= ~ATH9K_INT_TIM_TIMER;
Felix Fietkau72d874c2011-10-08 20:06:19 +02001134 ath9k_hw_set_interrupts(ah);
Senthil Balasubramanian845d7082010-10-05 20:36:41 +05301135 }
1136 }
Sujith Manoharanad128862012-04-24 10:23:20 +05301137 ath_dbg(common, PS, "PowerSave disabled\n");
Senthil Balasubramanian845d7082010-10-05 20:36:41 +05301138}
1139
Johannes Berge8975582008-10-09 12:18:51 +02001140static int ath9k_config(struct ieee80211_hw *hw, u32 changed)
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07001141{
Felix Fietkau9ac586152011-01-24 19:23:18 +01001142 struct ath_softc *sc = hw->priv;
Felix Fietkau34300982010-10-10 18:21:52 +02001143 struct ath_hw *ah = sc->sc_ah;
1144 struct ath_common *common = ath9k_hw_common(ah);
Johannes Berge8975582008-10-09 12:18:51 +02001145 struct ieee80211_conf *conf = &hw->conf;
Felix Fietkau75600ab2012-04-12 20:36:31 +02001146 bool reset_channel = false;
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07001147
Felix Fietkauc0c11742011-11-16 13:08:41 +01001148 ath9k_ps_wakeup(sc);
Sujithaa33de02008-12-18 11:40:16 +05301149 mutex_lock(&sc->mutex);
Sujith141b38b2009-02-04 08:10:07 +05301150
Felix Fietkaudaa1b6e2011-11-16 13:08:43 +01001151 if (changed & IEEE80211_CONF_CHANGE_IDLE) {
Felix Fietkau7545daf2011-01-24 19:23:16 +01001152 sc->ps_idle = !!(conf->flags & IEEE80211_CONF_IDLE);
Felix Fietkaudaa1b6e2011-11-16 13:08:43 +01001153 if (sc->ps_idle)
1154 ath_cancel_work(sc);
Felix Fietkau75600ab2012-04-12 20:36:31 +02001155 else
1156 /*
1157 * The chip needs a reset to properly wake up from
1158 * full sleep
1159 */
1160 reset_channel = ah->chip_fullsleep;
Felix Fietkaudaa1b6e2011-11-16 13:08:43 +01001161 }
Luis R. Rodriguez64839172009-07-14 20:22:53 -04001162
Luis R. Rodrigueze7824a52009-11-24 02:53:25 -05001163 /*
1164 * We just prepare to enable PS. We have to wait until our AP has
1165 * ACK'd our null data frame to disable RX otherwise we'll ignore
1166 * those ACKs and end up retransmitting the same null data frames.
1167 * IEEE80211_CONF_CHANGE_PS is only passed by mac80211 for STA mode.
1168 */
Vivek Natarajan3cbb5dd2009-01-20 11:17:08 +05301169 if (changed & IEEE80211_CONF_CHANGE_PS) {
Luis R. Rodriguez8ab2cd02010-09-16 15:12:26 -04001170 unsigned long flags;
1171 spin_lock_irqsave(&sc->sc_pm_lock, flags);
Senthil Balasubramanianfbab7392010-10-05 20:36:40 +05301172 if (conf->flags & IEEE80211_CONF_PS)
1173 ath9k_enable_ps(sc);
Senthil Balasubramanian845d7082010-10-05 20:36:41 +05301174 else
1175 ath9k_disable_ps(sc);
Luis R. Rodriguez8ab2cd02010-09-16 15:12:26 -04001176 spin_unlock_irqrestore(&sc->sc_pm_lock, flags);
Vivek Natarajan3cbb5dd2009-01-20 11:17:08 +05301177 }
1178
Sujith199afd92010-01-08 10:36:13 +05301179 if (changed & IEEE80211_CONF_CHANGE_MONITOR) {
1180 if (conf->flags & IEEE80211_CONF_MONITOR) {
Joe Perchesd2182b62011-12-15 14:55:53 -08001181 ath_dbg(common, CONFIG, "Monitor mode is enabled\n");
Rajkumar Manoharan5f841b42010-10-27 18:31:15 +05301182 sc->sc_ah->is_monitoring = true;
1183 } else {
Joe Perchesd2182b62011-12-15 14:55:53 -08001184 ath_dbg(common, CONFIG, "Monitor mode is disabled\n");
Rajkumar Manoharan5f841b42010-10-27 18:31:15 +05301185 sc->sc_ah->is_monitoring = false;
Sujith199afd92010-01-08 10:36:13 +05301186 }
1187 }
1188
Felix Fietkau75600ab2012-04-12 20:36:31 +02001189 if ((changed & IEEE80211_CONF_CHANGE_CHANNEL) || reset_channel) {
Sujith99405f92008-11-24 12:08:35 +05301190 struct ieee80211_channel *curchan = hw->conf.channel;
Luis R. Rodriguez5f8e0772009-01-22 15:16:48 -08001191 int pos = curchan->hw_value;
Felix Fietkau34300982010-10-10 18:21:52 +02001192 int old_pos = -1;
1193 unsigned long flags;
1194
1195 if (ah->curchan)
1196 old_pos = ah->curchan - &ah->channels[0];
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07001197
Joe Perchesd2182b62011-12-15 14:55:53 -08001198 ath_dbg(common, CONFIG, "Set channel: %d MHz type: %d\n",
Ben Greear8c79a612011-02-07 13:44:33 -08001199 curchan->center_freq, conf->channel_type);
Johannes Bergae5eb022008-10-14 16:58:37 +02001200
Felix Fietkau34300982010-10-10 18:21:52 +02001201 /* update survey stats for the old channel before switching */
1202 spin_lock_irqsave(&common->cc_lock, flags);
1203 ath_update_survey_stats(sc);
1204 spin_unlock_irqrestore(&common->cc_lock, flags);
1205
1206 /*
Rajkumar Manoharane338a852011-08-13 10:28:17 +05301207 * Preserve the current channel values, before updating
1208 * the same channel
1209 */
Rajkumar Manoharan1a19f772012-01-09 15:37:53 +05301210 if (ah->curchan && (old_pos == pos))
1211 ath9k_hw_getnf(ah, ah->curchan);
Rajkumar Manoharane338a852011-08-13 10:28:17 +05301212
1213 ath9k_cmn_update_ichannel(&sc->sc_ah->channels[pos],
1214 curchan, conf->channel_type);
1215
1216 /*
Felix Fietkau34300982010-10-10 18:21:52 +02001217 * If the operating channel changes, change the survey in-use flags
1218 * along with it.
1219 * Reset the survey data for the new channel, unless we're switching
1220 * back to the operating channel from an off-channel operation.
1221 */
1222 if (!(hw->conf.flags & IEEE80211_CONF_OFFCHANNEL) &&
1223 sc->cur_survey != &sc->survey[pos]) {
1224
1225 if (sc->cur_survey)
1226 sc->cur_survey->filled &= ~SURVEY_INFO_IN_USE;
1227
1228 sc->cur_survey = &sc->survey[pos];
1229
1230 memset(sc->cur_survey, 0, sizeof(struct survey_info));
1231 sc->cur_survey->filled |= SURVEY_INFO_IN_USE;
1232 } else if (!(sc->survey[pos].filled & SURVEY_INFO_IN_USE)) {
1233 memset(&sc->survey[pos], 0, sizeof(struct survey_info));
1234 }
1235
Jouni Malinen0e2dedf2009-03-03 19:23:32 +02001236 if (ath_set_channel(sc, hw, &sc->sc_ah->channels[pos]) < 0) {
Joe Perches38002762010-12-02 19:12:36 -08001237 ath_err(common, "Unable to set channel\n");
Sujithaa33de02008-12-18 11:40:16 +05301238 mutex_unlock(&sc->mutex);
Sujithe11602b2008-11-27 09:46:27 +05301239 return -EINVAL;
1240 }
Felix Fietkau34300982010-10-10 18:21:52 +02001241
1242 /*
1243 * The most recent snapshot of channel->noisefloor for the old
1244 * channel is only available after the hardware reset. Copy it to
1245 * the survey stats now.
1246 */
1247 if (old_pos >= 0)
1248 ath_update_survey_nf(sc, old_pos);
Sujith094d05d2008-12-12 11:57:43 +05301249 }
Sujith86b89ee2008-08-07 10:54:57 +05301250
Luis R. Rodriguezc9f6a652010-01-19 14:04:19 -05001251 if (changed & IEEE80211_CONF_CHANGE_POWER) {
Joe Perchesd2182b62011-12-15 14:55:53 -08001252 ath_dbg(common, CONFIG, "Set power: %d\n", conf->power_level);
Sujith17d79042009-02-09 13:27:03 +05301253 sc->config.txpowlimit = 2 * conf->power_level;
Rajkumar Manoharan5048e8c2011-01-31 23:47:44 +05301254 ath9k_cmn_update_txpow(ah, sc->curtxpow,
1255 sc->config.txpowlimit, &sc->curtxpow);
Luis R. Rodriguez64839172009-07-14 20:22:53 -04001256 }
1257
Sujithaa33de02008-12-18 11:40:16 +05301258 mutex_unlock(&sc->mutex);
Felix Fietkauc0c11742011-11-16 13:08:41 +01001259 ath9k_ps_restore(sc);
Sujith141b38b2009-02-04 08:10:07 +05301260
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07001261 return 0;
1262}
1263
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07001264#define SUPPORTED_FILTERS \
1265 (FIF_PROMISC_IN_BSS | \
1266 FIF_ALLMULTI | \
1267 FIF_CONTROL | \
Luis R. Rodriguezaf6a3fc2009-08-08 21:55:16 -04001268 FIF_PSPOLL | \
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07001269 FIF_OTHER_BSS | \
1270 FIF_BCN_PRBRESP_PROMISC | \
Jouni Malinen9c1d8e42010-10-13 17:29:31 +03001271 FIF_PROBE_REQ | \
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07001272 FIF_FCSFAIL)
1273
Sujith7dcfdcd2008-08-11 14:03:13 +05301274/* FIXME: sc->sc_full_reset ? */
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07001275static void ath9k_configure_filter(struct ieee80211_hw *hw,
1276 unsigned int changed_flags,
1277 unsigned int *total_flags,
Johannes Berg3ac64be2009-08-17 16:16:53 +02001278 u64 multicast)
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07001279{
Felix Fietkau9ac586152011-01-24 19:23:18 +01001280 struct ath_softc *sc = hw->priv;
Sujith7dcfdcd2008-08-11 14:03:13 +05301281 u32 rfilt;
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07001282
1283 changed_flags &= SUPPORTED_FILTERS;
1284 *total_flags &= SUPPORTED_FILTERS;
1285
Sujithb77f4832008-12-07 21:44:03 +05301286 sc->rx.rxfilter = *total_flags;
Jouni Malinenaa68aea2009-05-19 17:01:41 +03001287 ath9k_ps_wakeup(sc);
Sujith7dcfdcd2008-08-11 14:03:13 +05301288 rfilt = ath_calcrxfilter(sc);
1289 ath9k_hw_setrxfilter(sc->sc_ah, rfilt);
Jouni Malinenaa68aea2009-05-19 17:01:41 +03001290 ath9k_ps_restore(sc);
Sujith7dcfdcd2008-08-11 14:03:13 +05301291
Joe Perchesd2182b62011-12-15 14:55:53 -08001292 ath_dbg(ath9k_hw_common(sc->sc_ah), CONFIG, "Set HW RX filter: 0x%x\n",
1293 rfilt);
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07001294}
1295
Johannes Berg4ca77862010-02-19 19:06:56 +01001296static int ath9k_sta_add(struct ieee80211_hw *hw,
1297 struct ieee80211_vif *vif,
1298 struct ieee80211_sta *sta)
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07001299{
Felix Fietkau9ac586152011-01-24 19:23:18 +01001300 struct ath_softc *sc = hw->priv;
Felix Fietkau93ae2dd2011-04-17 23:28:10 +02001301 struct ath_common *common = ath9k_hw_common(sc->sc_ah);
1302 struct ath_node *an = (struct ath_node *) sta->drv_priv;
1303 struct ieee80211_key_conf ps_key = { };
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07001304
Ben Greear7e1e3862011-11-03 11:33:13 -07001305 ath_node_attach(sc, sta, vif);
Felix Fietkauf59a59f2011-05-10 20:52:22 +02001306
1307 if (vif->type != NL80211_IFTYPE_AP &&
1308 vif->type != NL80211_IFTYPE_AP_VLAN)
1309 return 0;
1310
Felix Fietkau93ae2dd2011-04-17 23:28:10 +02001311 an->ps_key = ath_key_config(common, vif, sta, &ps_key);
Johannes Berg4ca77862010-02-19 19:06:56 +01001312
1313 return 0;
1314}
1315
Felix Fietkau93ae2dd2011-04-17 23:28:10 +02001316static void ath9k_del_ps_key(struct ath_softc *sc,
1317 struct ieee80211_vif *vif,
1318 struct ieee80211_sta *sta)
1319{
1320 struct ath_common *common = ath9k_hw_common(sc->sc_ah);
1321 struct ath_node *an = (struct ath_node *) sta->drv_priv;
1322 struct ieee80211_key_conf ps_key = { .hw_key_idx = an->ps_key };
1323
1324 if (!an->ps_key)
1325 return;
1326
1327 ath_key_delete(common, &ps_key);
1328}
1329
Johannes Berg4ca77862010-02-19 19:06:56 +01001330static int ath9k_sta_remove(struct ieee80211_hw *hw,
1331 struct ieee80211_vif *vif,
1332 struct ieee80211_sta *sta)
1333{
Felix Fietkau9ac586152011-01-24 19:23:18 +01001334 struct ath_softc *sc = hw->priv;
Johannes Berg4ca77862010-02-19 19:06:56 +01001335
Felix Fietkau93ae2dd2011-04-17 23:28:10 +02001336 ath9k_del_ps_key(sc, vif, sta);
Johannes Berg4ca77862010-02-19 19:06:56 +01001337 ath_node_detach(sc, sta);
1338
1339 return 0;
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07001340}
1341
Felix Fietkau55195412011-04-17 23:28:09 +02001342static void ath9k_sta_notify(struct ieee80211_hw *hw,
1343 struct ieee80211_vif *vif,
1344 enum sta_notify_cmd cmd,
1345 struct ieee80211_sta *sta)
1346{
1347 struct ath_softc *sc = hw->priv;
1348 struct ath_node *an = (struct ath_node *) sta->drv_priv;
1349
Sujith Manoharan3d4e20f2012-03-14 14:40:58 +05301350 if (!sta->ht_cap.ht_supported)
Mohammed Shafi Shajakhanb25bfda2011-12-26 10:42:15 +05301351 return;
1352
Felix Fietkau55195412011-04-17 23:28:09 +02001353 switch (cmd) {
1354 case STA_NOTIFY_SLEEP:
1355 an->sleeping = true;
Johannes Berg042ec452011-09-29 16:04:26 +02001356 ath_tx_aggr_sleep(sta, sc, an);
Felix Fietkau55195412011-04-17 23:28:09 +02001357 break;
1358 case STA_NOTIFY_AWAKE:
1359 an->sleeping = false;
1360 ath_tx_aggr_wakeup(sc, an);
1361 break;
1362 }
1363}
1364
Eliad Peller8a3a3c82011-10-02 10:15:52 +02001365static int ath9k_conf_tx(struct ieee80211_hw *hw,
1366 struct ieee80211_vif *vif, u16 queue,
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07001367 const struct ieee80211_tx_queue_params *params)
1368{
Felix Fietkau9ac586152011-01-24 19:23:18 +01001369 struct ath_softc *sc = hw->priv;
Luis R. Rodriguezc46917b2009-09-13 02:42:02 -07001370 struct ath_common *common = ath9k_hw_common(sc->sc_ah);
Felix Fietkau066dae92010-11-07 14:59:39 +01001371 struct ath_txq *txq;
Sujithea9880f2008-08-07 10:53:10 +05301372 struct ath9k_tx_queue_info qi;
Felix Fietkau066dae92010-11-07 14:59:39 +01001373 int ret = 0;
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07001374
1375 if (queue >= WME_NUM_AC)
1376 return 0;
1377
Felix Fietkau066dae92010-11-07 14:59:39 +01001378 txq = sc->tx.txq_map[queue];
1379
Felix Fietkau96f372c2011-04-07 19:07:17 +02001380 ath9k_ps_wakeup(sc);
Sujith141b38b2009-02-04 08:10:07 +05301381 mutex_lock(&sc->mutex);
1382
Sujith1ffb0612009-03-30 15:28:46 +05301383 memset(&qi, 0, sizeof(struct ath9k_tx_queue_info));
1384
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07001385 qi.tqi_aifs = params->aifs;
1386 qi.tqi_cwmin = params->cw_min;
1387 qi.tqi_cwmax = params->cw_max;
1388 qi.tqi_burstTime = params->txop;
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07001389
Joe Perchesd2182b62011-12-15 14:55:53 -08001390 ath_dbg(common, CONFIG,
Joe Perches226afe62010-12-02 19:12:37 -08001391 "Configure tx [queue/halq] [%d/%d], aifs: %d, cw_min: %d, cw_max: %d, txop: %d\n",
1392 queue, txq->axq_qnum, params->aifs, params->cw_min,
1393 params->cw_max, params->txop);
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07001394
Felix Fietkau066dae92010-11-07 14:59:39 +01001395 ret = ath_txq_update(sc, txq->axq_qnum, &qi);
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07001396 if (ret)
Joe Perches38002762010-12-02 19:12:36 -08001397 ath_err(common, "TXQ Update failed\n");
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07001398
Vivek Natarajan94db2932009-11-25 12:01:54 +05301399 if (sc->sc_ah->opmode == NL80211_IFTYPE_ADHOC)
Felix Fietkau066dae92010-11-07 14:59:39 +01001400 if (queue == WME_AC_BE && !ret)
Vivek Natarajan94db2932009-11-25 12:01:54 +05301401 ath_beaconq_config(sc);
1402
Sujith141b38b2009-02-04 08:10:07 +05301403 mutex_unlock(&sc->mutex);
Felix Fietkau96f372c2011-04-07 19:07:17 +02001404 ath9k_ps_restore(sc);
Sujith141b38b2009-02-04 08:10:07 +05301405
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07001406 return ret;
1407}
1408
1409static int ath9k_set_key(struct ieee80211_hw *hw,
1410 enum set_key_cmd cmd,
Johannes Bergdc822b52008-12-29 12:55:09 +01001411 struct ieee80211_vif *vif,
1412 struct ieee80211_sta *sta,
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07001413 struct ieee80211_key_conf *key)
1414{
Felix Fietkau9ac586152011-01-24 19:23:18 +01001415 struct ath_softc *sc = hw->priv;
Luis R. Rodriguezc46917b2009-09-13 02:42:02 -07001416 struct ath_common *common = ath9k_hw_common(sc->sc_ah);
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07001417 int ret = 0;
1418
John W. Linville3e6109c2011-01-05 09:39:17 -05001419 if (ath9k_modparam_nohwcrypt)
Jouni Malinenb3bd89c2009-02-24 13:42:01 +02001420 return -ENOSPC;
1421
Chun-Yeow Yeoh5bd5e9a2011-12-07 12:45:46 -08001422 if ((vif->type == NL80211_IFTYPE_ADHOC ||
1423 vif->type == NL80211_IFTYPE_MESH_POINT) &&
Jouni Malinencfdc9a82011-03-23 14:52:19 +02001424 (key->cipher == WLAN_CIPHER_SUITE_TKIP ||
1425 key->cipher == WLAN_CIPHER_SUITE_CCMP) &&
1426 !(key->flags & IEEE80211_KEY_FLAG_PAIRWISE)) {
1427 /*
1428 * For now, disable hw crypto for the RSN IBSS group keys. This
1429 * could be optimized in the future to use a modified key cache
1430 * design to support per-STA RX GTK, but until that gets
1431 * implemented, use of software crypto for group addressed
1432 * frames is a acceptable to allow RSN IBSS to be used.
1433 */
1434 return -EOPNOTSUPP;
1435 }
1436
Sujith141b38b2009-02-04 08:10:07 +05301437 mutex_lock(&sc->mutex);
Vivek Natarajan3cbb5dd2009-01-20 11:17:08 +05301438 ath9k_ps_wakeup(sc);
Joe Perchesd2182b62011-12-15 14:55:53 -08001439 ath_dbg(common, CONFIG, "Set HW Key\n");
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07001440
1441 switch (cmd) {
1442 case SET_KEY:
Felix Fietkau93ae2dd2011-04-17 23:28:10 +02001443 if (sta)
1444 ath9k_del_ps_key(sc, vif, sta);
1445
Bruno Randolf040e5392010-09-08 16:05:04 +09001446 ret = ath_key_config(common, vif, sta, key);
Jouni Malinen6ace2892008-12-17 13:32:17 +02001447 if (ret >= 0) {
1448 key->hw_key_idx = ret;
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07001449 /* push IV and Michael MIC generation to stack */
1450 key->flags |= IEEE80211_KEY_FLAG_GENERATE_IV;
Johannes Berg97359d12010-08-10 09:46:38 +02001451 if (key->cipher == WLAN_CIPHER_SUITE_TKIP)
Senthil Balasubramanian1b961752008-09-01 19:45:21 +05301452 key->flags |= IEEE80211_KEY_FLAG_GENERATE_MMIC;
Johannes Berg97359d12010-08-10 09:46:38 +02001453 if (sc->sc_ah->sw_mgmt_crypto &&
1454 key->cipher == WLAN_CIPHER_SUITE_CCMP)
Jouni Malinen0ced0e12009-01-08 13:32:13 +02001455 key->flags |= IEEE80211_KEY_FLAG_SW_MGMT;
Jouni Malinen6ace2892008-12-17 13:32:17 +02001456 ret = 0;
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07001457 }
1458 break;
1459 case DISABLE_KEY:
Bruno Randolf040e5392010-09-08 16:05:04 +09001460 ath_key_delete(common, key);
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07001461 break;
1462 default:
1463 ret = -EINVAL;
1464 }
1465
Vivek Natarajan3cbb5dd2009-01-20 11:17:08 +05301466 ath9k_ps_restore(sc);
Sujith141b38b2009-02-04 08:10:07 +05301467 mutex_unlock(&sc->mutex);
1468
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07001469 return ret;
1470}
Rajkumar Manoharan4f5ef75b2011-04-04 22:56:18 +05301471static void ath9k_bss_iter(void *data, u8 *mac, struct ieee80211_vif *vif)
1472{
1473 struct ath_softc *sc = data;
1474 struct ath_common *common = ath9k_hw_common(sc->sc_ah);
1475 struct ieee80211_bss_conf *bss_conf = &vif->bss_conf;
1476 struct ath_vif *avp = (void *)vif->drv_priv;
1477
Rajkumar Manoharan2e5ef452011-05-20 17:52:12 +05301478 /*
1479 * Skip iteration if primary station vif's bss info
1480 * was not changed
1481 */
Sujith Manoharan781b14a2012-06-04 20:23:55 +05301482 if (test_bit(SC_OP_PRIM_STA_VIF, &sc->sc_flags))
Rajkumar Manoharan2e5ef452011-05-20 17:52:12 +05301483 return;
1484
1485 if (bss_conf->assoc) {
Sujith Manoharan781b14a2012-06-04 20:23:55 +05301486 set_bit(SC_OP_PRIM_STA_VIF, &sc->sc_flags);
Rajkumar Manoharan2e5ef452011-05-20 17:52:12 +05301487 avp->primary_sta_vif = true;
Rajkumar Manoharan4f5ef75b2011-04-04 22:56:18 +05301488 memcpy(common->curbssid, bss_conf->bssid, ETH_ALEN);
1489 common->curaid = bss_conf->aid;
1490 ath9k_hw_write_associd(sc->sc_ah);
Joe Perchesd2182b62011-12-15 14:55:53 -08001491 ath_dbg(common, CONFIG, "Bss Info ASSOC %d, bssid: %pM\n",
1492 bss_conf->aid, common->curbssid);
Rajkumar Manoharan2e5ef452011-05-20 17:52:12 +05301493 ath_beacon_config(sc, vif);
1494 /*
1495 * Request a re-configuration of Beacon related timers
1496 * on the receipt of the first Beacon frame (i.e.,
1497 * after time sync with the AP).
1498 */
1499 sc->ps_flags |= PS_BEACON_SYNC | PS_WAIT_FOR_BEACON;
1500 /* Reset rssi stats */
1501 sc->last_rssi = ATH_RSSI_DUMMY_MARKER;
1502 sc->sc_ah->stats.avgbrssi = ATH_RSSI_DUMMY_MARKER;
Rajkumar Manoharan99e4d432011-04-04 22:56:19 +05301503
Rajkumar Manoharan01e18912012-03-15 05:34:27 +05301504 ath_start_rx_poll(sc, 3);
1505
Mohammed Shafi Shajakhan05c0be22011-05-26 10:56:15 +05301506 if (!common->disable_ani) {
Sujith Manoharan781b14a2012-06-04 20:23:55 +05301507 set_bit(SC_OP_ANI_RUN, &sc->sc_flags);
Mohammed Shafi Shajakhan05c0be22011-05-26 10:56:15 +05301508 ath_start_ani(common);
1509 }
1510
Rajkumar Manoharan4f5ef75b2011-04-04 22:56:18 +05301511 }
1512}
1513
1514static void ath9k_config_bss(struct ath_softc *sc, struct ieee80211_vif *vif)
1515{
1516 struct ath_common *common = ath9k_hw_common(sc->sc_ah);
1517 struct ieee80211_bss_conf *bss_conf = &vif->bss_conf;
1518 struct ath_vif *avp = (void *)vif->drv_priv;
1519
Rajkumar Manoharan2e5ef452011-05-20 17:52:12 +05301520 if (sc->sc_ah->opmode != NL80211_IFTYPE_STATION)
1521 return;
1522
Rajkumar Manoharan4f5ef75b2011-04-04 22:56:18 +05301523 /* Reconfigure bss info */
1524 if (avp->primary_sta_vif && !bss_conf->assoc) {
Joe Perchesd2182b62011-12-15 14:55:53 -08001525 ath_dbg(common, CONFIG, "Bss Info DISASSOC %d, bssid %pM\n",
Rajkumar Manoharan99e4d432011-04-04 22:56:19 +05301526 common->curaid, common->curbssid);
Sujith Manoharan781b14a2012-06-04 20:23:55 +05301527 clear_bit(SC_OP_PRIM_STA_VIF, &sc->sc_flags);
1528 clear_bit(SC_OP_BEACONS, &sc->sc_flags);
Rajkumar Manoharan4f5ef75b2011-04-04 22:56:18 +05301529 avp->primary_sta_vif = false;
1530 memset(common->curbssid, 0, ETH_ALEN);
1531 common->curaid = 0;
1532 }
1533
1534 ieee80211_iterate_active_interfaces_atomic(
1535 sc->hw, ath9k_bss_iter, sc);
1536
1537 /*
1538 * None of station vifs are associated.
1539 * Clear bssid & aid
1540 */
Sujith Manoharan781b14a2012-06-04 20:23:55 +05301541 if (!test_bit(SC_OP_PRIM_STA_VIF, &sc->sc_flags)) {
Rajkumar Manoharan4f5ef75b2011-04-04 22:56:18 +05301542 ath9k_hw_write_associd(sc->sc_ah);
Sujith Manoharan781b14a2012-06-04 20:23:55 +05301543 clear_bit(SC_OP_ANI_RUN, &sc->sc_flags);
Rajkumar Manoharan99e4d432011-04-04 22:56:19 +05301544 del_timer_sync(&common->ani.timer);
Rajkumar Manoharan01e18912012-03-15 05:34:27 +05301545 del_timer_sync(&sc->rx_poll_timer);
Rajkumar Manoharand2c71c22011-09-15 19:02:54 +05301546 memset(&sc->caldata, 0, sizeof(sc->caldata));
Rajkumar Manoharan99e4d432011-04-04 22:56:19 +05301547 }
Rajkumar Manoharan4f5ef75b2011-04-04 22:56:18 +05301548}
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07001549
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07001550static void ath9k_bss_info_changed(struct ieee80211_hw *hw,
1551 struct ieee80211_vif *vif,
1552 struct ieee80211_bss_conf *bss_conf,
1553 u32 changed)
1554{
Felix Fietkau9ac586152011-01-24 19:23:18 +01001555 struct ath_softc *sc = hw->priv;
Johannes Berg2d0ddec2009-04-23 16:13:26 +02001556 struct ath_hw *ah = sc->sc_ah;
Luis R. Rodriguez15107182009-09-10 09:22:37 -07001557 struct ath_common *common = ath9k_hw_common(ah);
Johannes Berg2d0ddec2009-04-23 16:13:26 +02001558 struct ath_vif *avp = (void *)vif->drv_priv;
Felix Fietkau0005baf2010-01-15 02:33:40 +01001559 int slottime;
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07001560
Felix Fietkau96f372c2011-04-07 19:07:17 +02001561 ath9k_ps_wakeup(sc);
Sujith141b38b2009-02-04 08:10:07 +05301562 mutex_lock(&sc->mutex);
1563
Rajkumar Manoharan9f619032012-03-10 05:06:49 +05301564 if (changed & BSS_CHANGED_ASSOC) {
Rajkumar Manoharan4f5ef75b2011-04-04 22:56:18 +05301565 ath9k_config_bss(sc, vif);
Sujithc6089cc2009-11-16 11:40:48 +05301566
Joe Perchesd2182b62011-12-15 14:55:53 -08001567 ath_dbg(common, CONFIG, "BSSID: %pM aid: 0x%x\n",
Joe Perches226afe62010-12-02 19:12:37 -08001568 common->curbssid, common->curaid);
Johannes Berg2d0ddec2009-04-23 16:13:26 +02001569 }
1570
Rajkumar Manoharan2e5ef452011-05-20 17:52:12 +05301571 if (changed & BSS_CHANGED_IBSS) {
1572 /* There can be only one vif available */
1573 memcpy(common->curbssid, bss_conf->bssid, ETH_ALEN);
1574 common->curaid = bss_conf->aid;
1575 ath9k_hw_write_associd(sc->sc_ah);
1576
1577 if (bss_conf->ibss_joined) {
1578 sc->sc_ah->stats.avgbrssi = ATH_RSSI_DUMMY_MARKER;
Mohammed Shafi Shajakhan05c0be22011-05-26 10:56:15 +05301579
1580 if (!common->disable_ani) {
Sujith Manoharan781b14a2012-06-04 20:23:55 +05301581 set_bit(SC_OP_ANI_RUN, &sc->sc_flags);
Mohammed Shafi Shajakhan05c0be22011-05-26 10:56:15 +05301582 ath_start_ani(common);
1583 }
1584
Rajkumar Manoharan2e5ef452011-05-20 17:52:12 +05301585 } else {
Sujith Manoharan781b14a2012-06-04 20:23:55 +05301586 clear_bit(SC_OP_ANI_RUN, &sc->sc_flags);
Rajkumar Manoharan2e5ef452011-05-20 17:52:12 +05301587 del_timer_sync(&common->ani.timer);
Rajkumar Manoharan01e18912012-03-15 05:34:27 +05301588 del_timer_sync(&sc->rx_poll_timer);
Rajkumar Manoharan2e5ef452011-05-20 17:52:12 +05301589 }
1590 }
1591
Rajkumar Manoharaned2578c2012-04-19 19:13:51 +05301592 /*
1593 * In case of AP mode, the HW TSF has to be reset
1594 * when the beacon interval changes.
1595 */
1596 if ((changed & BSS_CHANGED_BEACON_INT) &&
1597 (vif->type == NL80211_IFTYPE_AP))
Sujith Manoharan781b14a2012-06-04 20:23:55 +05301598 set_bit(SC_OP_TSF_RESET, &sc->sc_flags);
Rajkumar Manoharaned2578c2012-04-19 19:13:51 +05301599
1600 /* Configure beaconing (AP, IBSS, MESH) */
1601 if (ath9k_uses_beacons(vif->type) &&
1602 ((changed & BSS_CHANGED_BEACON) ||
1603 (changed & BSS_CHANGED_BEACON_ENABLED) ||
1604 (changed & BSS_CHANGED_BEACON_INT))) {
Rajkumar Manoharan014cf3b2011-02-09 17:46:39 +05301605 ath9k_set_beaconing_status(sc, false);
Rajkumar Manoharaned2578c2012-04-19 19:13:51 +05301606 if (bss_conf->enable_beacon)
1607 ath_beacon_alloc(sc, vif);
1608 else
1609 avp->is_bslot_active = false;
1610 ath_beacon_config(sc, vif);
Rajkumar Manoharan014cf3b2011-02-09 17:46:39 +05301611 ath9k_set_beaconing_status(sc, true);
Johannes Berg2d0ddec2009-04-23 16:13:26 +02001612 }
1613
Felix Fietkau0005baf2010-01-15 02:33:40 +01001614 if (changed & BSS_CHANGED_ERP_SLOT) {
1615 if (bss_conf->use_short_slot)
1616 slottime = 9;
1617 else
1618 slottime = 20;
1619 if (vif->type == NL80211_IFTYPE_AP) {
1620 /*
1621 * Defer update, so that connected stations can adjust
1622 * their settings at the same time.
1623 * See beacon.c for more details
1624 */
1625 sc->beacon.slottime = slottime;
1626 sc->beacon.updateslot = UPDATE;
1627 } else {
1628 ah->slottime = slottime;
1629 ath9k_hw_init_global_settings(ah);
1630 }
1631 }
1632
Sujith141b38b2009-02-04 08:10:07 +05301633 mutex_unlock(&sc->mutex);
Felix Fietkau96f372c2011-04-07 19:07:17 +02001634 ath9k_ps_restore(sc);
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07001635}
1636
Eliad Peller37a41b42011-09-21 14:06:11 +03001637static u64 ath9k_get_tsf(struct ieee80211_hw *hw, struct ieee80211_vif *vif)
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07001638{
Felix Fietkau9ac586152011-01-24 19:23:18 +01001639 struct ath_softc *sc = hw->priv;
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07001640 u64 tsf;
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07001641
Sujith141b38b2009-02-04 08:10:07 +05301642 mutex_lock(&sc->mutex);
Sujith Manoharan9abbfb22010-12-10 11:27:06 +05301643 ath9k_ps_wakeup(sc);
Sujith141b38b2009-02-04 08:10:07 +05301644 tsf = ath9k_hw_gettsf64(sc->sc_ah);
Sujith Manoharan9abbfb22010-12-10 11:27:06 +05301645 ath9k_ps_restore(sc);
Sujith141b38b2009-02-04 08:10:07 +05301646 mutex_unlock(&sc->mutex);
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07001647
1648 return tsf;
1649}
1650
Eliad Peller37a41b42011-09-21 14:06:11 +03001651static void ath9k_set_tsf(struct ieee80211_hw *hw,
1652 struct ieee80211_vif *vif,
1653 u64 tsf)
Alina Friedrichsen3b5d6652009-01-24 07:09:59 +01001654{
Felix Fietkau9ac586152011-01-24 19:23:18 +01001655 struct ath_softc *sc = hw->priv;
Alina Friedrichsen3b5d6652009-01-24 07:09:59 +01001656
Sujith141b38b2009-02-04 08:10:07 +05301657 mutex_lock(&sc->mutex);
Sujith Manoharan9abbfb22010-12-10 11:27:06 +05301658 ath9k_ps_wakeup(sc);
Sujith141b38b2009-02-04 08:10:07 +05301659 ath9k_hw_settsf64(sc->sc_ah, tsf);
Sujith Manoharan9abbfb22010-12-10 11:27:06 +05301660 ath9k_ps_restore(sc);
Sujith141b38b2009-02-04 08:10:07 +05301661 mutex_unlock(&sc->mutex);
Alina Friedrichsen3b5d6652009-01-24 07:09:59 +01001662}
1663
Eliad Peller37a41b42011-09-21 14:06:11 +03001664static void ath9k_reset_tsf(struct ieee80211_hw *hw, struct ieee80211_vif *vif)
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07001665{
Felix Fietkau9ac586152011-01-24 19:23:18 +01001666 struct ath_softc *sc = hw->priv;
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07001667
Sujith141b38b2009-02-04 08:10:07 +05301668 mutex_lock(&sc->mutex);
Luis R. Rodriguez21526d52009-09-09 20:05:39 -07001669
1670 ath9k_ps_wakeup(sc);
Sujith141b38b2009-02-04 08:10:07 +05301671 ath9k_hw_reset_tsf(sc->sc_ah);
Luis R. Rodriguez21526d52009-09-09 20:05:39 -07001672 ath9k_ps_restore(sc);
1673
Sujith141b38b2009-02-04 08:10:07 +05301674 mutex_unlock(&sc->mutex);
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07001675}
1676
1677static int ath9k_ampdu_action(struct ieee80211_hw *hw,
Johannes Bergc951ad32009-11-16 12:00:38 +01001678 struct ieee80211_vif *vif,
Sujith141b38b2009-02-04 08:10:07 +05301679 enum ieee80211_ampdu_mlme_action action,
1680 struct ieee80211_sta *sta,
Johannes Berg0b01f032011-01-18 13:51:05 +01001681 u16 tid, u16 *ssn, u8 buf_size)
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07001682{
Felix Fietkau9ac586152011-01-24 19:23:18 +01001683 struct ath_softc *sc = hw->priv;
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07001684 int ret = 0;
1685
Johannes Berg85ad1812010-06-10 10:21:49 +02001686 local_bh_disable();
1687
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07001688 switch (action) {
1689 case IEEE80211_AMPDU_RX_START:
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07001690 break;
1691 case IEEE80211_AMPDU_RX_STOP:
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07001692 break;
1693 case IEEE80211_AMPDU_TX_START:
Luis R. Rodriguez8b685ba2009-12-23 20:03:29 -05001694 ath9k_ps_wakeup(sc);
Felix Fietkau231c3a12010-09-20 19:35:28 +02001695 ret = ath_tx_aggr_start(sc, sta, tid, ssn);
1696 if (!ret)
1697 ieee80211_start_tx_ba_cb_irqsafe(vif, sta->addr, tid);
Luis R. Rodriguez8b685ba2009-12-23 20:03:29 -05001698 ath9k_ps_restore(sc);
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07001699 break;
1700 case IEEE80211_AMPDU_TX_STOP:
Luis R. Rodriguez8b685ba2009-12-23 20:03:29 -05001701 ath9k_ps_wakeup(sc);
Sujithf83da962009-07-23 15:32:37 +05301702 ath_tx_aggr_stop(sc, sta, tid);
Johannes Bergc951ad32009-11-16 12:00:38 +01001703 ieee80211_stop_tx_ba_cb_irqsafe(vif, sta->addr, tid);
Luis R. Rodriguez8b685ba2009-12-23 20:03:29 -05001704 ath9k_ps_restore(sc);
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07001705 break;
Johannes Bergb1720232009-03-23 17:28:39 +01001706 case IEEE80211_AMPDU_TX_OPERATIONAL:
Luis R. Rodriguez8b685ba2009-12-23 20:03:29 -05001707 ath9k_ps_wakeup(sc);
Sujith8469cde2008-10-29 10:19:28 +05301708 ath_tx_aggr_resume(sc, sta, tid);
Luis R. Rodriguez8b685ba2009-12-23 20:03:29 -05001709 ath9k_ps_restore(sc);
Sujith8469cde2008-10-29 10:19:28 +05301710 break;
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07001711 default:
Joe Perches38002762010-12-02 19:12:36 -08001712 ath_err(ath9k_hw_common(sc->sc_ah), "Unknown AMPDU action\n");
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07001713 }
1714
Johannes Berg85ad1812010-06-10 10:21:49 +02001715 local_bh_enable();
1716
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07001717 return ret;
1718}
1719
Benoit Papillault62dad5b2010-04-28 00:08:24 +02001720static int ath9k_get_survey(struct ieee80211_hw *hw, int idx,
1721 struct survey_info *survey)
1722{
Felix Fietkau9ac586152011-01-24 19:23:18 +01001723 struct ath_softc *sc = hw->priv;
Felix Fietkau34300982010-10-10 18:21:52 +02001724 struct ath_common *common = ath9k_hw_common(sc->sc_ah);
Felix Fietkau39162db2010-09-29 19:12:06 +02001725 struct ieee80211_supported_band *sband;
Felix Fietkau34300982010-10-10 18:21:52 +02001726 struct ieee80211_channel *chan;
1727 unsigned long flags;
1728 int pos;
1729
1730 spin_lock_irqsave(&common->cc_lock, flags);
1731 if (idx == 0)
1732 ath_update_survey_stats(sc);
Benoit Papillault62dad5b2010-04-28 00:08:24 +02001733
Felix Fietkau39162db2010-09-29 19:12:06 +02001734 sband = hw->wiphy->bands[IEEE80211_BAND_2GHZ];
1735 if (sband && idx >= sband->n_channels) {
1736 idx -= sband->n_channels;
1737 sband = NULL;
1738 }
Benoit Papillault62dad5b2010-04-28 00:08:24 +02001739
Felix Fietkau39162db2010-09-29 19:12:06 +02001740 if (!sband)
1741 sband = hw->wiphy->bands[IEEE80211_BAND_5GHZ];
1742
Felix Fietkau34300982010-10-10 18:21:52 +02001743 if (!sband || idx >= sband->n_channels) {
1744 spin_unlock_irqrestore(&common->cc_lock, flags);
1745 return -ENOENT;
Felix Fietkau4f1a5a42010-09-29 17:15:28 +02001746 }
Benoit Papillault62dad5b2010-04-28 00:08:24 +02001747
Felix Fietkau34300982010-10-10 18:21:52 +02001748 chan = &sband->channels[idx];
1749 pos = chan->hw_value;
1750 memcpy(survey, &sc->survey[pos], sizeof(*survey));
1751 survey->channel = chan;
1752 spin_unlock_irqrestore(&common->cc_lock, flags);
1753
Benoit Papillault62dad5b2010-04-28 00:08:24 +02001754 return 0;
1755}
1756
Felix Fietkaue239d852010-01-15 02:34:58 +01001757static void ath9k_set_coverage_class(struct ieee80211_hw *hw, u8 coverage_class)
1758{
Felix Fietkau9ac586152011-01-24 19:23:18 +01001759 struct ath_softc *sc = hw->priv;
Felix Fietkaue239d852010-01-15 02:34:58 +01001760 struct ath_hw *ah = sc->sc_ah;
1761
1762 mutex_lock(&sc->mutex);
1763 ah->coverage_class = coverage_class;
Mohammed Shafi Shajakhan8b2a38272011-08-24 21:38:07 +05301764
1765 ath9k_ps_wakeup(sc);
Felix Fietkaue239d852010-01-15 02:34:58 +01001766 ath9k_hw_init_global_settings(ah);
Mohammed Shafi Shajakhan8b2a38272011-08-24 21:38:07 +05301767 ath9k_ps_restore(sc);
1768
Felix Fietkaue239d852010-01-15 02:34:58 +01001769 mutex_unlock(&sc->mutex);
1770}
1771
Vasanthakumar Thiagarajan69081622011-02-19 01:13:42 -08001772static void ath9k_flush(struct ieee80211_hw *hw, bool drop)
1773{
Vasanthakumar Thiagarajan69081622011-02-19 01:13:42 -08001774 struct ath_softc *sc = hw->priv;
Mohammed Shafi Shajakhan99aa55b2011-05-06 20:43:11 +05301775 struct ath_hw *ah = sc->sc_ah;
1776 struct ath_common *common = ath9k_hw_common(ah);
Felix Fietkau86271e42011-03-11 21:38:19 +01001777 int timeout = 200; /* ms */
1778 int i, j;
Rajkumar Manoharan2f6fc352011-04-28 15:31:57 +05301779 bool drain_txq;
Vasanthakumar Thiagarajan69081622011-02-19 01:13:42 -08001780
1781 mutex_lock(&sc->mutex);
Vasanthakumar Thiagarajan69081622011-02-19 01:13:42 -08001782 cancel_delayed_work_sync(&sc->tx_complete_work);
1783
Mohammed Shafi Shajakhan6a6b3f32011-09-09 10:41:08 +05301784 if (ah->ah_flags & AH_UNPLUGGED) {
Joe Perchesd2182b62011-12-15 14:55:53 -08001785 ath_dbg(common, ANY, "Device has been unplugged!\n");
Mohammed Shafi Shajakhan6a6b3f32011-09-09 10:41:08 +05301786 mutex_unlock(&sc->mutex);
1787 return;
1788 }
1789
Sujith Manoharan781b14a2012-06-04 20:23:55 +05301790 if (test_bit(SC_OP_INVALID, &sc->sc_flags)) {
Joe Perchesd2182b62011-12-15 14:55:53 -08001791 ath_dbg(common, ANY, "Device not present\n");
Mohammed Shafi Shajakhan99aa55b2011-05-06 20:43:11 +05301792 mutex_unlock(&sc->mutex);
1793 return;
1794 }
1795
Felix Fietkau86271e42011-03-11 21:38:19 +01001796 for (j = 0; j < timeout; j++) {
Mohammed Shafi Shajakhan108697c2011-05-13 20:59:42 +05301797 bool npend = false;
Felix Fietkau86271e42011-03-11 21:38:19 +01001798
1799 if (j)
1800 usleep_range(1000, 2000);
1801
1802 for (i = 0; i < ATH9K_NUM_TX_QUEUES; i++) {
1803 if (!ATH_TXQ_SETUP(sc, i))
1804 continue;
1805
Mohammed Shafi Shajakhan108697c2011-05-13 20:59:42 +05301806 npend = ath9k_has_pending_frames(sc, &sc->tx.txq[i]);
1807
1808 if (npend)
1809 break;
Vasanthakumar Thiagarajan69081622011-02-19 01:13:42 -08001810 }
1811
Felix Fietkau86271e42011-03-11 21:38:19 +01001812 if (!npend)
Felix Fietkau9df0d6a2011-11-16 13:08:42 +01001813 break;
Vasanthakumar Thiagarajan69081622011-02-19 01:13:42 -08001814 }
1815
Felix Fietkau9df0d6a2011-11-16 13:08:42 +01001816 if (drop) {
1817 ath9k_ps_wakeup(sc);
1818 spin_lock_bh(&sc->sc_pcu_lock);
1819 drain_txq = ath_drain_all_txq(sc, false);
1820 spin_unlock_bh(&sc->sc_pcu_lock);
Felix Fietkau9adcf442011-09-03 01:40:26 +02001821
Felix Fietkau9df0d6a2011-11-16 13:08:42 +01001822 if (!drain_txq)
1823 ath_reset(sc, false);
Felix Fietkau9adcf442011-09-03 01:40:26 +02001824
Felix Fietkau9df0d6a2011-11-16 13:08:42 +01001825 ath9k_ps_restore(sc);
1826 ieee80211_wake_queues(hw);
1827 }
Senthil Balasubramaniand78f4b32011-03-23 23:07:22 +05301828
Vasanthakumar Thiagarajan69081622011-02-19 01:13:42 -08001829 ieee80211_queue_delayed_work(hw, &sc->tx_complete_work, 0);
1830 mutex_unlock(&sc->mutex);
1831}
1832
Vivek Natarajan15b91e82011-04-06 11:41:11 +05301833static bool ath9k_tx_frames_pending(struct ieee80211_hw *hw)
1834{
1835 struct ath_softc *sc = hw->priv;
1836 int i;
1837
1838 for (i = 0; i < ATH9K_NUM_TX_QUEUES; i++) {
1839 if (!ATH_TXQ_SETUP(sc, i))
1840 continue;
1841
1842 if (ath9k_has_pending_frames(sc, &sc->tx.txq[i]))
1843 return true;
1844 }
1845 return false;
1846}
1847
Mohammed Shafi Shajakhan5595f112011-05-19 18:08:57 +05301848static int ath9k_tx_last_beacon(struct ieee80211_hw *hw)
Felix Fietkauba4903f2011-05-17 21:09:54 +02001849{
1850 struct ath_softc *sc = hw->priv;
1851 struct ath_hw *ah = sc->sc_ah;
1852 struct ieee80211_vif *vif;
1853 struct ath_vif *avp;
1854 struct ath_buf *bf;
1855 struct ath_tx_status ts;
Felix Fietkau4286df62012-02-27 19:58:40 +01001856 bool edma = !!(ah->caps.hw_caps & ATH9K_HW_CAP_EDMA);
Felix Fietkauba4903f2011-05-17 21:09:54 +02001857 int status;
1858
1859 vif = sc->beacon.bslot[0];
1860 if (!vif)
1861 return 0;
1862
1863 avp = (void *)vif->drv_priv;
1864 if (!avp->is_bslot_active)
1865 return 0;
1866
Felix Fietkau4286df62012-02-27 19:58:40 +01001867 if (!sc->beacon.tx_processed && !edma) {
Felix Fietkauba4903f2011-05-17 21:09:54 +02001868 tasklet_disable(&sc->bcon_tasklet);
1869
1870 bf = avp->av_bcbuf;
1871 if (!bf || !bf->bf_mpdu)
1872 goto skip;
1873
1874 status = ath9k_hw_txprocdesc(ah, bf->bf_desc, &ts);
1875 if (status == -EINPROGRESS)
1876 goto skip;
1877
1878 sc->beacon.tx_processed = true;
1879 sc->beacon.tx_last = !(ts.ts_status & ATH9K_TXERR_MASK);
1880
1881skip:
1882 tasklet_enable(&sc->bcon_tasklet);
1883 }
1884
1885 return sc->beacon.tx_last;
1886}
1887
Mohammed Shafi Shajakhan52c94f42011-08-20 17:21:42 +05301888static int ath9k_get_stats(struct ieee80211_hw *hw,
1889 struct ieee80211_low_level_stats *stats)
1890{
1891 struct ath_softc *sc = hw->priv;
1892 struct ath_hw *ah = sc->sc_ah;
1893 struct ath9k_mib_stats *mib_stats = &ah->ah_mibStats;
1894
1895 stats->dot11ACKFailureCount = mib_stats->ackrcv_bad;
1896 stats->dot11RTSFailureCount = mib_stats->rts_bad;
1897 stats->dot11FCSErrorCount = mib_stats->fcs_bad;
1898 stats->dot11RTSSuccessCount = mib_stats->rts_good;
1899 return 0;
1900}
1901
Felix Fietkau43c35282011-09-03 01:40:27 +02001902static u32 fill_chainmask(u32 cap, u32 new)
1903{
1904 u32 filled = 0;
1905 int i;
1906
1907 for (i = 0; cap && new; i++, cap >>= 1) {
1908 if (!(cap & BIT(0)))
1909 continue;
1910
1911 if (new & BIT(0))
1912 filled |= BIT(i);
1913
1914 new >>= 1;
1915 }
1916
1917 return filled;
1918}
1919
1920static int ath9k_set_antenna(struct ieee80211_hw *hw, u32 tx_ant, u32 rx_ant)
1921{
1922 struct ath_softc *sc = hw->priv;
1923 struct ath_hw *ah = sc->sc_ah;
1924
1925 if (!rx_ant || !tx_ant)
1926 return -EINVAL;
1927
1928 sc->ant_rx = rx_ant;
1929 sc->ant_tx = tx_ant;
1930
1931 if (ah->caps.rx_chainmask == 1)
1932 return 0;
1933
1934 /* AR9100 runs into calibration issues if not all rx chains are enabled */
1935 if (AR_SREV_9100(ah))
1936 ah->rxchainmask = 0x7;
1937 else
1938 ah->rxchainmask = fill_chainmask(ah->caps.rx_chainmask, rx_ant);
1939
1940 ah->txchainmask = fill_chainmask(ah->caps.tx_chainmask, tx_ant);
1941 ath9k_reload_chainmask_settings(sc);
1942
1943 return 0;
1944}
1945
1946static int ath9k_get_antenna(struct ieee80211_hw *hw, u32 *tx_ant, u32 *rx_ant)
1947{
1948 struct ath_softc *sc = hw->priv;
1949
1950 *tx_ant = sc->ant_tx;
1951 *rx_ant = sc->ant_rx;
1952 return 0;
1953}
1954
Ben Greearb90bd9d2012-05-15 15:33:25 -07001955#ifdef CONFIG_ATH9K_DEBUGFS
1956
1957/* Ethtool support for get-stats */
1958
1959#define AMKSTR(nm) #nm "_BE", #nm "_BK", #nm "_VI", #nm "_VO"
1960static const char ath9k_gstrings_stats[][ETH_GSTRING_LEN] = {
1961 "tx_pkts_nic",
1962 "tx_bytes_nic",
1963 "rx_pkts_nic",
1964 "rx_bytes_nic",
1965 AMKSTR(d_tx_pkts),
1966 AMKSTR(d_tx_bytes),
1967 AMKSTR(d_tx_mpdus_queued),
1968 AMKSTR(d_tx_mpdus_completed),
1969 AMKSTR(d_tx_mpdu_xretries),
1970 AMKSTR(d_tx_aggregates),
1971 AMKSTR(d_tx_ampdus_queued_hw),
1972 AMKSTR(d_tx_ampdus_queued_sw),
1973 AMKSTR(d_tx_ampdus_completed),
1974 AMKSTR(d_tx_ampdu_retries),
1975 AMKSTR(d_tx_ampdu_xretries),
1976 AMKSTR(d_tx_fifo_underrun),
1977 AMKSTR(d_tx_op_exceeded),
1978 AMKSTR(d_tx_timer_expiry),
1979 AMKSTR(d_tx_desc_cfg_err),
1980 AMKSTR(d_tx_data_underrun),
1981 AMKSTR(d_tx_delim_underrun),
1982
1983 "d_rx_decrypt_crc_err",
1984 "d_rx_phy_err",
1985 "d_rx_mic_err",
1986 "d_rx_pre_delim_crc_err",
1987 "d_rx_post_delim_crc_err",
1988 "d_rx_decrypt_busy_err",
1989
1990 "d_rx_phyerr_radar",
1991 "d_rx_phyerr_ofdm_timing",
1992 "d_rx_phyerr_cck_timing",
1993
1994};
1995#define ATH9K_SSTATS_LEN ARRAY_SIZE(ath9k_gstrings_stats)
1996
1997static void ath9k_get_et_strings(struct ieee80211_hw *hw,
1998 struct ieee80211_vif *vif,
1999 u32 sset, u8 *data)
2000{
2001 if (sset == ETH_SS_STATS)
2002 memcpy(data, *ath9k_gstrings_stats,
2003 sizeof(ath9k_gstrings_stats));
2004}
2005
2006static int ath9k_get_et_sset_count(struct ieee80211_hw *hw,
2007 struct ieee80211_vif *vif, int sset)
2008{
2009 if (sset == ETH_SS_STATS)
2010 return ATH9K_SSTATS_LEN;
2011 return 0;
2012}
2013
2014#define PR_QNUM(_n) (sc->tx.txq_map[_n]->axq_qnum)
2015#define AWDATA(elem) \
2016 do { \
2017 data[i++] = sc->debug.stats.txstats[PR_QNUM(WME_AC_BE)].elem; \
2018 data[i++] = sc->debug.stats.txstats[PR_QNUM(WME_AC_BK)].elem; \
2019 data[i++] = sc->debug.stats.txstats[PR_QNUM(WME_AC_VI)].elem; \
2020 data[i++] = sc->debug.stats.txstats[PR_QNUM(WME_AC_VO)].elem; \
2021 } while (0)
2022
2023#define AWDATA_RX(elem) \
2024 do { \
2025 data[i++] = sc->debug.stats.rxstats.elem; \
2026 } while (0)
2027
2028static void ath9k_get_et_stats(struct ieee80211_hw *hw,
2029 struct ieee80211_vif *vif,
2030 struct ethtool_stats *stats, u64 *data)
2031{
2032 struct ath_softc *sc = hw->priv;
2033 int i = 0;
2034
2035 data[i++] = (sc->debug.stats.txstats[PR_QNUM(WME_AC_BE)].tx_pkts_all +
2036 sc->debug.stats.txstats[PR_QNUM(WME_AC_BK)].tx_pkts_all +
2037 sc->debug.stats.txstats[PR_QNUM(WME_AC_VI)].tx_pkts_all +
2038 sc->debug.stats.txstats[PR_QNUM(WME_AC_VO)].tx_pkts_all);
2039 data[i++] = (sc->debug.stats.txstats[PR_QNUM(WME_AC_BE)].tx_bytes_all +
2040 sc->debug.stats.txstats[PR_QNUM(WME_AC_BK)].tx_bytes_all +
2041 sc->debug.stats.txstats[PR_QNUM(WME_AC_VI)].tx_bytes_all +
2042 sc->debug.stats.txstats[PR_QNUM(WME_AC_VO)].tx_bytes_all);
2043 AWDATA_RX(rx_pkts_all);
2044 AWDATA_RX(rx_bytes_all);
2045
2046 AWDATA(tx_pkts_all);
2047 AWDATA(tx_bytes_all);
2048 AWDATA(queued);
2049 AWDATA(completed);
2050 AWDATA(xretries);
2051 AWDATA(a_aggr);
2052 AWDATA(a_queued_hw);
2053 AWDATA(a_queued_sw);
2054 AWDATA(a_completed);
2055 AWDATA(a_retries);
2056 AWDATA(a_xretries);
2057 AWDATA(fifo_underrun);
2058 AWDATA(xtxop);
2059 AWDATA(timer_exp);
2060 AWDATA(desc_cfg_err);
2061 AWDATA(data_underrun);
2062 AWDATA(delim_underrun);
2063
2064 AWDATA_RX(decrypt_crc_err);
2065 AWDATA_RX(phy_err);
2066 AWDATA_RX(mic_err);
2067 AWDATA_RX(pre_delim_crc_err);
2068 AWDATA_RX(post_delim_crc_err);
2069 AWDATA_RX(decrypt_busy_err);
2070
2071 AWDATA_RX(phy_err_stats[ATH9K_PHYERR_RADAR]);
2072 AWDATA_RX(phy_err_stats[ATH9K_PHYERR_OFDM_TIMING]);
2073 AWDATA_RX(phy_err_stats[ATH9K_PHYERR_CCK_TIMING]);
2074
2075 WARN_ON(i != ATH9K_SSTATS_LEN);
2076}
2077
2078/* End of ethtool get-stats functions */
2079
2080#endif
2081
2082
Gabor Juhos6baff7f2009-01-14 20:17:06 +01002083struct ieee80211_ops ath9k_ops = {
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07002084 .tx = ath9k_tx,
2085 .start = ath9k_start,
2086 .stop = ath9k_stop,
2087 .add_interface = ath9k_add_interface,
Rajkumar Manoharan6b3b9912010-12-08 19:38:55 +05302088 .change_interface = ath9k_change_interface,
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07002089 .remove_interface = ath9k_remove_interface,
2090 .config = ath9k_config,
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07002091 .configure_filter = ath9k_configure_filter,
Johannes Berg4ca77862010-02-19 19:06:56 +01002092 .sta_add = ath9k_sta_add,
2093 .sta_remove = ath9k_sta_remove,
Felix Fietkau55195412011-04-17 23:28:09 +02002094 .sta_notify = ath9k_sta_notify,
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07002095 .conf_tx = ath9k_conf_tx,
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07002096 .bss_info_changed = ath9k_bss_info_changed,
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07002097 .set_key = ath9k_set_key,
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07002098 .get_tsf = ath9k_get_tsf,
Alina Friedrichsen3b5d6652009-01-24 07:09:59 +01002099 .set_tsf = ath9k_set_tsf,
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07002100 .reset_tsf = ath9k_reset_tsf,
Johannes Berg4233df62008-10-13 13:35:05 +02002101 .ampdu_action = ath9k_ampdu_action,
Benoit Papillault62dad5b2010-04-28 00:08:24 +02002102 .get_survey = ath9k_get_survey,
Johannes Berg3b319aa2009-06-13 14:50:26 +05302103 .rfkill_poll = ath9k_rfkill_poll_state,
Felix Fietkaue239d852010-01-15 02:34:58 +01002104 .set_coverage_class = ath9k_set_coverage_class,
Vasanthakumar Thiagarajan69081622011-02-19 01:13:42 -08002105 .flush = ath9k_flush,
Vivek Natarajan15b91e82011-04-06 11:41:11 +05302106 .tx_frames_pending = ath9k_tx_frames_pending,
Mohammed Shafi Shajakhan52c94f42011-08-20 17:21:42 +05302107 .tx_last_beacon = ath9k_tx_last_beacon,
2108 .get_stats = ath9k_get_stats,
Felix Fietkau43c35282011-09-03 01:40:27 +02002109 .set_antenna = ath9k_set_antenna,
2110 .get_antenna = ath9k_get_antenna,
Ben Greearb90bd9d2012-05-15 15:33:25 -07002111
2112#ifdef CONFIG_ATH9K_DEBUGFS
2113 .get_et_sset_count = ath9k_get_et_sset_count,
2114 .get_et_stats = ath9k_get_et_stats,
2115 .get_et_strings = ath9k_get_et_strings,
2116#endif
Luis R. Rodriguezf078f202008-08-04 00:16:41 -07002117};