Make sure that any open relayfs files are closed before
unregistering with mac80211, otherwise this crash is seen:
[ 1331.097846] BUG: unable to handle kernel paging request at 6b6b6b8b
[ 1331.098170] IP: [<c063d0d6>] debugfs_remove+0x26/0x80
[ 1331.098170] *pdpt = 000000002f9aa001 *pde = 0000000000000000
[ 1331.098170] Oops: 0000 [#1] PREEMPT SMP
[ 1331.098170] Modules linked in: iptable_raw xt_CT nf_conntrack_ipv4 nf_defrag]
[ 1331.098170] Pid: 4794, comm: rmmod Tainted: G        WC   3.9.1+ #5 To Be Fi.
[ 1331.098170] EIP: 0060:[<c063d0d6>] EFLAGS: 00010202 CPU: 0
[ 1331.098170] EIP is at debugfs_remove+0x26/0x80
[ 1331.098170] EAX: f2f3acd0 EBX: f2f3acd0 ECX: 00000006 EDX: f8622348
[ 1331.098170] ESI: 6b6b6b6b EDI: 00000001 EBP: ee251e14 ESP: ee251e0c
[ 1331.098170]  DS: 007b ES: 007b FS: 00d8 GS: 00e0 SS: 0068
[ 1331.098170] CR0: 8005003b CR2: 6b6b6b8b CR3: 2e7b7000 CR4: 000007e0
[ 1331.098170] DR0: 00000000 DR1: 00000000 DR2: 00000000 DR3: 00000000
[ 1331.098170] DR6: ffff0ff0 DR7: 00000400
[ 1331.098170] Process rmmod (pid: 4794, ti=ee250000 task=efaa2560 task.ti=ee25)
[ 1331.098170] Stack:
[ 1331.098170]  f241e170 0000000a ee251e1c f861394d ee251e28 c04e3088 f241e170 4
[ 1331.098170]  c04e30fe f45482b0 ee251e54 c04e3187 f25e86b0 ee251e54 f8618748 0
[ 1331.098170]  0000000a 00000001 ee251e68 f860065b f2509e20 f25085a0 f5b6e8a4 8
[ 1331.098170] Call Trace:
[ 1331.098170]  [<f861394d>] remove_buf_file_handler+0xd/0x20 [ath9k]
[ 1331.098170]  [<c04e3088>] relay_remove_buf+0x18/0x30
[ 1331.098170]  [<c04e30fe>] relay_close_buf+0x2e/0x40
[ 1331.098170]  [<c04e3187>] relay_close+0x77/0xf0
[ 1331.098170]  [<f8618748>] ? dpd_exit+0x38/0x40 [ath9k]
[ 1331.098170]  [<f860065b>] ath9k_deinit_softc+0x8b/0xa0 [ath9k]
[ 1331.098170]  [<f86006b8>] ath9k_deinit_device+0x48/0x60 [ath9k]
[ 1331.098170]  [<f86107f1>] ath_pci_remove+0x31/0x50 [ath9k]
[ 1331.098170]  [<c06dbff8>] pci_device_remove+0x38/0xc0
[ 1331.098170]  [<c079daa4>] __device_release_driver+0x64/0xc0
[ 1331.098170]  [<c079db97>] driver_detach+0x97/0xa0
[ 1331.098170]  [<c079cacc>] bus_remove_driver+0x6c/0xe0
[ 1331.098170]  [<c079c197>] ? bus_put+0x17/0x20
[ 1331.098170]  [<c079cae3>] ? bus_remove_driver+0x83/0xe0
[ 1331.098170]  [<c079e709>] driver_unregister+0x49/0x80
[ 1331.098170]  [<c06dc138>] pci_unregister_driver+0x18/0x80
[ 1331.098170]  [<f8610602>] ath_pci_exit+0x12/0x20 [ath9k]
[ 1331.098170]  [<f8619ce0>] ath9k_exit+0x17/0x337 [ath9k]
[ 1331.098170]  [<c09e537d>] ? mutex_unlock+0xd/0x10
[ 1331.098170]  [<c04bd36c>] sys_delete_module+0x17c/0x250
[ 1331.098170]  [<c0540dc4>] ? do_munmap+0x244/0x2d0
[ 1331.098170]  [<c0540e96>] ? vm_munmap+0x46/0x60
[ 1331.098170]  [<c09e8dc4>] ? restore_all+0xf/0xf
[ 1331.098170]  [<c09ebf50>] ? __do_page_fault+0x4c0/0x4c0
[ 1331.098170]  [<c04b18e4>] ? trace_hardirqs_on_caller+0xf4/0x180
[ 1331.098170]  [<c09ef28d>] sysenter_do_call+0x12/0x38
[ 1331.098170] Code: 90 8d 74 26 00 55 89 e5 83 ec 08 89 1c 24 89 74 24 04 3e 82
[ 1331.098170] EIP: [<c063d0d6>] debugfs_remove+0x26/0x80 SS:ESP 0068:ee251e0c
[ 1331.098170] CR2: 000000006b6b6b8b
[ 1331.727971] ---[ end trace b5bb9f2066cef7f9 ]---
Cc: <stable@vger.kernel.org>
Acked-by: Simon Wunderlich <siwu@hrz.tu-chemnitz.de>
Tested-by: Ben Greear <greearb@candelatech.com>
Signed-off-by: Sujith Manoharan <c_manoha@qca.qualcomm.com>
Signed-off-by: John W. Linville <linville@tuxdriver.com>
		
	
			
		
			
				
	
	
		
			381 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			381 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * Copyright (c) 2008-2011 Atheros Communications Inc.
 | |
