ath9k: Use atomic reference count for interrupt ops
Let us enable/disable interrupts based on reference count. By doing this we can ensure that interrupts are never be enabled in the middle of tasklet processing. Instead of addressing corner cases like "ath9k: avoid enabling interrupts while processing rx", this approach handles it in generic manner. Signed-off-by: Rajkumar Manoharan <rmanohar@qca.qualcomm.com> Signed-off-by: John W. Linville <linville@tuxdriver.com>
This commit is contained in:
parent
a844adfd7b
commit
e8fe733684
@ -567,7 +567,6 @@ struct ath_ant_comb {
|
|||||||
#define PS_WAIT_FOR_PSPOLL_DATA BIT(2)
|
#define PS_WAIT_FOR_PSPOLL_DATA BIT(2)
|
||||||
#define PS_WAIT_FOR_TX_ACK BIT(3)
|
#define PS_WAIT_FOR_TX_ACK BIT(3)
|
||||||
#define PS_BEACON_SYNC BIT(4)
|
#define PS_BEACON_SYNC BIT(4)
|
||||||
#define PS_TSFOOR_SYNC BIT(5)
|
|
||||||
|
|
||||||
struct ath_rate_table;
|
struct ath_rate_table;
|
||||||
|
|
||||||
|
@ -649,14 +649,8 @@ static void ath_beacon_config_sta(struct ath_softc *sc,
|
|||||||
ath9k_hw_set_sta_beacon_timers(ah, &bs);
|
ath9k_hw_set_sta_beacon_timers(ah, &bs);
|
||||||
ah->imask |= ATH9K_INT_BMISS;
|
ah->imask |= ATH9K_INT_BMISS;
|
||||||
|
|
||||||
/*
|
ath9k_hw_set_interrupts(ah, ah->imask);
|
||||||
* If the beacon config is called beacause of TSFOOR,
|
ath9k_hw_enable_interrupts(ah);
|
||||||
* Interrupts will be enabled back at the end of ath9k_tasklet
|
|
||||||
*/
|
|
||||||
if (!(sc->ps_flags & PS_TSFOOR_SYNC)) {
|
|
||||||
ath9k_hw_set_interrupts(ah, ah->imask);
|
|
||||||
ath9k_hw_enable_interrupts(ah);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void ath_beacon_config_adhoc(struct ath_softc *sc,
|
static void ath_beacon_config_adhoc(struct ath_softc *sc,
|
||||||
@ -690,14 +684,9 @@ static void ath_beacon_config_adhoc(struct ath_softc *sc,
|
|||||||
ath9k_hw_disable_interrupts(ah);
|
ath9k_hw_disable_interrupts(ah);
|
||||||
ath9k_beacon_init(sc, nexttbtt, intval);
|
ath9k_beacon_init(sc, nexttbtt, intval);
|
||||||
sc->beacon.bmisscnt = 0;
|
sc->beacon.bmisscnt = 0;
|
||||||
/*
|
|
||||||
* If the beacon config is called beacause of TSFOOR,
|
ath9k_hw_set_interrupts(ah, ah->imask);
|
||||||
* Interrupts will be enabled back at the end of ath9k_tasklet
|
ath9k_hw_enable_interrupts(ah);
|
||||||
*/
|
|
||||||
if (!(sc->ps_flags & PS_TSFOOR_SYNC)) {
|
|
||||||
ath9k_hw_set_interrupts(ah, ah->imask);
|
|
||||||
ath9k_hw_enable_interrupts(ah);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool ath9k_allow_beacon_config(struct ath_softc *sc,
|
static bool ath9k_allow_beacon_config(struct ath_softc *sc,
|
||||||
|
@ -709,6 +709,7 @@ struct ath_hw {
|
|||||||
u32 txdesc_interrupt_mask;
|
u32 txdesc_interrupt_mask;
|
||||||
u32 txeol_interrupt_mask;
|
u32 txeol_interrupt_mask;
|
||||||
u32 txurn_interrupt_mask;
|
u32 txurn_interrupt_mask;
|
||||||
|
atomic_t intr_ref_cnt;
|
||||||
bool chip_fullsleep;
|
bool chip_fullsleep;
|
||||||
u32 atim_window;
|
u32 atim_window;
|
||||||
|
|
||||||
|
@ -566,6 +566,7 @@ static int ath9k_init_softc(u16 devid, struct ath_softc *sc,
|
|||||||
ah->reg_ops.read = ath9k_ioread32;
|
ah->reg_ops.read = ath9k_ioread32;
|
||||||
ah->reg_ops.write = ath9k_iowrite32;
|
ah->reg_ops.write = ath9k_iowrite32;
|
||||||
ah->reg_ops.rmw = ath9k_reg_rmw;
|
ah->reg_ops.rmw = ath9k_reg_rmw;
|
||||||
|
atomic_set(&ah->intr_ref_cnt, -1);
|
||||||
sc->sc_ah = ah;
|
sc->sc_ah = ah;
|
||||||
|
|
||||||
if (!pdata) {
|
if (!pdata) {
|
||||||
|
@ -800,6 +800,11 @@ void ath9k_hw_disable_interrupts(struct ath_hw *ah)
|
|||||||
{
|
{
|
||||||
struct ath_common *common = ath9k_hw_common(ah);
|
struct ath_common *common = ath9k_hw_common(ah);
|
||||||
|
|
||||||
|
if (!(ah->imask & ATH9K_INT_GLOBAL))
|
||||||
|
atomic_set(&ah->intr_ref_cnt, -1);
|
||||||
|
else
|
||||||
|
atomic_dec(&ah->intr_ref_cnt);
|
||||||
|
|
||||||
ath_dbg(common, ATH_DBG_INTERRUPT, "disable IER\n");
|
ath_dbg(common, ATH_DBG_INTERRUPT, "disable IER\n");
|
||||||
REG_WRITE(ah, AR_IER, AR_IER_DISABLE);
|
REG_WRITE(ah, AR_IER, AR_IER_DISABLE);
|
||||||
(void) REG_READ(ah, AR_IER);
|
(void) REG_READ(ah, AR_IER);
|
||||||
@ -821,6 +826,13 @@ void ath9k_hw_enable_interrupts(struct ath_hw *ah)
|
|||||||
if (!(ah->imask & ATH9K_INT_GLOBAL))
|
if (!(ah->imask & ATH9K_INT_GLOBAL))
|
||||||
return;
|
return;
|
||||||
|
|
||||||
|
if (!atomic_inc_and_test(&ah->intr_ref_cnt)) {
|
||||||
|
ath_dbg(common, ATH_DBG_INTERRUPT,
|
||||||
|
"Do not enable IER ref count %d\n",
|
||||||
|
atomic_read(&ah->intr_ref_cnt));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
if (AR_SREV_9340(ah))
|
if (AR_SREV_9340(ah))
|
||||||
sync_default &= ~AR_INTR_SYNC_HOST1_FATAL;
|
sync_default &= ~AR_INTR_SYNC_HOST1_FATAL;
|
||||||
|
|
||||||
@ -852,7 +864,6 @@ void ath9k_hw_set_interrupts(struct ath_hw *ah, enum ath9k_int ints)
|
|||||||
|
|
||||||
ath_dbg(common, ATH_DBG_INTERRUPT, "0x%x => 0x%x\n", omask, ints);
|
ath_dbg(common, ATH_DBG_INTERRUPT, "0x%x => 0x%x\n", omask, ints);
|
||||||
|
|
||||||
/* TODO: global int Ref count */
|
|
||||||
mask = ints & ATH9K_INT_COMMON;
|
mask = ints & ATH9K_INT_COMMON;
|
||||||
mask2 = 0;
|
mask2 = 0;
|
||||||
|
|
||||||
|
@ -707,8 +707,7 @@ void ath9k_tasklet(unsigned long data)
|
|||||||
*/
|
*/
|
||||||
ath_dbg(common, ATH_DBG_PS,
|
ath_dbg(common, ATH_DBG_PS,
|
||||||
"TSFOOR - Sync with next Beacon\n");
|
"TSFOOR - Sync with next Beacon\n");
|
||||||
sc->ps_flags |= PS_WAIT_FOR_BEACON | PS_BEACON_SYNC |
|
sc->ps_flags |= PS_WAIT_FOR_BEACON | PS_BEACON_SYNC;
|
||||||
PS_TSFOOR_SYNC;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (ah->caps.hw_caps & ATH9K_HW_CAP_EDMA)
|
if (ah->caps.hw_caps & ATH9K_HW_CAP_EDMA)
|
||||||
@ -887,6 +886,7 @@ static void ath_radio_enable(struct ath_softc *sc, struct ieee80211_hw *hw)
|
|||||||
|
|
||||||
ath9k_ps_wakeup(sc);
|
ath9k_ps_wakeup(sc);
|
||||||
spin_lock_bh(&sc->sc_pcu_lock);
|
spin_lock_bh(&sc->sc_pcu_lock);
|
||||||
|
atomic_set(&ah->intr_ref_cnt, -1);
|
||||||
|
|
||||||
ath9k_hw_configpcipowersave(ah, 0, 0);
|
ath9k_hw_configpcipowersave(ah, 0, 0);
|
||||||
|
|
||||||
|
@ -601,7 +601,6 @@ static void ath_rx_ps_beacon(struct ath_softc *sc, struct sk_buff *skb)
|
|||||||
ath_dbg(common, ATH_DBG_PS,
|
ath_dbg(common, ATH_DBG_PS,
|
||||||
"Reconfigure Beacon timers based on timestamp from the AP\n");
|
"Reconfigure Beacon timers based on timestamp from the AP\n");
|
||||||
ath_set_beacon(sc);
|
ath_set_beacon(sc);
|
||||||
sc->ps_flags &= ~PS_TSFOOR_SYNC;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (ath_beacon_dtim_pending_cab(skb)) {
|
if (ath_beacon_dtim_pending_cab(skb)) {
|
||||||
|
Loading…
x
Reference in New Issue
Block a user