1 // SPDX-License-Identifier: BSD-3-Clause-Clear
2 /*
3 * Copyright (c) 2018-2019 The Linux Foundation. All rights reserved.
4 */
5
6 #include <linux/vmalloc.h>
7
8 #include "debugfs_sta.h"
9 #include "core.h"
10 #include "peer.h"
11 #include "debug.h"
12 #include "dp_tx.h"
13 #include "debugfs_htt_stats.h"
14
ath11k_debugfs_sta_add_tx_stats(struct ath11k_sta * arsta,struct ath11k_per_peer_tx_stats * peer_stats,u8 legacy_rate_idx)15 void ath11k_debugfs_sta_add_tx_stats(struct ath11k_sta *arsta,
16 struct ath11k_per_peer_tx_stats *peer_stats,
17 u8 legacy_rate_idx)
18 {
19 struct rate_info *txrate = &arsta->txrate;
20 struct ath11k_htt_tx_stats *tx_stats;
21 int gi, mcs, bw, nss;
22
23 if (!arsta->tx_stats)
24 return;
25
26 tx_stats = arsta->tx_stats;
27 gi = FIELD_GET(RATE_INFO_FLAGS_SHORT_GI, arsta->txrate.flags);
28 mcs = txrate->mcs;
29 bw = ath11k_mac_mac80211_bw_to_ath11k_bw(txrate->bw);
30 nss = txrate->nss - 1;
31
32 #define STATS_OP_FMT(name) tx_stats->stats[ATH11K_STATS_TYPE_##name]
33
34 if (txrate->flags & RATE_INFO_FLAGS_HE_MCS) {
35 STATS_OP_FMT(SUCC).he[0][mcs] += peer_stats->succ_bytes;
36 STATS_OP_FMT(SUCC).he[1][mcs] += peer_stats->succ_pkts;
37 STATS_OP_FMT(FAIL).he[0][mcs] += peer_stats->failed_bytes;
38 STATS_OP_FMT(FAIL).he[1][mcs] += peer_stats->failed_pkts;
39 STATS_OP_FMT(RETRY).he[0][mcs] += peer_stats->retry_bytes;
40 STATS_OP_FMT(RETRY).he[1][mcs] += peer_stats->retry_pkts;
41 } else if (txrate->flags & RATE_INFO_FLAGS_VHT_MCS) {
42 STATS_OP_FMT(SUCC).vht[0][mcs] += peer_stats->succ_bytes;
43 STATS_OP_FMT(SUCC).vht[1][mcs] += peer_stats->succ_pkts;
44 STATS_OP_FMT(FAIL).vht[0][mcs] += peer_stats->failed_bytes;
45 STATS_OP_FMT(FAIL).vht[1][mcs] += peer_stats->failed_pkts;
46 STATS_OP_FMT(RETRY).vht[0][mcs] += peer_stats->retry_bytes;
47 STATS_OP_FMT(RETRY).vht[1][mcs] += peer_stats->retry_pkts;
48 } else if (txrate->flags & RATE_INFO_FLAGS_MCS) {
49 STATS_OP_FMT(SUCC).ht[0][mcs] += peer_stats->succ_bytes;
50 STATS_OP_FMT(SUCC).ht[1][mcs] += peer_stats->succ_pkts;
51 STATS_OP_FMT(FAIL).ht[0][mcs] += peer_stats->failed_bytes;
52 STATS_OP_FMT(FAIL).ht[1][mcs] += peer_stats->failed_pkts;
53 STATS_OP_FMT(RETRY).ht[0][mcs] += peer_stats->retry_bytes;
54 STATS_OP_FMT(RETRY).ht[1][mcs] += peer_stats->retry_pkts;
55 } else {
56 mcs = legacy_rate_idx;
57
58 STATS_OP_FMT(SUCC).legacy[0][mcs] += peer_stats->succ_bytes;
59 STATS_OP_FMT(SUCC).legacy[1][mcs] += peer_stats->succ_pkts;
60 STATS_OP_FMT(FAIL).legacy[0][mcs] += peer_stats->failed_bytes;
61 STATS_OP_FMT(FAIL).legacy[1][mcs] += peer_stats->failed_pkts;
62 STATS_OP_FMT(RETRY).legacy[0][mcs] += peer_stats->retry_bytes;
63 STATS_OP_FMT(RETRY).legacy[1][mcs] += peer_stats->retry_pkts;
64 }
65
66 if (peer_stats->is_ampdu) {
67 tx_stats->ba_fails += peer_stats->ba_fails;
68
69 if (txrate->flags & RATE_INFO_FLAGS_HE_MCS) {
70 STATS_OP_FMT(AMPDU).he[0][mcs] +=
71 peer_stats->succ_bytes + peer_stats->retry_bytes;
72 STATS_OP_FMT(AMPDU).he[1][mcs] +=
73 peer_stats->succ_pkts + peer_stats->retry_pkts;
74 } else if (txrate->flags & RATE_INFO_FLAGS_MCS) {
75 STATS_OP_FMT(AMPDU).ht[0][mcs] +=
76 peer_stats->succ_bytes + peer_stats->retry_bytes;
77 STATS_OP_FMT(AMPDU).ht[1][mcs] +=
78 peer_stats->succ_pkts + peer_stats->retry_pkts;
79 } else {
80 STATS_OP_FMT(AMPDU).vht[0][mcs] +=
81 peer_stats->succ_bytes + peer_stats->retry_bytes;
82 STATS_OP_FMT(AMPDU).vht[1][mcs] +=
83 peer_stats->succ_pkts + peer_stats->retry_pkts;
84 }
85 STATS_OP_FMT(AMPDU).bw[0][bw] +=
86 peer_stats->succ_bytes + peer_stats->retry_bytes;
87 STATS_OP_FMT(AMPDU).nss[0][nss] +=
88 peer_stats->succ_bytes + peer_stats->retry_bytes;
89 STATS_OP_FMT(AMPDU).gi[0][gi] +=
90 peer_stats->succ_bytes + peer_stats->retry_bytes;
91 STATS_OP_FMT(AMPDU).bw[1][bw] +=
92 peer_stats->succ_pkts + peer_stats->retry_pkts;
93 STATS_OP_FMT(AMPDU).nss[1][nss] +=
94 peer_stats->succ_pkts + peer_stats->retry_pkts;
95 STATS_OP_FMT(AMPDU).gi[1][gi] +=
96 peer_stats->succ_pkts + peer_stats->retry_pkts;
97 } else {
98 tx_stats->ack_fails += peer_stats->ba_fails;
99 }
100
101 STATS_OP_FMT(SUCC).bw[0][bw] += peer_stats->succ_bytes;
102 STATS_OP_FMT(SUCC).nss[0][nss] += peer_stats->succ_bytes;
103 STATS_OP_FMT(SUCC).gi[0][gi] += peer_stats->succ_bytes;
104
105 STATS_OP_FMT(SUCC).bw[1][bw] += peer_stats->succ_pkts;
106 STATS_OP_FMT(SUCC).nss[1][nss] += peer_stats->succ_pkts;
107 STATS_OP_FMT(SUCC).gi[1][gi] += peer_stats->succ_pkts;
108
109 STATS_OP_FMT(FAIL).bw[0][bw] += peer_stats->failed_bytes;
110 STATS_OP_FMT(FAIL).nss[0][nss] += peer_stats->failed_bytes;
111 STATS_OP_FMT(FAIL).gi[0][gi] += peer_stats->failed_bytes;
112
113 STATS_OP_FMT(FAIL).bw[1][bw] += peer_stats->failed_pkts;
114 STATS_OP_FMT(FAIL).nss[1][nss] += peer_stats->failed_pkts;
115 STATS_OP_FMT(FAIL).gi[1][gi] += peer_stats->failed_pkts;
116
117 STATS_OP_FMT(RETRY).bw[0][bw] += peer_stats->retry_bytes;
118 STATS_OP_FMT(RETRY).nss[0][nss] += peer_stats->retry_bytes;
119 STATS_OP_FMT(RETRY).gi[0][gi] += peer_stats->retry_bytes;
120
121 STATS_OP_FMT(RETRY).bw[1][bw] += peer_stats->retry_pkts;
122 STATS_OP_FMT(RETRY).nss[1][nss] += peer_stats->retry_pkts;
123 STATS_OP_FMT(RETRY).gi[1][gi] += peer_stats->retry_pkts;
124
125 tx_stats->tx_duration += peer_stats->duration;
126 }
127
ath11k_debugfs_sta_update_txcompl(struct ath11k * ar,struct hal_tx_status * ts)128 void ath11k_debugfs_sta_update_txcompl(struct ath11k *ar,
129 struct hal_tx_status *ts)
130 {
131 ath11k_dp_tx_update_txcompl(ar, ts);
132 }
133
ath11k_dbg_sta_dump_tx_stats(struct file * file,char __user * user_buf,size_t count,loff_t * ppos)134 static ssize_t ath11k_dbg_sta_dump_tx_stats(struct file *file,
135 char __user *user_buf,
136 size_t count, loff_t *ppos)
137 {
138 struct ieee80211_sta *sta = file->private_data;
139 struct ath11k_sta *arsta = (struct ath11k_sta *)sta->drv_priv;
140 struct ath11k *ar = arsta->arvif->ar;
141 struct ath11k_htt_data_stats *stats;
142 static const char *str_name[ATH11K_STATS_TYPE_MAX] = {"succ", "fail",
143 "retry", "ampdu"};
144 static const char *str[ATH11K_COUNTER_TYPE_MAX] = {"bytes", "packets"};
145 int len = 0, i, j, k, retval = 0;
146 const int size = 2 * 4096;
147 char *buf;
148
149 if (!arsta->tx_stats)
150 return -ENOENT;
151
152 buf = kzalloc(size, GFP_KERNEL);
153 if (!buf)
154 return -ENOMEM;
155
156 mutex_lock(&ar->conf_mutex);
157
158 spin_lock_bh(&ar->data_lock);
159 for (k = 0; k < ATH11K_STATS_TYPE_MAX; k++) {
160 for (j = 0; j < ATH11K_COUNTER_TYPE_MAX; j++) {
161 stats = &arsta->tx_stats->stats[k];
162 len += scnprintf(buf + len, size - len, "%s_%s\n",
163 str_name[k],
164 str[j]);
165 len += scnprintf(buf + len, size - len,
166 " HE MCS %s\n",
167 str[j]);
168 for (i = 0; i < ATH11K_HE_MCS_NUM; i++)
169 len += scnprintf(buf + len, size - len,
170 " %llu ",
171 stats->he[j][i]);
172 len += scnprintf(buf + len, size - len, "\n");
173 len += scnprintf(buf + len, size - len,
174 " VHT MCS %s\n",
175 str[j]);
176 for (i = 0; i < ATH11K_VHT_MCS_NUM; i++)
177 len += scnprintf(buf + len, size - len,
178 " %llu ",
179 stats->vht[j][i]);
180 len += scnprintf(buf + len, size - len, "\n");
181 len += scnprintf(buf + len, size - len, " HT MCS %s\n",
182 str[j]);
183 for (i = 0; i < ATH11K_HT_MCS_NUM; i++)
184 len += scnprintf(buf + len, size - len,
185 " %llu ", stats->ht[j][i]);
186 len += scnprintf(buf + len, size - len, "\n");
187 len += scnprintf(buf + len, size - len,
188 " BW %s (20,40,80,160 MHz)\n", str[j]);
189 len += scnprintf(buf + len, size - len,
190 " %llu %llu %llu %llu\n",
191 stats->bw[j][0], stats->bw[j][1],
192 stats->bw[j][2], stats->bw[j][3]);
193 len += scnprintf(buf + len, size - len,
194 " NSS %s (1x1,2x2,3x3,4x4)\n", str[j]);
195 len += scnprintf(buf + len, size - len,
196 " %llu %llu %llu %llu\n",
197 stats->nss[j][0], stats->nss[j][1],
198 stats->nss[j][2], stats->nss[j][3]);
199 len += scnprintf(buf + len, size - len,
200 " GI %s (0.4us,0.8us,1.6us,3.2us)\n",
201 str[j]);
202 len += scnprintf(buf + len, size - len,
203 " %llu %llu %llu %llu\n",
204 stats->gi[j][0], stats->gi[j][1],
205 stats->gi[j][2], stats->gi[j][3]);
206 len += scnprintf(buf + len, size - len,
207 " legacy rate %s (1,2 ... Mbps)\n ",
208 str[j]);
209 for (i = 0; i < ATH11K_LEGACY_NUM; i++)
210 len += scnprintf(buf + len, size - len, "%llu ",
211 stats->legacy[j][i]);
212 len += scnprintf(buf + len, size - len, "\n");
213 }
214 }
215
216 len += scnprintf(buf + len, size - len,
217 "\nTX duration\n %llu usecs\n",
218 arsta->tx_stats->tx_duration);
219 len += scnprintf(buf + len, size - len,
220 "BA fails\n %llu\n", arsta->tx_stats->ba_fails);
221 len += scnprintf(buf + len, size - len,
222 "ack fails\n %llu\n", arsta->tx_stats->ack_fails);
223 spin_unlock_bh(&ar->data_lock);
224
225 if (len > size)
226 len = size;
227 retval = simple_read_from_buffer(user_buf, count, ppos, buf, len);
228 kfree(buf);
229
230 mutex_unlock(&ar->conf_mutex);
231 return retval;
232 }
233
234 static const struct file_operations fops_tx_stats = {
235 .read = ath11k_dbg_sta_dump_tx_stats,
236 .open = simple_open,
237 .owner = THIS_MODULE,
238 .llseek = default_llseek,
239 };
240
ath11k_dbg_sta_dump_rx_stats(struct file * file,char __user * user_buf,size_t count,loff_t * ppos)241 static ssize_t ath11k_dbg_sta_dump_rx_stats(struct file *file,
242 char __user *user_buf,
243 size_t count, loff_t *ppos)
244 {
245 struct ieee80211_sta *sta = file->private_data;
246 struct ath11k_sta *arsta = (struct ath11k_sta *)sta->drv_priv;
247 struct ath11k *ar = arsta->arvif->ar;
248 struct ath11k_rx_peer_stats *rx_stats = arsta->rx_stats;
249 int len = 0, i, retval = 0;
250 const int size = 4096;
251 char *buf;
252
253 if (!rx_stats)
254 return -ENOENT;
255
256 buf = kzalloc(size, GFP_KERNEL);
257 if (!buf)
258 return -ENOMEM;
259
260 mutex_lock(&ar->conf_mutex);
261 spin_lock_bh(&ar->ab->base_lock);
262
263 len += scnprintf(buf + len, size - len, "RX peer stats:\n");
264 len += scnprintf(buf + len, size - len, "Num of MSDUs: %llu\n",
265 rx_stats->num_msdu);
266 len += scnprintf(buf + len, size - len, "Num of MSDUs with TCP L4: %llu\n",
267 rx_stats->tcp_msdu_count);
268 len += scnprintf(buf + len, size - len, "Num of MSDUs with UDP L4: %llu\n",
269 rx_stats->udp_msdu_count);
270 len += scnprintf(buf + len, size - len, "Num of MSDUs part of AMPDU: %llu\n",
271 rx_stats->ampdu_msdu_count);
272 len += scnprintf(buf + len, size - len, "Num of MSDUs not part of AMPDU: %llu\n",
273 rx_stats->non_ampdu_msdu_count);
274 len += scnprintf(buf + len, size - len, "Num of MSDUs using STBC: %llu\n",
275 rx_stats->stbc_count);
276 len += scnprintf(buf + len, size - len, "Num of MSDUs beamformed: %llu\n",
277 rx_stats->beamformed_count);
278 len += scnprintf(buf + len, size - len, "Num of MPDUs with FCS ok: %llu\n",
279 rx_stats->num_mpdu_fcs_ok);
280 len += scnprintf(buf + len, size - len, "Num of MPDUs with FCS error: %llu\n",
281 rx_stats->num_mpdu_fcs_err);
282 len += scnprintf(buf + len, size - len,
283 "GI: 0.8us %llu 0.4us %llu 1.6us %llu 3.2us %llu\n",
284 rx_stats->gi_count[0], rx_stats->gi_count[1],
285 rx_stats->gi_count[2], rx_stats->gi_count[3]);
286 len += scnprintf(buf + len, size - len,
287 "BW: 20Mhz %llu 40Mhz %llu 80Mhz %llu 160Mhz %llu\n",
288 rx_stats->bw_count[0], rx_stats->bw_count[1],
289 rx_stats->bw_count[2], rx_stats->bw_count[3]);
290 len += scnprintf(buf + len, size - len, "BCC %llu LDPC %llu\n",
291 rx_stats->coding_count[0], rx_stats->coding_count[1]);
292 len += scnprintf(buf + len, size - len,
293 "preamble: 11A %llu 11B %llu 11N %llu 11AC %llu 11AX %llu\n",
294 rx_stats->pream_cnt[0], rx_stats->pream_cnt[1],
295 rx_stats->pream_cnt[2], rx_stats->pream_cnt[3],
296 rx_stats->pream_cnt[4]);
297 len += scnprintf(buf + len, size - len,
298 "reception type: SU %llu MU_MIMO %llu MU_OFDMA %llu MU_OFDMA_MIMO %llu\n",
299 rx_stats->reception_type[0], rx_stats->reception_type[1],
300 rx_stats->reception_type[2], rx_stats->reception_type[3]);
301 len += scnprintf(buf + len, size - len, "TID(0-15) Legacy TID(16):");
302 for (i = 0; i <= IEEE80211_NUM_TIDS; i++)
303 len += scnprintf(buf + len, size - len, "%llu ", rx_stats->tid_count[i]);
304 len += scnprintf(buf + len, size - len, "\nMCS(0-11) Legacy MCS(12):");
305 for (i = 0; i < HAL_RX_MAX_MCS + 1; i++)
306 len += scnprintf(buf + len, size - len, "%llu ", rx_stats->mcs_count[i]);
307 len += scnprintf(buf + len, size - len, "\nNSS(1-8):");
308 for (i = 0; i < HAL_RX_MAX_NSS; i++)
309 len += scnprintf(buf + len, size - len, "%llu ", rx_stats->nss_count[i]);
310 len += scnprintf(buf + len, size - len, "\nRX Duration:%llu ",
311 rx_stats->rx_duration);
312 len += scnprintf(buf + len, size - len,
313 "\nDCM: %llu\nRU: 26 %llu 52: %llu 106: %llu 242: %llu 484: %llu 996: %llu\n",
314 rx_stats->dcm_count, rx_stats->ru_alloc_cnt[0],
315 rx_stats->ru_alloc_cnt[1], rx_stats->ru_alloc_cnt[2],
316 rx_stats->ru_alloc_cnt[3], rx_stats->ru_alloc_cnt[4],
317 rx_stats->ru_alloc_cnt[5]);
318
319 len += scnprintf(buf + len, size - len, "\n");
320
321 spin_unlock_bh(&ar->ab->base_lock);
322
323 if (len > size)
324 len = size;
325 retval = simple_read_from_buffer(user_buf, count, ppos, buf, len);
326 kfree(buf);
327
328 mutex_unlock(&ar->conf_mutex);
329 return retval;
330 }
331
332 static const struct file_operations fops_rx_stats = {
333 .read = ath11k_dbg_sta_dump_rx_stats,
334 .open = simple_open,
335 .owner = THIS_MODULE,
336 .llseek = default_llseek,
337 };
338
339 static int
ath11k_dbg_sta_open_htt_peer_stats(struct inode * inode,struct file * file)340 ath11k_dbg_sta_open_htt_peer_stats(struct inode *inode, struct file *file)
341 {
342 struct ieee80211_sta *sta = inode->i_private;
343 struct ath11k_sta *arsta = (struct ath11k_sta *)sta->drv_priv;
344 struct ath11k *ar = arsta->arvif->ar;
345 struct debug_htt_stats_req *stats_req;
346 int type = ar->debug.htt_stats.type;
347 int ret;
348
349 if ((type != ATH11K_DBG_HTT_EXT_STATS_PEER_INFO &&
350 type != ATH11K_DBG_HTT_EXT_STATS_PEER_CTRL_PATH_TXRX_STATS) ||
351 type == ATH11K_DBG_HTT_EXT_STATS_RESET)
352 return -EPERM;
353
354 stats_req = vzalloc(sizeof(*stats_req) + ATH11K_HTT_STATS_BUF_SIZE);
355 if (!stats_req)
356 return -ENOMEM;
357
358 mutex_lock(&ar->conf_mutex);
359 ar->debug.htt_stats.stats_req = stats_req;
360 stats_req->type = type;
361 memcpy(stats_req->peer_addr, sta->addr, ETH_ALEN);
362 ret = ath11k_debugfs_htt_stats_req(ar);
363 mutex_unlock(&ar->conf_mutex);
364 if (ret < 0)
365 goto out;
366
367 file->private_data = stats_req;
368 return 0;
369 out:
370 vfree(stats_req);
371 ar->debug.htt_stats.stats_req = NULL;
372 return ret;
373 }
374
375 static int
ath11k_dbg_sta_release_htt_peer_stats(struct inode * inode,struct file * file)376 ath11k_dbg_sta_release_htt_peer_stats(struct inode *inode, struct file *file)
377 {
378 struct ieee80211_sta *sta = inode->i_private;
379 struct ath11k_sta *arsta = (struct ath11k_sta *)sta->drv_priv;
380 struct ath11k *ar = arsta->arvif->ar;
381
382 mutex_lock(&ar->conf_mutex);
383 vfree(file->private_data);
384 ar->debug.htt_stats.stats_req = NULL;
385 mutex_unlock(&ar->conf_mutex);
386
387 return 0;
388 }
389
ath11k_dbg_sta_read_htt_peer_stats(struct file * file,char __user * user_buf,size_t count,loff_t * ppos)390 static ssize_t ath11k_dbg_sta_read_htt_peer_stats(struct file *file,
391 char __user *user_buf,
392 size_t count, loff_t *ppos)
393 {
394 struct debug_htt_stats_req *stats_req = file->private_data;
395 char *buf;
396 u32 length = 0;
397
398 buf = stats_req->buf;
399 length = min_t(u32, stats_req->buf_len, ATH11K_HTT_STATS_BUF_SIZE);
400 return simple_read_from_buffer(user_buf, count, ppos, buf, length);
401 }
402
403 static const struct file_operations fops_htt_peer_stats = {
404 .open = ath11k_dbg_sta_open_htt_peer_stats,
405 .release = ath11k_dbg_sta_release_htt_peer_stats,
406 .read = ath11k_dbg_sta_read_htt_peer_stats,
407 .owner = THIS_MODULE,
408 .llseek = default_llseek,
409 };
410
ath11k_dbg_sta_write_peer_pktlog(struct file * file,const char __user * buf,size_t count,loff_t * ppos)411 static ssize_t ath11k_dbg_sta_write_peer_pktlog(struct file *file,
412 const char __user *buf,
413 size_t count, loff_t *ppos)
414 {
415 struct ieee80211_sta *sta = file->private_data;
416 struct ath11k_sta *arsta = (struct ath11k_sta *)sta->drv_priv;
417 struct ath11k *ar = arsta->arvif->ar;
418 int ret, enable;
419
420 mutex_lock(&ar->conf_mutex);
421
422 if (ar->state != ATH11K_STATE_ON) {
423 ret = -ENETDOWN;
424 goto out;
425 }
426
427 ret = kstrtoint_from_user(buf, count, 0, &enable);
428 if (ret)
429 goto out;
430
431 ar->debug.pktlog_peer_valid = enable;
432 memcpy(ar->debug.pktlog_peer_addr, sta->addr, ETH_ALEN);
433
434 /* Send peer based pktlog enable/disable */
435 ret = ath11k_wmi_pdev_peer_pktlog_filter(ar, sta->addr, enable);
436 if (ret) {
437 ath11k_warn(ar->ab, "failed to set peer pktlog filter %pM: %d\n",
438 sta->addr, ret);
439 goto out;
440 }
441
442 ath11k_dbg(ar->ab, ATH11K_DBG_WMI, "peer pktlog filter set to %d\n",
443 enable);
444 ret = count;
445
446 out:
447 mutex_unlock(&ar->conf_mutex);
448 return ret;
449 }
450
ath11k_dbg_sta_read_peer_pktlog(struct file * file,char __user * ubuf,size_t count,loff_t * ppos)451 static ssize_t ath11k_dbg_sta_read_peer_pktlog(struct file *file,
452 char __user *ubuf,
453 size_t count, loff_t *ppos)
454 {
455 struct ieee80211_sta *sta = file->private_data;
456 struct ath11k_sta *arsta = (struct ath11k_sta *)sta->drv_priv;
457 struct ath11k *ar = arsta->arvif->ar;
458 char buf[32] = {0};
459 int len;
460
461 mutex_lock(&ar->conf_mutex);
462 len = scnprintf(buf, sizeof(buf), "%08x %pM\n",
463 ar->debug.pktlog_peer_valid,
464 ar->debug.pktlog_peer_addr);
465 mutex_unlock(&ar->conf_mutex);
466
467 return simple_read_from_buffer(ubuf, count, ppos, buf, len);
468 }
469
470 static const struct file_operations fops_peer_pktlog = {
471 .write = ath11k_dbg_sta_write_peer_pktlog,
472 .read = ath11k_dbg_sta_read_peer_pktlog,
473 .open = simple_open,
474 .owner = THIS_MODULE,
475 .llseek = default_llseek,
476 };
477
ath11k_dbg_sta_write_delba(struct file * file,const char __user * user_buf,size_t count,loff_t * ppos)478 static ssize_t ath11k_dbg_sta_write_delba(struct file *file,
479 const char __user *user_buf,
480 size_t count, loff_t *ppos)
481 {
482 struct ieee80211_sta *sta = file->private_data;
483 struct ath11k_sta *arsta = (struct ath11k_sta *)sta->drv_priv;
484 struct ath11k *ar = arsta->arvif->ar;
485 u32 tid, initiator, reason;
486 int ret;
487 char buf[64] = {0};
488
489 ret = simple_write_to_buffer(buf, sizeof(buf) - 1, ppos,
490 user_buf, count);
491 if (ret <= 0)
492 return ret;
493
494 ret = sscanf(buf, "%u %u %u", &tid, &initiator, &reason);
495 if (ret != 3)
496 return -EINVAL;
497
498 /* Valid TID values are 0 through 15 */
499 if (tid > HAL_DESC_REO_NON_QOS_TID - 1)
500 return -EINVAL;
501
502 mutex_lock(&ar->conf_mutex);
503 if (ar->state != ATH11K_STATE_ON ||
504 arsta->aggr_mode != ATH11K_DBG_AGGR_MODE_MANUAL) {
505 ret = count;
506 goto out;
507 }
508
509 ret = ath11k_wmi_delba_send(ar, arsta->arvif->vdev_id, sta->addr,
510 tid, initiator, reason);
511 if (ret) {
512 ath11k_warn(ar->ab, "failed to send delba: vdev_id %u peer %pM tid %u initiator %u reason %u\n",
513 arsta->arvif->vdev_id, sta->addr, tid, initiator,
514 reason);
515 }
516 ret = count;
517 out:
518 mutex_unlock(&ar->conf_mutex);
519 return ret;
520 }
521
522 static const struct file_operations fops_delba = {
523 .write = ath11k_dbg_sta_write_delba,
524 .open = simple_open,
525 .owner = THIS_MODULE,
526 .llseek = default_llseek,
527 };
528
ath11k_dbg_sta_write_addba_resp(struct file * file,const char __user * user_buf,size_t count,loff_t * ppos)529 static ssize_t ath11k_dbg_sta_write_addba_resp(struct file *file,
530 const char __user *user_buf,
531 size_t count, loff_t *ppos)
532 {
533 struct ieee80211_sta *sta = file->private_data;
534 struct ath11k_sta *arsta = (struct ath11k_sta *)sta->drv_priv;
535 struct ath11k *ar = arsta->arvif->ar;
536 u32 tid, status;
537 int ret;
538 char buf[64] = {0};
539
540 ret = simple_write_to_buffer(buf, sizeof(buf) - 1, ppos,
541 user_buf, count);
542 if (ret <= 0)
543 return ret;
544
545 ret = sscanf(buf, "%u %u", &tid, &status);
546 if (ret != 2)
547 return -EINVAL;
548
549 /* Valid TID values are 0 through 15 */
550 if (tid > HAL_DESC_REO_NON_QOS_TID - 1)
551 return -EINVAL;
552
553 mutex_lock(&ar->conf_mutex);
554 if (ar->state != ATH11K_STATE_ON ||
555 arsta->aggr_mode != ATH11K_DBG_AGGR_MODE_MANUAL) {
556 ret = count;
557 goto out;
558 }
559
560 ret = ath11k_wmi_addba_set_resp(ar, arsta->arvif->vdev_id, sta->addr,
561 tid, status);
562 if (ret) {
563 ath11k_warn(ar->ab, "failed to send addba response: vdev_id %u peer %pM tid %u status%u\n",
564 arsta->arvif->vdev_id, sta->addr, tid, status);
565 }
566 ret = count;
567 out:
568 mutex_unlock(&ar->conf_mutex);
569 return ret;
570 }
571
572 static const struct file_operations fops_addba_resp = {
573 .write = ath11k_dbg_sta_write_addba_resp,
574 .open = simple_open,
575 .owner = THIS_MODULE,
576 .llseek = default_llseek,
577 };
578
ath11k_dbg_sta_write_addba(struct file * file,const char __user * user_buf,size_t count,loff_t * ppos)579 static ssize_t ath11k_dbg_sta_write_addba(struct file *file,
580 const char __user *user_buf,
581 size_t count, loff_t *ppos)
582 {
583 struct ieee80211_sta *sta = file->private_data;
584 struct ath11k_sta *arsta = (struct ath11k_sta *)sta->drv_priv;
585 struct ath11k *ar = arsta->arvif->ar;
586 u32 tid, buf_size;
587 int ret;
588 char buf[64] = {0};
589
590 ret = simple_write_to_buffer(buf, sizeof(buf) - 1, ppos,
591 user_buf, count);
592 if (ret <= 0)
593 return ret;
594
595 ret = sscanf(buf, "%u %u", &tid, &buf_size);
596 if (ret != 2)
597 return -EINVAL;
598
599 /* Valid TID values are 0 through 15 */
600 if (tid > HAL_DESC_REO_NON_QOS_TID - 1)
601 return -EINVAL;
602
603 mutex_lock(&ar->conf_mutex);
604 if (ar->state != ATH11K_STATE_ON ||
605 arsta->aggr_mode != ATH11K_DBG_AGGR_MODE_MANUAL) {
606 ret = count;
607 goto out;
608 }
609
610 ret = ath11k_wmi_addba_send(ar, arsta->arvif->vdev_id, sta->addr,
611 tid, buf_size);
612 if (ret) {
613 ath11k_warn(ar->ab, "failed to send addba request: vdev_id %u peer %pM tid %u buf_size %u\n",
614 arsta->arvif->vdev_id, sta->addr, tid, buf_size);
615 }
616
617 ret = count;
618 out:
619 mutex_unlock(&ar->conf_mutex);
620 return ret;
621 }
622
623 static const struct file_operations fops_addba = {
624 .write = ath11k_dbg_sta_write_addba,
625 .open = simple_open,
626 .owner = THIS_MODULE,
627 .llseek = default_llseek,
628 };
629
ath11k_dbg_sta_read_aggr_mode(struct file * file,char __user * user_buf,size_t count,loff_t * ppos)630 static ssize_t ath11k_dbg_sta_read_aggr_mode(struct file *file,
631 char __user *user_buf,
632 size_t count, loff_t *ppos)
633 {
634 struct ieee80211_sta *sta = file->private_data;
635 struct ath11k_sta *arsta = (struct ath11k_sta *)sta->drv_priv;
636 struct ath11k *ar = arsta->arvif->ar;
637 char buf[64];
638 int len = 0;
639
640 mutex_lock(&ar->conf_mutex);
641 len = scnprintf(buf, sizeof(buf) - len,
642 "aggregation mode: %s\n\n%s\n%s\n",
643 (arsta->aggr_mode == ATH11K_DBG_AGGR_MODE_AUTO) ?
644 "auto" : "manual", "auto = 0", "manual = 1");
645 mutex_unlock(&ar->conf_mutex);
646
647 return simple_read_from_buffer(user_buf, count, ppos, buf, len);
648 }
649
ath11k_dbg_sta_write_aggr_mode(struct file * file,const char __user * user_buf,size_t count,loff_t * ppos)650 static ssize_t ath11k_dbg_sta_write_aggr_mode(struct file *file,
651 const char __user *user_buf,
652 size_t count, loff_t *ppos)
653 {
654 struct ieee80211_sta *sta = file->private_data;
655 struct ath11k_sta *arsta = (struct ath11k_sta *)sta->drv_priv;
656 struct ath11k *ar = arsta->arvif->ar;
657 u32 aggr_mode;
658 int ret;
659
660 if (kstrtouint_from_user(user_buf, count, 0, &aggr_mode))
661 return -EINVAL;
662
663 if (aggr_mode >= ATH11K_DBG_AGGR_MODE_MAX)
664 return -EINVAL;
665
666 mutex_lock(&ar->conf_mutex);
667 if (ar->state != ATH11K_STATE_ON ||
668 aggr_mode == arsta->aggr_mode) {
669 ret = count;
670 goto out;
671 }
672
673 ret = ath11k_wmi_addba_clear_resp(ar, arsta->arvif->vdev_id, sta->addr);
674 if (ret) {
675 ath11k_warn(ar->ab, "failed to clear addba session ret: %d\n",
676 ret);
677 goto out;
678 }
679
680 arsta->aggr_mode = aggr_mode;
681 out:
682 mutex_unlock(&ar->conf_mutex);
683 return ret;
684 }
685
686 static const struct file_operations fops_aggr_mode = {
687 .read = ath11k_dbg_sta_read_aggr_mode,
688 .write = ath11k_dbg_sta_write_aggr_mode,
689 .open = simple_open,
690 .owner = THIS_MODULE,
691 .llseek = default_llseek,
692 };
693
694 static ssize_t
ath11k_write_htt_peer_stats_reset(struct file * file,const char __user * user_buf,size_t count,loff_t * ppos)695 ath11k_write_htt_peer_stats_reset(struct file *file,
696 const char __user *user_buf,
697 size_t count, loff_t *ppos)
698 {
699 struct ieee80211_sta *sta = file->private_data;
700 struct ath11k_sta *arsta = (struct ath11k_sta *)sta->drv_priv;
701 struct ath11k *ar = arsta->arvif->ar;
702 struct htt_ext_stats_cfg_params cfg_params = { 0 };
703 int ret;
704 u8 type;
705
706 ret = kstrtou8_from_user(user_buf, count, 0, &type);
707 if (ret)
708 return ret;
709
710 if (!type)
711 return ret;
712
713 mutex_lock(&ar->conf_mutex);
714 cfg_params.cfg0 = HTT_STAT_PEER_INFO_MAC_ADDR;
715 cfg_params.cfg0 |= FIELD_PREP(GENMASK(15, 1),
716 HTT_PEER_STATS_REQ_MODE_FLUSH_TQM);
717
718 cfg_params.cfg1 = HTT_STAT_DEFAULT_PEER_REQ_TYPE;
719
720 cfg_params.cfg2 |= FIELD_PREP(GENMASK(7, 0), sta->addr[0]);
721 cfg_params.cfg2 |= FIELD_PREP(GENMASK(15, 8), sta->addr[1]);
722 cfg_params.cfg2 |= FIELD_PREP(GENMASK(23, 16), sta->addr[2]);
723 cfg_params.cfg2 |= FIELD_PREP(GENMASK(31, 24), sta->addr[3]);
724
725 cfg_params.cfg3 |= FIELD_PREP(GENMASK(7, 0), sta->addr[4]);
726 cfg_params.cfg3 |= FIELD_PREP(GENMASK(15, 8), sta->addr[5]);
727
728 cfg_params.cfg3 |= ATH11K_HTT_PEER_STATS_RESET;
729
730 ret = ath11k_dp_tx_htt_h2t_ext_stats_req(ar,
731 ATH11K_DBG_HTT_EXT_STATS_PEER_INFO,
732 &cfg_params,
733 0ULL);
734 if (ret) {
735 ath11k_warn(ar->ab, "failed to send htt peer stats request: %d\n", ret);
736 mutex_unlock(&ar->conf_mutex);
737 return ret;
738 }
739
740 mutex_unlock(&ar->conf_mutex);
741
742 ret = count;
743
744 return ret;
745 }
746
747 static const struct file_operations fops_htt_peer_stats_reset = {
748 .write = ath11k_write_htt_peer_stats_reset,
749 .open = simple_open,
750 .owner = THIS_MODULE,
751 .llseek = default_llseek,
752 };
753
ath11k_debugfs_sta_op_add(struct ieee80211_hw * hw,struct ieee80211_vif * vif,struct ieee80211_sta * sta,struct dentry * dir)754 void ath11k_debugfs_sta_op_add(struct ieee80211_hw *hw, struct ieee80211_vif *vif,
755 struct ieee80211_sta *sta, struct dentry *dir)
756 {
757 struct ath11k *ar = hw->priv;
758
759 if (ath11k_debugfs_is_extd_tx_stats_enabled(ar))
760 debugfs_create_file("tx_stats", 0400, dir, sta,
761 &fops_tx_stats);
762 if (ath11k_debugfs_is_extd_rx_stats_enabled(ar))
763 debugfs_create_file("rx_stats", 0400, dir, sta,
764 &fops_rx_stats);
765
766 debugfs_create_file("htt_peer_stats", 0400, dir, sta,
767 &fops_htt_peer_stats);
768
769 debugfs_create_file("peer_pktlog", 0644, dir, sta,
770 &fops_peer_pktlog);
771
772 debugfs_create_file("aggr_mode", 0644, dir, sta, &fops_aggr_mode);
773 debugfs_create_file("addba", 0200, dir, sta, &fops_addba);
774 debugfs_create_file("addba_resp", 0200, dir, sta, &fops_addba_resp);
775 debugfs_create_file("delba", 0200, dir, sta, &fops_delba);
776
777 if (test_bit(WMI_TLV_SERVICE_PER_PEER_HTT_STATS_RESET,
778 ar->ab->wmi_ab.svc_map))
779 debugfs_create_file("htt_peer_stats_reset", 0600, dir, sta,
780 &fops_htt_peer_stats_reset);
781 }
782