|  *
 | |
|  * Permission to use, copy, modify, and/or distribute this software for any
 | |
|  * purpose with or without fee is hereby granted, provided that the above
 | |
|  * copyright notice and this permission notice appear in all copies.
 | |
|  *
 | |
|  * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
 | |
|  * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
 | |
|  * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
 | |
|  * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
 | |
|  * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
 | |
|  * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
 | |
|  * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
 | |
|  */
 | |
| 
 | |
| #ifndef DEBUG_H
 | |
| #define DEBUG_H
 | |
| 
 | |
| #include "hw.h"
 | |
| #include "rc.h"
 | |
| #include "dfs_debug.h"
 | |
| 
 | |
| struct ath_txq;
 | |
| struct ath_buf;
 | |
| struct fft_sample_tlv;
 | |
| 
 | |
| #ifdef CONFIG_ATH9K_DEBUGFS
 | |
| #define TX_STAT_INC(q, c) sc->debug.stats.txstats[q].c++
 | |
| #define RESET_STAT_INC(sc, type) sc->debug.stats.reset[type]++
 | |
| #else
 | |
| #define TX_STAT_INC(q, c) do { } while (0)
 | |
| #define RESET_STAT_INC(sc, type) do { } while (0)
 | |
| #endif
 | |
| 
 | |
| enum ath_reset_type {
 | |
| 	RESET_TYPE_BB_HANG,
 | |
| 	RESET_TYPE_BB_WATCHDOG,
 | |
| 	RESET_TYPE_FATAL_INT,
 | |
| 	RESET_TYPE_TX_ERROR,
 | |
| 	RESET_TYPE_TX_HANG,
 | |
| 	RESET_TYPE_PLL_HANG,
 | |
| 	RESET_TYPE_MAC_HANG,
 | |
| 	RESET_TYPE_BEACON_STUCK,
 | |
| 	RESET_TYPE_MCI,
 | |
| 	__RESET_TYPE_MAX
 | |
| };
 | |
| 
 | |
| #ifdef CONFIG_ATH9K_DEBUGFS
 | |
| 
 | |
