ath9k: do not access hardware on IRQs during reset
Instead of killing interrupts during reset when the first one happens, kill them before issuing the reset. This fixes an easy to reproduce crash with multiple cards sharing the same IRQ. Signed-off-by: Felix Fietkau <nbd@openwrt.org> Signed-off-by: John W. Linville <linville@tuxdriver.com>
This commit is contained in:
		
					parent
					
						
							
								ef739ab6aa
							
						
					
				
			
			
				commit
				
					
						872b5d814f
					
				
			
		
					 1 changed files with 9 additions and 7 deletions
				
			
		|  | @ -512,15 +512,12 @@ irqreturn_t ath_isr(int irq, void *dev) | ||||||
| 	if (!ah || test_bit(ATH_OP_INVALID, &common->op_flags)) | 	if (!ah || test_bit(ATH_OP_INVALID, &common->op_flags)) | ||||||
| 		return IRQ_NONE; | 		return IRQ_NONE; | ||||||
| 
 | 
 | ||||||
| 	/* shared irq, not for us */ | 	if (!AR_SREV_9100(ah) && test_bit(ATH_OP_HW_RESET, &common->op_flags)) | ||||||
| 
 |  | ||||||
| 	if (!ath9k_hw_intrpend(ah)) |  | ||||||
| 		return IRQ_NONE; | 		return IRQ_NONE; | ||||||
| 
 | 
 | ||||||
| 	if (test_bit(ATH_OP_HW_RESET, &common->op_flags)) { | 	/* shared irq, not for us */ | ||||||
| 		ath9k_hw_kill_interrupts(ah); | 	if (!ath9k_hw_intrpend(ah)) | ||||||
| 		return IRQ_HANDLED; | 		return IRQ_NONE; | ||||||
| 	} |  | ||||||
| 
 | 
 | ||||||
| 	/*
 | 	/*
 | ||||||
| 	 * Figure out the reason(s) for the interrupt.  Note | 	 * Figure out the reason(s) for the interrupt.  Note | ||||||
|  | @ -532,6 +529,9 @@ irqreturn_t ath_isr(int irq, void *dev) | ||||||
| 	ath9k_debug_sync_cause(sc, sync_cause); | 	ath9k_debug_sync_cause(sc, sync_cause); | ||||||
| 	status &= ah->imask;	/* discard unasked-for bits */ | 	status &= ah->imask;	/* discard unasked-for bits */ | ||||||
| 
 | 
 | ||||||
|  | 	if (AR_SREV_9100(ah) && test_bit(ATH_OP_HW_RESET, &common->op_flags)) | ||||||
|  | 		return IRQ_HANDLED; | ||||||
|  | 
 | ||||||
| 	/*
 | 	/*
 | ||||||
| 	 * If there are no status bits set, then this interrupt was not | 	 * If there are no status bits set, then this interrupt was not | ||||||
| 	 * for me (should have been caught above). | 	 * for me (should have been caught above). | ||||||
|  | @ -613,6 +613,7 @@ int ath_reset(struct ath_softc *sc, struct ath9k_channel *hchan) | ||||||
| 	struct ath_common *common = ath9k_hw_common(sc->sc_ah); | 	struct ath_common *common = ath9k_hw_common(sc->sc_ah); | ||||||
| 	int r; | 	int r; | ||||||
| 
 | 
 | ||||||
|  | 	ath9k_hw_kill_interrupts(sc->sc_ah); | ||||||
| 	set_bit(ATH_OP_HW_RESET, &common->op_flags); | 	set_bit(ATH_OP_HW_RESET, &common->op_flags); | ||||||
| 
 | 
 | ||||||
| 	ath9k_ps_wakeup(sc); | 	ath9k_ps_wakeup(sc); | ||||||
|  | @ -633,6 +634,7 @@ void ath9k_queue_reset(struct ath_softc *sc, enum ath_reset_type type) | ||||||
| #ifdef CONFIG_ATH9K_DEBUGFS | #ifdef CONFIG_ATH9K_DEBUGFS | ||||||
| 	RESET_STAT_INC(sc, type); | 	RESET_STAT_INC(sc, type); | ||||||
| #endif | #endif | ||||||
|  | 	ath9k_hw_kill_interrupts(sc->sc_ah); | ||||||
| 	set_bit(ATH_OP_HW_RESET, &common->op_flags); | 	set_bit(ATH_OP_HW_RESET, &common->op_flags); | ||||||
| 	ieee80211_queue_work(sc->hw, &sc->hw_reset_work); | 	ieee80211_queue_work(sc->hw, &sc->hw_reset_work); | ||||||
| } | } | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 Felix Fietkau
				Felix Fietkau