debug.c 28.0 KB
Newer Older
1
/*
2
 * Copyright (c) 2008-2009 Atheros Communications Inc.
3 4 5 6 7 8 9 10 11 12 13 14 15 16
 *
 * Permission to use, copy, modify, and/or distribute this software for any
 * purpose with or without fee is hereby granted, provided that the above
 * copyright notice and this permission notice appear in all copies.
 *
 * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
 * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
 * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
 * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
 * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
 * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
 * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
 */

17
#include <linux/slab.h>
18 19
#include <asm/unaligned.h>

S
Sujith 已提交
20
#include "ath9k.h"
21

22 23 24 25 26
#define REG_WRITE_D(_ah, _reg, _val) \
	ath9k_hw_common(_ah)->ops->write((_ah), (_val), (_reg))
#define REG_READ_D(_ah, _reg) \
	ath9k_hw_common(_ah)->ops->read((_ah), (_reg))

27 28
static struct dentry *ath9k_debugfs_root;

29 30 31 32 33 34
static int ath9k_debugfs_open(struct inode *inode, struct file *file)
{
	file->private_data = inode->i_private;
	return 0;
}

35 36
#ifdef CONFIG_ATH_DEBUG

J
Jeff Hansen 已提交
37 38 39 40
static ssize_t read_file_debug(struct file *file, char __user *user_buf,
			     size_t count, loff_t *ppos)
{
	struct ath_softc *sc = file->private_data;
41
	struct ath_common *common = ath9k_hw_common(sc->sc_ah);
J
Jeff Hansen 已提交
42
	char buf[32];
43 44
	unsigned int len;

45
	len = sprintf(buf, "0x%08x\n", common->debug_mask);
J
Jeff Hansen 已提交
46 47 48 49 50 51 52
	return simple_read_from_buffer(user_buf, count, ppos, buf, len);
}

static ssize_t write_file_debug(struct file *file, const char __user *user_buf,
			     size_t count, loff_t *ppos)
{
	struct ath_softc *sc = file->private_data;
53
	struct ath_common *common = ath9k_hw_common(sc->sc_ah);
J
Jeff Hansen 已提交
54 55
	unsigned long mask;
	char buf[32];
56 57 58 59
	ssize_t len;

	len = min(count, sizeof(buf) - 1);
	if (copy_from_user(buf, user_buf, len))
60
		return -EFAULT;
61 62 63 64 65

	buf[len] = '\0';
	if (strict_strtoul(buf, 0, &mask))
		return -EINVAL;

66
	common->debug_mask = mask;
J
Jeff Hansen 已提交
67 68 69 70 71 72 73
	return count;
}

static const struct file_operations fops_debug = {
	.read = read_file_debug,
	.write = write_file_debug,
	.open = ath9k_debugfs_open,
74 75
	.owner = THIS_MODULE,
	.llseek = default_llseek,
J
Jeff Hansen 已提交
76 77
};

78 79
#endif

80 81
#define DMA_BUF_LEN 1024

82 83 84 85 86 87 88 89
static ssize_t read_file_tx_chainmask(struct file *file, char __user *user_buf,
			     size_t count, loff_t *ppos)
{
	struct ath_softc *sc = file->private_data;
	struct ath_common *common = ath9k_hw_common(sc->sc_ah);
	char buf[32];
	unsigned int len;

90
	len = sprintf(buf, "0x%08x\n", common->tx_chainmask);
91 92 93 94 95 96 97 98 99 100 101 102 103 104
	return simple_read_from_buffer(user_buf, count, ppos, buf, len);
}

static ssize_t write_file_tx_chainmask(struct file *file, const char __user *user_buf,
			     size_t count, loff_t *ppos)
{
	struct ath_softc *sc = file->private_data;
	struct ath_common *common = ath9k_hw_common(sc->sc_ah);
	unsigned long mask;
	char buf[32];
	ssize_t len;

	len = min(count, sizeof(buf) - 1);
	if (copy_from_user(buf, user_buf, len))
105
		return -EFAULT;
106 107 108 109 110 111 112 113 114 115 116 117 118 119

	buf[len] = '\0';
	if (strict_strtoul(buf, 0, &mask))
		return -EINVAL;

	common->tx_chainmask = mask;
	sc->sc_ah->caps.tx_chainmask = mask;
	return count;
}

static const struct file_operations fops_tx_chainmask = {
	.read = read_file_tx_chainmask,
	.write = write_file_tx_chainmask,
	.open = ath9k_debugfs_open,
120 121
	.owner = THIS_MODULE,
	.llseek = default_llseek,
122 123 124 125 126 127 128 129 130 131 132
};


static ssize_t read_file_rx_chainmask(struct file *file, char __user *user_buf,
			     size_t count, loff_t *ppos)
{
	struct ath_softc *sc = file->private_data;
	struct ath_common *common = ath9k_hw_common(sc->sc_ah);
	char buf[32];
	unsigned int len;

133
	len = sprintf(buf, "0x%08x\n", common->rx_chainmask);
134 135 136 137 138 139 140 141 142 143 144 145 146 147
	return simple_read_from_buffer(user_buf, count, ppos, buf, len);
}

static ssize_t write_file_rx_chainmask(struct file *file, const char __user *user_buf,
			     size_t count, loff_t *ppos)
{
	struct ath_softc *sc = file->private_data;
	struct ath_common *common = ath9k_hw_common(sc->sc_ah);
	unsigned long mask;
	char buf[32];
	ssize_t len;

	len = min(count, sizeof(buf) - 1);
	if (copy_from_user(buf, user_buf, len))
148
		return -EFAULT;
149 150 151 152 153 154 155 156 157 158 159 160 161 162

	buf[len] = '\0';
	if (strict_strtoul(buf, 0, &mask))
		return -EINVAL;

	common->rx_chainmask = mask;
	sc->sc_ah->caps.rx_chainmask = mask;
	return count;
}