| /**
 | |
|  * struct ath_interrupt_stats - Contains statistics about interrupts
 | |
|  * @total: Total no. of interrupts generated so far
 | |
|  * @rxok: RX with no errors
 | |
|  * @rxlp: RX with low priority RX
 | |
|  * @rxhp: RX with high priority, uapsd only
 | |
|  * @rxeol: RX with no more RXDESC available
 | |
|  * @rxorn: RX FIFO overrun
 | |
|  * @txok: TX completed at the requested rate
 | |
|  * @txurn: TX FIFO underrun
 | |
|  * @mib: MIB regs reaching its threshold
 | |
|  * @rxphyerr: RX with phy errors
 | |
|  * @rx_keycache_miss: RX with key cache misses
 | |
|  * @swba: Software Beacon Alert
 | |
|  * @bmiss: Beacon Miss
 | |
|  * @bnr: Beacon Not Ready
 | |
|  * @cst: Carrier Sense TImeout
 | |
|  * @gtt: Global TX Timeout
 | |
|  * @tim: RX beacon TIM occurrence
 | |
|  * @cabend: RX End of CAB traffic
 | |
|  * @dtimsync: DTIM sync lossage
 | |
|  * @dtim: RX Beacon with DTIM
 | |
|  * @bb_watchdog: Baseband watchdog
 | |
|  * @tsfoor: TSF out of range, indicates that the corrected TSF received
 | |
|  * from a beacon differs from the PCU's internal TSF by more than a
 | |
|  * (programmable) threshold
 | |
|  * @local_timeout: Internal bus timeout.
 | |
|  * @mci: MCI interrupt, specific to MCI based BTCOEX chipsets
 | |
|  * @gen_timer: Generic hardware timer interrupt
 | |
|  */
 | |
| struct ath_interrupt_stats {
 | |
| 	u32 total;
 | |
| 	u32 rxok;
 | |
| 	u32 rxlp;
 | |
| 	u32 rxhp;
 | |
| 	u32 rxeol;
 | |
| 	u32 rxorn;
 | |
| 	u32 txok;
 | |
| 	u32 txeol;
 | |
| 	u32 txurn;
 | |
| 	u32 mib;
 | |
| 	u32 rxphyerr;
 | |
| 	u32 rx_keycache_miss;
 | |
| 	u32 swba;
 | |
| 	u32 bmiss;
 | |
| 	u32 bnr;
 | |
| 	u32 cst;
 | |
| 	u32 gtt;
 | |
| 	u32 tim;
 | |
| 	u32 cabend;
 | |
| 	u32 dtimsync;
 | |
| 	u32 dtim;
 | |
| 	u32 bb_watchdog;
 | |
| 	u32 tsfoor;
 | |
| 	u32 mci;
 | |
| 	u32 gen_timer;
 | |
| 
 | |
| 	/* Sync-cause stats */
 | |
| 	u32 sync_cause_all;
 | |
| 	u32 sync_rtc_irq;
 | |
| 	u32 sync_mac_irq;
 | |
| 	u32 eeprom_illegal_access;
 | |
| 	u32 apb_timeout;
 | |
| 	u32 pci_mode_conflict;
 | |
| 	u32 host1_fatal;
 | |
| 	u32 host1_perr;
 | |
| 	u32 trcv_fifo_perr;
 | |
| 	u32 radm_cpl_ep;
 | |
| 	u32 radm_cpl_dllp_abort;
 | |
| 	u32 radm_cpl_tlp_abort;
 | |
| 	u32 radm_cpl_ecrc_err;
 | |
| 	u32 radm_cpl_timeout;
 | |
| 	u32 local_timeout;
 | |
| 	u32 pm_access;
 | |
| 	u32 mac_awake;
 | |
| 	u32 mac_asleep;
 | |
| 	u32 mac_sleep_access;
 | |
| };
 | |
| 
 | |
| 
 | |
