diff options
Diffstat (limited to 'net/mac80211/debugfs_sta.c')
| -rw-r--r-- | net/mac80211/debugfs_sta.c | 147 |
1 files changed, 102 insertions, 45 deletions
diff --git a/net/mac80211/debugfs_sta.c b/net/mac80211/debugfs_sta.c index 33a2e892115b..d92800bb2d2f 100644 --- a/net/mac80211/debugfs_sta.c +++ b/net/mac80211/debugfs_sta.c | |||
| @@ -44,7 +44,7 @@ static const struct file_operations sta_ ##name## _ops = { \ | |||
| 44 | STA_OPS(name) | 44 | STA_OPS(name) |
| 45 | 45 | ||
| 46 | STA_FILE(aid, sta.aid, D); | 46 | STA_FILE(aid, sta.aid, D); |
| 47 | STA_FILE(dev, sdata->dev->name, S); | 47 | STA_FILE(dev, sdata->name, S); |
| 48 | STA_FILE(rx_packets, rx_packets, LU); | 48 | STA_FILE(rx_packets, rx_packets, LU); |
| 49 | STA_FILE(tx_packets, tx_packets, LU); | 49 | STA_FILE(tx_packets, tx_packets, LU); |
| 50 | STA_FILE(rx_bytes, rx_bytes, LU); | 50 | STA_FILE(rx_bytes, rx_bytes, LU); |
| @@ -57,7 +57,6 @@ STA_FILE(tx_filtered, tx_filtered_count, LU); | |||
| 57 | STA_FILE(tx_retry_failed, tx_retry_failed, LU); | 57 | STA_FILE(tx_retry_failed, tx_retry_failed, LU); |
| 58 | STA_FILE(tx_retry_count, tx_retry_count, LU); | 58 | STA_FILE(tx_retry_count, tx_retry_count, LU); |
| 59 | STA_FILE(last_signal, last_signal, D); | 59 | STA_FILE(last_signal, last_signal, D); |
| 60 | STA_FILE(last_qual, last_qual, D); | ||
| 61 | STA_FILE(last_noise, last_noise, D); | 60 | STA_FILE(last_noise, last_noise, D); |
| 62 | STA_FILE(wep_weak_iv_count, wep_weak_iv_count, LU); | 61 | STA_FILE(wep_weak_iv_count, wep_weak_iv_count, LU); |
| 63 | 62 | ||
| @@ -67,10 +66,11 @@ static ssize_t sta_flags_read(struct file *file, char __user *userbuf, | |||
| 67 | char buf[100]; | 66 | char buf[100]; |
| 68 | struct sta_info *sta = file->private_data; | 67 | struct sta_info *sta = file->private_data; |
| 69 | u32 staflags = get_sta_flags(sta); | 68 | u32 staflags = get_sta_flags(sta); |
| 70 | int res = scnprintf(buf, sizeof(buf), "%s%s%s%s%s%s%s%s", | 69 | int res = scnprintf(buf, sizeof(buf), "%s%s%s%s%s%s%s%s%s", |
| 71 | staflags & WLAN_STA_AUTH ? "AUTH\n" : "", | 70 | staflags & WLAN_STA_AUTH ? "AUTH\n" : "", |
| 72 | staflags & WLAN_STA_ASSOC ? "ASSOC\n" : "", | 71 | staflags & WLAN_STA_ASSOC ? "ASSOC\n" : "", |
| 73 | staflags & WLAN_STA_PS ? "PS\n" : "", | 72 | staflags & WLAN_STA_PS_STA ? "PS (sta)\n" : "", |
| 73 | staflags & WLAN_STA_PS_DRIVER ? "PS (driver)\n" : "", | ||
| 74 | staflags & WLAN_STA_AUTHORIZED ? "AUTHORIZED\n" : "", | 74 | staflags & WLAN_STA_AUTHORIZED ? "AUTHORIZED\n" : "", |
| 75 | staflags & WLAN_STA_SHORT_PREAMBLE ? "SHORT PREAMBLE\n" : "", | 75 | staflags & WLAN_STA_SHORT_PREAMBLE ? "SHORT PREAMBLE\n" : "", |
| 76 | staflags & WLAN_STA_WME ? "WME\n" : "", | 76 | staflags & WLAN_STA_WME ? "WME\n" : "", |
| @@ -120,36 +120,38 @@ STA_OPS(last_seq_ctrl); | |||
| 120 | static ssize_t sta_agg_status_read(struct file *file, char __user *userbuf, | 120 | static ssize_t sta_agg_status_read(struct file *file, char __user *userbuf, |
| 121 | size_t count, loff_t *ppos) | 121 | size_t count, loff_t *ppos) |
| 122 | { | 122 | { |
| 123 | char buf[30 + STA_TID_NUM * 70], *p = buf; | 123 | char buf[64 + STA_TID_NUM * 40], *p = buf; |
| 124 | int i; | 124 | int i; |
| 125 | struct sta_info *sta = file->private_data; | 125 | struct sta_info *sta = file->private_data; |
| 126 | 126 | ||
| 127 | spin_lock_bh(&sta->lock); | 127 | spin_lock_bh(&sta->lock); |
| 128 | p += scnprintf(p, sizeof(buf)+buf-p, "next dialog_token is %#02x\n", | 128 | p += scnprintf(p, sizeof(buf) + buf - p, "next dialog_token: %#02x\n", |
| 129 | sta->ampdu_mlme.dialog_token_allocator + 1); | 129 | sta->ampdu_mlme.dialog_token_allocator + 1); |
| 130 | p += scnprintf(p, sizeof(buf) + buf - p, | ||
| 131 | "TID\t\tRX\tDTKN\tSSN\t\tTX\tDTKN\tSSN\tpending\n"); | ||
| 130 | for (i = 0; i < STA_TID_NUM; i++) { | 132 | for (i = 0; i < STA_TID_NUM; i++) { |
| 131 | p += scnprintf(p, sizeof(buf)+buf-p, "TID %02d:", i); | 133 | p += scnprintf(p, sizeof(buf) + buf - p, "%02d", i); |
| 132 | p += scnprintf(p, sizeof(buf)+buf-p, " RX=%x", | 134 | p += scnprintf(p, sizeof(buf) + buf - p, "\t\t%x", |
| 133 | sta->ampdu_mlme.tid_state_rx[i]); | 135 | sta->ampdu_mlme.tid_state_rx[i]); |
| 134 | p += scnprintf(p, sizeof(buf)+buf-p, "/DTKN=%#.2x", | 136 | p += scnprintf(p, sizeof(buf) + buf - p, "\t%#.2x", |
| 135 | sta->ampdu_mlme.tid_state_rx[i] ? | 137 | sta->ampdu_mlme.tid_state_rx[i] ? |
| 136 | sta->ampdu_mlme.tid_rx[i]->dialog_token : 0); | 138 | sta->ampdu_mlme.tid_rx[i]->dialog_token : 0); |
| 137 | p += scnprintf(p, sizeof(buf)+buf-p, "/SSN=%#.3x", | 139 | p += scnprintf(p, sizeof(buf) + buf - p, "\t%#.3x", |
| 138 | sta->ampdu_mlme.tid_state_rx[i] ? | 140 | sta->ampdu_mlme.tid_state_rx[i] ? |
| 139 | sta->ampdu_mlme.tid_rx[i]->ssn : 0); | 141 | sta->ampdu_mlme.tid_rx[i]->ssn : 0); |
| 140 | 142 | ||
| 141 | p += scnprintf(p, sizeof(buf)+buf-p, " TX=%x", | 143 | p += scnprintf(p, sizeof(buf) + buf - p, "\t\t%x", |
| 142 | sta->ampdu_mlme.tid_state_tx[i]); | 144 | sta->ampdu_mlme.tid_state_tx[i]); |
| 143 | p += scnprintf(p, sizeof(buf)+buf-p, "/DTKN=%#.2x", | 145 | p += scnprintf(p, sizeof(buf) + buf - p, "\t%#.2x", |
| 144 | sta->ampdu_mlme.tid_state_tx[i] ? | 146 | sta->ampdu_mlme.tid_state_tx[i] ? |
| 145 | sta->ampdu_mlme.tid_tx[i]->dialog_token : 0); | 147 | sta->ampdu_mlme.tid_tx[i]->dialog_token : 0); |
| 146 | p += scnprintf(p, sizeof(buf)+buf-p, "/SSN=%#.3x", | 148 | p += scnprintf(p, sizeof(buf) + buf - p, "\t%#.3x", |
| 147 | sta->ampdu_mlme.tid_state_tx[i] ? | 149 | sta->ampdu_mlme.tid_state_tx[i] ? |
| 148 | sta->ampdu_mlme.tid_tx[i]->ssn : 0); | 150 | sta->ampdu_mlme.tid_tx[i]->ssn : 0); |
| 149 | p += scnprintf(p, sizeof(buf)+buf-p, "/pending=%03d", | 151 | p += scnprintf(p, sizeof(buf) + buf - p, "\t%03d", |
| 150 | sta->ampdu_mlme.tid_state_tx[i] ? | 152 | sta->ampdu_mlme.tid_state_tx[i] ? |
| 151 | skb_queue_len(&sta->ampdu_mlme.tid_tx[i]->pending) : 0); | 153 | skb_queue_len(&sta->ampdu_mlme.tid_tx[i]->pending) : 0); |
| 152 | p += scnprintf(p, sizeof(buf)+buf-p, "\n"); | 154 | p += scnprintf(p, sizeof(buf) + buf - p, "\n"); |
| 153 | } | 155 | } |
| 154 | spin_unlock_bh(&sta->lock); | 156 | spin_unlock_bh(&sta->lock); |
| 155 | 157 | ||
| @@ -157,14 +159,92 @@ static ssize_t sta_agg_status_read(struct file *file, char __user *userbuf, | |||
| 157 | } | 159 | } |
| 158 | STA_OPS(agg_status); | 160 | STA_OPS(agg_status); |
| 159 | 161 | ||
| 162 | static ssize_t sta_ht_capa_read(struct file *file, char __user *userbuf, | ||
| 163 | size_t count, loff_t *ppos) | ||
| 164 | { | ||
| 165 | #define PRINT_HT_CAP(_cond, _str) \ | ||
| 166 | do { \ | ||
| 167 | if (_cond) \ | ||
| 168 | p += scnprintf(p, sizeof(buf)+buf-p, "\t" _str "\n"); \ | ||
| 169 | } while (0) | ||
| 170 | char buf[512], *p = buf; | ||
| 171 | int i; | ||
| 172 | struct sta_info *sta = file->private_data; | ||
| 173 | struct ieee80211_sta_ht_cap *htc = &sta->sta.ht_cap; | ||
| 174 | |||
| 175 | p += scnprintf(p, sizeof(buf) + buf - p, "ht %ssupported\n", | ||
| 176 | htc->ht_supported ? "" : "not "); | ||
| 177 | if (htc->ht_supported) { | ||
| 178 | p += scnprintf(p, sizeof(buf)+buf-p, "cap: %#.4x\n", htc->cap); | ||
| 179 | |||
| 180 | PRINT_HT_CAP((htc->cap & BIT(0)), "RX LDCP"); | ||
| 181 | PRINT_HT_CAP((htc->cap & BIT(1)), "HT20/HT40"); | ||
| 182 | PRINT_HT_CAP(!(htc->cap & BIT(1)), "HT20"); | ||
| 183 | |||
| 184 | PRINT_HT_CAP(((htc->cap >> 2) & 0x3) == 0, "Static SM Power Save"); | ||
| 185 | PRINT_HT_CAP(((htc->cap >> 2) & 0x3) == 1, "Dynamic SM Power Save"); | ||
| 186 | PRINT_HT_CAP(((htc->cap >> 2) & 0x3) == 3, "SM Power Save disabled"); | ||
| 187 | |||
| 188 | PRINT_HT_CAP((htc->cap & BIT(4)), "RX Greenfield"); | ||
| 189 | PRINT_HT_CAP((htc->cap & BIT(5)), "RX HT20 SGI"); | ||
| 190 | PRINT_HT_CAP((htc->cap & BIT(6)), "RX HT40 SGI"); | ||
| 191 | PRINT_HT_CAP((htc->cap & BIT(7)), "TX STBC"); | ||
| 192 | |||
| 193 | PRINT_HT_CAP(((htc->cap >> 8) & 0x3) == 0, "No RX STBC"); | ||
| 194 | PRINT_HT_CAP(((htc->cap >> 8) & 0x3) == 1, "RX STBC 1-stream"); | ||
| 195 | PRINT_HT_CAP(((htc->cap >> 8) & 0x3) == 2, "RX STBC 2-streams"); | ||
| 196 | PRINT_HT_CAP(((htc->cap >> 8) & 0x3) == 3, "RX STBC 3-streams"); | ||
| 197 | |||
| 198 | PRINT_HT_CAP((htc->cap & BIT(10)), "HT Delayed Block Ack"); | ||
| 199 | |||
| 200 | PRINT_HT_CAP((htc->cap & BIT(11)), "Max AMSDU length: " | ||
| 201 | "3839 bytes"); | ||
| 202 | PRINT_HT_CAP(!(htc->cap & BIT(11)), "Max AMSDU length: " | ||
| 203 | "7935 bytes"); | ||
| 204 | |||
| 205 | /* | ||
| 206 | * For beacons and probe response this would mean the BSS | ||
| 207 | * does or does not allow the usage of DSSS/CCK HT40. | ||
| 208 | * Otherwise it means the STA does or does not use | ||
| 209 | * DSSS/CCK HT40. | ||
| 210 | */ | ||
| 211 | PRINT_HT_CAP((htc->cap & BIT(12)), "DSSS/CCK HT40"); | ||
| 212 | PRINT_HT_CAP(!(htc->cap & BIT(12)), "No DSSS/CCK HT40"); | ||
| 213 | |||
| 214 | /* BIT(13) is reserved */ | ||
| 215 | |||
| 216 | PRINT_HT_CAP((htc->cap & BIT(14)), "40 MHz Intolerant"); | ||
| 217 | |||
| 218 | PRINT_HT_CAP((htc->cap & BIT(15)), "L-SIG TXOP protection"); | ||
| 219 | |||
| 220 | p += scnprintf(p, sizeof(buf)+buf-p, "ampdu factor/density: %d/%d\n", | ||
| 221 | htc->ampdu_factor, htc->ampdu_density); | ||
| 222 | p += scnprintf(p, sizeof(buf)+buf-p, "MCS mask:"); | ||
| 223 | |||
| 224 | for (i = 0; i < IEEE80211_HT_MCS_MASK_LEN; i++) | ||
| 225 | p += scnprintf(p, sizeof(buf)+buf-p, " %.2x", | ||
| 226 | htc->mcs.rx_mask[i]); | ||
| 227 | p += scnprintf(p, sizeof(buf)+buf-p, "\n"); | ||
| 228 | |||
| 229 | /* If not set this is meaningless */ | ||
| 230 | if (le16_to_cpu(htc->mcs.rx_highest)) { | ||
| 231 | p += scnprintf(p, sizeof(buf)+buf-p, | ||
| 232 | "MCS rx highest: %d Mbps\n", | ||
| 233 | le16_to_cpu(htc->mcs.rx_highest)); | ||
| 234 | } | ||
| 235 | |||
| 236 | p += scnprintf(p, sizeof(buf)+buf-p, "MCS tx params: %x\n", | ||
| 237 | htc->mcs.tx_params); | ||
| 238 | } | ||
| 239 | |||
| 240 | return simple_read_from_buffer(userbuf, count, ppos, buf, p - buf); | ||
| 241 | } | ||
| 242 | STA_OPS(ht_capa); | ||
| 243 | |||
| 160 | #define DEBUGFS_ADD(name) \ | 244 | #define DEBUGFS_ADD(name) \ |
| 161 | sta->debugfs.name = debugfs_create_file(#name, 0400, \ | 245 | debugfs_create_file(#name, 0400, \ |
| 162 | sta->debugfs.dir, sta, &sta_ ##name## _ops); | 246 | sta->debugfs.dir, sta, &sta_ ##name## _ops); |
| 163 | 247 | ||
| 164 | #define DEBUGFS_DEL(name) \ | ||
| 165 | debugfs_remove(sta->debugfs.name);\ | ||
| 166 | sta->debugfs.name = NULL; | ||
| 167 | |||
| 168 | 248 | ||
| 169 | void ieee80211_sta_debugfs_add(struct sta_info *sta) | 249 | void ieee80211_sta_debugfs_add(struct sta_info *sta) |
| 170 | { | 250 | { |
| @@ -209,36 +289,13 @@ void ieee80211_sta_debugfs_add(struct sta_info *sta) | |||
| 209 | DEBUGFS_ADD(tx_retry_failed); | 289 | DEBUGFS_ADD(tx_retry_failed); |
| 210 | DEBUGFS_ADD(tx_retry_count); | 290 | DEBUGFS_ADD(tx_retry_count); |
| 211 | DEBUGFS_ADD(last_signal); | 291 | DEBUGFS_ADD(last_signal); |
| 212 | DEBUGFS_ADD(last_qual); | ||
| 213 | DEBUGFS_ADD(last_noise); | 292 | DEBUGFS_ADD(last_noise); |
| 214 | DEBUGFS_ADD(wep_weak_iv_count); | 293 | DEBUGFS_ADD(wep_weak_iv_count); |
| 294 | DEBUGFS_ADD(ht_capa); | ||
| 215 | } | 295 | } |
| 216 | 296 | ||
| 217 | void ieee80211_sta_debugfs_remove(struct sta_info *sta) | 297 | void ieee80211_sta_debugfs_remove(struct sta_info *sta) |
| 218 | { | 298 | { |
| 219 | DEBUGFS_DEL(flags); | 299 | debugfs_remove_recursive(sta->debugfs.dir); |
| 220 | DEBUGFS_DEL(num_ps_buf_frames); | ||
| 221 | DEBUGFS_DEL(inactive_ms); | ||
| 222 | DEBUGFS_DEL(last_seq_ctrl); | ||
| 223 | DEBUGFS_DEL(agg_status); | ||
| 224 | DEBUGFS_DEL(aid); | ||
| 225 | DEBUGFS_DEL(dev); | ||
| 226 | DEBUGFS_DEL(rx_packets); | ||
| 227 | DEBUGFS_DEL(tx_packets); | ||
| 228 | DEBUGFS_DEL(rx_bytes); | ||
| 229 | DEBUGFS_DEL(tx_bytes); | ||
| 230 | DEBUGFS_DEL(rx_duplicates); | ||
| 231 | DEBUGFS_DEL(rx_fragments); | ||
| 232 | DEBUGFS_DEL(rx_dropped); | ||
| 233 | DEBUGFS_DEL(tx_fragments); | ||
| 234 | DEBUGFS_DEL(tx_filtered); | ||
| 235 | DEBUGFS_DEL(tx_retry_failed); | ||
| 236 | DEBUGFS_DEL(tx_retry_count); | ||
| 237 | DEBUGFS_DEL(last_signal); | ||
| 238 | DEBUGFS_DEL(last_qual); | ||
| 239 | DEBUGFS_DEL(last_noise); | ||
| 240 | DEBUGFS_DEL(wep_weak_iv_count); | ||
| 241 | |||
| 242 | debugfs_remove(sta->debugfs.dir); | ||
| 243 | sta->debugfs.dir = NULL; | 300 | sta->debugfs.dir = NULL; |
| 244 | } | 301 | } |