static const struct file_operations fops_rx_chainmask = {
	.read = read_file_rx_chainmask,
	.write = write_file_rx_chainmask,
	.open = ath9k_debugfs_open,
163 164
	.owner = THIS_MODULE,
	.llseek = default_llseek,
165 166 167
};


168 169 170 171
static ssize_t read_file_dma(struct file *file, char __user *user_buf,
			     size_t count, loff_t *ppos)
{
	struct ath_softc *sc = file->private_data;
172
	struct ath_hw *ah = sc->sc_ah;
173 174
	char *buf;
	int retval;
175 176 177 178 179
	unsigned int len = 0;
	u32 val[ATH9K_NUM_DMA_DEBUG_REGS];
	int i, qcuOffset = 0, dcuOffset = 0;
	u32 *qcuBase = &val[0], *dcuBase = &val[4];

180 181
	buf = kmalloc(DMA_BUF_LEN, GFP_KERNEL);
	if (!buf)
182
		return -ENOMEM;
183

184 185
	ath9k_ps_wakeup(sc);

186
	REG_WRITE_D(ah, AR_MACMISC,
187 188 189 190
		  ((AR_MACMISC_DMA_OBS_LINE_8 << AR_MACMISC_DMA_OBS_S) |
		   (AR_MACMISC_MISC_OBS_BUS_1 <<
		    AR_MACMISC_MISC_OBS_BUS_MSB_S)));

191
	len += snprintf(buf + len, DMA_BUF_LEN - len,
192 193 194 195
			"Raw DMA Debug values:\n");

	for (i = 0; i < ATH9K_NUM_DMA_DEBUG_REGS; i++) {
		if (i % 4 == 0)
196
			len += snprintf(buf + len, DMA_BUF_LEN - len, "\n");
197

198
		val[i] = REG_READ_D(ah, AR_DMADBG_0 + (i * sizeof(u32)));
199
		len += snprintf(buf + len, DMA_BUF_LEN - len, "%d: %08x ",
200 201 202
				i, val[i]);
	}

203 204
	len += snprintf(buf + len, DMA_BUF_LEN - len, "\n\n");
	len += snprintf(buf + len, DMA_BUF_LEN - len,
205 206 207 208 209 210 211 212 213 214 215 216 217
			"Num QCU: chain_st fsp_ok fsp_st DCU: chain_st\n");

	for (i = 0; i < ATH9K_NUM_QUEUES; i++, qcuOffset += 4, dcuOffset += 5) {
		if (i == 8) {
			qcuOffset = 0;
			qcuBase++;
		}

		if (i == 6) {
			dcuOffset = 0;
			dcuBase++;
		}

218
		len += snprintf(buf + len, DMA_BUF_LEN - len,
219 220 221 222 223 224 225
			"%2d          %2x      %1x     %2x           %2x\n",
			i, (*qcuBase & (0x7 << qcuOffset)) >> qcuOffset,
			(*qcuBase & (0x8 << qcuOffset)) >> (qcuOffset + 3),
			val[2] & (0x7 << (i * 3)) >> (i * 3),
			(*dcuBase & (0x1f << dcuOffset)) >> dcuOffset);
	}

226
	len += snprintf(buf + len, DMA_BUF_LEN - len, "\n");
227

228
	len += snprintf(buf + len, DMA_BUF_LEN - len,
229 230
		"qcu_stitch state:   %2x    qcu_fetch state:        %2x\n",
		(val[3] & 0x003c0000) >> 18, (val[3] & 0x03c00000) >> 22);
231
	len += snprintf(buf + len, DMA_BUF_LEN - len,
232 233
		"qcu_complete state: %2x    dcu_complete state:     %2x\n",
		(val[3] & 0x1c000000) >> 26, (val[6] & 0x3));
234
	len += snprintf(buf + len, DMA_BUF_LEN - len,
235 236
		"dcu_arb state:      %2x    dcu_fp state:           %2x\n",
		(val[5] & 0x06000000) >> 25, (val[5] & 0x38000000) >> 27);
237
	len += snprintf(buf + len, DMA_BUF_LEN - len,
238 239
		"chan_idle_dur:     %3d    chan_idle_dur_valid:     %1d\n",
		(val[6] & 0x000003fc) >> 2, (val[6] & 0x00000400) >> 10);
240
	len += snprintf(buf + len, DMA_BUF_LEN - len,
241 242
		"txfifo_valid_0:      %1d    txfifo_valid_1:          %1d\n",
		(val[6] & 0x00000800) >> 11, (val[6] & 0x00001000) >> 12);
243
	len += snprintf(buf + len, DMA_BUF_LEN - len,
244 245 246
		"txfifo_dcu_num_0:   %2d    txfifo_dcu_num_1:       %2d\n",
		(val[6] & 0x0001e000) >> 13, (val[6] & 0x001e0000) >> 17);

247
	len += snprintf(buf + len, DMA_BUF_LEN - len, "pcu observe: 0x%x\n",
248
			REG_READ_D(ah, AR_OBS_BUS_1));
249
	len += snprintf(buf + len, DMA_BUF_LEN - len,
250
			"AR_CR: 0x%x\n", REG_READ_D(ah, AR_CR));
251

252 253
	ath9k_ps_restore(sc);

254 255 256
	if (len > DMA_BUF_LEN)
		len = DMA_BUF_LEN;

257 258 259
	retval = simple_read_from_buffer(user_buf, count, ppos, buf, len);
	kfree(buf);
	return retval;
260 261 262 263 264
}

static const struct file_operations fops_dma = {
	.read = read_file_dma,
	.open = ath9k_debugfs_open,
265 266
	.owner = THIS_MODULE,
	.llseek = default_llseek,
267 268
};

269 270 271 272