| /**
 | |
|  * struct ath_tx_stats - Statistics about TX
 | |
|  * @tx_pkts_all:  No. of total frames transmitted, including ones that
 | |
| 	may have had errors.
 | |
|  * @tx_bytes_all:  No. of total bytes transmitted, including ones that
 | |
| 	may have had errors.
 | |
|  * @queued: Total MPDUs (non-aggr) queued
 | |
|  * @completed: Total MPDUs (non-aggr) completed
 | |
|  * @a_aggr: Total no. of aggregates queued
 | |
|  * @a_queued_hw: Total AMPDUs queued to hardware
 | |
|  * @a_queued_sw: Total AMPDUs queued to software queues
 | |
|  * @a_completed: Total AMPDUs completed
 | |
|  * @a_retries: No. of AMPDUs retried (SW)
 | |
|  * @a_xretries: No. of AMPDUs dropped due to xretries
 | |
|  * @txerr_filtered: No. of frames with TXERR_FILT flag set.
 | |
|  * @fifo_underrun: FIFO underrun occurrences
 | |
| 	Valid only for:
 | |
| 		- non-aggregate condition.
 | |
| 		- first packet of aggregate.
 | |
|  * @xtxop: No. of frames filtered because of TXOP limit
 | |
|  * @timer_exp: Transmit timer expiry
 | |
|  * @desc_cfg_err: Descriptor configuration errors
 | |
|  * @data_urn: TX data underrun errors
 | |
|  * @delim_urn: TX delimiter underrun errors
 | |
|  * @puttxbuf: Number of times hardware was given txbuf to write.
 | |
|  * @txstart:  Number of times hardware was told to start tx.
 | |
|  * @txprocdesc:  Number of times tx descriptor was processed
 | |
|  * @txfailed:  Out-of-memory or other errors in xmit path.
 | |
|  */
 | |
| struct ath_tx_stats {
 | |
| 	u32 tx_pkts_all;
 | |
| 	u32 tx_bytes_all;
 | |
| 	u32 queued;
 | |
| 	u32 completed;
 | |
| 	u32 xretries;
 | |
| 	u32 a_aggr;
 | |
| 	u32 a_queued_hw;
 | |
| 	u32 a_queued_sw;
 | |
| 	u32 a_completed;
 | |
| 	u32 a_retries;
 | |
| 	u32 a_xretries;
 | |
| 	u32 txerr_filtered;
 | |
| 	u32 fifo_underrun;
 | |
| 	u32 xtxop;
 | |
| 	u32 timer_exp;
 | |
| 	u32 desc_cfg_err;
 | |
| 	u32 data_underrun;
 | |
| 	u32 delim_underrun;
 | |
| 	u32 puttxbuf;
 | |
| 	u32 txstart;
 | |
| 	u32 txprocdesc;
 | |
| 	u32 txfailed;
 | |
| };
 | |
| 
 | |
| /*
 | |
|  * Various utility macros to print TX/Queue counters.
 | |
|  */
 | |
| #define PR_QNUM(_n) sc->tx.txq_map[_n]->axq_qnum
 | |
| #define TXSTATS sc->debug.stats.txstats
 | |
| #define PR(str, elem)							\
 | |
| 	do {								\
 | |
| 		len += snprintf(buf + len, size - len,			\
 | |
| 				"%s%13u%11u%10u%10u\n", str,		\
 | |
| 				TXSTATS[PR_QNUM(IEEE80211_AC_BE)].elem,	\
 | |
| 				TXSTATS[PR_QNUM(IEEE80211_AC_BK)].elem,	\
 | |
| 				TXSTATS[PR_QNUM(IEEE80211_AC_VI)].elem,	\
 | |
| 				TXSTATS[PR_QNUM(IEEE80211_AC_VO)].elem); \
 | |
| 	} while(0)
 | |
| 
 | |
| #define RX_STAT_INC(c) (sc->debug.stats.rxstats.c++)
 | |
| 
 | |
| /**
 | |
|  * struct ath_rx_stats - RX Statistics
 | |
|  * @rx_pkts_all:  No. of total frames received, including ones that
 | |
| 	may have had errors.
 | |
|  * @rx_bytes_all:  No. of total bytes received, including ones that
 | |
| 	may have had errors.
 | |
|  * @crc_err: No. of frames with incorrect CRC value
 | |
|  * @decrypt_crc_err: No. of frames whose CRC check failed after
 | |
| 	decryption process completed
 | |
|  * @phy_err: No. of frames whose reception failed because the PHY
 | |
| 	encountered an error
 | |
|  * @mic_err: No. of frames with incorrect TKIP MIC verification failure
 | |
|  * @pre_delim_crc_err: Pre-Frame delimiter CRC error detections
 | |
|  * @post_delim_crc_err: Post-Frame delimiter CRC error detections
 | |
|  * @decrypt_busy_err: Decryption interruptions counter
 | |
|  * @phy_err_stats: Individual PHY error statistics
 | |
|  * @rx_len_err:  No. of frames discarded due to bad length.
 | |
|  * @rx_oom_err:  No. of frames dropped due to OOM issues.
 | |
|  * @rx_rate_err:  No. of frames dropped due to rate errors.
 | |
|  * @rx_too_many_frags_err:  Frames dropped due to too-many-frags received.
 | |
|  * @rx_beacons:  No. of beacons received.
 | |
|  * @rx_frags:  No. of rx-fragements received.
 | |
|  * @rx_spectral: No of spectral packets received.
 | |
|  */
 | |
