1 /*
2  * Copyright (c) 2012-2017 Qualcomm Atheros, Inc.
3  * Copyright (c) 2018-2019, The Linux Foundation. All rights reserved.
4  *
5  * Permission to use, copy, modify, and/or distribute this software for any
6  * purpose with or without fee is hereby granted, provided that the above
7  * copyright notice and this permission notice appear in all copies.
8  *
9  * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
10  * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
11  * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
12  * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
13  * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
14  * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
15  * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
16  */
17 
18 #include <linux/module.h>
19 #include <linux/debugfs.h>
20 #include <linux/seq_file.h>
21 #include <linux/pci.h>
22 #include <linux/rtnetlink.h>
23 #include <linux/power_supply.h>
24 #include "wil6210.h"
25 #include "wmi.h"
26 #include "txrx.h"
27 #include "pmc.h"
28 
29 /* Nasty hack. Better have per device instances */
30 static u32 mem_addr;
31 static u32 dbg_txdesc_index;
32 static u32 dbg_ring_index; /* 24+ for Rx, 0..23 for Tx */
33 static u32 dbg_status_msg_index;
34 /* 0..wil->num_rx_status_rings-1 for Rx, wil->tx_sring_idx for Tx */
35 static u32 dbg_sring_index;
36 
37 enum dbg_off_type {
38 	doff_u32 = 0,
39 	doff_x32 = 1,
40 	doff_ulong = 2,
41 	doff_io32 = 3,
42 	doff_u8 = 4
43 };
44 
45 /* offset to "wil" */
46 struct dbg_off {
47 	const char *name;
48 	umode_t mode;
49 	ulong off;
50 	enum dbg_off_type type;
51 };
52 
wil_print_desc_edma(struct seq_file * s,struct wil6210_priv * wil,struct wil_ring * ring,char _s,char _h,int idx)53 static void wil_print_desc_edma(struct seq_file *s, struct wil6210_priv *wil,
54 				struct wil_ring *ring,
55 				char _s, char _h, int idx)
56 {
57 	u8 num_of_descs;
58 	bool has_skb = false;
59 
60 	if (ring->is_rx) {
61 		struct wil_rx_enhanced_desc *rx_d =
62 			(struct wil_rx_enhanced_desc *)
63 			&ring->va[idx].rx.enhanced;
64 		u16 buff_id = le16_to_cpu(rx_d->mac.buff_id);
65 
66 		if (wil->rx_buff_mgmt.buff_arr &&
67 		    wil_val_in_range(buff_id, 0, wil->rx_buff_mgmt.size))
68 			has_skb = wil->rx_buff_mgmt.buff_arr[buff_id].skb;
69 		seq_printf(s, "%c", (has_skb) ? _h : _s);
70 	} else {
71 		struct wil_tx_enhanced_desc *d =
72 			(struct wil_tx_enhanced_desc *)
73 			&ring->va[idx].tx.enhanced;
74 
75 		num_of_descs = (u8)d->mac.d[2];
76 		has_skb = ring->ctx && ring->ctx[idx].skb;
77 		if (num_of_descs >= 1)
78 			seq_printf(s, "%c", has_skb ? _h : _s);
79 		else
80 			/* num_of_descs == 0, it's a frag in a list of descs */
81 			seq_printf(s, "%c", has_skb ? 'h' : _s);
82 	}
83 }
84 
wil_print_ring(struct seq_file * s,struct wil6210_priv * wil,const char * name,struct wil_ring * ring,char _s,char _h)85 static void wil_print_ring(struct seq_file *s, struct wil6210_priv *wil,
86 			   const char *name, struct wil_ring *ring,
87 			   char _s, char _h)
88 {
89 	void __iomem *x;
90 	u32 v;
91 
92 	seq_printf(s, "RING %s = {\n", name);
93 	seq_printf(s, "  pa     = %pad\n", &ring->pa);
94 	seq_printf(s, "  va     = 0x%p\n", ring->va);
95 	seq_printf(s, "  size   = %d\n", ring->size);
96 	if (wil->use_enhanced_dma_hw && ring->is_rx)
97 		seq_printf(s, "  swtail = %u\n", *ring->edma_rx_swtail.va);
98 	else
99 		seq_printf(s, "  swtail = %d\n", ring->swtail);
100 	seq_printf(s, "  swhead = %d\n", ring->swhead);
101 	if (wil->use_enhanced_dma_hw) {
102 		int ring_id = ring->is_rx ?
103 			WIL_RX_DESC_RING_ID : ring - wil->ring_tx;
104 		/* SUBQ_CONS is a table of 32 entries, one for each Q pair.
105 		 * lower 16bits are for even ring_id and upper 16bits are for
106 		 * odd ring_id
107 		 */
108 		x = wmi_addr(wil, RGF_DMA_SCM_SUBQ_CONS + 4 * (ring_id / 2));
109 		v = readl_relaxed(x);
110 
111 		v = (ring_id % 2 ? (v >> 16) : (v & 0xffff));
112 		seq_printf(s, "  hwhead = %u\n", v);
113 	}
114 	seq_printf(s, "  hwtail = [0x%08x] -> ", ring->hwtail);
115 	x = wmi_addr(wil, ring->hwtail);
116 	if (x) {
117 		v = readl(x);
118 		seq_printf(s, "0x%08x = %d\n", v, v);
119 	} else {
120 		seq_puts(s, "???\n");
121 	}
122 
123 	if (ring->va && (ring->size <= (1 << WIL_RING_SIZE_ORDER_MAX))) {
124 		uint i;
125 
126 		for (i = 0; i < ring->size; i++) {
127 			if ((i % 128) == 0 && i != 0)
128 				seq_puts(s, "\n");
129 			if (wil->use_enhanced_dma_hw) {
130 				wil_print_desc_edma(s, wil, ring, _s, _h, i);
131 			} else {
132 				volatile struct vring_tx_desc *d =
133 					&ring->va[i].tx.legacy;
134 				seq_printf(s, "%c", (d->dma.status & BIT(0)) ?
135 					   _s : (ring->ctx[i].skb ? _h : 'h'));
136 			}
137 		}
138 		seq_puts(s, "\n");
139 	}
140 	seq_puts(s, "}\n");
141 }
142 
ring_show(struct seq_file * s,void * data)143 static int ring_show(struct seq_file *s, void *data)
144 {
145 	uint i;
146 	struct wil6210_priv *wil = s->private;
147 
148 	wil_print_ring(s, wil, "rx", &wil->ring_rx, 'S', '_');
149 
150 	for (i = 0; i < ARRAY_SIZE(wil->ring_tx); i++) {
151 		struct wil_ring *ring = &wil->ring_tx[i];
152 		struct wil_ring_tx_data *txdata = &wil->ring_tx_data[i];
153 
154 		if (ring->va) {
155 			int cid = wil->ring2cid_tid[i][0];
156 			int tid = wil->ring2cid_tid[i][1];
157 			u32 swhead = ring->swhead;
158 			u32 swtail = ring->swtail;
159 			int used = (ring->size + swhead - swtail)
160 				   % ring->size;
161 			int avail = ring->size - used - 1;
162 			char name[10];
163 			char sidle[10];
164 			/* performance monitoring */
165 			cycles_t now = get_cycles();
166 			uint64_t idle = txdata->idle * 100;
167 			uint64_t total = now - txdata->begin;
168 
169 			if (total != 0) {
170 				do_div(idle, total);
171 				snprintf(sidle, sizeof(sidle), "%3d%%",
172 					 (int)idle);
173 			} else {
174 				snprintf(sidle, sizeof(sidle), "N/A");
175 			}
176 			txdata->begin = now;
177 			txdata->idle = 0ULL;
178 
179 			snprintf(name, sizeof(name), "tx_%2d", i);
180 
181 			if (cid < wil->max_assoc_sta)
182 				seq_printf(s,
183 					   "\n%pM CID %d TID %d 1x%s BACK([%u] %u TU A%s) [%3d|%3d] idle %s\n",
184 					   wil->sta[cid].addr, cid, tid,
185 					   txdata->dot1x_open ? "+" : "-",
186 					   txdata->agg_wsize,
187 					   txdata->agg_timeout,
188 					   txdata->agg_amsdu ? "+" : "-",
189 					   used, avail, sidle);
190 			else
191 				seq_printf(s,
192 					   "\nBroadcast 1x%s [%3d|%3d] idle %s\n",
193 					   txdata->dot1x_open ? "+" : "-",
194 					   used, avail, sidle);
195 
196 			wil_print_ring(s, wil, name, ring, '_', 'H');
197 		}
198 	}
199 
200 	return 0;
201 }
202 DEFINE_SHOW_ATTRIBUTE(ring);
203 
wil_print_sring(struct seq_file * s,struct wil6210_priv * wil,struct wil_status_ring * sring)204 static void wil_print_sring(struct seq_file *s, struct wil6210_priv *wil,
205 			    struct wil_status_ring *sring)
206 {
207 	void __iomem *x;
208 	int sring_idx = sring - wil->srings;
209 	u32 v;
210 
211 	seq_printf(s, "Status Ring %s [ %d ] = {\n",
212 		   sring->is_rx ? "RX" : "TX", sring_idx);
213 	seq_printf(s, "  pa     = %pad\n", &sring->pa);
214 	seq_printf(s, "  va     = 0x%pK\n", sring->va);
215 	seq_printf(s, "  size   = %d\n", sring->size);
216 	seq_printf(s, "  elem_size   = %zu\n", sring->elem_size);
217 	seq_printf(s, "  swhead = %d\n", sring->swhead);
218 	if (wil->use_enhanced_dma_hw) {
219 		/* COMPQ_PROD is a table of 32 entries, one for each Q pair.
220 		 * lower 16bits are for even ring_id and upper 16bits are for
221 		 * odd ring_id
222 		 */
223 		x = wmi_addr(wil, RGF_DMA_SCM_COMPQ_PROD + 4 * (sring_idx / 2));
224 		v = readl_relaxed(x);
225 
226 		v = (sring_idx % 2 ? (v >> 16) : (v & 0xffff));
227 		seq_printf(s, "  hwhead = %u\n", v);
228 	}
229 	seq_printf(s, "  hwtail = [0x%08x] -> ", sring->hwtail);
230 	x = wmi_addr(wil, sring->hwtail);
231 	if (x) {
232 		v = readl_relaxed(x);
233 		seq_printf(s, "0x%08x = %d\n", v, v);
234 	} else {
235 		seq_puts(s, "???\n");
236 	}
237 	seq_printf(s, "  desc_rdy_pol   = %d\n", sring->desc_rdy_pol);
238 	seq_printf(s, "  invalid_buff_id_cnt   = %d\n",
239 		   sring->invalid_buff_id_cnt);
240 
241 	if (sring->va && (sring->size <= (1 << WIL_RING_SIZE_ORDER_MAX))) {
242 		uint i;
243 
244 		for (i = 0; i < sring->size; i++) {
245 			u32 *sdword_0 =
246 				(u32 *)(sring->va + (sring->elem_size * i));
247 
248 			if ((i % 128) == 0 && i != 0)
249 				seq_puts(s, "\n");
250 			if (i == sring->swhead)
251 				seq_printf(s, "%c", (*sdword_0 & BIT(31)) ?
252 					   'X' : 'x');
253 			else
254 				seq_printf(s, "%c", (*sdword_0 & BIT(31)) ?
255 					   '1' : '0');
256 		}
257 		seq_puts(s, "\n");
258 	}
259 	seq_puts(s, "}\n");
260 }
261 
srings_show(struct seq_file * s,void * data)262 static int srings_show(struct seq_file *s, void *data)
263 {
264 	struct wil6210_priv *wil = s->private;
265 	int i = 0;
266 
267 	for (i = 0; i < WIL6210_MAX_STATUS_RINGS; i++)
268 		if (wil->srings[i].va)
269 			wil_print_sring(s, wil, &wil->srings[i]);
270 
271 	return 0;
272 }
273 DEFINE_SHOW_ATTRIBUTE(srings);
274 
wil_seq_hexdump(struct seq_file * s,void * p,int len,const char * prefix)275 static void wil_seq_hexdump(struct seq_file *s, void *p, int len,
276 			    const char *prefix)
277 {
278 	seq_hex_dump(s, prefix, DUMP_PREFIX_NONE, 16, 1, p, len, false);
279 }
280 
wil_print_mbox_ring(struct seq_file * s,const char * prefix,void __iomem * off)281 static void wil_print_mbox_ring(struct seq_file *s, const char *prefix,
282 				void __iomem *off)
283 {
284 	struct wil6210_priv *wil = s->private;
285 	struct wil6210_mbox_ring r;
286 	int rsize;
287 	uint i;
288 
289 	wil_halp_vote(wil);
290 
291 	if (wil_mem_access_lock(wil)) {
292 		wil_halp_unvote(wil);
293 		return;
294 	}
295 
296 	wil_memcpy_fromio_32(&r, off, sizeof(r));
297 	wil_mbox_ring_le2cpus(&r);
298 	/*
299 	 * we just read memory block from NIC. This memory may be
300 	 * garbage. Check validity before using it.
301 	 */
302 	rsize = r.size / sizeof(struct wil6210_mbox_ring_desc);
303 
304 	seq_printf(s, "ring %s = {\n", prefix);
305 	seq_printf(s, "  base = 0x%08x\n", r.base);
306 	seq_printf(s, "  size = 0x%04x bytes -> %d entries\n", r.size, rsize);
307 	seq_printf(s, "  tail = 0x%08x\n", r.tail);
308 	seq_printf(s, "  head = 0x%08x\n", r.head);
309 	seq_printf(s, "  entry size = %d\n", r.entry_size);
310 
311 	if (r.size % sizeof(struct wil6210_mbox_ring_desc)) {
312 		seq_printf(s, "  ??? size is not multiple of %zd, garbage?\n",
313 			   sizeof(struct wil6210_mbox_ring_desc));
314 		goto out;
315 	}
316 
317 	if (!wmi_addr(wil, r.base) ||
318 	    !wmi_addr(wil, r.tail) ||
319 	    !wmi_addr(wil, r.head)) {
320 		seq_puts(s, "  ??? pointers are garbage?\n");
321 		goto out;
322 	}
323 
324 	for (i = 0; i < rsize; i++) {
325 		struct wil6210_mbox_ring_desc d;
326 		struct wil6210_mbox_hdr hdr;
327 		size_t delta = i * sizeof(d);
328 		void __iomem *x = wil->csr + HOSTADDR(r.base) + delta;
329 
330 		wil_memcpy_fromio_32(&d, x, sizeof(d));
331 
332 		seq_printf(s, "  [%2x] %s %s%s 0x%08x", i,
333 			   d.sync ? "F" : "E",
334 			   (r.tail - r.base == delta) ? "t" : " ",
335 			   (r.head - r.base == delta) ? "h" : " ",
336 			   le32_to_cpu(d.addr));
337 		if (0 == wmi_read_hdr(wil, d.addr, &hdr)) {
338 			u16 len = le16_to_cpu(hdr.len);
339 
340 			seq_printf(s, " -> %04x %04x %04x %02x\n",
341 				   le16_to_cpu(hdr.seq), len,
342 				   le16_to_cpu(hdr.type), hdr.flags);
343 			if (len <= MAX_MBOXITEM_SIZE) {
344 				unsigned char databuf[MAX_MBOXITEM_SIZE];
345 				void __iomem *src = wmi_buffer(wil, d.addr) +
346 					sizeof(struct wil6210_mbox_hdr);
347 				/*
348 				 * No need to check @src for validity -
349 				 * we already validated @d.addr while
350 				 * reading header
351 				 */
352 				wil_memcpy_fromio_32(databuf, src, len);
353 				wil_seq_hexdump(s, databuf, len, "      : ");
354 			}
355 		} else {
356 			seq_puts(s, "\n");
357 		}
358 	}
359  out:
360 	seq_puts(s, "}\n");
361 	wil_mem_access_unlock(wil);
362 	wil_halp_unvote(wil);
363 }
364 
mbox_show(struct seq_file * s,void * data)365 static int mbox_show(struct seq_file *s, void *data)
366 {
367 	struct wil6210_priv *wil = s->private;
368 	int ret;
369 
370 	ret = wil_pm_runtime_get(wil);
371 	if (ret < 0)
372 		return ret;
373 
374 	wil_print_mbox_ring(s, "tx", wil->csr + HOST_MBOX +
375 		       offsetof(struct wil6210_mbox_ctl, tx));
376 	wil_print_mbox_ring(s, "rx", wil->csr + HOST_MBOX +
377 		       offsetof(struct wil6210_mbox_ctl, rx));
378 
379 	wil_pm_runtime_put(wil);
380 
381 	return 0;
382 }
383 DEFINE_SHOW_ATTRIBUTE(mbox);
384 
wil_debugfs_iomem_x32_set(void * data,u64 val)385 static int wil_debugfs_iomem_x32_set(void *data, u64 val)
386 {
387 	struct wil_debugfs_iomem_data *d = (struct
388 					    wil_debugfs_iomem_data *)data;
389 	struct wil6210_priv *wil = d->wil;
390 	int ret;
391 
392 	ret = wil_pm_runtime_get(wil);
393 	if (ret < 0)
394 		return ret;
395 
396 	writel_relaxed(val, (void __iomem *)d->offset);
397 
398 	wmb(); /* make sure write propagated to HW */
399 
400 	wil_pm_runtime_put(wil);
401 
402 	return 0;
403 }
404 
wil_debugfs_iomem_x32_get(void * data,u64 * val)405 static int wil_debugfs_iomem_x32_get(void *data, u64 *val)
406 {
407 	struct wil_debugfs_iomem_data *d = (struct
408 					    wil_debugfs_iomem_data *)data;
409 	struct wil6210_priv *wil = d->wil;
410 	int ret;
411 
412 	ret = wil_pm_runtime_get(wil);
413 	if (ret < 0)
414 		return ret;
415 
416 	*val = readl((void __iomem *)d->offset);
417 
418 	wil_pm_runtime_put(wil);
419 
420 	return 0;
421 }
422 
423 DEFINE_DEBUGFS_ATTRIBUTE(fops_iomem_x32, wil_debugfs_iomem_x32_get,
424 			 wil_debugfs_iomem_x32_set, "0x%08llx\n");
425 
wil_debugfs_create_iomem_x32(const char * name,umode_t mode,struct dentry * parent,void * value,struct wil6210_priv * wil)426 static void wil_debugfs_create_iomem_x32(const char *name, umode_t mode,
427 					 struct dentry *parent, void *value,
428 					 struct wil6210_priv *wil)
429 {
430 	struct wil_debugfs_iomem_data *data = &wil->dbg_data.data_arr[
431 					      wil->dbg_data.iomem_data_count];
432 
433 	data->wil = wil;
434 	data->offset = value;
435 
436 	debugfs_create_file_unsafe(name, mode, parent, data, &fops_iomem_x32);
437 	wil->dbg_data.iomem_data_count++;
438 }
439 
wil_debugfs_ulong_set(void * data,u64 val)440 static int wil_debugfs_ulong_set(void *data, u64 val)
441 {
442 	*(ulong *)data = val;
443 	return 0;
444 }
445 
wil_debugfs_ulong_get(void * data,u64 * val)446 static int wil_debugfs_ulong_get(void *data, u64 *val)
447 {
448 	*val = *(ulong *)data;
449 	return 0;
450 }
451 
452 DEFINE_DEBUGFS_ATTRIBUTE(wil_fops_ulong, wil_debugfs_ulong_get,
453 			 wil_debugfs_ulong_set, "0x%llx\n");
454 
455 /**
456  * wil6210_debugfs_init_offset - create set of debugfs files
457  * @wil - driver's context, used for printing
458  * @dbg - directory on the debugfs, where files will be created
459  * @base - base address used in address calculation
460  * @tbl - table with file descriptions. Should be terminated with empty element.
461  *
462  * Creates files accordingly to the @tbl.
463  */
wil6210_debugfs_init_offset(struct wil6210_priv * wil,struct dentry * dbg,void * base,const struct dbg_off * const tbl)464 static void wil6210_debugfs_init_offset(struct wil6210_priv *wil,
465 					struct dentry *dbg, void *base,
466 					const struct dbg_off * const tbl)
467 {
468 	int i;
469 
470 	for (i = 0; tbl[i].name; i++) {
471 		switch (tbl[i].type) {
472 		case doff_u32:
473 			debugfs_create_u32(tbl[i].name, tbl[i].mode, dbg,
474 					   base + tbl[i].off);
475 			break;
476 		case doff_x32:
477 			debugfs_create_x32(tbl[i].name, tbl[i].mode, dbg,
478 					   base + tbl[i].off);
479 			break;
480 		case doff_ulong:
481 			debugfs_create_file_unsafe(tbl[i].name, tbl[i].mode,
482 						   dbg, base + tbl[i].off,
483 						   &wil_fops_ulong);
484 			break;
485 		case doff_io32:
486 			wil_debugfs_create_iomem_x32(tbl[i].name, tbl[i].mode,
487 						     dbg, base + tbl[i].off,
488 						     wil);
489 			break;
490 		case doff_u8:
491 			debugfs_create_u8(tbl[i].name, tbl[i].mode, dbg,
492 					  base + tbl[i].off);
493 			break;
494 		}
495 	}
496 }
497 
498 static const struct dbg_off isr_off[] = {
499 	{"ICC", 0644, offsetof(struct RGF_ICR, ICC), doff_io32},
500 	{"ICR", 0644, offsetof(struct RGF_ICR, ICR), doff_io32},
501 	{"ICM", 0644, offsetof(struct RGF_ICR, ICM), doff_io32},
502 	{"ICS",	0244, offsetof(struct RGF_ICR, ICS), doff_io32},
503 	{"IMV", 0644, offsetof(struct RGF_ICR, IMV), doff_io32},
504 	{"IMS",	0244, offsetof(struct RGF_ICR, IMS), doff_io32},
505 	{"IMC",	0244, offsetof(struct RGF_ICR, IMC), doff_io32},
506 	{},
507 };
508 
wil6210_debugfs_create_ISR(struct wil6210_priv * wil,const char * name,struct dentry * parent,u32 off)509 static void wil6210_debugfs_create_ISR(struct wil6210_priv *wil,
510 				       const char *name, struct dentry *parent,
511 				       u32 off)
512 {
513 	struct dentry *d = debugfs_create_dir(name, parent);
514 
515 	wil6210_debugfs_init_offset(wil, d, (void * __force)wil->csr + off,
516 				    isr_off);
517 }
518 
519 static const struct dbg_off pseudo_isr_off[] = {
520 	{"CAUSE",   0444, HOSTADDR(RGF_DMA_PSEUDO_CAUSE), doff_io32},
521 	{"MASK_SW", 0444, HOSTADDR(RGF_DMA_PSEUDO_CAUSE_MASK_SW), doff_io32},
522 	{"MASK_FW", 0444, HOSTADDR(RGF_DMA_PSEUDO_CAUSE_MASK_FW), doff_io32},
523 	{},
524 };
525 
wil6210_debugfs_create_pseudo_ISR(struct wil6210_priv * wil,struct dentry * parent)526 static void wil6210_debugfs_create_pseudo_ISR(struct wil6210_priv *wil,
527 					      struct dentry *parent)
528 {
529 	struct dentry *d = debugfs_create_dir("PSEUDO_ISR", parent);
530 
531 	wil6210_debugfs_init_offset(wil, d, (void * __force)wil->csr,
532 				    pseudo_isr_off);
533 }
534 
535 static const struct dbg_off lgc_itr_cnt_off[] = {
536 	{"TRSH", 0644, HOSTADDR(RGF_DMA_ITR_CNT_TRSH), doff_io32},
537 	{"DATA", 0644, HOSTADDR(RGF_DMA_ITR_CNT_DATA), doff_io32},
538 	{"CTL",  0644, HOSTADDR(RGF_DMA_ITR_CNT_CRL), doff_io32},
539 	{},
540 };
541 
542 static const struct dbg_off tx_itr_cnt_off[] = {
543 	{"TRSH", 0644, HOSTADDR(RGF_DMA_ITR_TX_CNT_TRSH),
544 	 doff_io32},
545 	{"DATA", 0644, HOSTADDR(RGF_DMA_ITR_TX_CNT_DATA),
546 	 doff_io32},
547 	{"CTL",  0644, HOSTADDR(RGF_DMA_ITR_TX_CNT_CTL),
548 	 doff_io32},
549 	{"IDL_TRSH", 0644, HOSTADDR(RGF_DMA_ITR_TX_IDL_CNT_TRSH),
550 	 doff_io32},
551 	{"IDL_DATA", 0644, HOSTADDR(RGF_DMA_ITR_TX_IDL_CNT_DATA),
552 	 doff_io32},
553 	{"IDL_CTL",  0644, HOSTADDR(RGF_DMA_ITR_TX_IDL_CNT_CTL),
554 	 doff_io32},
555 	{},
556 };
557 
558 static const struct dbg_off rx_itr_cnt_off[] = {
559 	{"TRSH", 0644, HOSTADDR(RGF_DMA_ITR_RX_CNT_TRSH),
560 	 doff_io32},
561 	{"DATA", 0644, HOSTADDR(RGF_DMA_ITR_RX_CNT_DATA),
562 	 doff_io32},
563 	{"CTL",  0644, HOSTADDR(RGF_DMA_ITR_RX_CNT_CTL),
564 	 doff_io32},
565 	{"IDL_TRSH", 0644, HOSTADDR(RGF_DMA_ITR_RX_IDL_CNT_TRSH),
566 	 doff_io32},
567 	{"IDL_DATA", 0644, HOSTADDR(RGF_DMA_ITR_RX_IDL_CNT_DATA),
568 	 doff_io32},
569 	{"IDL_CTL",  0644, HOSTADDR(RGF_DMA_ITR_RX_IDL_CNT_CTL),
570 	 doff_io32},
571 	{},
572 };
573 
wil6210_debugfs_create_ITR_CNT(struct wil6210_priv * wil,struct dentry * parent)574 static int wil6210_debugfs_create_ITR_CNT(struct wil6210_priv *wil,
575 					  struct dentry *parent)
576 {
577 	struct dentry *d, *dtx, *drx;
578 
579 	d = debugfs_create_dir("ITR_CNT", parent);
580 
581 	dtx = debugfs_create_dir("TX", d);
582 	drx = debugfs_create_dir("RX", d);
583 
584 	wil6210_debugfs_init_offset(wil, d, (void * __force)wil->csr,
585 				    lgc_itr_cnt_off);
586 
587 	wil6210_debugfs_init_offset(wil, dtx, (void * __force)wil->csr,
588 				    tx_itr_cnt_off);
589 
590 	wil6210_debugfs_init_offset(wil, drx, (void * __force)wil->csr,
591 				    rx_itr_cnt_off);
592 	return 0;
593 }
594 
memread_show(struct seq_file * s,void * data)595 static int memread_show(struct seq_file *s, void *data)
596 {
597 	struct wil6210_priv *wil = s->private;
598 	void __iomem *a;
599 	int ret;
600 
601 	ret = wil_pm_runtime_get(wil);
602 	if (ret < 0)
603 		return ret;
604 
605 	ret = wil_mem_access_lock(wil);
606 	if (ret) {
607 		wil_pm_runtime_put(wil);
608 		return ret;
609 	}
610 
611 	a = wmi_buffer(wil, cpu_to_le32(mem_addr));
612 
613 	if (a)
614 		seq_printf(s, "[0x%08x] = 0x%08x\n", mem_addr, readl(a));
615 	else
616 		seq_printf(s, "[0x%08x] = INVALID\n", mem_addr);
617 
618 	wil_mem_access_unlock(wil);
619 	wil_pm_runtime_put(wil);
620 
621 	return 0;
622 }
623 DEFINE_SHOW_ATTRIBUTE(memread);
624 
wil_read_file_ioblob(struct file * file,char __user * user_buf,size_t count,loff_t * ppos)625 static ssize_t wil_read_file_ioblob(struct file *file, char __user *user_buf,
626 				    size_t count, loff_t *ppos)
627 {
628 	enum { max_count = 4096 };
629 	struct wil_blob_wrapper *wil_blob = file->private_data;
630 	struct wil6210_priv *wil = wil_blob->wil;
631 	loff_t aligned_pos, pos = *ppos;
632 	size_t available = wil_blob->blob.size;
633 	void *buf;
634 	size_t unaligned_bytes, aligned_count, ret;
635 	int rc;
636 
637 	if (pos < 0)
638 		return -EINVAL;
639 
640 	if (pos >= available || !count)
641 		return 0;
642 
643 	if (count > available - pos)
644 		count = available - pos;
645 	if (count > max_count)
646 		count = max_count;
647 
648 	/* set pos to 4 bytes aligned */
649 	unaligned_bytes = pos % 4;
650 	aligned_pos = pos - unaligned_bytes;
651 	aligned_count = count + unaligned_bytes;
652 
653 	buf = kmalloc(aligned_count, GFP_KERNEL);
654 	if (!buf)
655 		return -ENOMEM;
656 
657 	rc = wil_pm_runtime_get(wil);
658 	if (rc < 0) {
659 		kfree(buf);
660 		return rc;
661 	}
662 
663 	rc = wil_mem_access_lock(wil);
664 	if (rc) {
665 		kfree(buf);
666 		wil_pm_runtime_put(wil);
667 		return rc;
668 	}
669 
670 	wil_memcpy_fromio_32(buf, (const void __iomem *)
671 			     wil_blob->blob.data + aligned_pos, aligned_count);
672 
673 	ret = copy_to_user(user_buf, buf + unaligned_bytes, count);
674 
675 	wil_mem_access_unlock(wil);
676 	wil_pm_runtime_put(wil);
677 
678 	kfree(buf);
679 	if (ret == count)
680 		return -EFAULT;
681 
682 	count -= ret;
683 	*ppos = pos + count;
684 
685 	return count;
686 }
687 
688 static const struct file_operations fops_ioblob = {
689 	.read =		wil_read_file_ioblob,
690 	.open =		simple_open,
691 	.llseek =	default_llseek,
692 };
693 
694 static
wil_debugfs_create_ioblob(const char * name,umode_t mode,struct dentry * parent,struct wil_blob_wrapper * wil_blob)695 struct dentry *wil_debugfs_create_ioblob(const char *name,
696 					 umode_t mode,
697 					 struct dentry *parent,
698 					 struct wil_blob_wrapper *wil_blob)
699 {
700 	return debugfs_create_file(name, mode, parent, wil_blob, &fops_ioblob);
701 }
702 
703 /*---write channel 1..4 to rxon for it, 0 to rxoff---*/
wil_write_file_rxon(struct file * file,const char __user * buf,size_t len,loff_t * ppos)704 static ssize_t wil_write_file_rxon(struct file *file, const char __user *buf,
705 				   size_t len, loff_t *ppos)
706 {
707 	struct wil6210_priv *wil = file->private_data;
708 	int rc;
709 	long channel;
710 	bool on;
711 
712 	char *kbuf = memdup_user_nul(buf, len);
713 
714 	if (IS_ERR(kbuf))
715 		return PTR_ERR(kbuf);
716 	rc = kstrtol(kbuf, 0, &channel);
717 	kfree(kbuf);
718 	if (rc)
719 		return rc;
720 
721 	if ((channel < 0) || (channel > 4)) {
722 		wil_err(wil, "Invalid channel %ld\n", channel);
723 		return -EINVAL;
724 	}
725 	on = !!channel;
726 
727 	if (on) {
728 		rc = wmi_set_channel(wil, (int)channel);
729 		if (rc)
730 			return rc;
731 	}
732 
733 	rc = wmi_rxon(wil, on);
734 	if (rc)
735 		return rc;
736 
737 	return len;
738 }
739 
740 static const struct file_operations fops_rxon = {
741 	.write = wil_write_file_rxon,
742 	.open  = simple_open,
743 };
744 
wil_write_file_rbufcap(struct file * file,const char __user * buf,size_t count,loff_t * ppos)745 static ssize_t wil_write_file_rbufcap(struct file *file,
746 				      const char __user *buf,
747 				      size_t count, loff_t *ppos)
748 {
749 	struct wil6210_priv *wil = file->private_data;
750 	int val;
751 	int rc;
752 
753 	rc = kstrtoint_from_user(buf, count, 0, &val);
754 	if (rc) {
755 		wil_err(wil, "Invalid argument\n");
756 		return rc;
757 	}
758 	/* input value: negative to disable, 0 to use system default,
759 	 * 1..ring size to set descriptor threshold
760 	 */
761 	wil_info(wil, "%s RBUFCAP, descriptors threshold - %d\n",
762 		 val < 0 ? "Disabling" : "Enabling", val);
763 
764 	if (!wil->ring_rx.va || val > wil->ring_rx.size) {
765 		wil_err(wil, "Invalid descriptors threshold, %d\n", val);
766 		return -EINVAL;
767 	}
768 
769 	rc = wmi_rbufcap_cfg(wil, val < 0 ? 0 : 1, val < 0 ? 0 : val);
770 	if (rc) {
771 		wil_err(wil, "RBUFCAP config failed: %d\n", rc);
772 		return rc;
773 	}
774 
775 	return count;
776 }
777 
778 static const struct file_operations fops_rbufcap = {
779 	.write = wil_write_file_rbufcap,
780 	.open  = simple_open,
781 };
782 
783 /* block ack control, write:
784  * - "add <ringid> <agg_size> <timeout>" to trigger ADDBA
785  * - "del_tx <ringid> <reason>" to trigger DELBA for Tx side
786  * - "del_rx <CID> <TID> <reason>" to trigger DELBA for Rx side
787  */
wil_write_back(struct file * file,const char __user * buf,size_t len,loff_t * ppos)788 static ssize_t wil_write_back(struct file *file, const char __user *buf,
789 			      size_t len, loff_t *ppos)
790 {
791 	struct wil6210_priv *wil = file->private_data;
792 	int rc;
793 	char *kbuf = kmalloc(len + 1, GFP_KERNEL);
794 	char cmd[9];
795 	int p1, p2, p3;
796 
797 	if (!kbuf)
798 		return -ENOMEM;
799 
800 	rc = simple_write_to_buffer(kbuf, len, ppos, buf, len);
801 	if (rc != len) {
802 		kfree(kbuf);
803 		return rc >= 0 ? -EIO : rc;
804 	}
805 
806 	kbuf[len] = '\0';
807 	rc = sscanf(kbuf, "%8s %d %d %d", cmd, &p1, &p2, &p3);
808 	kfree(kbuf);
809 
810 	if (rc < 0)
811 		return rc;
812 	if (rc < 2)
813 		return -EINVAL;
814 
815 	if ((strcmp(cmd, "add") == 0) ||
816 	    (strcmp(cmd, "del_tx") == 0)) {
817 		struct wil_ring_tx_data *txdata;
818 
819 		if (p1 < 0 || p1 >= WIL6210_MAX_TX_RINGS) {
820 			wil_err(wil, "BACK: invalid ring id %d\n", p1);
821 			return -EINVAL;
822 		}
823 		txdata = &wil->ring_tx_data[p1];
824 		if (strcmp(cmd, "add") == 0) {
825 			if (rc < 3) {
826 				wil_err(wil, "BACK: add require at least 2 params\n");
827 				return -EINVAL;
828 			}
829 			if (rc < 4)
830 				p3 = 0;
831 			wmi_addba(wil, txdata->mid, p1, p2, p3);
832 		} else {
833 			if (rc < 3)
834 				p2 = WLAN_REASON_QSTA_LEAVE_QBSS;
835 			wmi_delba_tx(wil, txdata->mid, p1, p2);
836 		}
837 	} else if (strcmp(cmd, "del_rx") == 0) {
838 		struct wil_sta_info *sta;
839 
840 		if (rc < 3) {
841 			wil_err(wil,
842 				"BACK: del_rx require at least 2 params\n");
843 			return -EINVAL;
844 		}
845 		if (p1 < 0 || p1 >= wil->max_assoc_sta) {
846 			wil_err(wil, "BACK: invalid CID %d\n", p1);
847 			return -EINVAL;
848 		}
849 		if (rc < 4)
850 			p3 = WLAN_REASON_QSTA_LEAVE_QBSS;
851 		sta = &wil->sta[p1];
852 		wmi_delba_rx(wil, sta->mid, p1, p2, p3);
853 	} else {
854 		wil_err(wil, "BACK: Unrecognized command \"%s\"\n", cmd);
855 		return -EINVAL;
856 	}
857 
858 	return len;
859 }
860 
wil_read_back(struct file * file,char __user * user_buf,size_t count,loff_t * ppos)861 static ssize_t wil_read_back(struct file *file, char __user *user_buf,
862 			     size_t count, loff_t *ppos)
863 {
864 	static const char text[] = "block ack control, write:\n"
865 	" - \"add <ringid> <agg_size> <timeout>\" to trigger ADDBA\n"
866 	"If missing, <timeout> defaults to 0\n"
867 	" - \"del_tx <ringid> <reason>\" to trigger DELBA for Tx side\n"
868 	" - \"del_rx <CID> <TID> <reason>\" to trigger DELBA for Rx side\n"
869 	"If missing, <reason> set to \"STA_LEAVING\" (36)\n";
870 
871 	return simple_read_from_buffer(user_buf, count, ppos, text,
872 				       sizeof(text));
873 }
874 
875 static const struct file_operations fops_back = {
876 	.read = wil_read_back,
877 	.write = wil_write_back,
878 	.open  = simple_open,
879 };
880 
881 /* pmc control, write:
882  * - "alloc <num descriptors> <descriptor_size>" to allocate PMC
883  * - "free" to release memory allocated for PMC
884  */
wil_write_pmccfg(struct file * file,const char __user * buf,size_t len,loff_t * ppos)885 static ssize_t wil_write_pmccfg(struct file *file, const char __user *buf,
886 				size_t len, loff_t *ppos)
887 {
888 	struct wil6210_priv *wil = file->private_data;
889 	int rc;
890 	char *kbuf = kmalloc(len + 1, GFP_KERNEL);
891 	char cmd[9];
892 	int num_descs, desc_size;
893 
894 	if (!kbuf)
895 		return -ENOMEM;
896 
897 	rc = simple_write_to_buffer(kbuf, len, ppos, buf, len);
898 	if (rc != len) {
899 		kfree(kbuf);
900 		return rc >= 0 ? -EIO : rc;
901 	}
902 
903 	kbuf[len] = '\0';
904 	rc = sscanf(kbuf, "%8s %d %d", cmd, &num_descs, &desc_size);
905 	kfree(kbuf);
906 
907 	if (rc < 0)
908 		return rc;
909 
910 	if (rc < 1) {
911 		wil_err(wil, "pmccfg: no params given\n");
912 		return -EINVAL;
913 	}
914 
915 	if (0 == strcmp(cmd, "alloc")) {
916 		if (rc != 3) {
917 			wil_err(wil, "pmccfg: alloc requires 2 params\n");
918 			return -EINVAL;
919 		}
920 		wil_pmc_alloc(wil, num_descs, desc_size);
921 	} else if (0 == strcmp(cmd, "free")) {
922 		if (rc != 1) {
923 			wil_err(wil, "pmccfg: free does not have any params\n");
924 			return -EINVAL;
925 		}
926 		wil_pmc_free(wil, true);
927 	} else {
928 		wil_err(wil, "pmccfg: Unrecognized command \"%s\"\n", cmd);
929 		return -EINVAL;
930 	}
931 
932 	return len;
933 }
934 
wil_read_pmccfg(struct file * file,char __user * user_buf,size_t count,loff_t * ppos)935 static ssize_t wil_read_pmccfg(struct file *file, char __user *user_buf,
936 			       size_t count, loff_t *ppos)
937 {
938 	struct wil6210_priv *wil = file->private_data;
939 	char text[256];
940 	char help[] = "pmc control, write:\n"
941 	" - \"alloc <num descriptors> <descriptor_size>\" to allocate pmc\n"
942 	" - \"free\" to free memory allocated for pmc\n";
943 
944 	snprintf(text, sizeof(text), "Last command status: %d\n\n%s",
945 		 wil_pmc_last_cmd_status(wil), help);
946 
947 	return simple_read_from_buffer(user_buf, count, ppos, text,
948 				       strlen(text) + 1);
949 }
950 
951 static const struct file_operations fops_pmccfg = {
952 	.read = wil_read_pmccfg,
953 	.write = wil_write_pmccfg,
954 	.open  = simple_open,
955 };
956 
957 static const struct file_operations fops_pmcdata = {
958 	.open		= simple_open,
959 	.read		= wil_pmc_read,
960 	.llseek		= wil_pmc_llseek,
961 };
962 
wil_pmcring_seq_open(struct inode * inode,struct file * file)963 static int wil_pmcring_seq_open(struct inode *inode, struct file *file)
964 {
965 	return single_open(file, wil_pmcring_read, inode->i_private);
966 }
967 
968 static const struct file_operations fops_pmcring = {
969 	.open		= wil_pmcring_seq_open,
970 	.release	= single_release,
971 	.read		= seq_read,
972 	.llseek		= seq_lseek,
973 };
974 
975 /*---tx_mgmt---*/
976 /* Write mgmt frame to this file to send it */
wil_write_file_txmgmt(struct file * file,const char __user * buf,size_t len,loff_t * ppos)977 static ssize_t wil_write_file_txmgmt(struct file *file, const char __user *buf,
978 				     size_t len, loff_t *ppos)
979 {
980 	struct wil6210_priv *wil = file->private_data;
981 	struct wiphy *wiphy = wil_to_wiphy(wil);
982 	struct wireless_dev *wdev = wil->main_ndev->ieee80211_ptr;
983 	struct cfg80211_mgmt_tx_params params;
984 	int rc;
985 	void *frame;
986 
987 	memset(&params, 0, sizeof(params));
988 
989 	if (!len)
990 		return -EINVAL;
991 
992 	frame = memdup_user(buf, len);
993 	if (IS_ERR(frame))
994 		return PTR_ERR(frame);
995 
996 	params.buf = frame;
997 	params.len = len;
998 
999 	rc = wil_cfg80211_mgmt_tx(wiphy, wdev, &params, NULL);
1000 
1001 	kfree(frame);
1002 	wil_info(wil, "-> %d\n", rc);
1003 
1004 	return len;
1005 }
1006 
1007 static const struct file_operations fops_txmgmt = {
1008 	.write = wil_write_file_txmgmt,
1009 	.open  = simple_open,
1010 };
1011 
1012 /* Write WMI command (w/o mbox header) to this file to send it
1013  * WMI starts from wil6210_mbox_hdr_wmi header
1014  */
wil_write_file_wmi(struct file * file,const char __user * buf,size_t len,loff_t * ppos)1015 static ssize_t wil_write_file_wmi(struct file *file, const char __user *buf,
1016 				  size_t len, loff_t *ppos)
1017 {
1018 	struct wil6210_priv *wil = file->private_data;
1019 	struct wil6210_vif *vif = ndev_to_vif(wil->main_ndev);
1020 	struct wmi_cmd_hdr *wmi;
1021 	void *cmd;
1022 	int cmdlen = len - sizeof(struct wmi_cmd_hdr);
1023 	u16 cmdid;
1024 	int rc, rc1;
1025 
1026 	if (cmdlen < 0)
1027 		return -EINVAL;
1028 
1029 	wmi = kmalloc(len, GFP_KERNEL);
1030 	if (!wmi)
1031 		return -ENOMEM;
1032 
1033 	rc = simple_write_to_buffer(wmi, len, ppos, buf, len);
1034 	if (rc < 0) {
1035 		kfree(wmi);
1036 		return rc;
1037 	}
1038 
1039 	cmd = (cmdlen > 0) ? &wmi[1] : NULL;
1040 	cmdid = le16_to_cpu(wmi->command_id);
1041 
1042 	rc1 = wmi_send(wil, cmdid, vif->mid, cmd, cmdlen);
1043 	kfree(wmi);
1044 
1045 	wil_info(wil, "0x%04x[%d] -> %d\n", cmdid, cmdlen, rc1);
1046 
1047 	return rc;
1048 }
1049 
1050 static const struct file_operations fops_wmi = {
1051 	.write = wil_write_file_wmi,
1052 	.open  = simple_open,
1053 };
1054 
wil_seq_print_skb(struct seq_file * s,struct sk_buff * skb)1055 static void wil_seq_print_skb(struct seq_file *s, struct sk_buff *skb)
1056 {
1057 	int i = 0;
1058 	int len = skb_headlen(skb);
1059 	void *p = skb->data;
1060 	int nr_frags = skb_shinfo(skb)->nr_frags;
1061 
1062 	seq_printf(s, "    len = %d\n", len);
1063 	wil_seq_hexdump(s, p, len, "      : ");
1064 
1065 	if (nr_frags) {
1066 		seq_printf(s, "    nr_frags = %d\n", nr_frags);
1067 		for (i = 0; i < nr_frags; i++) {
1068 			const skb_frag_t *frag = &skb_shinfo(skb)->frags[i];
1069 
1070 			len = skb_frag_size(frag);
1071 			p = skb_frag_address_safe(frag);
1072 			seq_printf(s, "    [%2d] : len = %d\n", i, len);
1073 			wil_seq_hexdump(s, p, len, "      : ");
1074 		}
1075 	}
1076 }
1077 
1078 /*---------Tx/Rx descriptor------------*/
txdesc_show(struct seq_file * s,void * data)1079 static int txdesc_show(struct seq_file *s, void *data)
1080 {
1081 	struct wil6210_priv *wil = s->private;
1082 	struct wil_ring *ring;
1083 	bool tx;
1084 	int ring_idx = dbg_ring_index;
1085 	int txdesc_idx = dbg_txdesc_index;
1086 	volatile struct vring_tx_desc *d;
1087 	volatile u32 *u;
1088 	struct sk_buff *skb;
1089 
1090 	if (wil->use_enhanced_dma_hw) {
1091 		/* RX ring index == 0 */
1092 		if (ring_idx >= WIL6210_MAX_TX_RINGS) {
1093 			seq_printf(s, "invalid ring index %d\n", ring_idx);
1094 			return 0;
1095 		}
1096 		tx = ring_idx > 0; /* desc ring 0 is reserved for RX */
1097 	} else {
1098 		/* RX ring index == WIL6210_MAX_TX_RINGS */
1099 		if (ring_idx > WIL6210_MAX_TX_RINGS) {
1100 			seq_printf(s, "invalid ring index %d\n", ring_idx);
1101 			return 0;
1102 		}
1103 		tx = (ring_idx < WIL6210_MAX_TX_RINGS);
1104 	}
1105 
1106 	ring = tx ? &wil->ring_tx[ring_idx] : &wil->ring_rx;
1107 
1108 	if (!ring->va) {
1109 		if (tx)
1110 			seq_printf(s, "No Tx[%2d] RING\n", ring_idx);
1111 		else
1112 			seq_puts(s, "No Rx RING\n");
1113 		return 0;
1114 	}
1115 
1116 	if (txdesc_idx >= ring->size) {
1117 		if (tx)
1118 			seq_printf(s, "[%2d] TxDesc index (%d) >= size (%d)\n",
1119 				   ring_idx, txdesc_idx, ring->size);
1120 		else
1121 			seq_printf(s, "RxDesc index (%d) >= size (%d)\n",
1122 				   txdesc_idx, ring->size);
1123 		return 0;
1124 	}
1125 
1126 	/* use struct vring_tx_desc for Rx as well,
1127 	 * only field used, .dma.length, is the same
1128 	 */
1129 	d = &ring->va[txdesc_idx].tx.legacy;
1130 	u = (volatile u32 *)d;
1131 	skb = NULL;
1132 
1133 	if (wil->use_enhanced_dma_hw) {
1134 		if (tx) {
1135 			skb = ring->ctx ? ring->ctx[txdesc_idx].skb : NULL;
1136 		} else if (wil->rx_buff_mgmt.buff_arr) {
1137 			struct wil_rx_enhanced_desc *rx_d =
1138 				(struct wil_rx_enhanced_desc *)
1139 				&ring->va[txdesc_idx].rx.enhanced;
1140 			u16 buff_id = le16_to_cpu(rx_d->mac.buff_id);
1141 
1142 			if (!wil_val_in_range(buff_id, 0,
1143 					      wil->rx_buff_mgmt.size))
1144 				seq_printf(s, "invalid buff_id %d\n", buff_id);
1145 			else
1146 				skb = wil->rx_buff_mgmt.buff_arr[buff_id].skb;
1147 		}
1148 	} else {
1149 		skb = ring->ctx[txdesc_idx].skb;
1150 	}
1151 	if (tx)
1152 		seq_printf(s, "Tx[%2d][%3d] = {\n", ring_idx,
1153 			   txdesc_idx);
1154 	else
1155 		seq_printf(s, "Rx[%3d] = {\n", txdesc_idx);
1156 	seq_printf(s, "  MAC = 0x%08x 0x%08x 0x%08x 0x%08x\n",
1157 		   u[0], u[1], u[2], u[3]);
1158 	seq_printf(s, "  DMA = 0x%08x 0x%08x 0x%08x 0x%08x\n",
1159 		   u[4], u[5], u[6], u[7]);
1160 	seq_printf(s, "  SKB = 0x%p\n", skb);
1161 
1162 	if (skb) {
1163 		skb_get(skb);
1164 		wil_seq_print_skb(s, skb);
1165 		kfree_skb(skb);
1166 	}
1167 	seq_puts(s, "}\n");
1168 
1169 	return 0;
1170 }
1171 DEFINE_SHOW_ATTRIBUTE(txdesc);
1172 
1173 /*---------Tx/Rx status message------------*/
status_msg_show(struct seq_file * s,void * data)1174 static int status_msg_show(struct seq_file *s, void *data)
1175 {
1176 	struct wil6210_priv *wil = s->private;
1177 	int sring_idx = dbg_sring_index;
1178 	struct wil_status_ring *sring;
1179 	bool tx;
1180 	u32 status_msg_idx = dbg_status_msg_index;
1181 	u32 *u;
1182 
1183 	if (sring_idx >= WIL6210_MAX_STATUS_RINGS) {
1184 		seq_printf(s, "invalid status ring index %d\n", sring_idx);
1185 		return 0;
1186 	}
1187 
1188 	sring = &wil->srings[sring_idx];
1189 	tx = !sring->is_rx;
1190 
1191 	if (!sring->va) {
1192 		seq_printf(s, "No %cX status ring\n", tx ? 'T' : 'R');
1193 		return 0;
1194 	}
1195 
1196 	if (status_msg_idx >= sring->size) {
1197 		seq_printf(s, "%cxDesc index (%d) >= size (%d)\n",
1198 			   tx ? 'T' : 'R', status_msg_idx, sring->size);
1199 		return 0;
1200 	}
1201 
1202 	u = sring->va + (sring->elem_size * status_msg_idx);
1203 
1204 	seq_printf(s, "%cx[%d][%3d] = {\n",
1205 		   tx ? 'T' : 'R', sring_idx, status_msg_idx);
1206 
1207 	seq_printf(s, "  0x%08x 0x%08x 0x%08x 0x%08x\n",
1208 		   u[0], u[1], u[2], u[3]);
1209 	if (!tx && !wil->use_compressed_rx_status)
1210 		seq_printf(s, "  0x%08x 0x%08x 0x%08x 0x%08x\n",
1211 			   u[4], u[5], u[6], u[7]);
1212 
1213 	seq_puts(s, "}\n");
1214 
1215 	return 0;
1216 }
1217 DEFINE_SHOW_ATTRIBUTE(status_msg);
1218 
wil_print_rx_buff(struct seq_file * s,struct list_head * lh)1219 static int wil_print_rx_buff(struct seq_file *s, struct list_head *lh)
1220 {
1221 	struct wil_rx_buff *it;
1222 	int i = 0;
1223 
1224 	list_for_each_entry(it, lh, list) {
1225 		if ((i % 16) == 0 && i != 0)
1226 			seq_puts(s, "\n    ");
1227 		seq_printf(s, "[%4d] ", it->id);
1228 		i++;
1229 	}
1230 	seq_printf(s, "\nNumber of buffers: %u\n", i);
1231 
1232 	return i;
1233 }
1234 
rx_buff_mgmt_show(struct seq_file * s,void * data)1235 static int rx_buff_mgmt_show(struct seq_file *s, void *data)
1236 {
1237 	struct wil6210_priv *wil = s->private;
1238 	struct wil_rx_buff_mgmt *rbm = &wil->rx_buff_mgmt;
1239 	int num_active;
1240 	int num_free;
1241 
1242 	if (!rbm->buff_arr)
1243 		return -EINVAL;
1244 
1245 	seq_printf(s, "  size = %zu\n", rbm->size);
1246 	seq_printf(s, "  free_list_empty_cnt = %lu\n",
1247 		   rbm->free_list_empty_cnt);
1248 
1249 	/* Print active list */
1250 	seq_puts(s, "  Active list:\n");
1251 	num_active = wil_print_rx_buff(s, &rbm->active);
1252 	seq_puts(s, "\n  Free list:\n");
1253 	num_free = wil_print_rx_buff(s, &rbm->free);
1254 
1255 	seq_printf(s, "  Total number of buffers: %u\n",
1256 		   num_active + num_free);
1257 
1258 	return 0;
1259 }
1260 DEFINE_SHOW_ATTRIBUTE(rx_buff_mgmt);
1261 
1262 /*---------beamforming------------*/
wil_bfstatus_str(u32 status)1263 static char *wil_bfstatus_str(u32 status)
1264 {
1265 	switch (status) {
1266 	case 0:
1267 		return "Failed";
1268 	case 1:
1269 		return "OK";
1270 	case 2:
1271 		return "Retrying";
1272 	default:
1273 		return "??";
1274 	}
1275 }
1276 
is_all_zeros(void * const x_,size_t sz)1277 static bool is_all_zeros(void * const x_, size_t sz)
1278 {
1279 	/* if reply is all-0, ignore this CID */
1280 	u32 *x = x_;
1281 	int n;
1282 
1283 	for (n = 0; n < sz / sizeof(*x); n++)
1284 		if (x[n])
1285 			return false;
1286 
1287 	return true;
1288 }
1289 
bf_show(struct seq_file * s,void * data)1290 static int bf_show(struct seq_file *s, void *data)
1291 {
1292 	int rc;
1293 	int i;
1294 	struct wil6210_priv *wil = s->private;
1295 	struct wil6210_vif *vif = ndev_to_vif(wil->main_ndev);
1296 	struct wmi_notify_req_cmd cmd = {
1297 		.interval_usec = 0,
1298 	};
1299 	struct {
1300 		struct wmi_cmd_hdr wmi;
1301 		struct wmi_notify_req_done_event evt;
1302 	} __packed reply;
1303 
1304 	memset(&reply, 0, sizeof(reply));
1305 
1306 	for (i = 0; i < wil->max_assoc_sta; i++) {
1307 		u32 status;
1308 
1309 		cmd.cid = i;
1310 		rc = wmi_call(wil, WMI_NOTIFY_REQ_CMDID, vif->mid,
1311 			      &cmd, sizeof(cmd),
1312 			      WMI_NOTIFY_REQ_DONE_EVENTID, &reply,
1313 			      sizeof(reply), WIL_WMI_CALL_GENERAL_TO_MS);
1314 		/* if reply is all-0, ignore this CID */
1315 		if (rc || is_all_zeros(&reply.evt, sizeof(reply.evt)))
1316 			continue;
1317 
1318 		status = le32_to_cpu(reply.evt.status);
1319 		seq_printf(s, "CID %d {\n"
1320 			   "  TSF = 0x%016llx\n"
1321 			   "  TxMCS = %2d TxTpt = %4d\n"
1322 			   "  SQI = %4d\n"
1323 			   "  RSSI = %4d\n"
1324 			   "  Status = 0x%08x %s\n"
1325 			   "  Sectors(rx:tx) my %2d:%2d peer %2d:%2d\n"
1326 			   "  Goodput(rx:tx) %4d:%4d\n"
1327 			   "}\n",
1328 			   i,
1329 			   le64_to_cpu(reply.evt.tsf),
1330 			   le16_to_cpu(reply.evt.bf_mcs),
1331 			   le32_to_cpu(reply.evt.tx_tpt),
1332 			   reply.evt.sqi,
1333 			   reply.evt.rssi,
1334 			   status, wil_bfstatus_str(status),
1335 			   le16_to_cpu(reply.evt.my_rx_sector),
1336 			   le16_to_cpu(reply.evt.my_tx_sector),
1337 			   le16_to_cpu(reply.evt.other_rx_sector),
1338 			   le16_to_cpu(reply.evt.other_tx_sector),
1339 			   le32_to_cpu(reply.evt.rx_goodput),
1340 			   le32_to_cpu(reply.evt.tx_goodput));
1341 	}
1342 	return 0;
1343 }
1344 DEFINE_SHOW_ATTRIBUTE(bf);
1345 
1346 /*---------temp------------*/
print_temp(struct seq_file * s,const char * prefix,s32 t)1347 static void print_temp(struct seq_file *s, const char *prefix, s32 t)
1348 {
1349 	switch (t) {
1350 	case 0:
1351 	case WMI_INVALID_TEMPERATURE:
1352 		seq_printf(s, "%s N/A\n", prefix);
1353 	break;
1354 	default:
1355 		seq_printf(s, "%s %s%d.%03d\n", prefix, (t < 0 ? "-" : ""),
1356 			   abs(t / 1000), abs(t % 1000));
1357 		break;
1358 	}
1359 }
1360 
temp_show(struct seq_file * s,void * data)1361 static int temp_show(struct seq_file *s, void *data)
1362 {
1363 	struct wil6210_priv *wil = s->private;
1364 	int rc, i;
1365 
1366 	if (test_bit(WMI_FW_CAPABILITY_TEMPERATURE_ALL_RF,
1367 		     wil->fw_capabilities)) {
1368 		struct wmi_temp_sense_all_done_event sense_all_evt;
1369 
1370 		wil_dbg_misc(wil,
1371 			     "WMI_FW_CAPABILITY_TEMPERATURE_ALL_RF is supported");
1372 		rc = wmi_get_all_temperatures(wil, &sense_all_evt);
1373 		if (rc) {
1374 			seq_puts(s, "Failed\n");
1375 			return 0;
1376 		}
1377 		print_temp(s, "T_mac   =",
1378 			   le32_to_cpu(sense_all_evt.baseband_t1000));
1379 		seq_printf(s, "Connected RFs [0x%08x]\n",
1380 			   sense_all_evt.rf_bitmap);
1381 		for (i = 0; i < WMI_MAX_XIF_PORTS_NUM; i++) {
1382 			seq_printf(s, "RF[%d]   = ", i);
1383 			print_temp(s, "",
1384 				   le32_to_cpu(sense_all_evt.rf_t1000[i]));
1385 		}
1386 	} else {
1387 		s32 t_m, t_r;
1388 
1389 		wil_dbg_misc(wil,
1390 			     "WMI_FW_CAPABILITY_TEMPERATURE_ALL_RF is not supported");
1391 		rc = wmi_get_temperature(wil, &t_m, &t_r);
1392 		if (rc) {
1393 			seq_puts(s, "Failed\n");
1394 			return 0;
1395 		}
1396 		print_temp(s, "T_mac   =", t_m);
1397 		print_temp(s, "T_radio =", t_r);
1398 	}
1399 	return 0;
1400 }
1401 DEFINE_SHOW_ATTRIBUTE(temp);
1402 
1403 /*---------freq------------*/
freq_show(struct seq_file * s,void * data)1404 static int freq_show(struct seq_file *s, void *data)
1405 {
1406 	struct wil6210_priv *wil = s->private;
1407 	struct wireless_dev *wdev = wil->main_ndev->ieee80211_ptr;
1408 	u32 freq = wdev->chandef.chan ? wdev->chandef.chan->center_freq : 0;
1409 
1410 	seq_printf(s, "Freq = %d\n", freq);
1411 
1412 	return 0;
1413 }
1414 DEFINE_SHOW_ATTRIBUTE(freq);
1415 
1416 /*---------link------------*/
link_show(struct seq_file * s,void * data)1417 static int link_show(struct seq_file *s, void *data)
1418 {
1419 	struct wil6210_priv *wil = s->private;
1420 	struct station_info *sinfo;
1421 	int i, rc = 0;
1422 
1423 	sinfo = kzalloc(sizeof(*sinfo), GFP_KERNEL);
1424 	if (!sinfo)
1425 		return -ENOMEM;
1426 
1427 	for (i = 0; i < wil->max_assoc_sta; i++) {
1428 		struct wil_sta_info *p = &wil->sta[i];
1429 		char *status = "unknown";
1430 		struct wil6210_vif *vif;
1431 		u8 mid;
1432 
1433 		switch (p->status) {
1434 		case wil_sta_unused:
1435 			status = "unused   ";
1436 			break;
1437 		case wil_sta_conn_pending:
1438 			status = "pending  ";
1439 			break;
1440 		case wil_sta_connected:
1441 			status = "connected";
1442 			break;
1443 		}
1444 		mid = (p->status != wil_sta_unused) ? p->mid : U8_MAX;
1445 		seq_printf(s, "[%d][MID %d] %pM %s\n",
1446 			   i, mid, p->addr, status);
1447 
1448 		if (p->status != wil_sta_connected)
1449 			continue;
1450 
1451 		vif = (mid < GET_MAX_VIFS(wil)) ? wil->vifs[mid] : NULL;
1452 		if (vif) {
1453 			rc = wil_cid_fill_sinfo(vif, i, sinfo);
1454 			if (rc)
1455 				goto out;
1456 
1457 			seq_printf(s, "  Tx_mcs = %d\n", sinfo->txrate.mcs);
1458 			seq_printf(s, "  Rx_mcs = %d\n", sinfo->rxrate.mcs);
1459 			seq_printf(s, "  SQ     = %d\n", sinfo->signal);
1460 		} else {
1461 			seq_puts(s, "  INVALID MID\n");
1462 		}
1463 	}
1464 
1465 out:
1466 	kfree(sinfo);
1467 	return rc;
1468 }
1469 DEFINE_SHOW_ATTRIBUTE(link);
1470 
1471 /*---------info------------*/
info_show(struct seq_file * s,void * data)1472 static int info_show(struct seq_file *s, void *data)
1473 {
1474 	struct wil6210_priv *wil = s->private;
1475 	struct net_device *ndev = wil->main_ndev;
1476 	int is_ac = power_supply_is_system_supplied();
1477 	int rx = atomic_xchg(&wil->isr_count_rx, 0);
1478 	int tx = atomic_xchg(&wil->isr_count_tx, 0);
1479 	static ulong rxf_old, txf_old;
1480 	ulong rxf = ndev->stats.rx_packets;
1481 	ulong txf = ndev->stats.tx_packets;
1482 	unsigned int i;
1483 
1484 	/* >0 : AC; 0 : battery; <0 : error */
1485 	seq_printf(s, "AC powered : %d\n", is_ac);
1486 	seq_printf(s, "Rx irqs:packets : %8d : %8ld\n", rx, rxf - rxf_old);
1487 	seq_printf(s, "Tx irqs:packets : %8d : %8ld\n", tx, txf - txf_old);
1488 	rxf_old = rxf;
1489 	txf_old = txf;
1490 
1491 #define CHECK_QSTATE(x) (state & BIT(__QUEUE_STATE_ ## x)) ? \
1492 	" " __stringify(x) : ""
1493 
1494 	for (i = 0; i < ndev->num_tx_queues; i++) {
1495 		struct netdev_queue *txq = netdev_get_tx_queue(ndev, i);
1496 		unsigned long state = txq->state;
1497 
1498 		seq_printf(s, "Tx queue[%i] state : 0x%lx%s%s%s\n", i, state,
1499 			   CHECK_QSTATE(DRV_XOFF),
1500 			   CHECK_QSTATE(STACK_XOFF),
1501 			   CHECK_QSTATE(FROZEN)
1502 			  );
1503 	}
1504 #undef CHECK_QSTATE
1505 	return 0;
1506 }
1507 DEFINE_SHOW_ATTRIBUTE(info);
1508 
1509 /*---------recovery------------*/
1510 /* mode = [manual|auto]
1511  * state = [idle|pending|running]
1512  */
wil_read_file_recovery(struct file * file,char __user * user_buf,size_t count,loff_t * ppos)1513 static ssize_t wil_read_file_recovery(struct file *file, char __user *user_buf,
1514 				      size_t count, loff_t *ppos)
1515 {
1516 	struct wil6210_priv *wil = file->private_data;
1517 	char buf[80];
1518 	int n;
1519 	static const char * const sstate[] = {"idle", "pending", "running"};
1520 
1521 	n = snprintf(buf, sizeof(buf), "mode = %s\nstate = %s\n",
1522 		     no_fw_recovery ? "manual" : "auto",
1523 		     sstate[wil->recovery_state]);
1524 
1525 	n = min_t(int, n, sizeof(buf));
1526 
1527 	return simple_read_from_buffer(user_buf, count, ppos,
1528 				       buf, n);
1529 }
1530 
wil_write_file_recovery(struct file * file,const char __user * buf_,size_t count,loff_t * ppos)1531 static ssize_t wil_write_file_recovery(struct file *file,
1532 				       const char __user *buf_,
1533 				       size_t count, loff_t *ppos)
1534 {
1535 	struct wil6210_priv *wil = file->private_data;
1536 	static const char run_command[] = "run";
1537 	char buf[sizeof(run_command) + 1]; /* to detect "runx" */
1538 	ssize_t rc;
1539 
1540 	if (wil->recovery_state != fw_recovery_pending) {
1541 		wil_err(wil, "No recovery pending\n");
1542 		return -EINVAL;
1543 	}
1544 
1545 	if (*ppos != 0) {
1546 		wil_err(wil, "Offset [%d]\n", (int)*ppos);
1547 		return -EINVAL;
1548 	}
1549 
1550 	if (count > sizeof(buf)) {
1551 		wil_err(wil, "Input too long, len = %d\n", (int)count);
1552 		return -EINVAL;
1553 	}
1554 
1555 	rc = simple_write_to_buffer(buf, sizeof(buf) - 1, ppos, buf_, count);
1556 	if (rc < 0)
1557 		return rc;
1558 
1559 	buf[rc] = '\0';
1560 	if (0 == strcmp(buf, run_command))
1561 		wil_set_recovery_state(wil, fw_recovery_running);
1562 	else
1563 		wil_err(wil, "Bad recovery command \"%s\"\n", buf);
1564 
1565 	return rc;
1566 }
1567 
1568 static const struct file_operations fops_recovery = {
1569 	.read = wil_read_file_recovery,
1570 	.write = wil_write_file_recovery,
1571 	.open  = simple_open,
1572 };
1573 
1574 /*---------Station matrix------------*/
wil_print_rxtid(struct seq_file * s,struct wil_tid_ampdu_rx * r)1575 static void wil_print_rxtid(struct seq_file *s, struct wil_tid_ampdu_rx *r)
1576 {
1577 	int i;
1578 	u16 index = ((r->head_seq_num - r->ssn) & 0xfff) % r->buf_size;
1579 	unsigned long long drop_dup = r->drop_dup, drop_old = r->drop_old;
1580 	unsigned long long drop_dup_mcast = r->drop_dup_mcast;
1581 
1582 	seq_printf(s, "([%2d]) 0x%03x [", r->buf_size, r->head_seq_num);
1583 	for (i = 0; i < r->buf_size; i++) {
1584 		if (i == index)
1585 			seq_printf(s, "%c", r->reorder_buf[i] ? 'O' : '|');
1586 		else
1587 			seq_printf(s, "%c", r->reorder_buf[i] ? '*' : '_');
1588 	}
1589 	seq_printf(s,
1590 		   "] total %llu drop %llu (dup %llu + old %llu + dup mcast %llu) last 0x%03x\n",
1591 		   r->total, drop_dup + drop_old + drop_dup_mcast, drop_dup,
1592 		   drop_old, drop_dup_mcast, r->ssn_last_drop);
1593 }
1594 
wil_print_rxtid_crypto(struct seq_file * s,int tid,struct wil_tid_crypto_rx * c)1595 static void wil_print_rxtid_crypto(struct seq_file *s, int tid,
1596 				   struct wil_tid_crypto_rx *c)
1597 {
1598 	int i;
1599 
1600 	for (i = 0; i < 4; i++) {
1601 		struct wil_tid_crypto_rx_single *cc = &c->key_id[i];
1602 
1603 		if (cc->key_set)
1604 			goto has_keys;
1605 	}
1606 	return;
1607 
1608 has_keys:
1609 	if (tid < WIL_STA_TID_NUM)
1610 		seq_printf(s, "  [%2d] PN", tid);
1611 	else
1612 		seq_puts(s, "  [GR] PN");
1613 
1614 	for (i = 0; i < 4; i++) {
1615 		struct wil_tid_crypto_rx_single *cc = &c->key_id[i];
1616 
1617 		seq_printf(s, " [%i%s]%6phN", i, cc->key_set ? "+" : "-",
1618 			   cc->pn);
1619 	}
1620 	seq_puts(s, "\n");
1621 }
1622 
sta_show(struct seq_file * s,void * data)1623 static int sta_show(struct seq_file *s, void *data)
1624 __acquires(&p->tid_rx_lock) __releases(&p->tid_rx_lock)
1625 {
1626 	struct wil6210_priv *wil = s->private;
1627 	int i, tid, mcs;
1628 
1629 	for (i = 0; i < wil->max_assoc_sta; i++) {
1630 		struct wil_sta_info *p = &wil->sta[i];
1631 		char *status = "unknown";
1632 		u8 aid = 0;
1633 		u8 mid;
1634 		bool sta_connected = false;
1635 
1636 		switch (p->status) {
1637 		case wil_sta_unused:
1638 			status = "unused   ";
1639 			break;
1640 		case wil_sta_conn_pending:
1641 			status = "pending  ";
1642 			break;
1643 		case wil_sta_connected:
1644 			status = "connected";
1645 			aid = p->aid;
1646 			break;
1647 		}
1648 		mid = (p->status != wil_sta_unused) ? p->mid : U8_MAX;
1649 		if (mid < GET_MAX_VIFS(wil)) {
1650 			struct wil6210_vif *vif = wil->vifs[mid];
1651 
1652 			if (vif->wdev.iftype == NL80211_IFTYPE_STATION &&
1653 			    p->status == wil_sta_connected)
1654 				sta_connected = true;
1655 		}
1656 		/* print roam counter only for connected stations */
1657 		if (sta_connected)
1658 			seq_printf(s, "[%d] %pM connected (roam counter %d) MID %d AID %d\n",
1659 				   i, p->addr, p->stats.ft_roams, mid, aid);
1660 		else
1661 			seq_printf(s, "[%d] %pM %s MID %d AID %d\n", i,
1662 				   p->addr, status, mid, aid);
1663 
1664 		if (p->status == wil_sta_connected) {
1665 			spin_lock_bh(&p->tid_rx_lock);
1666 			for (tid = 0; tid < WIL_STA_TID_NUM; tid++) {
1667 				struct wil_tid_ampdu_rx *r = p->tid_rx[tid];
1668 				struct wil_tid_crypto_rx *c =
1669 						&p->tid_crypto_rx[tid];
1670 
1671 				if (r) {
1672 					seq_printf(s, "  [%2d] ", tid);
1673 					wil_print_rxtid(s, r);
1674 				}
1675 
1676 				wil_print_rxtid_crypto(s, tid, c);
1677 			}
1678 			wil_print_rxtid_crypto(s, WIL_STA_TID_NUM,
1679 					       &p->group_crypto_rx);
1680 			spin_unlock_bh(&p->tid_rx_lock);
1681 			seq_printf(s,
1682 				   "Rx invalid frame: non-data %lu, short %lu, large %lu, replay %lu\n",
1683 				   p->stats.rx_non_data_frame,
1684 				   p->stats.rx_short_frame,
1685 				   p->stats.rx_large_frame,
1686 				   p->stats.rx_replay);
1687 			seq_printf(s,
1688 				   "mic error %lu, key error %lu, amsdu error %lu, csum error %lu\n",
1689 				   p->stats.rx_mic_error,
1690 				   p->stats.rx_key_error,
1691 				   p->stats.rx_amsdu_error,
1692 				   p->stats.rx_csum_err);
1693 
1694 			seq_puts(s, "Rx/MCS:");
1695 			for (mcs = 0; mcs < ARRAY_SIZE(p->stats.rx_per_mcs);
1696 			     mcs++)
1697 				seq_printf(s, " %lld",
1698 					   p->stats.rx_per_mcs[mcs]);
1699 			seq_puts(s, "\n");
1700 		}
1701 	}
1702 
1703 	return 0;
1704 }
1705 DEFINE_SHOW_ATTRIBUTE(sta);
1706 
mids_show(struct seq_file * s,void * data)1707 static int mids_show(struct seq_file *s, void *data)
1708 {
1709 	struct wil6210_priv *wil = s->private;
1710 	struct wil6210_vif *vif;
1711 	struct net_device *ndev;
1712 	int i;
1713 
1714 	mutex_lock(&wil->vif_mutex);
1715 	for (i = 0; i < GET_MAX_VIFS(wil); i++) {
1716 		vif = wil->vifs[i];
1717 
1718 		if (vif) {
1719 			ndev = vif_to_ndev(vif);
1720 			seq_printf(s, "[%d] %pM %s\n", i, ndev->dev_addr,
1721 				   ndev->name);
1722 		} else {
1723 			seq_printf(s, "[%d] unused\n", i);
1724 		}
1725 	}
1726 	mutex_unlock(&wil->vif_mutex);
1727 
1728 	return 0;
1729 }
1730 DEFINE_SHOW_ATTRIBUTE(mids);
1731 
wil_tx_latency_debugfs_show(struct seq_file * s,void * data)1732 static int wil_tx_latency_debugfs_show(struct seq_file *s, void *data)
1733 __acquires(&p->tid_rx_lock) __releases(&p->tid_rx_lock)
1734 {
1735 	struct wil6210_priv *wil = s->private;
1736 	int i, bin;
1737 
1738 	for (i = 0; i < wil->max_assoc_sta; i++) {
1739 		struct wil_sta_info *p = &wil->sta[i];
1740 		char *status = "unknown";
1741 		u8 aid = 0;
1742 		u8 mid;
1743 
1744 		if (!p->tx_latency_bins)
1745 			continue;
1746 
1747 		switch (p->status) {
1748 		case wil_sta_unused:
1749 			status = "unused   ";
1750 			break;
1751 		case wil_sta_conn_pending:
1752 			status = "pending  ";
1753 			break;
1754 		case wil_sta_connected:
1755 			status = "connected";
1756 			aid = p->aid;
1757 			break;
1758 		}
1759 		mid = (p->status != wil_sta_unused) ? p->mid : U8_MAX;
1760 		seq_printf(s, "[%d] %pM %s MID %d AID %d\n", i, p->addr, status,
1761 			   mid, aid);
1762 
1763 		if (p->status == wil_sta_connected) {
1764 			u64 num_packets = 0;
1765 			u64 tx_latency_avg = p->stats.tx_latency_total_us;
1766 
1767 			seq_puts(s, "Tx/Latency bin:");
1768 			for (bin = 0; bin < WIL_NUM_LATENCY_BINS; bin++) {
1769 				seq_printf(s, " %lld",
1770 					   p->tx_latency_bins[bin]);
1771 				num_packets += p->tx_latency_bins[bin];
1772 			}
1773 			seq_puts(s, "\n");
1774 			if (!num_packets)
1775 				continue;
1776 			do_div(tx_latency_avg, num_packets);
1777 			seq_printf(s, "Tx/Latency min/avg/max (us): %d/%lld/%d",
1778 				   p->stats.tx_latency_min_us,
1779 				   tx_latency_avg,
1780 				   p->stats.tx_latency_max_us);
1781 
1782 			seq_puts(s, "\n");
1783 		}
1784 	}
1785 
1786 	return 0;
1787 }
1788 
wil_tx_latency_seq_open(struct inode * inode,struct file * file)1789 static int wil_tx_latency_seq_open(struct inode *inode, struct file *file)
1790 {
1791 	return single_open(file, wil_tx_latency_debugfs_show,
1792 			   inode->i_private);
1793 }
1794 
wil_tx_latency_write(struct file * file,const char __user * buf,size_t len,loff_t * ppos)1795 static ssize_t wil_tx_latency_write(struct file *file, const char __user *buf,
1796 				    size_t len, loff_t *ppos)
1797 {
1798 	struct seq_file *s = file->private_data;
1799 	struct wil6210_priv *wil = s->private;
1800 	int val, rc, i;
1801 	bool enable;
1802 
1803 	rc = kstrtoint_from_user(buf, len, 0, &val);
1804 	if (rc) {
1805 		wil_err(wil, "Invalid argument\n");
1806 		return rc;
1807 	}
1808 	if (val == 1)
1809 		/* default resolution */
1810 		val = 500;
1811 	if (val && (val < 50 || val > 1000)) {
1812 		wil_err(wil, "Invalid resolution %d\n", val);
1813 		return -EINVAL;
1814 	}
1815 
1816 	enable = !!val;
1817 	if (wil->tx_latency == enable)
1818 		return len;
1819 
1820 	wil_info(wil, "%s TX latency measurements (resolution %dusec)\n",
1821 		 enable ? "Enabling" : "Disabling", val);
1822 
1823 	if (enable) {
1824 		size_t sz = sizeof(u64) * WIL_NUM_LATENCY_BINS;
1825 
1826 		wil->tx_latency_res = val;
1827 		for (i = 0; i < wil->max_assoc_sta; i++) {
1828 			struct wil_sta_info *sta = &wil->sta[i];
1829 
1830 			kfree(sta->tx_latency_bins);
1831 			sta->tx_latency_bins = kzalloc(sz, GFP_KERNEL);
1832 			if (!sta->tx_latency_bins)
1833 				return -ENOMEM;
1834 			sta->stats.tx_latency_min_us = U32_MAX;
1835 			sta->stats.tx_latency_max_us = 0;
1836 			sta->stats.tx_latency_total_us = 0;
1837 		}
1838 	}
1839 	wil->tx_latency = enable;
1840 
1841 	return len;
1842 }
1843 
1844 static const struct file_operations fops_tx_latency = {
1845 	.open		= wil_tx_latency_seq_open,
1846 	.release	= single_release,
1847 	.read		= seq_read,
1848 	.write		= wil_tx_latency_write,
1849 	.llseek		= seq_lseek,
1850 };
1851 
wil_link_stats_print_basic(struct wil6210_vif * vif,struct seq_file * s,struct wmi_link_stats_basic * basic)1852 static void wil_link_stats_print_basic(struct wil6210_vif *vif,
1853 				       struct seq_file *s,
1854 				       struct wmi_link_stats_basic *basic)
1855 {
1856 	char per[5] = "?";
1857 
1858 	if (basic->per_average != 0xff)
1859 		snprintf(per, sizeof(per), "%d%%", basic->per_average);
1860 
1861 	seq_printf(s, "CID %d {\n"
1862 		   "\tTxMCS %d TxTpt %d\n"
1863 		   "\tGoodput(rx:tx) %d:%d\n"
1864 		   "\tRxBcastFrames %d\n"
1865 		   "\tRSSI %d SQI %d SNR %d PER %s\n"
1866 		   "\tRx RFC %d Ant num %d\n"
1867 		   "\tSectors(rx:tx) my %d:%d peer %d:%d\n"
1868 		   "}\n",
1869 		   basic->cid,
1870 		   basic->bf_mcs, le32_to_cpu(basic->tx_tpt),
1871 		   le32_to_cpu(basic->rx_goodput),
1872 		   le32_to_cpu(basic->tx_goodput),
1873 		   le32_to_cpu(basic->rx_bcast_frames),
1874 		   basic->rssi, basic->sqi, basic->snr, per,
1875 		   basic->selected_rfc, basic->rx_effective_ant_num,
1876 		   basic->my_rx_sector, basic->my_tx_sector,
1877 		   basic->other_rx_sector, basic->other_tx_sector);
1878 }
1879 
wil_link_stats_print_global(struct wil6210_priv * wil,struct seq_file * s,struct wmi_link_stats_global * global)1880 static void wil_link_stats_print_global(struct wil6210_priv *wil,
1881 					struct seq_file *s,
1882 					struct wmi_link_stats_global *global)
1883 {
1884 	seq_printf(s, "Frames(rx:tx) %d:%d\n"
1885 		   "BA Frames(rx:tx) %d:%d\n"
1886 		   "Beacons %d\n"
1887 		   "Rx Errors (MIC:CRC) %d:%d\n"
1888 		   "Tx Errors (no ack) %d\n",
1889 		   le32_to_cpu(global->rx_frames),
1890 		   le32_to_cpu(global->tx_frames),
1891 		   le32_to_cpu(global->rx_ba_frames),
1892 		   le32_to_cpu(global->tx_ba_frames),
1893 		   le32_to_cpu(global->tx_beacons),
1894 		   le32_to_cpu(global->rx_mic_errors),
1895 		   le32_to_cpu(global->rx_crc_errors),
1896 		   le32_to_cpu(global->tx_fail_no_ack));
1897 }
1898 
wil_link_stats_debugfs_show_vif(struct wil6210_vif * vif,struct seq_file * s)1899 static void wil_link_stats_debugfs_show_vif(struct wil6210_vif *vif,
1900 					    struct seq_file *s)
1901 {
1902 	struct wil6210_priv *wil = vif_to_wil(vif);
1903 	struct wmi_link_stats_basic *stats;
1904 	int i;
1905 
1906 	if (!vif->fw_stats_ready) {
1907 		seq_puts(s, "no statistics\n");
1908 		return;
1909 	}
1910 
1911 	seq_printf(s, "TSF %lld\n", vif->fw_stats_tsf);
1912 	for (i = 0; i < wil->max_assoc_sta; i++) {
1913 		if (wil->sta[i].status == wil_sta_unused)
1914 			continue;
1915 		if (wil->sta[i].mid != vif->mid)
1916 			continue;
1917 
1918 		stats = &wil->sta[i].fw_stats_basic;
1919 		wil_link_stats_print_basic(vif, s, stats);
1920 	}
1921 }
1922 
wil_link_stats_debugfs_show(struct seq_file * s,void * data)1923 static int wil_link_stats_debugfs_show(struct seq_file *s, void *data)
1924 {
1925 	struct wil6210_priv *wil = s->private;
1926 	struct wil6210_vif *vif;
1927 	int i, rc;
1928 
1929 	rc = mutex_lock_interruptible(&wil->vif_mutex);
1930 	if (rc)
1931 		return rc;
1932 
1933 	/* iterate over all MIDs and show per-cid statistics. Then show the
1934 	 * global statistics
1935 	 */
1936 	for (i = 0; i < GET_MAX_VIFS(wil); i++) {
1937 		vif = wil->vifs[i];
1938 
1939 		seq_printf(s, "MID %d ", i);
1940 		if (!vif) {
1941 			seq_puts(s, "unused\n");
1942 			continue;
1943 		}
1944 
1945 		wil_link_stats_debugfs_show_vif(vif, s);
1946 	}
1947 
1948 	mutex_unlock(&wil->vif_mutex);
1949 
1950 	return 0;
1951 }
1952 
wil_link_stats_seq_open(struct inode * inode,struct file * file)1953 static int wil_link_stats_seq_open(struct inode *inode, struct file *file)
1954 {
1955 	return single_open(file, wil_link_stats_debugfs_show, inode->i_private);
1956 }
1957 
wil_link_stats_write(struct file * file,const char __user * buf,size_t len,loff_t * ppos)1958 static ssize_t wil_link_stats_write(struct file *file, const char __user *buf,
1959 				    size_t len, loff_t *ppos)
1960 {
1961 	struct seq_file *s = file->private_data;
1962 	struct wil6210_priv *wil = s->private;
1963 	int cid, interval, rc, i;
1964 	struct wil6210_vif *vif;
1965 	char *kbuf = kmalloc(len + 1, GFP_KERNEL);
1966 
1967 	if (!kbuf)
1968 		return -ENOMEM;
1969 
1970 	rc = simple_write_to_buffer(kbuf, len, ppos, buf, len);
1971 	if (rc != len) {
1972 		kfree(kbuf);
1973 		return rc >= 0 ? -EIO : rc;
1974 	}
1975 
1976 	kbuf[len] = '\0';
1977 	/* specify cid (use -1 for all cids) and snapshot interval in ms */
1978 	rc = sscanf(kbuf, "%d %d", &cid, &interval);
1979 	kfree(kbuf);
1980 	if (rc < 0)
1981 		return rc;
1982 	if (rc < 2 || interval < 0)
1983 		return -EINVAL;
1984 
1985 	wil_info(wil, "request link statistics, cid %d interval %d\n",
1986 		 cid, interval);
1987 
1988 	rc = mutex_lock_interruptible(&wil->vif_mutex);
1989 	if (rc)
1990 		return rc;
1991 
1992 	for (i = 0; i < GET_MAX_VIFS(wil); i++) {
1993 		vif = wil->vifs[i];
1994 		if (!vif)
1995 			continue;
1996 
1997 		rc = wmi_link_stats_cfg(vif, WMI_LINK_STATS_TYPE_BASIC,
1998 					(cid == -1 ? 0xff : cid), interval);
1999 		if (rc)
2000 			wil_err(wil, "link statistics failed for mid %d\n", i);
2001 	}
2002 	mutex_unlock(&wil->vif_mutex);
2003 
2004 	return len;
2005 }
2006 
2007 static const struct file_operations fops_link_stats = {
2008 	.open		= wil_link_stats_seq_open,
2009 	.release	= single_release,
2010 	.read		= seq_read,
2011 	.write		= wil_link_stats_write,
2012 	.llseek		= seq_lseek,
2013 };
2014 
2015 static int
wil_link_stats_global_debugfs_show(struct seq_file * s,void * data)2016 wil_link_stats_global_debugfs_show(struct seq_file *s, void *data)
2017 {
2018 	struct wil6210_priv *wil = s->private;
2019 
2020 	if (!wil->fw_stats_global.ready)
2021 		return 0;
2022 
2023 	seq_printf(s, "TSF %lld\n", wil->fw_stats_global.tsf);
2024 	wil_link_stats_print_global(wil, s, &wil->fw_stats_global.stats);
2025 
2026 	return 0;
2027 }
2028 
2029 static int
wil_link_stats_global_seq_open(struct inode * inode,struct file * file)2030 wil_link_stats_global_seq_open(struct inode *inode, struct file *file)
2031 {
2032 	return single_open(file, wil_link_stats_global_debugfs_show,
2033 			   inode->i_private);
2034 }
2035 
2036 static ssize_t
wil_link_stats_global_write(struct file * file,const char __user * buf,size_t len,loff_t * ppos)2037 wil_link_stats_global_write(struct file *file, const char __user *buf,
2038 			    size_t len, loff_t *ppos)
2039 {
2040 	struct seq_file *s = file->private_data;
2041 	struct wil6210_priv *wil = s->private;
2042 	int interval, rc;
2043 	struct wil6210_vif *vif = ndev_to_vif(wil->main_ndev);
2044 
2045 	/* specify snapshot interval in ms */
2046 	rc = kstrtoint_from_user(buf, len, 0, &interval);
2047 	if (rc || interval < 0) {
2048 		wil_err(wil, "Invalid argument\n");
2049 		return -EINVAL;
2050 	}
2051 
2052 	wil_info(wil, "request global link stats, interval %d\n", interval);
2053 
2054 	rc = wmi_link_stats_cfg(vif, WMI_LINK_STATS_TYPE_GLOBAL, 0, interval);
2055 	if (rc)
2056 		wil_err(wil, "global link stats failed %d\n", rc);
2057 
2058 	return rc ? rc : len;
2059 }
2060 
2061 static const struct file_operations fops_link_stats_global = {
2062 	.open		= wil_link_stats_global_seq_open,
2063 	.release	= single_release,
2064 	.read		= seq_read,
2065 	.write		= wil_link_stats_global_write,
2066 	.llseek		= seq_lseek,
2067 };
2068 
wil_read_file_led_cfg(struct file * file,char __user * user_buf,size_t count,loff_t * ppos)2069 static ssize_t wil_read_file_led_cfg(struct file *file, char __user *user_buf,
2070 				     size_t count, loff_t *ppos)
2071 {
2072 	char buf[80];
2073 	int n;
2074 
2075 	n = snprintf(buf, sizeof(buf),
2076 		     "led_id is set to %d, echo 1 to enable, 0 to disable\n",
2077 		     led_id);
2078 
2079 	n = min_t(int, n, sizeof(buf));
2080 
2081 	return simple_read_from_buffer(user_buf, count, ppos,
2082 				       buf, n);
2083 }
2084 
wil_write_file_led_cfg(struct file * file,const char __user * buf_,size_t count,loff_t * ppos)2085 static ssize_t wil_write_file_led_cfg(struct file *file,
2086 				      const char __user *buf_,
2087 				      size_t count, loff_t *ppos)
2088 {
2089 	struct wil6210_priv *wil = file->private_data;
2090 	int val;
2091 	int rc;
2092 
2093 	rc = kstrtoint_from_user(buf_, count, 0, &val);
2094 	if (rc) {
2095 		wil_err(wil, "Invalid argument\n");
2096 		return rc;
2097 	}
2098 
2099 	wil_info(wil, "%s led %d\n", val ? "Enabling" : "Disabling", led_id);
2100 	rc = wmi_led_cfg(wil, val);
2101 	if (rc) {
2102 		wil_info(wil, "%s led %d failed\n",
2103 			 val ? "Enabling" : "Disabling", led_id);
2104 		return rc;
2105 	}
2106 
2107 	return count;
2108 }
2109 
2110 static const struct file_operations fops_led_cfg = {
2111 	.read = wil_read_file_led_cfg,
2112 	.write = wil_write_file_led_cfg,
2113 	.open  = simple_open,
2114 };
2115 
2116 /* led_blink_time, write:
2117  * "<blink_on_slow> <blink_off_slow> <blink_on_med> <blink_off_med> <blink_on_fast> <blink_off_fast>
2118  */
wil_write_led_blink_time(struct file * file,const char __user * buf,size_t len,loff_t * ppos)2119 static ssize_t wil_write_led_blink_time(struct file *file,
2120 					const char __user *buf,
2121 					size_t len, loff_t *ppos)
2122 {
2123 	int rc;
2124 	char *kbuf = kmalloc(len + 1, GFP_KERNEL);
2125 
2126 	if (!kbuf)
2127 		return -ENOMEM;
2128 
2129 	rc = simple_write_to_buffer(kbuf, len, ppos, buf, len);
2130 	if (rc != len) {
2131 		kfree(kbuf);
2132 		return rc >= 0 ? -EIO : rc;
2133 	}
2134 
2135 	kbuf[len] = '\0';
2136 	rc = sscanf(kbuf, "%d %d %d %d %d %d",
2137 		    &led_blink_time[WIL_LED_TIME_SLOW].on_ms,
2138 		    &led_blink_time[WIL_LED_TIME_SLOW].off_ms,
2139 		    &led_blink_time[WIL_LED_TIME_MED].on_ms,
2140 		    &led_blink_time[WIL_LED_TIME_MED].off_ms,
2141 		    &led_blink_time[WIL_LED_TIME_FAST].on_ms,
2142 		    &led_blink_time[WIL_LED_TIME_FAST].off_ms);
2143 	kfree(kbuf);
2144 
2145 	if (rc < 0)
2146 		return rc;
2147 	if (rc < 6)
2148 		return -EINVAL;
2149 
2150 	return len;
2151 }
2152 
wil_read_led_blink_time(struct file * file,char __user * user_buf,size_t count,loff_t * ppos)2153 static ssize_t wil_read_led_blink_time(struct file *file, char __user *user_buf,
2154 				       size_t count, loff_t *ppos)
2155 {
2156 	static char text[400];
2157 
2158 	snprintf(text, sizeof(text),
2159 		 "To set led blink on/off time variables write:\n"
2160 		 "<blink_on_slow> <blink_off_slow> <blink_on_med> "
2161 		 "<blink_off_med> <blink_on_fast> <blink_off_fast>\n"
2162 		 "The current values are:\n"
2163 		 "%d %d %d %d %d %d\n",
2164 		 led_blink_time[WIL_LED_TIME_SLOW].on_ms,
2165 		 led_blink_time[WIL_LED_TIME_SLOW].off_ms,
2166 		 led_blink_time[WIL_LED_TIME_MED].on_ms,
2167 		 led_blink_time[WIL_LED_TIME_MED].off_ms,
2168 		 led_blink_time[WIL_LED_TIME_FAST].on_ms,
2169 		 led_blink_time[WIL_LED_TIME_FAST].off_ms);
2170 
2171 	return simple_read_from_buffer(user_buf, count, ppos, text,
2172 				       sizeof(text));
2173 }
2174 
2175 static const struct file_operations fops_led_blink_time = {
2176 	.read = wil_read_led_blink_time,
2177 	.write = wil_write_led_blink_time,
2178 	.open  = simple_open,
2179 };
2180 
2181 /*---------FW capabilities------------*/
wil_fw_capabilities_debugfs_show(struct seq_file * s,void * data)2182 static int wil_fw_capabilities_debugfs_show(struct seq_file *s, void *data)
2183 {
2184 	struct wil6210_priv *wil = s->private;
2185 
2186 	seq_printf(s, "fw_capabilities : %*pb\n", WMI_FW_CAPABILITY_MAX,
2187 		   wil->fw_capabilities);
2188 
2189 	return 0;
2190 }
2191 
wil_fw_capabilities_seq_open(struct inode * inode,struct file * file)2192 static int wil_fw_capabilities_seq_open(struct inode *inode, struct file *file)
2193 {
2194 	return single_open(file, wil_fw_capabilities_debugfs_show,
2195 			   inode->i_private);
2196 }
2197 
2198 static const struct file_operations fops_fw_capabilities = {
2199 	.open		= wil_fw_capabilities_seq_open,
2200 	.release	= single_release,
2201 	.read		= seq_read,
2202 	.llseek		= seq_lseek,
2203 };
2204 
2205 /*---------FW version------------*/
wil_fw_version_debugfs_show(struct seq_file * s,void * data)2206 static int wil_fw_version_debugfs_show(struct seq_file *s, void *data)
2207 {
2208 	struct wil6210_priv *wil = s->private;
2209 
2210 	if (wil->fw_version[0])
2211 		seq_printf(s, "%s\n", wil->fw_version);
2212 	else
2213 		seq_puts(s, "N/A\n");
2214 
2215 	return 0;
2216 }
2217 
wil_fw_version_seq_open(struct inode * inode,struct file * file)2218 static int wil_fw_version_seq_open(struct inode *inode, struct file *file)
2219 {
2220 	return single_open(file, wil_fw_version_debugfs_show,
2221 			   inode->i_private);
2222 }
2223 
2224 static const struct file_operations fops_fw_version = {
2225 	.open		= wil_fw_version_seq_open,
2226 	.release	= single_release,
2227 	.read		= seq_read,
2228 	.llseek		= seq_lseek,
2229 };
2230 
2231 /*---------suspend_stats---------*/
wil_write_suspend_stats(struct file * file,const char __user * buf,size_t len,loff_t * ppos)2232 static ssize_t wil_write_suspend_stats(struct file *file,
2233 				       const char __user *buf,
2234 				       size_t len, loff_t *ppos)
2235 {
2236 	struct wil6210_priv *wil = file->private_data;
2237 
2238 	memset(&wil->suspend_stats, 0, sizeof(wil->suspend_stats));
2239 
2240 	return len;
2241 }
2242 
wil_read_suspend_stats(struct file * file,char __user * user_buf,size_t count,loff_t * ppos)2243 static ssize_t wil_read_suspend_stats(struct file *file,
2244 				      char __user *user_buf,
2245 				      size_t count, loff_t *ppos)
2246 {
2247 	struct wil6210_priv *wil = file->private_data;
2248 	char *text;
2249 	int n, ret, text_size = 500;
2250 
2251 	text = kmalloc(text_size, GFP_KERNEL);
2252 	if (!text)
2253 		return -ENOMEM;
2254 
2255 	n = snprintf(text, text_size,
2256 		     "Radio on suspend statistics:\n"
2257 		     "successful suspends:%ld failed suspends:%ld\n"
2258 		     "successful resumes:%ld failed resumes:%ld\n"
2259 		     "rejected by device:%ld\n"
2260 		     "Radio off suspend statistics:\n"
2261 		     "successful suspends:%ld failed suspends:%ld\n"
2262 		     "successful resumes:%ld failed resumes:%ld\n"
2263 		     "General statistics:\n"
2264 		     "rejected by host:%ld\n",
2265 		     wil->suspend_stats.r_on.successful_suspends,
2266 		     wil->suspend_stats.r_on.failed_suspends,
2267 		     wil->suspend_stats.r_on.successful_resumes,
2268 		     wil->suspend_stats.r_on.failed_resumes,
2269 		     wil->suspend_stats.rejected_by_device,
2270 		     wil->suspend_stats.r_off.successful_suspends,
2271 		     wil->suspend_stats.r_off.failed_suspends,
2272 		     wil->suspend_stats.r_off.successful_resumes,
2273 		     wil->suspend_stats.r_off.failed_resumes,
2274 		     wil->suspend_stats.rejected_by_host);
2275 
2276 	n = min_t(int, n, text_size);
2277 
2278 	ret = simple_read_from_buffer(user_buf, count, ppos, text, n);
2279 
2280 	kfree(text);
2281 
2282 	return ret;
2283 }
2284 
2285 static const struct file_operations fops_suspend_stats = {
2286 	.read = wil_read_suspend_stats,
2287 	.write = wil_write_suspend_stats,
2288 	.open  = simple_open,
2289 };
2290 
2291 /*---------compressed_rx_status---------*/
wil_compressed_rx_status_write(struct file * file,const char __user * buf,size_t len,loff_t * ppos)2292 static ssize_t wil_compressed_rx_status_write(struct file *file,
2293 					      const char __user *buf,
2294 					      size_t len, loff_t *ppos)
2295 {
2296 	struct seq_file *s = file->private_data;
2297 	struct wil6210_priv *wil = s->private;
2298 	int compressed_rx_status;
2299 	int rc;
2300 
2301 	rc = kstrtoint_from_user(buf, len, 0, &compressed_rx_status);
2302 	if (rc) {
2303 		wil_err(wil, "Invalid argument\n");
2304 		return rc;
2305 	}
2306 
2307 	if (wil_has_active_ifaces(wil, true, false)) {
2308 		wil_err(wil, "cannot change edma config after iface is up\n");
2309 		return -EPERM;
2310 	}
2311 
2312 	wil_info(wil, "%sable compressed_rx_status\n",
2313 		 compressed_rx_status ? "En" : "Dis");
2314 
2315 	wil->use_compressed_rx_status = compressed_rx_status;
2316 
2317 	return len;
2318 }
2319 
2320 static int
wil_compressed_rx_status_show(struct seq_file * s,void * data)2321 wil_compressed_rx_status_show(struct seq_file *s, void *data)
2322 {
2323 	struct wil6210_priv *wil = s->private;
2324 
2325 	seq_printf(s, "%d\n", wil->use_compressed_rx_status);
2326 
2327 	return 0;
2328 }
2329 
2330 static int
wil_compressed_rx_status_seq_open(struct inode * inode,struct file * file)2331 wil_compressed_rx_status_seq_open(struct inode *inode, struct file *file)
2332 {
2333 	return single_open(file, wil_compressed_rx_status_show,
2334 			   inode->i_private);
2335 }
2336 
2337 static const struct file_operations fops_compressed_rx_status = {
2338 	.open  = wil_compressed_rx_status_seq_open,
2339 	.release = single_release,
2340 	.read = seq_read,
2341 	.write = wil_compressed_rx_status_write,
2342 	.llseek	= seq_lseek,
2343 };
2344 
2345 /*----------------*/
wil6210_debugfs_init_blobs(struct wil6210_priv * wil,struct dentry * dbg)2346 static void wil6210_debugfs_init_blobs(struct wil6210_priv *wil,
2347 				       struct dentry *dbg)
2348 {
2349 	int i;
2350 	char name[32];
2351 
2352 	for (i = 0; i < ARRAY_SIZE(fw_mapping); i++) {
2353 		struct wil_blob_wrapper *wil_blob = &wil->blobs[i];
2354 		struct debugfs_blob_wrapper *blob = &wil_blob->blob;
2355 		const struct fw_map *map = &fw_mapping[i];
2356 
2357 		if (!map->name)
2358 			continue;
2359 
2360 		wil_blob->wil = wil;
2361 		blob->data = (void * __force)wil->csr + HOSTADDR(map->host);
2362 		blob->size = map->to - map->from;
2363 		snprintf(name, sizeof(name), "blob_%s", map->name);
2364 		wil_debugfs_create_ioblob(name, 0444, dbg, wil_blob);
2365 	}
2366 }
2367 
2368 /* misc files */
2369 static const struct {
2370 	const char *name;
2371 	umode_t mode;
2372 	const struct file_operations *fops;
2373 } dbg_files[] = {
2374 	{"mbox",	0444,		&mbox_fops},
2375 	{"rings",	0444,		&ring_fops},
2376 	{"stations", 0444,		&sta_fops},
2377 	{"mids",	0444,		&mids_fops},
2378 	{"desc",	0444,		&txdesc_fops},
2379 	{"bf",		0444,		&bf_fops},
2380 	{"mem_val",	0644,		&memread_fops},
2381 	{"rxon",	0244,		&fops_rxon},
2382 	{"tx_mgmt",	0244,		&fops_txmgmt},
2383 	{"wmi_send", 0244,		&fops_wmi},
2384 	{"back",	0644,		&fops_back},
2385 	{"pmccfg",	0644,		&fops_pmccfg},
2386 	{"pmcdata",	0444,		&fops_pmcdata},
2387 	{"pmcring",	0444,		&fops_pmcring},
2388 	{"temp",	0444,		&temp_fops},
2389 	{"freq",	0444,		&freq_fops},
2390 	{"link",	0444,		&link_fops},
2391 	{"info",	0444,		&info_fops},
2392 	{"recovery", 0644,		&fops_recovery},
2393 	{"led_cfg",	0644,		&fops_led_cfg},
2394 	{"led_blink_time",	0644,	&fops_led_blink_time},
2395 	{"fw_capabilities",	0444,	&fops_fw_capabilities},
2396 	{"fw_version",	0444,		&fops_fw_version},
2397 	{"suspend_stats",	0644,	&fops_suspend_stats},
2398 	{"compressed_rx_status", 0644,	&fops_compressed_rx_status},
2399 	{"srings",	0444,		&srings_fops},
2400 	{"status_msg",	0444,		&status_msg_fops},
2401 	{"rx_buff_mgmt",	0444,	&rx_buff_mgmt_fops},
2402 	{"tx_latency",	0644,		&fops_tx_latency},
2403 	{"link_stats",	0644,		&fops_link_stats},
2404 	{"link_stats_global",	0644,	&fops_link_stats_global},
2405 	{"rbufcap",	0244,		&fops_rbufcap},
2406 };
2407 
wil6210_debugfs_init_files(struct wil6210_priv * wil,struct dentry * dbg)2408 static void wil6210_debugfs_init_files(struct wil6210_priv *wil,
2409 				       struct dentry *dbg)
2410 {
2411 	int i;
2412 
2413 	for (i = 0; i < ARRAY_SIZE(dbg_files); i++)
2414 		debugfs_create_file(dbg_files[i].name, dbg_files[i].mode, dbg,
2415 				    wil, dbg_files[i].fops);
2416 }
2417 
2418 /* interrupt control blocks */
2419 static const struct {
2420 	const char *name;
2421 	u32 icr_off;
2422 } dbg_icr[] = {
2423 	{"USER_ICR",		HOSTADDR(RGF_USER_USER_ICR)},
2424 	{"DMA_EP_TX_ICR",	HOSTADDR(RGF_DMA_EP_TX_ICR)},
2425 	{"DMA_EP_RX_ICR",	HOSTADDR(RGF_DMA_EP_RX_ICR)},
2426 	{"DMA_EP_MISC_ICR",	HOSTADDR(RGF_DMA_EP_MISC_ICR)},
2427 };
2428 
wil6210_debugfs_init_isr(struct wil6210_priv * wil,struct dentry * dbg)2429 static void wil6210_debugfs_init_isr(struct wil6210_priv *wil,
2430 				     struct dentry *dbg)
2431 {
2432 	int i;
2433 
2434 	for (i = 0; i < ARRAY_SIZE(dbg_icr); i++)
2435 		wil6210_debugfs_create_ISR(wil, dbg_icr[i].name, dbg,
2436 					   dbg_icr[i].icr_off);
2437 }
2438 
2439 #define WIL_FIELD(name, mode, type) { __stringify(name), mode, \
2440 	offsetof(struct wil6210_priv, name), type}
2441 
2442 /* fields in struct wil6210_priv */
2443 static const struct dbg_off dbg_wil_off[] = {
2444 	WIL_FIELD(status[0],	0644,	doff_ulong),
2445 	WIL_FIELD(hw_version,	0444,	doff_x32),
2446 	WIL_FIELD(recovery_count, 0444,	doff_u32),
2447 	WIL_FIELD(discovery_mode, 0644,	doff_u8),
2448 	WIL_FIELD(chip_revision, 0444,	doff_u8),
2449 	WIL_FIELD(abft_len, 0644,		doff_u8),
2450 	WIL_FIELD(wakeup_trigger, 0644,		doff_u8),
2451 	WIL_FIELD(ring_idle_trsh, 0644,	doff_u32),
2452 	WIL_FIELD(num_rx_status_rings, 0644,	doff_u8),
2453 	WIL_FIELD(rx_status_ring_order, 0644,	doff_u32),
2454 	WIL_FIELD(tx_status_ring_order, 0644,	doff_u32),
2455 	WIL_FIELD(rx_buff_id_count, 0644,	doff_u32),
2456 	WIL_FIELD(amsdu_en, 0644,	doff_u8),
2457 	{},
2458 };
2459 
2460 static const struct dbg_off dbg_wil_regs[] = {
2461 	{"RGF_MAC_MTRL_COUNTER_0", 0444, HOSTADDR(RGF_MAC_MTRL_COUNTER_0),
2462 		doff_io32},
2463 	{"RGF_USER_USAGE_1", 0444, HOSTADDR(RGF_USER_USAGE_1), doff_io32},
2464 	{"RGF_USER_USAGE_2", 0444, HOSTADDR(RGF_USER_USAGE_2), doff_io32},
2465 	{},
2466 };
2467 
2468 /* static parameters */
2469 static const struct dbg_off dbg_statics[] = {
2470 	{"desc_index",	0644, (ulong)&dbg_txdesc_index, doff_u32},
2471 	{"ring_index",	0644, (ulong)&dbg_ring_index, doff_u32},
2472 	{"mem_addr",	0644, (ulong)&mem_addr, doff_u32},
2473 	{"led_polarity", 0644, (ulong)&led_polarity, doff_u8},
2474 	{"status_index", 0644, (ulong)&dbg_status_msg_index, doff_u32},
2475 	{"sring_index",	0644, (ulong)&dbg_sring_index, doff_u32},
2476 	{"drop_if_ring_full", 0644, (ulong)&drop_if_ring_full, doff_u8},
2477 	{},
2478 };
2479 
2480 static const int dbg_off_count = 4 * (ARRAY_SIZE(isr_off) - 1) +
2481 				ARRAY_SIZE(dbg_wil_regs) - 1 +
2482 				ARRAY_SIZE(pseudo_isr_off) - 1 +
2483 				ARRAY_SIZE(lgc_itr_cnt_off) - 1 +
2484 				ARRAY_SIZE(tx_itr_cnt_off) - 1 +
2485 				ARRAY_SIZE(rx_itr_cnt_off) - 1;
2486 
wil6210_debugfs_init(struct wil6210_priv * wil)2487 int wil6210_debugfs_init(struct wil6210_priv *wil)
2488 {
2489 	struct dentry *dbg = wil->debug = debugfs_create_dir(WIL_NAME,
2490 			wil_to_wiphy(wil)->debugfsdir);
2491 	if (IS_ERR_OR_NULL(dbg))
2492 		return -ENODEV;
2493 
2494 	wil->dbg_data.data_arr = kcalloc(dbg_off_count,
2495 					 sizeof(struct wil_debugfs_iomem_data),
2496 					 GFP_KERNEL);
2497 	if (!wil->dbg_data.data_arr) {
2498 		debugfs_remove_recursive(dbg);
2499 		wil->debug = NULL;
2500 		return -ENOMEM;
2501 	}
2502 
2503 	wil->dbg_data.iomem_data_count = 0;
2504 
2505 	wil_pmc_init(wil);
2506 
2507 	wil6210_debugfs_init_files(wil, dbg);
2508 	wil6210_debugfs_init_isr(wil, dbg);
2509 	wil6210_debugfs_init_blobs(wil, dbg);
2510 	wil6210_debugfs_init_offset(wil, dbg, wil, dbg_wil_off);
2511 	wil6210_debugfs_init_offset(wil, dbg, (void * __force)wil->csr,
2512 				    dbg_wil_regs);
2513 	wil6210_debugfs_init_offset(wil, dbg, NULL, dbg_statics);
2514 
2515 	wil6210_debugfs_create_pseudo_ISR(wil, dbg);
2516 
2517 	wil6210_debugfs_create_ITR_CNT(wil, dbg);
2518 
2519 	return 0;
2520 }
2521 
wil6210_debugfs_remove(struct wil6210_priv * wil)2522 void wil6210_debugfs_remove(struct wil6210_priv *wil)
2523 {
2524 	int i;
2525 
2526 	debugfs_remove_recursive(wil->debug);
2527 	wil->debug = NULL;
2528 
2529 	kfree(wil->dbg_data.data_arr);
2530 	for (i = 0; i < wil->max_assoc_sta; i++)
2531 		kfree(wil->sta[i].tx_latency_bins);
2532 
2533 	/* free pmc memory without sending command to fw, as it will
2534 	 * be reset on the way down anyway
2535 	 */
2536 	wil_pmc_free(wil, false);
2537 }
2538