void ath_debug_stat_interrupt(struct ath_softc *sc, enum ath9k_int status)
{
	if (status)
S
Sujith 已提交
273
		sc->debug.stats.istats.total++;
274 275 276 277 278
	if (sc->sc_ah->caps.hw_caps & ATH9K_HW_CAP_EDMA) {
		if (status & ATH9K_INT_RXLP)
			sc->debug.stats.istats.rxlp++;
		if (status & ATH9K_INT_RXHP)
			sc->debug.stats.istats.rxhp++;
279 280
		if (status & ATH9K_INT_BB_WATCHDOG)
			sc->debug.stats.istats.bb_watchdog++;
281 282 283 284
	} else {
		if (status & ATH9K_INT_RX)
			sc->debug.stats.istats.rxok++;
	}
285
	if (status & ATH9K_INT_RXEOL)
S
Sujith 已提交
286
		sc->debug.stats.istats.rxeol++;
287
	if (status & ATH9K_INT_RXORN)
S
Sujith 已提交
288
		sc->debug.stats.istats.rxorn++;
289
	if (status & ATH9K_INT_TX)
S
Sujith 已提交
290
		sc->debug.stats.istats.txok++;
291
	if (status & ATH9K_INT_TXURN)
S
Sujith 已提交
292
		sc->debug.stats.istats.txurn++;
293
	if (status & ATH9K_INT_MIB)
S
Sujith 已提交
294
		sc->debug.stats.istats.mib++;
295
	if (status & ATH9K_INT_RXPHY)
S
Sujith 已提交
296
		sc->debug.stats.istats.rxphyerr++;
297
	if (status & ATH9K_INT_RXKCM)
S
Sujith 已提交
298
		sc->debug.stats.istats.rx_keycache_miss++;
299
	if (status & ATH9K_INT_SWBA)
S
Sujith 已提交
300
		sc->debug.stats.istats.swba++;
301
	if (status & ATH9K_INT_BMISS)
S
Sujith 已提交
302
		sc->debug.stats.istats.bmiss++;
303
	if (status & ATH9K_INT_BNR)
S
Sujith 已提交
304
		sc->debug.stats.istats.bnr++;
305
	if (status & ATH9K_INT_CST)
S
Sujith 已提交
306
		sc->debug.stats.istats.cst++;
307
	if (status & ATH9K_INT_GTT)
S
Sujith 已提交
308
		sc->debug.stats.istats.gtt++;
309
	if (status & ATH9K_INT_TIM)
S
Sujith 已提交
310
		sc->debug.stats.istats.tim++;
311
	if (status & ATH9K_INT_CABEND)
S
Sujith 已提交
312
		sc->debug.stats.istats.cabend++;
313
	if (status & ATH9K_INT_DTIMSYNC)
S
Sujith 已提交
314
		sc->debug.stats.istats.dtimsync++;
315
	if (status & ATH9K_INT_DTIM)
S
Sujith 已提交
316
		sc->debug.stats.istats.dtim++;
317 318 319 320 321 322 323 324 325
}

static ssize_t read_file_interrupt(struct file *file, char __user *user_buf,
				   size_t count, loff_t *ppos)
{
	struct ath_softc *sc = file->private_data;
	char buf[512];
	unsigned int len = 0;

326 327 328 329 330
	if (sc->sc_ah->caps.hw_caps & ATH9K_HW_CAP_EDMA) {
		len += snprintf(buf + len, sizeof(buf) - len,
			"%8s: %10u\n", "RXLP", sc->debug.stats.istats.rxlp);
		len += snprintf(buf + len, sizeof(buf) - len,
			"%8s: %10u\n", "RXHP", sc->debug.stats.istats.rxhp);
331 332 333
		len += snprintf(buf + len, sizeof(buf) - len,
			"%8s: %10u\n", "WATCHDOG",
			sc->debug.stats.istats.bb_watchdog);
334 335 336 337
	} else {
		len += snprintf(buf + len, sizeof(buf) - len,
			"%8s: %10u\n", "RX", sc->debug.stats.istats.rxok);
	}
338
	len += snprintf(buf + len, sizeof(buf) - len,
S
Sujith 已提交
339
		"%8s: %10u\n", "RXEOL", sc->debug.stats.istats.rxeol);
340
	len += snprintf(buf + len, sizeof(buf) - len,
S
Sujith 已提交
341
		"%8s: %10u\n", "RXORN", sc->debug.stats.istats.rxorn);
342
	len += snprintf(buf + len, sizeof(buf) - len,
S
Sujith 已提交
343
		"%8s: %10u\n", "TX", sc->debug.stats.istats.txok);
344
	len += snprintf(buf + len, sizeof(buf) - len,
S
Sujith 已提交
345
		"%8s: %10u\n", "TXURN", sc->debug.stats.istats.txurn);
346
	len += snprintf(buf + len, sizeof(buf) - len,
S
Sujith 已提交
347
		"%8s: %10u\n", "MIB", sc->debug.stats.istats.mib);
348
	len += snprintf(buf + len, sizeof(buf) - len,
S
Sujith 已提交
349
		"%8s: %10u\n", "RXPHY", sc->debug.stats.istats.rxphyerr);
350
	len += snprintf(buf + len, sizeof(buf) - len,
S
Sujith 已提交
351
		"%8s: %10u\n", "RXKCM", sc->debug.stats.istats.rx_keycache_miss);
352
	len += snprintf(buf + len, sizeof(buf) - len,
S
Sujith 已提交
353
		"%8s: %10u\n", "SWBA", sc->debug.stats.istats.swba);
354
	len += snprintf(buf + len, sizeof(buf) - len,
S
Sujith 已提交
355
		"%8s: %10u\n", "BMISS", sc->debug.stats.istats.bmiss);
356
	len += snprintf(buf + len, sizeof(buf) - len,
S
Sujith 已提交
357
		"%8s: %10u\n", "BNR", sc->debug.stats.istats.bnr);
358
	len += snprintf(buf + len, sizeof(buf) - len,
S
Sujith 已提交
359
		"%8s: %10u\n", "CST", sc->debug.stats.istats.cst);
360
	len += snprintf(buf + len, sizeof(buf) - len,
S
Sujith 已提交
361
		"%8s: %10u\n", "GTT", sc->debug.stats.istats.gtt);
362
	len += snprintf(buf + len, sizeof(buf) - len,
S
Sujith 已提交
363
		"%8s: %10u\n", "TIM", sc->debug.stats.istats.tim);
364
	len += snprintf(buf + len, sizeof(buf) - len,
S
Sujith 已提交
365
		"%8s: %10u\n", "CABEND", sc->debug.stats.istats.cabend);
366
	len += snprintf(buf + len, sizeof(buf) - len,
S
Sujith 已提交
367
		"%8s: %10u\n", "DTIMSYNC", sc->debug.stats.istats.dtimsync);
368
	len += snprintf(buf + len, sizeof(buf) - len,
S
Sujith 已提交
369
		"%8s: %10u\n", "DTIM", sc->debug.stats.istats.dtim);
370
	len += snprintf(buf + len, sizeof(buf) - len,
S
Sujith 已提交
371
		"%8s: %10u\n", "TOTAL", sc->debug.stats.istats.total);
372

373 374 375
	if (len > sizeof(buf))
		len = sizeof(buf);

376 377 378 379 380 381
	return simple_read_from_buffer(user_buf, count, ppos, buf, len);
}