| struct ath_rx_stats {
 | |
| 	u32 rx_pkts_all;
 | |
| 	u32 rx_bytes_all;
 | |
| 	u32 crc_err;
 | |
| 	u32 decrypt_crc_err;
 | |
| 	u32 phy_err;
 | |
| 	u32 mic_err;
 | |
| 	u32 pre_delim_crc_err;
 | |
| 	u32 post_delim_crc_err;
 | |
| 	u32 decrypt_busy_err;
 | |
| 	u32 phy_err_stats[ATH9K_PHYERR_MAX];
 | |
| 	u32 rx_len_err;
 | |
| 	u32 rx_oom_err;
 | |
| 	u32 rx_rate_err;
 | |
| 	u32 rx_too_many_frags_err;
 | |
| 	u32 rx_beacons;
 | |
| 	u32 rx_frags;
 | |
| 	u32 rx_spectral;
 | |
| };
 | |
| 
 | |
| struct ath_stats {
 | |
| 	struct ath_interrupt_stats istats;
 | |
| 	struct ath_tx_stats txstats[ATH9K_NUM_TX_QUEUES];
 | |
| 	struct ath_rx_stats rxstats;
 | |
| 	struct ath_dfs_stats dfs_stats;
 | |
| 	u32 reset[__RESET_TYPE_MAX];
 | |
| };
 | |
| 
 | |
| #define ATH_DBG_MAX_SAMPLES	10
 | |
| struct ath_dbg_bb_mac_samp {
 | |
| 	u32 dma_dbg_reg_vals[ATH9K_NUM_DMA_DEBUG_REGS];
 | |
| 	u32 pcu_obs, pcu_cr, noise;
 | |
| 	struct {
 | |
| 		u64 jiffies;
 | |
| 		int8_t rssi_ctl0;
 | |
| 		int8_t rssi_ctl1;
 | |
| 		int8_t rssi_ctl2;
 | |
| 		int8_t rssi_ext0;
 | |
| 		int8_t rssi_ext1;
 | |
| 		int8_t rssi_ext2;
 | |
| 		int8_t rssi;
 | |
| 		bool isok;
 | |
| 		u8 rts_fail_cnt;
 | |
| 		u8 data_fail_cnt;
 | |
| 		u8 rateindex;
 | |
| 		u8 qid;
 | |
| 		u8 tid;
 | |
| 		u32 ba_low;
 | |
| 		u32 ba_high;
 | |
| 	} ts[ATH_DBG_MAX_SAMPLES];
 | |
| 	struct {
 | |
| 		u64 jiffies;
 | |
| 		int8_t rssi_ctl0;
 | |
| 		int8_t rssi_ctl1;
 | |
| 		int8_t rssi_ctl2;
 | |
| 		int8_t rssi_ext0;
 | |
| 		int8_t rssi_ext1;
 | |
| 		int8_t rssi_ext2;
 | |
| 		int8_t rssi;
 | |
| 		bool is_mybeacon;
 | |
| 		u8 antenna;
 | |
| 		u8 rate;
 | |
| 	} rs[ATH_DBG_MAX_SAMPLES];
 | |
| 	struct ath_cycle_counters cc;
 | |
| 	struct ath9k_nfcal_hist nfCalHist[NUM_NF_READINGS];
 | |
| };
 | |
| 
 | |
