[s3c24xx] bump to 2.6.30-rc6
[openwrt/svn-archive/archive.git] / target / linux / s3c24xx / files-2.6.30 / drivers / ar6000 / htc / ar6k_events.c
1 /*
2 * AR6K Driver layer event handling (i.e. interrupts, message polling)
3 *
4 * Copyright (c) 2007 Atheros Communications Inc.
5 * All rights reserved.
6 *
7 *
8 * This program is free software; you can redistribute it and/or modify
9 * it under the terms of the GNU General Public License version 2 as
10 * published by the Free Software Foundation;
11 *
12 * Software distributed under the License is distributed on an "AS
13 * IS" basis, WITHOUT WARRANTY OF ANY KIND, either express or
14 * implied. See the License for the specific language governing
15 * rights and limitations under the License.
16 *
17 *
18 *
19 */
20 #include "a_config.h"
21 #include "athdefs.h"
22 #include "a_types.h"
23 #include "AR6Khwreg.h"
24 #include "a_osapi.h"
25 #include "a_debug.h"
26 #include "hif.h"
27 #include "htc_packet.h"
28 #include "ar6k.h"
29
30 extern void AR6KFreeIOPacket(AR6K_DEVICE *pDev, HTC_PACKET *pPacket);
31 extern HTC_PACKET *AR6KAllocIOPacket(AR6K_DEVICE *pDev);
32
33 static A_STATUS DevServiceDebugInterrupt(AR6K_DEVICE *pDev);
34
35 #define DELAY_PER_INTERVAL_MS 10 /* 10 MS delay per polling interval */
36
37 /* completion routine for ALL HIF layer async I/O */
38 A_STATUS DevRWCompletionHandler(void *context, A_STATUS status)
39 {
40 HTC_PACKET *pPacket = (HTC_PACKET *)context;
41
42 COMPLETE_HTC_PACKET(pPacket,status);
43
44 return A_OK;
45 }
46
47 /* mailbox recv message polling */
48 A_STATUS DevPollMboxMsgRecv(AR6K_DEVICE *pDev,
49 A_UINT32 *pLookAhead,
50 int TimeoutMS)
51 {
52 A_STATUS status = A_OK;
53 int timeout = TimeoutMS/DELAY_PER_INTERVAL_MS;
54
55 AR_DEBUG_ASSERT(timeout > 0);
56
57 AR_DEBUG_PRINTF(ATH_DEBUG_RECV,("+DevPollMboxMsgRecv \n"));
58
59 while (TRUE) {
60
61 if (pDev->GetPendingEventsFunc != NULL)
62 {
63
64 HIF_PENDING_EVENTS_INFO events;
65
66 /* the HIF layer uses a special mechanism to get events, do this
67 * synchronously */
68 status = pDev->GetPendingEventsFunc(pDev->HIFDevice,
69 &events,
70 NULL);
71 if (A_FAILED(status))
72 {
73 AR_DEBUG_PRINTF(ATH_DEBUG_ERR,("Failed to get pending events \n"));
74 break;
75 }
76
77 if (events.Events & HIF_RECV_MSG_AVAIL)
78 {
79 /* there is a message available, the lookahead should be valid now */
80 *pLookAhead = events.LookAhead;
81
82 break;
83 }
84 }
85 else
86 {
87
88 /* this is the standard HIF way.... */
89 /* load the register table */
90 status = HIFReadWrite(pDev->HIFDevice,
91 HOST_INT_STATUS_ADDRESS,
92 (A_UINT8 *)&pDev->IrqProcRegisters,
93 AR6K_IRQ_PROC_REGS_SIZE,
94 HIF_RD_SYNC_BYTE_INC,
95 NULL);
96
97 if (A_FAILED(status))
98 {
99 AR_DEBUG_PRINTF(ATH_DEBUG_ERR,("Failed to read register table \n"));
100 break;
101 }
102
103 /* check for MBOX data and valid lookahead */
104 if (pDev->IrqProcRegisters.host_int_status & (1 << HTC_MAILBOX))
105 {
106 if (pDev->IrqProcRegisters.rx_lookahead_valid & (1 << HTC_MAILBOX))
107 {
108 /* mailbox has a message and the look ahead is valid */
109 *pLookAhead = pDev->IrqProcRegisters.rx_lookahead[HTC_MAILBOX];
110 break;
111 }
112 }
113
114 }
115
116 timeout--;
117
118 if (timeout <= 0)
119 {
120 AR_DEBUG_PRINTF(ATH_DEBUG_ERR, (" Timeout waiting for recv message \n"));
121 status = A_ERROR;
122
123 /* check if the target asserted */
124 if ( pDev->IrqProcRegisters.counter_int_status & AR6K_TARGET_DEBUG_INTR_MASK) {
125 /* target signaled an assert, process this pending interrupt
126 * this will call the target failure handler */
127 DevServiceDebugInterrupt(pDev);
128 }
129
130 break;
131 }
132
133 /* delay a little */
134 msleep(DELAY_PER_INTERVAL_MS);
135 AR_DEBUG_PRINTF(ATH_DEBUG_RECV,(" Retry Mbox Poll : %d \n",timeout));
136 }
137
138 AR_DEBUG_PRINTF(ATH_DEBUG_RECV,("-DevPollMboxMsgRecv \n"));
139
140 return status;
141 }
142
143 static A_STATUS DevServiceCPUInterrupt(AR6K_DEVICE *pDev)
144 {
145 A_STATUS status;
146 A_UINT8 cpu_int_status;
147 A_UINT8 regBuffer[4];
148
149 AR_DEBUG_PRINTF(ATH_DEBUG_IRQ, ("CPU Interrupt\n"));
150 cpu_int_status = pDev->IrqProcRegisters.cpu_int_status &
151 pDev->IrqEnableRegisters.cpu_int_status_enable;
152 AR_DEBUG_ASSERT(cpu_int_status);
153 AR_DEBUG_PRINTF(ATH_DEBUG_IRQ,
154 ("Valid interrupt source(s) in CPU_INT_STATUS: 0x%x\n",
155 cpu_int_status));
156
157 /* Clear the interrupt */
158 pDev->IrqProcRegisters.cpu_int_status &= ~cpu_int_status; /* W1C */
159
160 /* set up the register transfer buffer to hit the register 4 times , this is done
161 * to make the access 4-byte aligned to mitigate issues with host bus interconnects that
162 * restrict bus transfer lengths to be a multiple of 4-bytes */
163
164 /* set W1C value to clear the interrupt, this hits the register first */
165 regBuffer[0] = cpu_int_status;
166 /* the remaining 4 values are set to zero which have no-effect */
167 regBuffer[1] = 0;
168 regBuffer[2] = 0;
169 regBuffer[3] = 0;
170
171 status = HIFReadWrite(pDev->HIFDevice,
172 CPU_INT_STATUS_ADDRESS,
173 regBuffer,
174 4,
175 HIF_WR_SYNC_BYTE_FIX,
176 NULL);
177
178 AR_DEBUG_ASSERT(status == A_OK);
179 return status;
180 }
181
182
183 static A_STATUS DevServiceErrorInterrupt(AR6K_DEVICE *pDev)
184 {
185 A_STATUS status;
186 A_UINT8 error_int_status;
187 A_UINT8 regBuffer[4];
188
189 AR_DEBUG_PRINTF(ATH_DEBUG_IRQ, ("Error Interrupt\n"));
190 error_int_status = pDev->IrqProcRegisters.error_int_status & 0x0F;
191 AR_DEBUG_ASSERT(error_int_status);
192 AR_DEBUG_PRINTF(ATH_DEBUG_IRQ,
193 ("Valid interrupt source(s) in ERROR_INT_STATUS: 0x%x\n",
194 error_int_status));
195
196 if (ERROR_INT_STATUS_WAKEUP_GET(error_int_status)) {
197 /* Wakeup */
198 AR_DEBUG_PRINTF(ATH_DEBUG_IRQ, ("Error : Wakeup\n"));
199 }
200
201 if (ERROR_INT_STATUS_RX_UNDERFLOW_GET(error_int_status)) {
202 /* Rx Underflow */
203 AR_DEBUG_PRINTF(ATH_DEBUG_ERR, ("Error : Rx Underflow\n"));
204 }
205
206 if (ERROR_INT_STATUS_TX_OVERFLOW_GET(error_int_status)) {
207 /* Tx Overflow */
208 AR_DEBUG_PRINTF(ATH_DEBUG_ERR, ("Error : Tx Overflow\n"));
209 }
210
211 /* Clear the interrupt */
212 pDev->IrqProcRegisters.error_int_status &= ~error_int_status; /* W1C */
213
214 /* set up the register transfer buffer to hit the register 4 times , this is done
215 * to make the access 4-byte aligned to mitigate issues with host bus interconnects that
216 * restrict bus transfer lengths to be a multiple of 4-bytes */
217
218 /* set W1C value to clear the interrupt, this hits the register first */
219 regBuffer[0] = error_int_status;
220 /* the remaining 4 values are set to zero which have no-effect */
221 regBuffer[1] = 0;
222 regBuffer[2] = 0;
223 regBuffer[3] = 0;
224
225 status = HIFReadWrite(pDev->HIFDevice,
226 ERROR_INT_STATUS_ADDRESS,
227 regBuffer,
228 4,
229 HIF_WR_SYNC_BYTE_FIX,
230 NULL);
231
232 AR_DEBUG_ASSERT(status == A_OK);
233 return status;
234 }
235
236 static A_STATUS DevServiceDebugInterrupt(AR6K_DEVICE *pDev)
237 {
238 A_UINT32 dummy;
239 A_STATUS status;
240
241 /* Send a target failure event to the application */
242 AR_DEBUG_PRINTF(ATH_DEBUG_ERR, ("Target debug interrupt\n"));
243
244 if (pDev->TargetFailureCallback != NULL) {
245 pDev->TargetFailureCallback(pDev->HTCContext);
246 }
247
248 /* clear the interrupt , the debug error interrupt is
249 * counter 0 */
250 /* read counter to clear interrupt */
251 status = HIFReadWrite(pDev->HIFDevice,
252 COUNT_DEC_ADDRESS,
253 (A_UINT8 *)&dummy,
254 4,
255 HIF_RD_SYNC_BYTE_INC,
256 NULL);
257
258 AR_DEBUG_ASSERT(status == A_OK);
259 return status;
260 }
261
262 static A_STATUS DevServiceCounterInterrupt(AR6K_DEVICE *pDev)
263 {
264 A_UINT8 counter_int_status;
265
266 AR_DEBUG_PRINTF(ATH_DEBUG_IRQ, ("Counter Interrupt\n"));
267
268 counter_int_status = pDev->IrqProcRegisters.counter_int_status &
269 pDev->IrqEnableRegisters.counter_int_status_enable;
270
271 AR_DEBUG_ASSERT(counter_int_status);
272 AR_DEBUG_PRINTF(ATH_DEBUG_IRQ,
273 ("Valid interrupt source(s) in COUNTER_INT_STATUS: 0x%x\n",
274 counter_int_status));
275
276 /* Check if the debug interrupt is pending */
277 if (counter_int_status & AR6K_TARGET_DEBUG_INTR_MASK) {
278 return DevServiceDebugInterrupt(pDev);
279 }
280
281 return A_OK;
282 }
283
284 /* callback when our fetch to get interrupt status registers completes */
285 static void DevGetEventAsyncHandler(void *Context, HTC_PACKET *pPacket)
286 {
287 AR6K_DEVICE *pDev = (AR6K_DEVICE *)Context;
288 A_UINT32 lookAhead = 0;
289 A_BOOL otherInts = FALSE;
290
291 AR_DEBUG_PRINTF(ATH_DEBUG_IRQ,("+DevGetEventAsyncHandler: (dev: 0x%X)\n", (A_UINT32)pDev));
292
293 do {
294
295 if (A_FAILED(pPacket->Status)) {
296 AR_DEBUG_PRINTF(ATH_DEBUG_ERR,
297 (" GetEvents I/O request failed, status:%d \n", pPacket->Status));
298 /* bail out, don't unmask HIF interrupt */
299 break;
300 }
301
302 if (pDev->GetPendingEventsFunc != NULL) {
303 /* the HIF layer collected the information for us */
304 HIF_PENDING_EVENTS_INFO *pEvents = (HIF_PENDING_EVENTS_INFO *)pPacket->pBuffer;
305 if (pEvents->Events & HIF_RECV_MSG_AVAIL) {
306 lookAhead = pEvents->LookAhead;
307 if (0 == lookAhead) {
308 AR_DEBUG_PRINTF(ATH_DEBUG_ERR,(" DevGetEventAsyncHandler1, lookAhead is zero! \n"));
309 }
310 }
311 if (pEvents->Events & HIF_OTHER_EVENTS) {
312 otherInts = TRUE;
313 }
314 } else {
315 /* standard interrupt table handling.... */
316 AR6K_IRQ_PROC_REGISTERS *pReg = (AR6K_IRQ_PROC_REGISTERS *)pPacket->pBuffer;
317 A_UINT8 host_int_status;
318
319 host_int_status = pReg->host_int_status & pDev->IrqEnableRegisters.int_status_enable;
320
321 if (host_int_status & (1 << HTC_MAILBOX)) {
322 host_int_status &= ~(1 << HTC_MAILBOX);
323 if (pReg->rx_lookahead_valid & (1 << HTC_MAILBOX)) {
324 /* mailbox has a message and the look ahead is valid */
325 lookAhead = pReg->rx_lookahead[HTC_MAILBOX];
326 if (0 == lookAhead) {
327 AR_DEBUG_PRINTF(ATH_DEBUG_ERR,(" DevGetEventAsyncHandler2, lookAhead is zero! \n"));
328 }
329 }
330 }
331
332 if (host_int_status) {
333 /* there are other interrupts to handle */
334 otherInts = TRUE;
335 }
336 }
337
338 if (otherInts || (lookAhead == 0)) {
339 /* if there are other interrupts to process, we cannot do this in the async handler so
340 * ack the interrupt which will cause our sync handler to run again
341 * if however there are no more messages, we can now ack the interrupt */
342 AR_DEBUG_PRINTF(ATH_DEBUG_IRQ,
343 (" Acking interrupt from DevGetEventAsyncHandler (otherints:%d, lookahead:0x%X)\n",
344 otherInts, lookAhead));
345 HIFAckInterrupt(pDev->HIFDevice);
346 } else {
347 AR_DEBUG_PRINTF(ATH_DEBUG_IRQ,
348 (" DevGetEventAsyncHandler : detected another message, lookahead :0x%X \n",
349 lookAhead));
350 /* lookahead is non-zero and there are no other interrupts to service,
351 * go get the next message */
352 pDev->MessagePendingCallback(pDev->HTCContext, lookAhead, NULL);
353 }
354
355 } while (FALSE);
356
357 /* free this IO packet */
358 AR6KFreeIOPacket(pDev,pPacket);
359 AR_DEBUG_PRINTF(ATH_DEBUG_IRQ,("-DevGetEventAsyncHandler \n"));
360 }
361
362 /* called by the HTC layer when it wants us to check if the device has any more pending
363 * recv messages, this starts off a series of async requests to read interrupt registers */
364 A_STATUS DevCheckPendingRecvMsgsAsync(void *context)
365 {
366 AR6K_DEVICE *pDev = (AR6K_DEVICE *)context;
367 A_STATUS status = A_OK;
368 HTC_PACKET *pIOPacket;
369
370 /* this is called in an ASYNC only context, we may NOT block, sleep or call any apis that can
371 * cause us to switch contexts */
372
373 AR_DEBUG_PRINTF(ATH_DEBUG_IRQ,("+DevCheckPendingRecvMsgsAsync: (dev: 0x%X)\n", (A_UINT32)pDev));
374
375 do {
376
377 if (HIF_DEVICE_IRQ_SYNC_ONLY == pDev->HifIRQProcessingMode) {
378 /* break the async processing chain right here, no need to continue.
379 * The DevDsrHandler() will handle things in a loop when things are driven
380 * synchronously */
381 break;
382 }
383 /* first allocate one of our HTC packets we created for async I/O
384 * we reuse HTC packet definitions so that we can use the completion mechanism
385 * in DevRWCompletionHandler() */
386 pIOPacket = AR6KAllocIOPacket(pDev);
387
388 if (NULL == pIOPacket) {
389 /* there should be only 1 asynchronous request out at a time to read these registers
390 * so this should actually never happen */
391 status = A_NO_MEMORY;
392 AR_DEBUG_ASSERT(FALSE);
393 break;
394 }
395
396 /* stick in our completion routine when the I/O operation completes */
397 pIOPacket->Completion = DevGetEventAsyncHandler;
398 pIOPacket->pContext = pDev;
399
400 if (pDev->GetPendingEventsFunc) {
401 /* HIF layer has it's own mechanism, pass the IO to it.. */
402 status = pDev->GetPendingEventsFunc(pDev->HIFDevice,
403 (HIF_PENDING_EVENTS_INFO *)pIOPacket->pBuffer,
404 pIOPacket);
405
406 } else {
407 /* standard way, read the interrupt register table asynchronously again */
408 status = HIFReadWrite(pDev->HIFDevice,
409 HOST_INT_STATUS_ADDRESS,
410 pIOPacket->pBuffer,
411 AR6K_IRQ_PROC_REGS_SIZE,
412 HIF_RD_ASYNC_BYTE_INC,
413 pIOPacket);
414 }
415
416 AR_DEBUG_PRINTF(ATH_DEBUG_IRQ,(" Async IO issued to get interrupt status...\n"));
417 } while (FALSE);
418
419 AR_DEBUG_PRINTF(ATH_DEBUG_IRQ,("-DevCheckPendingRecvMsgsAsync \n"));
420
421 return status;
422 }
423
424 /* process pending interrupts synchronously */
425 static A_STATUS ProcessPendingIRQs(AR6K_DEVICE *pDev, A_BOOL *pDone, A_BOOL *pASyncProcessing)
426 {
427 A_STATUS status = A_OK;
428 A_UINT8 host_int_status = 0;
429 A_UINT32 lookAhead = 0;
430
431 AR_DEBUG_PRINTF(ATH_DEBUG_IRQ,("+ProcessPendingIRQs: (dev: 0x%X)\n", (A_UINT32)pDev));
432
433 /*** NOTE: the HIF implementation guarantees that the context of this call allows
434 * us to perform SYNCHRONOUS I/O, that is we can block, sleep or call any API that
435 * can block or switch thread/task ontexts.
436 * This is a fully schedulable context.
437 * */
438 do {
439
440 if (pDev->GetPendingEventsFunc != NULL) {
441 HIF_PENDING_EVENTS_INFO events;
442
443 /* the HIF layer uses a special mechanism to get events
444 * get this synchronously */
445 status = pDev->GetPendingEventsFunc(pDev->HIFDevice,
446 &events,
447 NULL);
448
449 if (A_FAILED(status)) {
450 break;
451 }
452
453 if (events.Events & HIF_RECV_MSG_AVAIL) {
454 lookAhead = events.LookAhead;
455 if (0 == lookAhead) {
456 AR_DEBUG_PRINTF(ATH_DEBUG_ERR,(" ProcessPendingIRQs1 lookAhead is zero! \n"));
457 }
458 }
459
460 if (!(events.Events & HIF_OTHER_EVENTS) ||
461 !(pDev->IrqEnableRegisters.int_status_enable & OTHER_INTS_ENABLED)) {
462 /* no need to read the register table, no other interesting interrupts.
463 * Some interfaces (like SPI) can shadow interrupt sources without
464 * requiring the host to do a full table read */
465 break;
466 }
467
468 /* otherwise fall through and read the register table */
469 }
470
471 /*
472 * Read the first 28 bytes of the HTC register table. This will yield us
473 * the value of different int status registers and the lookahead
474 * registers.
475 * length = sizeof(int_status) + sizeof(cpu_int_status) +
476 * sizeof(error_int_status) + sizeof(counter_int_status) +
477 * sizeof(mbox_frame) + sizeof(rx_lookahead_valid) +
478 * sizeof(hole) + sizeof(rx_lookahead) +
479 * sizeof(int_status_enable) + sizeof(cpu_int_status_enable) +
480 * sizeof(error_status_enable) +
481 * sizeof(counter_int_status_enable);
482 *
483 */
484 status = HIFReadWrite(pDev->HIFDevice,
485 HOST_INT_STATUS_ADDRESS,
486 (A_UINT8 *)&pDev->IrqProcRegisters,
487 AR6K_IRQ_PROC_REGS_SIZE,
488 HIF_RD_SYNC_BYTE_INC,
489 NULL);
490
491 if (A_FAILED(status)) {
492 break;
493 }
494
495 if (AR_DEBUG_LVL_CHECK(ATH_DEBUG_IRQ)) {
496 DevDumpRegisters(&pDev->IrqProcRegisters,
497 &pDev->IrqEnableRegisters);
498 }
499
500 /* Update only those registers that are enabled */
501 host_int_status = pDev->IrqProcRegisters.host_int_status &
502 pDev->IrqEnableRegisters.int_status_enable;
503
504 if (NULL == pDev->GetPendingEventsFunc) {
505 /* only look at mailbox status if the HIF layer did not provide this function,
506 * on some HIF interfaces reading the RX lookahead is not valid to do */
507 if (host_int_status & (1 << HTC_MAILBOX)) {
508 /* mask out pending mailbox value, we use "lookAhead" as the real flag for
509 * mailbox processing below */
510 host_int_status &= ~(1 << HTC_MAILBOX);
511 if (pDev->IrqProcRegisters.rx_lookahead_valid & (1 << HTC_MAILBOX)) {
512 /* mailbox has a message and the look ahead is valid */
513 lookAhead = pDev->IrqProcRegisters.rx_lookahead[HTC_MAILBOX];
514 if (0 == lookAhead) {
515 AR_DEBUG_PRINTF(ATH_DEBUG_ERR,(" ProcessPendingIRQs2, lookAhead is zero! \n"));
516 }
517 }
518 }
519 } else {
520 /* not valid to check if the HIF has another mechanism for reading mailbox pending status*/
521 host_int_status &= ~(1 << HTC_MAILBOX);
522 }
523
524 } while (FALSE);
525
526
527 do {
528
529 /* did the interrupt status fetches succeed? */
530 if (A_FAILED(status)) {
531 break;
532 }
533
534 if ((0 == host_int_status) && (0 == lookAhead)) {
535 /* nothing to process, the caller can use this to break out of a loop */
536 *pDone = TRUE;
537 break;
538 }
539
540 if (lookAhead != 0) {
541 AR_DEBUG_PRINTF(ATH_DEBUG_IRQ,("Pending mailbox message, LookAhead: 0x%X\n",lookAhead));
542 /* Mailbox Interrupt, the HTC layer may issue async requests to empty the
543 * mailbox...
544 * When emptying the recv mailbox we use the async handler above called from the
545 * completion routine of the callers read request. This can improve performance
546 * by reducing context switching when we rapidly pull packets */
547 status = pDev->MessagePendingCallback(pDev->HTCContext, lookAhead, pASyncProcessing);
548 if (A_FAILED(status)) {
549 break;
550 }
551 }
552
553 /* now handle the rest of them */
554 AR_DEBUG_PRINTF(ATH_DEBUG_IRQ,
555 (" Valid interrupt source(s) for OTHER interrupts: 0x%x\n",
556 host_int_status));
557
558 if (HOST_INT_STATUS_CPU_GET(host_int_status)) {
559 /* CPU Interrupt */
560 status = DevServiceCPUInterrupt(pDev);
561 if (A_FAILED(status)){
562 break;
563 }
564 }
565
566 if (HOST_INT_STATUS_ERROR_GET(host_int_status)) {
567 /* Error Interrupt */
568 status = DevServiceErrorInterrupt(pDev);
569 if (A_FAILED(status)){
570 break;
571 }
572 }
573
574 if (HOST_INT_STATUS_COUNTER_GET(host_int_status)) {
575 /* Counter Interrupt */
576 status = DevServiceCounterInterrupt(pDev);
577 if (A_FAILED(status)){
578 break;
579 }
580 }
581
582 } while (FALSE);
583
584 AR_DEBUG_PRINTF(ATH_DEBUG_IRQ,("-ProcessPendingIRQs: (done:%d, async:%d) status=%d \n",
585 *pDone, *pASyncProcessing, status));
586
587 return status;
588 }
589
590
591 /* Synchronousinterrupt handler, this handler kicks off all interrupt processing.*/
592 A_STATUS DevDsrHandler(void *context)
593 {
594 AR6K_DEVICE *pDev = (AR6K_DEVICE *)context;
595 A_STATUS status = A_OK;
596 A_BOOL done = FALSE;
597 A_BOOL asyncProc = FALSE;
598
599 AR_DEBUG_PRINTF(ATH_DEBUG_IRQ,("+DevDsrHandler: (dev: 0x%X)\n", (A_UINT32)pDev));
600
601
602 while (!done) {
603 status = ProcessPendingIRQs(pDev, &done, &asyncProc);
604 if (A_FAILED(status)) {
605 break;
606 }
607
608 if (HIF_DEVICE_IRQ_SYNC_ONLY == pDev->HifIRQProcessingMode) {
609 /* the HIF layer does not allow async IRQ processing, override the asyncProc flag */
610 asyncProc = FALSE;
611 /* this will cause us to re-enter ProcessPendingIRQ() and re-read interrupt status registers.
612 * this has a nice side effect of blocking us until all async read requests are completed.
613 * This behavior is required on some HIF implementations that do not allow ASYNC
614 * processing in interrupt handlers (like Windows CE) */
615 }
616
617 if (asyncProc) {
618 /* the function performed some async I/O for performance, we
619 need to exit the ISR immediately, the check below will prevent the interrupt from being
620 Ack'd while we handle it asynchronously */
621 break;
622 }
623
624 }
625
626 if (A_SUCCESS(status) && !asyncProc) {
627 /* Ack the interrupt only if :
628 * 1. we did not get any errors in processing interrupts
629 * 2. there are no outstanding async processing requests */
630 AR_DEBUG_PRINTF(ATH_DEBUG_IRQ,(" Acking interrupt from DevDsrHandler \n"));
631 HIFAckInterrupt(pDev->HIFDevice);
632 }
633
634 AR_DEBUG_PRINTF(ATH_DEBUG_IRQ,("-DevDsrHandler \n"));
635 return A_OK;
636 }
637
638