diff options
Diffstat (limited to 'drivers/net/wireless/ath/ath9k/main.c')
| -rw-r--r-- | drivers/net/wireless/ath/ath9k/main.c | 37 | 
1 files changed, 21 insertions, 16 deletions
diff --git a/drivers/net/wireless/ath/ath9k/main.c b/drivers/net/wireless/ath/ath9k/main.c index f90a6ca94a7..a09d15f7aa6 100644 --- a/drivers/net/wireless/ath/ath9k/main.c +++ b/drivers/net/wireless/ath/ath9k/main.c @@ -325,6 +325,8 @@ static bool ath_paprd_send_frame(struct ath_softc *sc, struct sk_buff *skb, int  {  	struct ieee80211_hw *hw = sc->hw;  	struct ieee80211_tx_info *tx_info = IEEE80211_SKB_CB(skb); +	struct ath_hw *ah = sc->sc_ah; +	struct ath_common *common = ath9k_hw_common(ah);  	struct ath_tx_control txctl;  	int time_left; @@ -340,14 +342,16 @@ static bool ath_paprd_send_frame(struct ath_softc *sc, struct sk_buff *skb, int  	tx_info->control.rates[1].idx = -1;  	init_completion(&sc->paprd_complete); -	sc->paprd_pending = true;  	txctl.paprd = BIT(chain); -	if (ath_tx_start(hw, skb, &txctl) != 0) + +	if (ath_tx_start(hw, skb, &txctl) != 0) { +		ath_dbg(common, ATH_DBG_XMIT, "PAPRD TX failed\n"); +		dev_kfree_skb_any(skb);  		return false; +	}  	time_left = wait_for_completion_timeout(&sc->paprd_complete,  			msecs_to_jiffies(ATH_PAPRD_TIMEOUT)); -	sc->paprd_pending = false;  	if (!time_left)  		ath_dbg(ath9k_hw_common(sc->sc_ah), ATH_DBG_CALIBRATE, @@ -592,14 +596,12 @@ void ath9k_tasklet(unsigned long data)  	u32 status = sc->intrstatus;  	u32 rxmask; -	ath9k_ps_wakeup(sc); -  	if (status & ATH9K_INT_FATAL) {  		ath_reset(sc, true); -		ath9k_ps_restore(sc);  		return;  	} +	ath9k_ps_wakeup(sc);  	spin_lock(&sc->sc_pcu_lock);  	if (!ath9k_hw_check_alive(ah)) @@ -955,8 +957,6 @@ void ath_radio_disable(struct ath_softc *sc, struct ieee80211_hw *hw)  	spin_unlock_bh(&sc->sc_pcu_lock);  	ath9k_ps_restore(sc); - -	ath9k_setpower(sc, ATH9K_PM_FULL_SLEEP);  }  int ath_reset(struct ath_softc *sc, bool retry_tx) @@ -969,6 +969,7 @@ int ath_reset(struct ath_softc *sc, bool retry_tx)  	/* Stop ANI */  	del_timer_sync(&common->ani.timer); +	ath9k_ps_wakeup(sc);  	spin_lock_bh(&sc->sc_pcu_lock);  	ieee80211_stop_queues(hw); @@ -1015,6 +1016,7 @@ int ath_reset(struct ath_softc *sc, bool retry_tx)  	/* Start ANI */  	ath_start_ani(common); +	ath9k_ps_restore(sc);  	return r;  } @@ -1171,12 +1173,6 @@ static int ath9k_start(struct ieee80211_hw *hw)  			ath9k_btcoex_timer_resume(sc);  	} -	/* User has the option to provide pm-qos value as a module -	 * parameter rather than using the default value of -	 * 'ATH9K_PM_QOS_DEFAULT_VALUE'. -	 */ -	pm_qos_update_request(&sc->pm_qos_req, ath9k_pm_qos_value); -  	if (ah->caps.pcie_lcr_extsync_en && common->bus_ops->extn_synch_en)  		common->bus_ops->extn_synch_en(common); @@ -1309,6 +1305,9 @@ static void ath9k_stop(struct ieee80211_hw *hw)  	spin_lock_bh(&sc->sc_pcu_lock); +	/* prevent tasklets to enable interrupts once we disable them */ +	ah->imask &= ~ATH9K_INT_GLOBAL; +  	/* make sure h/w will not generate any interrupt  	 * before setting the invalid flag. */  	ath9k_hw_disable_interrupts(ah); @@ -1326,6 +1325,12 @@ static void ath9k_stop(struct ieee80211_hw *hw)  	spin_unlock_bh(&sc->sc_pcu_lock); +	/* we can now sync irq and kill any running tasklets, since we already +	 * disabled interrupts and not holding a spin lock */ +	synchronize_irq(sc->irq); +	tasklet_kill(&sc->intr_tq); +	tasklet_kill(&sc->bcon_tasklet); +  	ath9k_ps_restore(sc);  	sc->ps_idle = true; @@ -1334,8 +1339,6 @@ static void ath9k_stop(struct ieee80211_hw *hw)  	sc->sc_flags |= SC_OP_INVALID; -	pm_qos_update_request(&sc->pm_qos_req, PM_QOS_DEFAULT_VALUE); -  	mutex_unlock(&sc->mutex);  	ath_dbg(common, ATH_DBG_CONFIG, "Driver halt\n"); @@ -1701,7 +1704,9 @@ static int ath9k_config(struct ieee80211_hw *hw, u32 changed)  skip_chan_change:  	if (changed & IEEE80211_CONF_CHANGE_POWER) {  		sc->config.txpowlimit = 2 * conf->power_level; +		ath9k_ps_wakeup(sc);  		ath_update_txpow(sc); +		ath9k_ps_restore(sc);  	}  	spin_lock_bh(&sc->wiphy_lock);  |