static const struct file_operations fops_interrupt = {
	.read = read_file_interrupt,
	.open = ath9k_debugfs_open,
382 383
	.owner = THIS_MODULE,
	.llseek = default_llseek,
384 385
};

386
void ath_debug_stat_rc(struct ath_softc *sc, int final_rate)
387
{
388
	struct ath_rc_stats *stats;
389

390
	stats = &sc->debug.stats.rcstats[final_rate];
391
	stats->success++;
392 393
}

394
void ath_debug_stat_retries(struct ath_softc *sc, int rix,
S
Sujith 已提交
395
			    int xretries, int retries, u8 per)
396
{
397
	struct ath_rc_stats *stats = &sc->debug.stats.rcstats[rix];
398

399 400 401
	stats->xretries += xretries;
	stats->retries += retries;
	stats->per = per;
402 403
}

404 405
static ssize_t read_file_rcstat(struct file *file, char __user *user_buf,
				size_t count, loff_t *ppos)
406 407
{
	struct ath_softc *sc = file->private_data;
408 409
	char *buf;
	unsigned int len = 0, max;
410
	int i = 0;
411
	ssize_t retval;
412

413 414
	if (sc->cur_rate_table == NULL)
		return 0;
415

416 417
	max = 80 + sc->cur_rate_table->rate_cnt * 1024 + 1;
	buf = kmalloc(max, GFP_KERNEL);
418
	if (buf == NULL)
419
		return -ENOMEM;
420

421 422 423 424
	len += sprintf(buf, "%6s %6s %6s "
		       "%10s %10s %10s %10s\n",
		       "HT", "MCS", "Rate",
		       "Success", "Retries", "XRetries", "PER");
425 426

	for (i = 0; i < sc->cur_rate_table->rate_cnt; i++) {
427 428
		u32 ratekbps = sc->cur_rate_table->info[i].ratekbps;
		struct ath_rc_stats *stats = &sc->debug.stats.rcstats[i];
429 430 431 432 433 434 435 436 437 438 439 440 441 442 443 444 445 446
		char mcs[5];
		char htmode[5];
		int used_mcs = 0, used_htmode = 0;

		if (WLAN_RC_PHY_HT(sc->cur_rate_table->info[i].phy)) {
			used_mcs = snprintf(mcs, 5, "%d",
				sc->cur_rate_table->info[i].ratecode);

			if (WLAN_RC_PHY_40(sc->cur_rate_table->info[i].phy))
				used_htmode = snprintf(htmode, 5, "HT40");
			else if (WLAN_RC_PHY_20(sc->cur_rate_table->info[i].phy))
				used_htmode = snprintf(htmode, 5, "HT20");
			else
				used_htmode = snprintf(htmode, 5, "????");
		}

		mcs[used_mcs] = '\0';
		htmode[used_htmode] = '\0';
447 448

		len += snprintf(buf + len, max - len,
449 450 451 452 453 454 455 456 457
			"%6s %6s %3u.%d: "
			"%10u %10u %10u %10u\n",
			htmode,
			mcs,
			ratekbps / 1000,
			(ratekbps % 1000) / 100,
			stats->success,
			stats->retries,
			stats->xretries,
458
			stats->per);
459 460
	}

461 462 463
	if (len > max)
		len = max;

464 465 466
	retval = simple_read_from_buffer(user_buf, count, ppos, buf, len);
	kfree(buf);
	return retval;
467 468 469 470 471
}

static const struct file_operations fops_rcstat = {
	.read = read_file_rcstat,
	.open = ath9k_debugfs_open,
472 473
	.owner = THIS_MODULE,
	.llseek = default_llseek,
474
};
475

476 477 478 479 480 481 482 483 484 485 486 487 488 489 490 491 492 493 494 495 496 497 498 499 500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516 517
static const char * ath_wiphy_state_str(enum ath_wiphy_state state)
{
	switch (state) {
	case ATH_WIPHY_INACTIVE:
		return "INACTIVE";
	case ATH_WIPHY_ACTIVE:
		return "ACTIVE";
	case ATH_WIPHY_PAUSING:
		return "PAUSING";
	case ATH_WIPHY_PAUSED:
		return "PAUSED";
	case ATH_WIPHY_SCAN:
		return "SCAN";
	}
	return "?";
}

