blob: 00c490833ba7d860f882f3826a9eef389275055b [file] [log] [blame]
Nick Kossifidisc6e387a2008-08-29 22:45:39 +03001/*
2 * Copyright (c) 2004-2008 Reyk Floeter <reyk@openbsd.org>
3 * Copyright (c) 2006-2008 Nick Kossifidis <mickflemm@gmail.com>
4 *
5 * Permission to use, copy, modify, and distribute this software for any
6 * purpose with or without fee is hereby granted, provided that the above
7 * copyright notice and this permission notice appear in all copies.
8 *
9 * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
10 * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
11 * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
12 * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
13 * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
14 * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
15 * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
16 *
17 */
18
19/********************************************\
20Queue Control Unit, DFS Control Unit Functions
21\********************************************/
22
23#include "ath5k.h"
24#include "reg.h"
25#include "debug.h"
26#include "base.h"
27
Nick Kossifidis9320b5c42010-11-23 20:36:45 +020028
29/******************\
30* Helper functions *
31\******************/
32
Nick Kossifidisc6e387a2008-08-29 22:45:39 +030033/*
Nick Kossifidis9320b5c42010-11-23 20:36:45 +020034 * Get number of pending frames
35 * for a specific queue [5211+]
Nick Kossifidisc6e387a2008-08-29 22:45:39 +030036 */
Nick Kossifidis9320b5c42010-11-23 20:36:45 +020037u32 ath5k_hw_num_tx_pending(struct ath5k_hw *ah, unsigned int queue)
Nick Kossifidisc6e387a2008-08-29 22:45:39 +030038{
Nick Kossifidis9320b5c42010-11-23 20:36:45 +020039 u32 pending;
40 AR5K_ASSERT_ENTRY(queue, ah->ah_capabilities.cap_queues.q_tx_num);
41
42 /* Return if queue is declared inactive */
43 if (ah->ah_txq[queue].tqi_type == AR5K_TX_QUEUE_INACTIVE)
44 return false;
45
46 /* XXX: How about AR5K_CFG_TXCNT ? */
47 if (ah->ah_version == AR5K_AR5210)
48 return false;
49
50 pending = ath5k_hw_reg_read(ah, AR5K_QUEUE_STATUS(queue));
51 pending &= AR5K_QCU_STS_FRMPENDCNT;
52
53 /* It's possible to have no frames pending even if TXE
54 * is set. To indicate that q has not stopped return
55 * true */
56 if (!pending && AR5K_REG_READ_Q(ah, AR5K_QCU_TXE, queue))
57 return true;
58
59 return pending;
60}
61
62/*
63 * Set a transmit queue inactive
64 */
65void ath5k_hw_release_tx_queue(struct ath5k_hw *ah, unsigned int queue)
66{
67 if (WARN_ON(queue >= ah->ah_capabilities.cap_queues.q_tx_num))
68 return;
69
70 /* This queue will be skipped in further operations */
71 ah->ah_txq[queue].tqi_type = AR5K_TX_QUEUE_INACTIVE;
72 /*For SIMR setup*/
73 AR5K_Q_DISABLE_BITS(ah->ah_txq_status, queue);
Nick Kossifidisc6e387a2008-08-29 22:45:39 +030074}
75
76/*
Bruno Randolfde8af452010-09-17 11:37:12 +090077 * Make sure cw is a power of 2 minus 1 and smaller than 1024
78 */
79static u16 ath5k_cw_validate(u16 cw_req)
80{
81 u32 cw = 1;
82 cw_req = min(cw_req, (u16)1023);
83
84 while (cw < cw_req)
85 cw = (cw << 1) | 1;
86
87 return cw;
88}
89
90/*
Nick Kossifidis9320b5c42010-11-23 20:36:45 +020091 * Get properties for a transmit queue
92 */
93int ath5k_hw_get_tx_queueprops(struct ath5k_hw *ah, int queue,
94 struct ath5k_txq_info *queue_info)
95{
96 memcpy(queue_info, &ah->ah_txq[queue], sizeof(struct ath5k_txq_info));
97 return 0;
98}
99
100/*
Nick Kossifidisc6e387a2008-08-29 22:45:39 +0300101 * Set properties for a transmit queue
102 */
103int ath5k_hw_set_tx_queueprops(struct ath5k_hw *ah, int queue,
Bruno Randolfde8af452010-09-17 11:37:12 +0900104 const struct ath5k_txq_info *qinfo)
Nick Kossifidisc6e387a2008-08-29 22:45:39 +0300105{
Bruno Randolfde8af452010-09-17 11:37:12 +0900106 struct ath5k_txq_info *qi;
107
Nick Kossifidisc6e387a2008-08-29 22:45:39 +0300108 AR5K_ASSERT_ENTRY(queue, ah->ah_capabilities.cap_queues.q_tx_num);
109
Bruno Randolfde8af452010-09-17 11:37:12 +0900110 qi = &ah->ah_txq[queue];
111
112 if (qi->tqi_type == AR5K_TX_QUEUE_INACTIVE)
Nick Kossifidisc6e387a2008-08-29 22:45:39 +0300113 return -EIO;
114
Bruno Randolfde8af452010-09-17 11:37:12 +0900115 /* copy and validate values */
116 qi->tqi_type = qinfo->tqi_type;
117 qi->tqi_subtype = qinfo->tqi_subtype;
118 qi->tqi_flags = qinfo->tqi_flags;
119 /*
120 * According to the docs: Although the AIFS field is 8 bit wide,
121 * the maximum supported value is 0xFC. Setting it higher than that
122 * will cause the DCU to hang.
123 */
124 qi->tqi_aifs = min(qinfo->tqi_aifs, (u8)0xFC);
125 qi->tqi_cw_min = ath5k_cw_validate(qinfo->tqi_cw_min);
126 qi->tqi_cw_max = ath5k_cw_validate(qinfo->tqi_cw_max);
127 qi->tqi_cbr_period = qinfo->tqi_cbr_period;
128 qi->tqi_cbr_overflow_limit = qinfo->tqi_cbr_overflow_limit;
129 qi->tqi_burst_time = qinfo->tqi_burst_time;
130 qi->tqi_ready_time = qinfo->tqi_ready_time;
Nick Kossifidisc6e387a2008-08-29 22:45:39 +0300131
132 /*XXX: Is this supported on 5210 ?*/
Bruno Randolfde8af452010-09-17 11:37:12 +0900133 /*XXX: Is this correct for AR5K_WME_AC_VI,VO ???*/
134 if ((qinfo->tqi_type == AR5K_TX_QUEUE_DATA &&
135 ((qinfo->tqi_subtype == AR5K_WME_AC_VI) ||
136 (qinfo->tqi_subtype == AR5K_WME_AC_VO))) ||
137 qinfo->tqi_type == AR5K_TX_QUEUE_UAPSD)
138 qi->tqi_flags |= AR5K_TXQ_FLAG_POST_FR_BKOFF_DIS;
Nick Kossifidisc6e387a2008-08-29 22:45:39 +0300139
140 return 0;
141}
142
143/*
144 * Initialize a transmit queue
145 */
146int ath5k_hw_setup_tx_queue(struct ath5k_hw *ah, enum ath5k_tx_queue queue_type,
147 struct ath5k_txq_info *queue_info)
148{
149 unsigned int queue;
150 int ret;
151
Nick Kossifidisc6e387a2008-08-29 22:45:39 +0300152 /*
153 * Get queue by type
154 */
155 /*5210 only has 2 queues*/
156 if (ah->ah_version == AR5K_AR5210) {
157 switch (queue_type) {
158 case AR5K_TX_QUEUE_DATA:
159 queue = AR5K_TX_QUEUE_ID_NOQCU_DATA;
160 break;
161 case AR5K_TX_QUEUE_BEACON:
162 case AR5K_TX_QUEUE_CAB:
163 queue = AR5K_TX_QUEUE_ID_NOQCU_BEACON;
164 break;
165 default:
166 return -EINVAL;
167 }
168 } else {
169 switch (queue_type) {
170 case AR5K_TX_QUEUE_DATA:
171 for (queue = AR5K_TX_QUEUE_ID_DATA_MIN;
172 ah->ah_txq[queue].tqi_type !=
173 AR5K_TX_QUEUE_INACTIVE; queue++) {
174
175 if (queue > AR5K_TX_QUEUE_ID_DATA_MAX)
176 return -EINVAL;
177 }
178 break;
179 case AR5K_TX_QUEUE_UAPSD:
180 queue = AR5K_TX_QUEUE_ID_UAPSD;
181 break;
182 case AR5K_TX_QUEUE_BEACON:
183 queue = AR5K_TX_QUEUE_ID_BEACON;
184 break;
185 case AR5K_TX_QUEUE_CAB:
186 queue = AR5K_TX_QUEUE_ID_CAB;
187 break;
188 case AR5K_TX_QUEUE_XR_DATA:
189 if (ah->ah_version != AR5K_AR5212)
190 ATH5K_ERR(ah->ah_sc,
191 "XR data queues only supported in"
192 " 5212!\n");
193 queue = AR5K_TX_QUEUE_ID_XR_DATA;
194 break;
195 default:
196 return -EINVAL;
197 }
198 }
199
200 /*
201 * Setup internal queue structure
202 */
203 memset(&ah->ah_txq[queue], 0, sizeof(struct ath5k_txq_info));
204 ah->ah_txq[queue].tqi_type = queue_type;
205
206 if (queue_info != NULL) {
207 queue_info->tqi_type = queue_type;
208 ret = ath5k_hw_set_tx_queueprops(ah, queue, queue_info);
209 if (ret)
210 return ret;
211 }
212
213 /*
214 * We use ah_txq_status to hold a temp value for
215 * the Secondary interrupt mask registers on 5211+
216 * check out ath5k_hw_reset_tx_queue
217 */
218 AR5K_Q_ENABLE_BITS(ah->ah_txq_status, queue);
219
220 return queue;
221}
222
Nick Kossifidisc6e387a2008-08-29 22:45:39 +0300223
Nick Kossifidis9320b5c42010-11-23 20:36:45 +0200224/*******************************\
225* Single QCU/DCU initialization *
226\*******************************/
Nick Kossifidisc6e387a2008-08-29 22:45:39 +0300227
228/*
229 * Set DFS properties for a transmit queue on DCU
230 */
231int ath5k_hw_reset_tx_queue(struct ath5k_hw *ah, unsigned int queue)
232{
Bruno Randolfde8af452010-09-17 11:37:12 +0900233 u32 retry_lg, retry_sh;
Nick Kossifidisc6e387a2008-08-29 22:45:39 +0300234 struct ath5k_txq_info *tq = &ah->ah_txq[queue];
235
Nick Kossifidisc6e387a2008-08-29 22:45:39 +0300236 AR5K_ASSERT_ENTRY(queue, ah->ah_capabilities.cap_queues.q_tx_num);
237
238 tq = &ah->ah_txq[queue];
239
240 if (tq->tqi_type == AR5K_TX_QUEUE_INACTIVE)
241 return 0;
242
243 if (ah->ah_version == AR5K_AR5210) {
244 /* Only handle data queues, others will be ignored */
245 if (tq->tqi_type != AR5K_TX_QUEUE_DATA)
246 return 0;
247
248 /* Set Slot time */
Nick Kossifidisfa3d2fe2010-11-23 20:58:34 +0200249 ath5k_hw_reg_write(ah, (ah->ah_bwmode == AR5K_BWMODE_40MHZ) ?
Nick Kossifidisc6e387a2008-08-29 22:45:39 +0300250 AR5K_INIT_SLOT_TIME_TURBO : AR5K_INIT_SLOT_TIME,
251 AR5K_SLOT_TIME);
252 /* Set ACK_CTS timeout */
Nick Kossifidisfa3d2fe2010-11-23 20:58:34 +0200253 ath5k_hw_reg_write(ah, (ah->ah_bwmode == AR5K_BWMODE_40MHZ) ?
Nick Kossifidisc6e387a2008-08-29 22:45:39 +0300254 AR5K_INIT_ACK_CTS_TIMEOUT_TURBO :
255 AR5K_INIT_ACK_CTS_TIMEOUT, AR5K_SLOT_TIME);
Nick Kossifidisc6e387a2008-08-29 22:45:39 +0300256
257 /* Set IFS0 */
Nick Kossifidisfa3d2fe2010-11-23 20:58:34 +0200258 if (ah->ah_bwmode == AR5K_BWMODE_40MHZ) {
259 ath5k_hw_reg_write(ah, ((AR5K_INIT_SIFS_TURBO +
Bruno Randolfde8af452010-09-17 11:37:12 +0900260 tq->tqi_aifs * AR5K_INIT_SLOT_TIME_TURBO) <<
Nick Kossifidisc6e387a2008-08-29 22:45:39 +0300261 AR5K_IFS0_DIFS_S) | AR5K_INIT_SIFS_TURBO,
262 AR5K_IFS0);
263 } else {
264 ath5k_hw_reg_write(ah, ((AR5K_INIT_SIFS +
Bruno Randolfde8af452010-09-17 11:37:12 +0900265 tq->tqi_aifs * AR5K_INIT_SLOT_TIME) <<
266 AR5K_IFS0_DIFS_S) |
Nick Kossifidisc6e387a2008-08-29 22:45:39 +0300267 AR5K_INIT_SIFS, AR5K_IFS0);
268 }
269
270 /* Set IFS1 */
Nick Kossifidisfa3d2fe2010-11-23 20:58:34 +0200271 ath5k_hw_reg_write(ah, (ah->ah_bwmode == AR5K_BWMODE_40MHZ) ?
Nick Kossifidisc6e387a2008-08-29 22:45:39 +0300272 AR5K_INIT_PROTO_TIME_CNTRL_TURBO :
273 AR5K_INIT_PROTO_TIME_CNTRL, AR5K_IFS1);
Nick Kossifidisc6e387a2008-08-29 22:45:39 +0300274 }
275
276 /*
Nick Kossifidisc6e387a2008-08-29 22:45:39 +0300277 * Calculate and set retry limits
278 */
279 if (ah->ah_software_retry) {
280 /* XXX Need to test this */
281 retry_lg = ah->ah_limit_tx_retries;
282 retry_sh = retry_lg = retry_lg > AR5K_DCU_RETRY_LMT_SH_RETRY ?
283 AR5K_DCU_RETRY_LMT_SH_RETRY : retry_lg;
284 } else {
285 retry_lg = AR5K_INIT_LG_RETRY;
286 retry_sh = AR5K_INIT_SH_RETRY;
287 }
288
289 /*No QCU/DCU [5210]*/
290 if (ah->ah_version == AR5K_AR5210) {
291 ath5k_hw_reg_write(ah,
Bruno Randolfde8af452010-09-17 11:37:12 +0900292 (tq->tqi_cw_min << AR5K_NODCU_RETRY_LMT_CW_MIN_S)
Nick Kossifidisc6e387a2008-08-29 22:45:39 +0300293 | AR5K_REG_SM(AR5K_INIT_SLG_RETRY,
294 AR5K_NODCU_RETRY_LMT_SLG_RETRY)
295 | AR5K_REG_SM(AR5K_INIT_SSH_RETRY,
296 AR5K_NODCU_RETRY_LMT_SSH_RETRY)
297 | AR5K_REG_SM(retry_lg, AR5K_NODCU_RETRY_LMT_LG_RETRY)
298 | AR5K_REG_SM(retry_sh, AR5K_NODCU_RETRY_LMT_SH_RETRY),
299 AR5K_NODCU_RETRY_LMT);
300 } else {
301 /*QCU/DCU [5211+]*/
302 ath5k_hw_reg_write(ah,
303 AR5K_REG_SM(AR5K_INIT_SLG_RETRY,
304 AR5K_DCU_RETRY_LMT_SLG_RETRY) |
305 AR5K_REG_SM(AR5K_INIT_SSH_RETRY,
306 AR5K_DCU_RETRY_LMT_SSH_RETRY) |
307 AR5K_REG_SM(retry_lg, AR5K_DCU_RETRY_LMT_LG_RETRY) |
308 AR5K_REG_SM(retry_sh, AR5K_DCU_RETRY_LMT_SH_RETRY),
309 AR5K_QUEUE_DFS_RETRY_LIMIT(queue));
310
311 /*===Rest is also for QCU/DCU only [5211+]===*/
312
313 /*
Bruno Randolfde8af452010-09-17 11:37:12 +0900314 * Set contention window (cw_min/cw_max)
Nick Kossifidisc6e387a2008-08-29 22:45:39 +0300315 * and arbitrated interframe space (aifs)...
316 */
317 ath5k_hw_reg_write(ah,
Bruno Randolfde8af452010-09-17 11:37:12 +0900318 AR5K_REG_SM(tq->tqi_cw_min, AR5K_DCU_LCL_IFS_CW_MIN) |
319 AR5K_REG_SM(tq->tqi_cw_max, AR5K_DCU_LCL_IFS_CW_MAX) |
320 AR5K_REG_SM(tq->tqi_aifs, AR5K_DCU_LCL_IFS_AIFS),
Nick Kossifidisc6e387a2008-08-29 22:45:39 +0300321 AR5K_QUEUE_DFS_LOCAL_IFS(queue));
322
323 /*
324 * Set misc registers
325 */
Nick Kossifidis846567622009-01-06 17:27:06 +0200326
327 /* Enable DCU to wait for next fragment from QCU */
328 AR5K_REG_ENABLE_BITS(ah, AR5K_QUEUE_DFS_MISC(queue),
329 AR5K_DCU_MISC_FRAG_WAIT);
330
331 /* On Maui and Spirit use the global seqnum on DCU */
332 if (ah->ah_mac_version < AR5K_SREV_AR5211)
333 AR5K_REG_ENABLE_BITS(ah, AR5K_QUEUE_DFS_MISC(queue),
334 AR5K_DCU_MISC_SEQNUM_CTL);
Nick Kossifidisc6e387a2008-08-29 22:45:39 +0300335
336 if (tq->tqi_cbr_period) {
337 ath5k_hw_reg_write(ah, AR5K_REG_SM(tq->tqi_cbr_period,
338 AR5K_QCU_CBRCFG_INTVAL) |
339 AR5K_REG_SM(tq->tqi_cbr_overflow_limit,
340 AR5K_QCU_CBRCFG_ORN_THRES),
341 AR5K_QUEUE_CBRCFG(queue));
342 AR5K_REG_ENABLE_BITS(ah, AR5K_QUEUE_MISC(queue),
343 AR5K_QCU_MISC_FRSHED_CBR);
344 if (tq->tqi_cbr_overflow_limit)
345 AR5K_REG_ENABLE_BITS(ah,
346 AR5K_QUEUE_MISC(queue),
347 AR5K_QCU_MISC_CBR_THRES_ENABLE);
348 }
349
Nick Kossifidis846567622009-01-06 17:27:06 +0200350 if (tq->tqi_ready_time &&
Julia Lawall4d30d302009-08-08 15:22:26 +0200351 (tq->tqi_type != AR5K_TX_QUEUE_CAB))
Nick Kossifidisc6e387a2008-08-29 22:45:39 +0300352 ath5k_hw_reg_write(ah, AR5K_REG_SM(tq->tqi_ready_time,
353 AR5K_QCU_RDYTIMECFG_INTVAL) |
354 AR5K_QCU_RDYTIMECFG_ENABLE,
355 AR5K_QUEUE_RDYTIMECFG(queue));
356
357 if (tq->tqi_burst_time) {
358 ath5k_hw_reg_write(ah, AR5K_REG_SM(tq->tqi_burst_time,
359 AR5K_DCU_CHAN_TIME_DUR) |
360 AR5K_DCU_CHAN_TIME_ENABLE,
361 AR5K_QUEUE_DFS_CHANNEL_TIME(queue));
362
363 if (tq->tqi_flags
364 & AR5K_TXQ_FLAG_RDYTIME_EXP_POLICY_ENABLE)
365 AR5K_REG_ENABLE_BITS(ah,
366 AR5K_QUEUE_MISC(queue),
367 AR5K_QCU_MISC_RDY_VEOL_POLICY);
368 }
369
370 if (tq->tqi_flags & AR5K_TXQ_FLAG_BACKOFF_DISABLE)
371 ath5k_hw_reg_write(ah, AR5K_DCU_MISC_POST_FR_BKOFF_DIS,
372 AR5K_QUEUE_DFS_MISC(queue));
373
374 if (tq->tqi_flags & AR5K_TXQ_FLAG_FRAG_BURST_BACKOFF_ENABLE)
375 ath5k_hw_reg_write(ah, AR5K_DCU_MISC_BACKOFF_FRAG,
376 AR5K_QUEUE_DFS_MISC(queue));
377
378 /*
379 * Set registers by queue type
380 */
381 switch (tq->tqi_type) {
382 case AR5K_TX_QUEUE_BEACON:
383 AR5K_REG_ENABLE_BITS(ah, AR5K_QUEUE_MISC(queue),
384 AR5K_QCU_MISC_FRSHED_DBA_GT |
Nick Kossifidis1bef0162008-09-29 02:09:09 +0300385 AR5K_QCU_MISC_CBREXP_BCN_DIS |
Nick Kossifidisc6e387a2008-08-29 22:45:39 +0300386 AR5K_QCU_MISC_BCN_ENABLE);
387
388 AR5K_REG_ENABLE_BITS(ah, AR5K_QUEUE_DFS_MISC(queue),
389 (AR5K_DCU_MISC_ARBLOCK_CTL_GLOBAL <<
390 AR5K_DCU_MISC_ARBLOCK_CTL_S) |
Nick Kossifidis428cbd42009-04-30 15:55:47 -0400391 AR5K_DCU_MISC_ARBLOCK_IGNORE |
Nick Kossifidisc6e387a2008-08-29 22:45:39 +0300392 AR5K_DCU_MISC_POST_FR_BKOFF_DIS |
393 AR5K_DCU_MISC_BCN_ENABLE);
Nick Kossifidisc6e387a2008-08-29 22:45:39 +0300394 break;
395
396 case AR5K_TX_QUEUE_CAB:
Bob Copelanda951ae22010-01-20 23:51:04 -0500397 /* XXX: use BCN_SENT_GT, if we can figure out how */
Nick Kossifidisc6e387a2008-08-29 22:45:39 +0300398 AR5K_REG_ENABLE_BITS(ah, AR5K_QUEUE_MISC(queue),
Bob Copelanda951ae22010-01-20 23:51:04 -0500399 AR5K_QCU_MISC_FRSHED_DBA_GT |
Nick Kossifidis1bef0162008-09-29 02:09:09 +0300400 AR5K_QCU_MISC_CBREXP_DIS |
401 AR5K_QCU_MISC_CBREXP_BCN_DIS);
Nick Kossifidisc6e387a2008-08-29 22:45:39 +0300402
Bob Copelanda951ae22010-01-20 23:51:04 -0500403 ath5k_hw_reg_write(ah, ((tq->tqi_ready_time -
Nick Kossifidis846567622009-01-06 17:27:06 +0200404 (AR5K_TUNE_SW_BEACON_RESP -
405 AR5K_TUNE_DMA_BEACON_RESP) -
406 AR5K_TUNE_ADDITIONAL_SWBA_BACKOFF) * 1024) |
407 AR5K_QCU_RDYTIMECFG_ENABLE,
408 AR5K_QUEUE_RDYTIMECFG(queue));
409
Nick Kossifidisc6e387a2008-08-29 22:45:39 +0300410 AR5K_REG_ENABLE_BITS(ah, AR5K_QUEUE_DFS_MISC(queue),
411 (AR5K_DCU_MISC_ARBLOCK_CTL_GLOBAL <<
412 AR5K_DCU_MISC_ARBLOCK_CTL_S));
413 break;
414
415 case AR5K_TX_QUEUE_UAPSD:
416 AR5K_REG_ENABLE_BITS(ah, AR5K_QUEUE_MISC(queue),
Nick Kossifidis1bef0162008-09-29 02:09:09 +0300417 AR5K_QCU_MISC_CBREXP_DIS);
Nick Kossifidisc6e387a2008-08-29 22:45:39 +0300418 break;
419
420 case AR5K_TX_QUEUE_DATA:
421 default:
422 break;
423 }
424
Nick Kossifidis846567622009-01-06 17:27:06 +0200425 /* TODO: Handle frame compression */
426
Nick Kossifidisc6e387a2008-08-29 22:45:39 +0300427 /*
428 * Enable interrupts for this tx queue
429 * in the secondary interrupt mask registers
430 */
431 if (tq->tqi_flags & AR5K_TXQ_FLAG_TXOKINT_ENABLE)
432 AR5K_Q_ENABLE_BITS(ah->ah_txq_imr_txok, queue);
433
434 if (tq->tqi_flags & AR5K_TXQ_FLAG_TXERRINT_ENABLE)
435 AR5K_Q_ENABLE_BITS(ah->ah_txq_imr_txerr, queue);
436
437 if (tq->tqi_flags & AR5K_TXQ_FLAG_TXURNINT_ENABLE)
438 AR5K_Q_ENABLE_BITS(ah->ah_txq_imr_txurn, queue);
439
440 if (tq->tqi_flags & AR5K_TXQ_FLAG_TXDESCINT_ENABLE)
441 AR5K_Q_ENABLE_BITS(ah->ah_txq_imr_txdesc, queue);
442
443 if (tq->tqi_flags & AR5K_TXQ_FLAG_TXEOLINT_ENABLE)
444 AR5K_Q_ENABLE_BITS(ah->ah_txq_imr_txeol, queue);
445
Nick Kossifidis4c674c62008-10-26 20:40:25 +0200446 if (tq->tqi_flags & AR5K_TXQ_FLAG_CBRORNINT_ENABLE)
447 AR5K_Q_ENABLE_BITS(ah->ah_txq_imr_cbrorn, queue);
448
449 if (tq->tqi_flags & AR5K_TXQ_FLAG_CBRURNINT_ENABLE)
450 AR5K_Q_ENABLE_BITS(ah->ah_txq_imr_cbrurn, queue);
451
452 if (tq->tqi_flags & AR5K_TXQ_FLAG_QTRIGINT_ENABLE)
453 AR5K_Q_ENABLE_BITS(ah->ah_txq_imr_qtrig, queue);
454
455 if (tq->tqi_flags & AR5K_TXQ_FLAG_TXNOFRMINT_ENABLE)
456 AR5K_Q_ENABLE_BITS(ah->ah_txq_imr_nofrm, queue);
Nick Kossifidisc6e387a2008-08-29 22:45:39 +0300457
458 /* Update secondary interrupt mask registers */
Nick Kossifidis4c674c62008-10-26 20:40:25 +0200459
460 /* Filter out inactive queues */
Nick Kossifidisc6e387a2008-08-29 22:45:39 +0300461 ah->ah_txq_imr_txok &= ah->ah_txq_status;
462 ah->ah_txq_imr_txerr &= ah->ah_txq_status;
463 ah->ah_txq_imr_txurn &= ah->ah_txq_status;
464 ah->ah_txq_imr_txdesc &= ah->ah_txq_status;
465 ah->ah_txq_imr_txeol &= ah->ah_txq_status;
Nick Kossifidis4c674c62008-10-26 20:40:25 +0200466 ah->ah_txq_imr_cbrorn &= ah->ah_txq_status;
467 ah->ah_txq_imr_cbrurn &= ah->ah_txq_status;
468 ah->ah_txq_imr_qtrig &= ah->ah_txq_status;
469 ah->ah_txq_imr_nofrm &= ah->ah_txq_status;
Nick Kossifidisc6e387a2008-08-29 22:45:39 +0300470
471 ath5k_hw_reg_write(ah, AR5K_REG_SM(ah->ah_txq_imr_txok,
472 AR5K_SIMR0_QCU_TXOK) |
473 AR5K_REG_SM(ah->ah_txq_imr_txdesc,
474 AR5K_SIMR0_QCU_TXDESC), AR5K_SIMR0);
475 ath5k_hw_reg_write(ah, AR5K_REG_SM(ah->ah_txq_imr_txerr,
476 AR5K_SIMR1_QCU_TXERR) |
477 AR5K_REG_SM(ah->ah_txq_imr_txeol,
478 AR5K_SIMR1_QCU_TXEOL), AR5K_SIMR1);
Nick Kossifidis4c674c62008-10-26 20:40:25 +0200479 /* Update simr2 but don't overwrite rest simr2 settings */
480 AR5K_REG_DISABLE_BITS(ah, AR5K_SIMR2, AR5K_SIMR2_QCU_TXURN);
481 AR5K_REG_ENABLE_BITS(ah, AR5K_SIMR2,
482 AR5K_REG_SM(ah->ah_txq_imr_txurn,
483 AR5K_SIMR2_QCU_TXURN));
484 ath5k_hw_reg_write(ah, AR5K_REG_SM(ah->ah_txq_imr_cbrorn,
485 AR5K_SIMR3_QCBRORN) |
486 AR5K_REG_SM(ah->ah_txq_imr_cbrurn,
487 AR5K_SIMR3_QCBRURN), AR5K_SIMR3);
488 ath5k_hw_reg_write(ah, AR5K_REG_SM(ah->ah_txq_imr_qtrig,
489 AR5K_SIMR4_QTRIG), AR5K_SIMR4);
490 /* Set TXNOFRM_QCU for the queues with TXNOFRM enabled */
491 ath5k_hw_reg_write(ah, AR5K_REG_SM(ah->ah_txq_imr_nofrm,
492 AR5K_TXNOFRM_QCU), AR5K_TXNOFRM);
493 /* No queue has TXNOFRM enabled, disable the interrupt
494 * by setting AR5K_TXNOFRM to zero */
495 if (ah->ah_txq_imr_nofrm == 0)
496 ath5k_hw_reg_write(ah, 0, AR5K_TXNOFRM);
Nick Kossifidis846567622009-01-06 17:27:06 +0200497
498 /* Set QCU mask for this DCU to save power */
499 AR5K_REG_WRITE_Q(ah, AR5K_QUEUE_QCUMASK(queue), queue);
Nick Kossifidisc6e387a2008-08-29 22:45:39 +0300500 }
501
502 return 0;
503}
504
Nick Kossifidis9320b5c42010-11-23 20:36:45 +0200505
506/**************************\
507* Global QCU/DCU functions *
508\**************************/
509
Nick Kossifidisc6e387a2008-08-29 22:45:39 +0300510/*
511 * Set slot time on DCU
512 */
513int ath5k_hw_set_slot_time(struct ath5k_hw *ah, unsigned int slot_time)
514{
Lukáš Turek3578e6e2009-12-21 22:50:50 +0100515 u32 slot_time_clock = ath5k_hw_htoclock(ah, slot_time);
Lukáš Tureke1aa3692009-12-21 22:50:49 +0100516
Lukáš Tureke1aa3692009-12-21 22:50:49 +0100517 if (slot_time < 6 || slot_time_clock > AR5K_SLOT_TIME_MAX)
Nick Kossifidisc6e387a2008-08-29 22:45:39 +0300518 return -EINVAL;
519
520 if (ah->ah_version == AR5K_AR5210)
Lukáš Tureke1aa3692009-12-21 22:50:49 +0100521 ath5k_hw_reg_write(ah, slot_time_clock, AR5K_SLOT_TIME);
Nick Kossifidisc6e387a2008-08-29 22:45:39 +0300522 else
Lukáš Tureke1aa3692009-12-21 22:50:49 +0100523 ath5k_hw_reg_write(ah, slot_time_clock, AR5K_DCU_GBL_IFS_SLOT);
Nick Kossifidisc6e387a2008-08-29 22:45:39 +0300524
525 return 0;
526}
527
Nick Kossifidis9320b5c42010-11-23 20:36:45 +0200528int ath5k_hw_init_queues(struct ath5k_hw *ah)
529{
530 int i, ret;
531
532 /* TODO: HW Compression support for data queues */
533 /* TODO: Burst prefetch for data queues */
534
535 /*
536 * Reset queues and start beacon timers at the end of the reset routine
537 * This also sets QCU mask on each DCU for 1:1 qcu to dcu mapping
538 * Note: If we want we can assign multiple qcus on one dcu.
539 */
540 for (i = 0; i < ah->ah_capabilities.cap_queues.q_tx_num; i++) {
541 ret = ath5k_hw_reset_tx_queue(ah, i);
542 if (ret) {
543 ATH5K_ERR(ah->ah_sc,
544 "failed to reset TX queue #%d\n", i);
545 return ret;
546 }
547 }
548
549 return 0;
550}