| struct ath9k_debug {
 | |
| 	struct dentry *debugfs_phy;
 | |
| 	u32 regidx;
 | |
| 	struct ath_stats stats;
 | |
| #ifdef CONFIG_ATH9K_MAC_DEBUG
 | |
| 	spinlock_t samp_lock;
 | |
| 	struct ath_dbg_bb_mac_samp bb_mac_samp[ATH_DBG_MAX_SAMPLES];
 | |
| 	u8 sampidx;
 | |
| 	u8 tsidx;
 | |
| 	u8 rsidx;
 | |
| #endif
 | |
| };
 | |
| 
 | |
| int ath9k_init_debug(struct ath_hw *ah);
 | |
| void ath9k_deinit_debug(struct ath_softc *sc);
 | |
| 
 | |
| void ath_debug_stat_interrupt(struct ath_softc *sc, enum ath9k_int status);
 | |
| void ath_debug_stat_tx(struct ath_softc *sc, struct ath_buf *bf,
 | |
| 		       struct ath_tx_status *ts, struct ath_txq *txq,
 | |
| 		       unsigned int flags);
 | |
| void ath_debug_stat_rx(struct ath_softc *sc, struct ath_rx_status *rs);
 | |
| int ath9k_get_et_sset_count(struct ieee80211_hw *hw,
 | |
| 			    struct ieee80211_vif *vif, int sset);
 | |
| void ath9k_get_et_stats(struct ieee80211_hw *hw,
 | |
| 			struct ieee80211_vif *vif,
 | |
| 			struct ethtool_stats *stats, u64 *data);
 | |
| void ath9k_get_et_strings(struct ieee80211_hw *hw,
 | |
| 			  struct ieee80211_vif *vif,
 | |
| 			  u32 sset, u8 *data);
 | |
| void ath9k_sta_add_debugfs(struct ieee80211_hw *hw,
 | |
| 			   struct ieee80211_vif *vif,
 | |
| 			   struct ieee80211_sta *sta,
 | |
| 			   struct dentry *dir);
 | |
| void ath9k_sta_remove_debugfs(struct ieee80211_hw *hw,
 | |
| 			      struct ieee80211_vif *vif,
 | |
| 			      struct ieee80211_sta *sta,
 | |
| 			      struct dentry *dir);
 | |
| 
 | |
| void ath_debug_send_fft_sample(struct ath_softc *sc,
 | |
| 			       struct fft_sample_tlv *fft_sample);
 | |
| 
 | |
| #else
 | |
| 
 | |
| #define RX_STAT_INC(c) /* NOP */
 | |
| 
 | |
| static inline int ath9k_init_debug(struct ath_hw *ah)
 | |
| {
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static inline void ath9k_deinit_debug(struct ath_softc *sc)
 | |
| {
 | |
| }
 | |
| 
 | |
| static inline void ath_debug_stat_interrupt(struct ath_softc *sc,
 | |
| 					    enum ath9k_int status)
 | |
| {
 | |
| }
 | |
| 
 | |
| static inline void ath_debug_stat_tx(struct ath_softc *sc,
 | |
| 				     struct ath_buf *bf,
 | |
| 				     struct ath_tx_status *ts,
 | |
| 				     struct ath_txq *txq,
 | |
| 				     unsigned int flags)
 | |
| {
 | |
| }
 | |
| 
 | |
| static inline void ath_debug_stat_rx(struct ath_softc *sc,
 | |
| 				     struct ath_rx_status *rs)
 | |
| {
 | |
| }
 | |
| 
 | |
| #endif /* CONFIG_ATH9K_DEBUGFS */
 | |
| 
 | |
| #ifdef CONFIG_ATH9K_MAC_DEBUG
 | |
| 
 | |
| void ath9k_debug_samp_bb_mac(struct ath_softc *sc);
 | |
| 
 | |
| #else
 | |
| 
 | |
| static inline void ath9k_debug_samp_bb_mac(struct ath_softc *sc)
 | |
| {
 | |
| }
 | |
| 
 | |
| #endif
 | |
| 
 | |
| 
 | |
| #endif /* DEBUG_H */
 |