static ssize_t read_file_wiphy(struct file *file, char __user *user_buf,
			       size_t count, loff_t *ppos)
{
	struct ath_softc *sc = file->private_data;
	char buf[512];
	unsigned int len = 0;
	int i;
	u8 addr[ETH_ALEN];

	len += snprintf(buf + len, sizeof(buf) - len,
			"primary: %s (%s chan=%d ht=%d)\n",
			wiphy_name(sc->pri_wiphy->hw->wiphy),
			ath_wiphy_state_str(sc->pri_wiphy->state),
			sc->pri_wiphy->chan_idx, sc->pri_wiphy->chan_is_ht);
	for (i = 0; i < sc->num_sec_wiphy; i++) {
		struct ath_wiphy *aphy = sc->sec_wiphy[i];
		if (aphy == NULL)
			continue;
		len += snprintf(buf + len, sizeof(buf) - len,
				"secondary: %s (%s chan=%d ht=%d)\n",
				wiphy_name(aphy->hw->wiphy),
				ath_wiphy_state_str(aphy->state),
				aphy->chan_idx, aphy->chan_is_ht);
	}

518 519
	put_unaligned_le32(REG_READ_D(sc->sc_ah, AR_STA_ID0), addr);
	put_unaligned_le16(REG_READ_D(sc->sc_ah, AR_STA_ID1) & 0xffff, addr + 4);
520 521
	len += snprintf(buf + len, sizeof(buf) - len,
			"addr: %pM\n", addr);
522 523
	put_unaligned_le32(REG_READ_D(sc->sc_ah, AR_BSSMSKL), addr);
	put_unaligned_le16(REG_READ_D(sc->sc_ah, AR_BSSMSKU) & 0xffff, addr + 4);
524 525 526
	len += snprintf(buf + len, sizeof(buf) - len,
			"addrmask: %pM\n", addr);

527 528 529
	if (len > sizeof(buf))
		len = sizeof(buf);

530 531 532 533 534 535 536 537 538 539 540 541 542 543 544 545 546 547 548 549 550 551 552 553 554 555 556 557 558 559 560 561 562 563 564 565 566 567 568 569 570 571 572 573 574 575 576 577 578 579 580 581 582 583 584 585 586 587 588 589 590 591 592 593 594 595 596 597 598 599 600 601 602 603 604 605 606 607 608 609 610 611 612 613 614 615 616 617 618 619 620 621 622 623 624 625 626 627 628 629 630 631
	return simple_read_from_buffer(user_buf, count, ppos, buf, len);
}

static struct ath_wiphy * get_wiphy(struct ath_softc *sc, const char *name)
{
	int i;
	if (strcmp(name, wiphy_name(sc->pri_wiphy->hw->wiphy)) == 0)
		return sc->pri_wiphy;
	for (i = 0; i < sc->num_sec_wiphy; i++) {
		struct ath_wiphy *aphy = sc->sec_wiphy[i];
		if (aphy && strcmp(name, wiphy_name(aphy->hw->wiphy)) == 0)
			return aphy;
	}
	return NULL;
}

static int del_wiphy(struct ath_softc *sc, const char *name)
{
	struct ath_wiphy *aphy = get_wiphy(sc, name);
	if (!aphy)
		return -ENOENT;
	return ath9k_wiphy_del(aphy);
}

static int pause_wiphy(struct ath_softc *sc, const char *name)
{
	struct ath_wiphy *aphy = get_wiphy(sc, name);
	if (!aphy)
		return -ENOENT;
	return ath9k_wiphy_pause(aphy);
}

static int unpause_wiphy(struct ath_softc *sc, const char *name)
{
	struct ath_wiphy *aphy = get_wiphy(sc, name);
	if (!aphy)
		return -ENOENT;
	return ath9k_wiphy_unpause(aphy);
}

static int select_wiphy(struct ath_softc *sc, const char *name)
{
	struct ath_wiphy *aphy = get_wiphy(sc, name);
	if (!aphy)
		return -ENOENT;
	return ath9k_wiphy_select(aphy);
}

static int schedule_wiphy(struct ath_softc *sc, const char *msec)
{
	ath9k_wiphy_set_scheduler(sc, simple_strtoul(msec, NULL, 0));
	return 0;
}

static ssize_t write_file_wiphy(struct file *file, const char __user *user_buf,
				size_t count, loff_t *ppos)
{
	struct ath_softc *sc = file->private_data;
	char buf[50];
	size_t len;

	len = min(count, sizeof(buf) - 1);
	if (copy_from_user(buf, user_buf, len))
		return -EFAULT;
	buf[len] = '\0';
	if (len > 0 && buf[len - 1] == '\n')
		buf[len - 1] = '\0';

	if (strncmp(buf, "add", 3) == 0) {
		int res = ath9k_wiphy_add(sc);
		if (res < 0)
			return res;
	} else if (strncmp(buf, "del=", 4) == 0) {
		int res = del_wiphy(sc, buf + 4);
		if (res < 0)
			return res;
	} else if (strncmp(buf, "pause=", 6) == 0) {
		int res = pause_wiphy(sc, buf + 6);
		if (res < 0)
			return res;
	} else if (strncmp(buf, "unpause=", 8) == 0) {
		int res = unpause_wiphy(sc, buf + 8);
		if (res < 0)
			return res;
	} else if (strncmp(buf, "select=", 7) == 0) {
		int res = select_wiphy(sc, buf + 7);
		if (res < 0)
			return res;
	} else if (strncmp(buf, "schedule=", 9) == 0) {
		int res = schedule_wiphy(sc, buf + 9);
		if (res < 0)
			return res;
	} else
		return -EOPNOTSUPP;

	return count;
}

static const struct file_operations fops_wiphy = {
	.read = read_file_wiphy,
	.write = write_file_wiphy,
	.open = ath9k_debugfs_open,
632 633
	.owner = THIS_MODULE,
	.llseek = default_llseek,
634 635
};

S
Sujith 已提交
636 637 638 639
#define PR(str, elem)							\
	do {								\
		len += snprintf(buf + len, size - len,			\
				"%s%13u%11u%10u%10u\n", str,		\
640 641 642 643
		sc->debug.stats.txstats[sc->tx.hwq_map[WME_AC_BE]].elem, \
		sc->debug.stats.txstats[sc->tx.hwq_map[WME_AC_BK]].elem, \
		sc->debug.stats.txstats[sc->tx.hwq_map[WME_AC_VI]].elem, \
		sc->debug.stats.txstats[sc->tx.hwq_map[WME_AC_VO]].elem); \
S
Sujith 已提交
644 645 646 647 648 649 650 651 652 653 654 655
} while(0)

