1 1.1 alc /* 2 1.1 alc * Copyright (c) 2002-2008 Sam Leffler, Errno Consulting 3 1.1 alc * Copyright (c) 2002-2004 Atheros Communications, Inc. 4 1.1 alc * 5 1.1 alc * Permission to use, copy, modify, and/or distribute this software for any 6 1.1 alc * purpose with or without fee is hereby granted, provided that the above 7 1.1 alc * copyright notice and this permission notice appear in all copies. 8 1.1 alc * 9 1.1 alc * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES 10 1.1 alc * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF 11 1.1 alc * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR 12 1.1 alc * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES 13 1.1 alc * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN 14 1.1 alc * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF 15 1.1 alc * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. 16 1.1 alc * 17 1.2 cegger * $Id: ar5210_interrupts.c,v 1.2 2011/03/07 11:25:42 cegger Exp $ 18 1.1 alc */ 19 1.1 alc #include "opt_ah.h" 20 1.1 alc 21 1.1 alc #include "ah.h" 22 1.1 alc #include "ah_internal.h" 23 1.1 alc 24 1.1 alc #include "ar5210/ar5210.h" 25 1.1 alc #include "ar5210/ar5210reg.h" 26 1.1 alc 27 1.1 alc /* 28 1.1 alc * Return non-zero if an interrupt is pending. 29 1.1 alc */ 30 1.1 alc HAL_BOOL 31 1.1 alc ar5210IsInterruptPending(struct ath_hal *ah) 32 1.1 alc { 33 1.1 alc return (OS_REG_READ(ah, AR_INTPEND) ? AH_TRUE : AH_FALSE); 34 1.1 alc } 35 1.1 alc 36 1.1 alc /* 37 1.1 alc * Read the Interrupt Status Register value and return 38 1.1 alc * an abstracted bitmask of the data found in the ISR. 39 1.1 alc * Note that reading the ISR clear pending interrupts. 40 1.1 alc */ 41 1.1 alc HAL_BOOL 42 1.1 alc ar5210GetPendingInterrupts(struct ath_hal *ah, HAL_INT *masked) 43 1.1 alc { 44 1.1 alc #define AR_FATAL_INT \ 45 1.1 alc (AR_ISR_MCABT_INT | AR_ISR_SSERR_INT | AR_ISR_DPERR_INT | AR_ISR_RXORN_INT) 46 1.1 alc struct ath_hal_5210 *ahp = AH5210(ah); 47 1.1 alc uint32_t isr; 48 1.1 alc 49 1.1 alc isr = OS_REG_READ(ah, AR_ISR); 50 1.1 alc if (isr == 0xffffffff) { 51 1.1 alc *masked = 0; 52 1.1 alc return AH_FALSE; 53 1.1 alc } 54 1.1 alc 55 1.1 alc /* 56 1.1 alc * Mask interrupts that have no device-independent 57 1.1 alc * representation; these are added back below. We 58 1.1 alc * also masked with the abstracted IMR to insure no 59 1.1 alc * status bits leak through that weren't requested 60 1.1 alc * (e.g. RXNOFRM) and that might confuse the caller. 61 1.1 alc */ 62 1.2 cegger *masked = (isr & (HAL_INT_COMMON - HAL_INT_BNR)) & ahp->ah_maskReg; 63 1.1 alc 64 1.1 alc if (isr & AR_FATAL_INT) 65 1.1 alc *masked |= HAL_INT_FATAL; 66 1.1 alc if (isr & (AR_ISR_RXOK_INT | AR_ISR_RXERR_INT)) 67 1.1 alc *masked |= HAL_INT_RX; 68 1.1 alc if (isr & (AR_ISR_TXOK_INT | AR_ISR_TXDESC_INT | AR_ISR_TXERR_INT | AR_ISR_TXEOL_INT)) 69 1.1 alc *masked |= HAL_INT_TX; 70 1.1 alc 71 1.1 alc /* 72 1.1 alc * On fatal errors collect ISR state for debugging. 73 1.1 alc */ 74 1.1 alc if (*masked & HAL_INT_FATAL) { 75 1.1 alc AH_PRIVATE(ah)->ah_fatalState[0] = isr; 76 1.1 alc } 77 1.1 alc 78 1.1 alc return AH_TRUE; 79 1.1 alc #undef AR_FATAL_INT 80 1.1 alc } 81 1.1 alc 82 1.1 alc HAL_INT 83 1.1 alc ar5210GetInterrupts(struct ath_hal *ah) 84 1.1 alc { 85 1.1 alc return AH5210(ah)->ah_maskReg; 86 1.1 alc } 87 1.1 alc 88 1.1 alc HAL_INT 89 1.1 alc ar5210SetInterrupts(struct ath_hal *ah, HAL_INT ints) 90 1.1 alc { 91 1.1 alc struct ath_hal_5210 *ahp = AH5210(ah); 92 1.1 alc uint32_t omask = ahp->ah_maskReg; 93 1.1 alc uint32_t mask; 94 1.1 alc 95 1.1 alc HALDEBUG(ah, HAL_DEBUG_INTERRUPT, "%s: 0x%x => 0x%x\n", 96 1.1 alc __func__, omask, ints); 97 1.1 alc 98 1.1 alc /* 99 1.1 alc * Disable interrupts here before reading & modifying 100 1.1 alc * the mask so that the ISR does not modify the mask 101 1.1 alc * out from under us. 102 1.1 alc */ 103 1.1 alc if (omask & HAL_INT_GLOBAL) { 104 1.1 alc HALDEBUG(ah, HAL_DEBUG_INTERRUPT, "%s: disable IER\n", __func__); 105 1.1 alc OS_REG_WRITE(ah, AR_IER, AR_IER_DISABLE); 106 1.1 alc } 107 1.1 alc 108 1.2 cegger mask = ints & (HAL_INT_COMMON - HAL_INT_BNR); 109 1.1 alc if (ints & HAL_INT_RX) 110 1.1 alc mask |= AR_IMR_RXOK_INT | AR_IMR_RXERR_INT; 111 1.1 alc if (ints & HAL_INT_TX) { 112 1.1 alc if (ahp->ah_txOkInterruptMask) 113 1.1 alc mask |= AR_IMR_TXOK_INT; 114 1.1 alc if (ahp->ah_txErrInterruptMask) 115 1.1 alc mask |= AR_IMR_TXERR_INT; 116 1.1 alc if (ahp->ah_txDescInterruptMask) 117 1.1 alc mask |= AR_IMR_TXDESC_INT; 118 1.1 alc if (ahp->ah_txEolInterruptMask) 119 1.1 alc mask |= AR_IMR_TXEOL_INT; 120 1.1 alc } 121 1.1 alc 122 1.1 alc /* Write the new IMR and store off our SW copy. */ 123 1.1 alc HALDEBUG(ah, HAL_DEBUG_INTERRUPT, "%s: new IMR 0x%x\n", __func__, mask); 124 1.1 alc OS_REG_WRITE(ah, AR_IMR, mask); 125 1.1 alc ahp->ah_maskReg = ints; 126 1.1 alc 127 1.1 alc /* Re-enable interrupts as appropriate. */ 128 1.1 alc if (ints & HAL_INT_GLOBAL) { 129 1.1 alc HALDEBUG(ah, HAL_DEBUG_INTERRUPT, "%s: enable IER\n", __func__); 130 1.1 alc OS_REG_WRITE(ah, AR_IER, AR_IER_ENABLE); 131 1.1 alc } 132 1.1 alc 133 1.1 alc return omask; 134 1.1 alc } 135