static ssize_t read_file_xmit(struct file *file, char __user *user_buf,
			      size_t count, loff_t *ppos)
{
	struct ath_softc *sc = file->private_data;
	char *buf;
	unsigned int len = 0, size = 2048;
	ssize_t retval = 0;

	buf = kzalloc(size, GFP_KERNEL);
	if (buf == NULL)
656
		return -ENOMEM;
S
Sujith 已提交
657 658 659 660 661 662 663 664 665 666 667 668 669 670 671 672 673

	len += sprintf(buf, "%30s %10s%10s%10s\n\n", "BE", "BK", "VI", "VO");

	PR("MPDUs Queued:    ", queued);
	PR("MPDUs Completed: ", completed);
	PR("Aggregates:      ", a_aggr);
	PR("AMPDUs Queued:   ", a_queued);
	PR("AMPDUs Completed:", a_completed);
	PR("AMPDUs Retried:  ", a_retries);
	PR("AMPDUs XRetried: ", a_xretries);
	PR("FIFO Underrun:   ", fifo_underrun);
	PR("TXOP Exceeded:   ", xtxop);
	PR("TXTIMER Expiry:  ", timer_exp);
	PR("DESC CFG Error:  ", desc_cfg_err);
	PR("DATA Underrun:   ", data_underrun);
	PR("DELIM Underrun:  ", delim_underrun);

674 675 676
	if (len > size)
		len = size;

S
Sujith 已提交
677 678 679 680 681 682 683
	retval = simple_read_from_buffer(user_buf, count, ppos, buf, len);
	kfree(buf);

	return retval;
}

void ath_debug_stat_tx(struct ath_softc *sc, struct ath_txq *txq,
684
		       struct ath_buf *bf, struct ath_tx_status *ts)
S
Sujith 已提交
685 686 687 688 689 690 691 692 693 694
{
	if (bf_isampdu(bf)) {
		if (bf_isxretried(bf))
			TX_STAT_INC(txq->axq_qnum, a_xretries);
		else
			TX_STAT_INC(txq->axq_qnum, a_completed);
	} else {
		TX_STAT_INC(txq->axq_qnum, completed);
	}

695
	if (ts->ts_status & ATH9K_TXERR_FIFO)
S
Sujith 已提交
696
		TX_STAT_INC(txq->axq_qnum, fifo_underrun);
697
	if (ts->ts_status & ATH9K_TXERR_XTXOP)
S
Sujith 已提交
698
		TX_STAT_INC(txq->axq_qnum, xtxop);
699
	if (ts->ts_status & ATH9K_TXERR_TIMER_EXPIRED)
S
Sujith 已提交
700
		TX_STAT_INC(txq->axq_qnum, timer_exp);
701
	if (ts->ts_flags & ATH9K_TX_DESC_CFG_ERR)
S
Sujith 已提交
702
		TX_STAT_INC(txq->axq_qnum, desc_cfg_err);
703
	if (ts->ts_flags & ATH9K_TX_DATA_UNDERRUN)
S
Sujith 已提交
704
		TX_STAT_INC(txq->axq_qnum, data_underrun);
705
	if (ts->ts_flags & ATH9K_TX_DELIM_UNDERRUN)
S
Sujith 已提交
706 707 708 709 710 711
		TX_STAT_INC(txq->axq_qnum, delim_underrun);
}

static const struct file_operations fops_xmit = {
	.read = read_file_xmit,
	.open = ath9k_debugfs_open,
712 713
	.owner = THIS_MODULE,
	.llseek = default_llseek,
S
Sujith 已提交
714
};
715

S
Sujith 已提交
716 717 718 719 720 721 722 723 724 725 726 727 728 729
static ssize_t read_file_recv(struct file *file, char __user *user_buf,
			      size_t count, loff_t *ppos)
{
#define PHY_ERR(s, p) \
	len += snprintf(buf + len, size - len, "%18s : %10u\n", s, \
			sc->debug.stats.rxstats.phy_err_stats[p]);

	struct ath_softc *sc = file->private_data;
	char *buf;
	unsigned int len = 0, size = 1152;
	ssize_t retval = 0;

	buf = kzalloc(size, GFP_KERNEL);
	if (buf == NULL)
730
		return -ENOMEM;
S
Sujith 已提交
731 732 733 734 735 736 737 738 739 740 741 742 743 744 745 746 747 748 749 750 751 752 753 754 755 756 757 758 759 760 761 762 763 764 765 766 767 768 769 770 771 772 773 774 775 776 777 778 779 780

	len += snprintf(buf + len, size - len,
			"%18s : %10u\n", "CRC ERR",
			sc->debug.stats.rxstats.crc_err);
	len += snprintf(buf + len, size - len,
			"%18s : %10u\n", "DECRYPT CRC ERR",
			sc->debug.stats.rxstats.decrypt_crc_err);
	len += snprintf(buf + len, size - len,
			"%18s : %10u\n", "PHY ERR",
			sc->debug.stats.rxstats.phy_err);
	len += snprintf(buf + len, size - len,
			"%18s : %10u\n", "MIC ERR",
			sc->debug.stats.rxstats.mic_err);
	len += snprintf(buf + len, size - len,
			"%18s : %10u\n", "PRE-DELIM CRC ERR",
			sc->debug.stats.rxstats.pre_delim_crc_err);
	len += snprintf(buf + len, size - len,
			"%18s : %10u\n", "POST-DELIM CRC ERR",
			sc->debug.stats.rxstats.post_delim_crc_err);
	len += snprintf(buf + len, size - len,
			"%18s : %10u\n", "DECRYPT BUSY ERR",
			sc->debug.stats.rxstats.decrypt_busy_err);

	PHY_ERR("UNDERRUN", ATH9K_PHYERR_UNDERRUN);
	PHY_ERR("TIMING", ATH9K_PHYERR_TIMING);
	PHY_ERR("PARITY", ATH9K_PHYERR_PARITY);
	PHY_ERR("RATE", ATH9K_PHYERR_RATE);
	PHY_ERR("LENGTH", ATH9K_PHYERR_LENGTH);
	PHY_ERR("RADAR", ATH9K_PHYERR_RADAR);
	PHY_ERR("SERVICE", ATH9K_PHYERR_SERVICE);
	PHY_ERR("TOR", ATH9K_PHYERR_TOR);
	PHY_ERR("OFDM-TIMING", ATH9K_PHYERR_OFDM_TIMING);
	PHY_ERR("OFDM-SIGNAL-PARITY", ATH9K_PHYERR_OFDM_SIGNAL_PARITY);
	PHY_ERR("OFDM-RATE", ATH9K_PHYERR_OFDM_RATE_ILLEGAL);
	PHY_ERR("OFDM-LENGTH", ATH9K_PHYERR_OFDM_LENGTH_ILLEGAL);
	PHY_ERR("OFDM-POWER-DROP", ATH9K_PHYERR_OFDM_POWER_DROP);
	PHY_ERR("OFDM-SERVICE", ATH9K_PHYERR_OFDM_SERVICE);
	PHY_ERR("OFDM-RESTART", ATH9K_PHYERR_OFDM_RESTART);
	PHY_ERR("FALSE-RADAR-EXT", ATH9K_PHYERR_FALSE_RADAR_EXT);
	PHY_ERR("CCK-TIMING", ATH9K_PHYERR_CCK_TIMING);
	PHY_ERR("CCK-HEADER-CRC", ATH9K_PHYERR_CCK_HEADER_CRC);
	PHY_ERR("CCK-RATE", ATH9K_PHYERR_CCK_RATE_ILLEGAL);
	PHY_ERR("CCK-SERVICE", ATH9K_PHYERR_CCK_SERVICE);
	PHY_ERR("CCK-RESTART", ATH9K_PHYERR_CCK_RESTART);
	PHY_ERR("CCK-LENGTH", ATH9K_PHYERR_CCK_LENGTH_ILLEGAL);
	PHY_ERR("CCK-POWER-DROP", ATH9K_PHYERR_CCK_POWER_DROP);
	PHY_ERR("HT-CRC", ATH9K_PHYERR_HT_CRC_ERROR);
	PHY_ERR("HT-LENGTH", ATH9K_PHYERR_HT_LENGTH_ILLEGAL);
	PHY_ERR("HT-RATE", ATH9K_PHYERR_HT_RATE_ILLEGAL);

781 782 783
	if (len > size)
		len = size;

S
Sujith 已提交
784 785 786 787 788 789 790 791
	retval = simple_read_from_buffer(user_buf, count, ppos, buf, len);
	kfree(buf);

	return retval;

#undef PHY_ERR
}

792
void ath_debug_stat_rx(struct ath_softc *sc, struct ath_rx_status *rs)
S
Sujith 已提交
793 794 795 796 797 798
{
#define RX_STAT_INC(c) sc->debug.stats.rxstats.c++
#define RX_PHY_ERR_INC(c) sc->debug.stats.rxstats.phy_err_stats[c]++

	u32 phyerr;

799
	if (rs->rs_status & ATH9K_RXERR_CRC)
S
Sujith 已提交
800
		RX_STAT_INC(crc_err);
801
	if (rs->rs_status & ATH9K_RXERR_DECRYPT)
S
Sujith 已提交
802
		RX_STAT_INC(decrypt_crc_err);
803
	if (rs->rs_status & ATH9K_RXERR_MIC)
S
Sujith 已提交
804
		RX_STAT_INC(mic_err);
805
	if (rs->rs_status & ATH9K_RX_DELIM_CRC_PRE)
S
Sujith 已提交
806
		RX_STAT_INC(pre_delim_crc_err);
807
	if (rs->rs_status & ATH9K_RX_DELIM_CRC_POST)
S
Sujith 已提交
808
		RX_STAT_INC(post_delim_crc_err);
809
	if (rs->rs_status & ATH9K_RX_DECRYPT_BUSY)
S
Sujith 已提交
810 811
		RX_STAT_INC(decrypt_busy_err);

812
	if (rs->rs_status & ATH9K_RXERR_PHY) {
S
Sujith 已提交
813
		RX_STAT_INC(phy_err);
814
		phyerr = rs->rs_phyerr & 0x24;
S
Sujith 已提交
815 816 817 818 819 820 821 822 823 824
		RX_PHY_ERR_INC(phyerr);
	}

#undef RX_STAT_INC
#undef RX_PHY_ERR_INC
}

static const struct file_operations fops_recv = {
	.read = read_file_recv,
	.open = ath9k_debugfs_open,
825 826
	.owner = THIS_MODULE,
	.llseek = default_llseek,
S
Sujith 已提交
827 828
};

829 830 831 832 833 834 835
static ssize_t read_file_regidx(struct file *file, char __user *user_buf,
                                size_t count, loff_t *ppos)
{
	struct ath_softc *sc = file->private_data;
	char buf[32];
	unsigned int len;

836
	len = sprintf(buf, "0x%08x\n", sc->debug.regidx);
837 838 839 840 841 842 843 844 845 846 847 848 849
	return simple_read_from_buffer(user_buf, count, ppos, buf, len);
}

static ssize_t write_file_regidx(struct file *file, const char __user *user_buf,
			     size_t count, loff_t *ppos)
{
	struct ath_softc *sc = file->private_data;
	unsigned long regidx;
	char buf[32];
	ssize_t len;

	len = min(count, sizeof(buf) - 1);
	if (copy_from_user(buf, user_buf, len))
850
		return -EFAULT;
851 852 853 854 855 856 857 858 859 860 861 862 863

	buf[len] = '\0';
	if (strict_strtoul(buf, 0, &regidx))
		return -EINVAL;

	sc->debug.regidx = regidx;
	return count;
}

static const struct file_operations fops_regidx = {
	.read = read_file_regidx,
	.write = write_file_regidx,
	.open = ath9k_debugfs_open,
864 865
	.owner = THIS_MODULE,
	.llseek = default_llseek,
866 867 868 869 870 871 872 873 874 875 876 877
};

static ssize_t read_file_regval(struct file *file, char __user *user_buf,
			     size_t count, loff_t *ppos)
{
	struct ath_softc *sc = file->private_data;
	struct ath_hw *ah = sc->sc_ah;
	char buf[32];
	unsigned int len;
	u32 regval;

	regval = REG_READ_D(ah, sc->debug.regidx);
878
	len = sprintf(buf, "0x%08x\n", regval);
879 880 881 882 883 884 885 886 887 888 889 890 891 892
	return simple_read_from_buffer(user_buf, count, ppos, buf, len);
}

static ssize_t write_file_regval(struct file *file, const char __user *user_buf,
			     size_t count, loff_t *ppos)
{
	struct ath_softc *sc = file->private_data;
	struct ath_hw *ah = sc->sc_ah;
	unsigned long regval;
	char buf[32];
	ssize_t len;

	len = min(count, sizeof(buf) - 1);
	if (copy_from_user(buf, user_buf, len))
893
		return -EFAULT;
894 895 896 897 898 899 900 901 902 903 904 905 906

	buf[len] = '\0';
	if (strict_strtoul(buf, 0, &regval))
		return -EINVAL;

	REG_WRITE_D(ah, sc->debug.regidx, regval);
	return count;
}

static const struct file_operations fops_regval = {
	.read = read_file_regval,
	.write = write_file_regval,
	.open = ath9k_debugfs_open,
907 908
	.owner = THIS_MODULE,
	.llseek = default_llseek,
909 910
};

911
int ath9k_init_debug(struct ath_hw *ah)
912
{
913 914
	struct ath_common *common = ath9k_hw_common(ah);
	struct ath_softc *sc = (struct ath_softc *) common->priv;
915

S
Sujith 已提交
916 917 918
	if (!ath9k_debugfs_root)
		return -ENOENT;

S
Sujith 已提交
919
	sc->debug.debugfs_phy = debugfs_create_dir(wiphy_name(sc->hw->wiphy),
920
						      ath9k_debugfs_root);
S
Sujith 已提交
921
	if (!sc->debug.debugfs_phy)
922
		return -ENOMEM;
923

924
#ifdef CONFIG_ATH_DEBUG
925 926
	if (!debugfs_create_file("debug", S_IRUSR | S_IWUSR,
			sc->debug.debugfs_phy, sc, &fops_debug))
J
Jeff Hansen 已提交
927
		goto err;
928
#endif
J
Jeff Hansen 已提交
929

930 931
	if (!debugfs_create_file("dma", S_IRUSR, sc->debug.debugfs_phy,
			sc, &fops_dma))
932 933
		goto err;

934 935
	if (!debugfs_create_file("interrupt", S_IRUSR, sc->debug.debugfs_phy,
			sc, &fops_interrupt))
936 937
		goto err;

938 939
	if (!debugfs_create_file("rcstat", S_IRUSR, sc->debug.debugfs_phy,
			sc, &fops_rcstat))
940 941
		goto err;

942 943
	if (!debugfs_create_file("wiphy", S_IRUSR | S_IWUSR,
			sc->debug.debugfs_phy, sc, &fops_wiphy))
944 945
		goto err;

946 947
	if (!debugfs_create_file("xmit", S_IRUSR, sc->debug.debugfs_phy,
			sc, &fops_xmit))
948 949
		goto err;

950 951
	if (!debugfs_create_file("recv", S_IRUSR, sc->debug.debugfs_phy,
			sc, &fops_recv))
952 953
		goto err;

954 955
	if (!debugfs_create_file("rx_chainmask", S_IRUSR | S_IWUSR,
			sc->debug.debugfs_phy, sc, &fops_rx_chainmask))
956 957
		goto err;

958 959
	if (!debugfs_create_file("tx_chainmask", S_IRUSR | S_IWUSR,
			sc->debug.debugfs_phy, sc, &fops_tx_chainmask))
960 961
		goto err;

962 963
	if (!debugfs_create_file("regidx", S_IRUSR | S_IWUSR,
			sc->debug.debugfs_phy, sc, &fops_regidx))
S
Sujith 已提交
964 965
		goto err;

966 967
	if (!debugfs_create_file("regval", S_IRUSR | S_IWUSR,
			sc->debug.debugfs_phy, sc, &fops_regval))
S
Sujith 已提交
968 969
		goto err;

970 971 972 973
	if (!debugfs_create_bool("ignore_extcca", S_IRUSR | S_IWUSR,
			sc->debug.debugfs_phy, &ah->config.cwm_ignore_extcca))
		goto err;

974
	sc->debug.regidx = 0;
975 976
	return 0;
err:
977
	ath9k_exit_debug(ah);
978 979 980
	return -ENOMEM;
}

981
void ath9k_exit_debug(struct ath_hw *ah)
982
{
983 984
	struct ath_common *common = ath9k_hw_common(ah);
	struct ath_softc *sc = (struct ath_softc *) common->priv;
985

986
	debugfs_remove_recursive(sc->debug.debugfs_phy);
987 988 989 990 991 992 993 994 995 996 997 998 999 1000 1001
}

int ath9k_debug_create_root(void)
{
	ath9k_debugfs_root = debugfs_create_dir(KBUILD_MODNAME, NULL);
	if (!ath9k_debugfs_root)
		return -ENOENT;

	return 0;
}

void ath9k_debug_remove_root(void)
{
	debugfs_remove(ath9k_debugfs_root);
	ath9k_debugfs_root = NULL;
1002
}