aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/net/wireless
diff options
context:
space:
mode:
authorJouni Malinen <jouni.malinen@atheros.com>2009-03-03 12:23:40 -0500
committerJohn W. Linville <linville@tuxdriver.com>2009-03-05 14:39:48 -0500
commit39d89cd34d9900cd2415f46e179b91cdd14b15fe (patch)
tree73e7e7978a9926d7d10aeb5de69defdedb4abc23 /drivers/net/wireless
parentf98c3bd24161e9aaa73b9cd4dc6b1742c085ac17 (diff)
ath9k: Add a debugfs interface for controlling virtual wiphys
debugfs ath9k/phy#/wiphy can be used to show the current list of virtual wiphys and to add/remove virtual wiphys. Eventually, this interface could be replaced with a cfg80211/nl80211 command that is passed through mac80211. For example: # cat /debug/ath9k/phy0/wiphy primary: phy0 # echo add > /debug/ath9k/phy0/wiphy # cat /debug/ath9k/phy0/wiphy primary: phy0 secondary: phy1 # echo del=phy1 > /debug/ath9k/phy0/wiphy # cat /debug/ath9k/phy0/wiphy primary: phy0 In addition, following commands can be used to test pausing and unpausing of the virtual wiphys: pause=phy1 unpause=phy1 select=phy1 (select pauses and unpauses wiphys automatically based on channel) schedule=500 (set wiphy scheduling interval in msec; 0 = disable; default value: 500) Signed-off-by: Jouni Malinen <jouni.malinen@atheros.com> Signed-off-by: John W. Linville <linville@tuxdriver.com>
Diffstat (limited to 'drivers/net/wireless')
-rw-r--r--drivers/net/wireless/ath9k/debug.c164
-rw-r--r--drivers/net/wireless/ath9k/debug.h1
2 files changed, 165 insertions, 0 deletions
diff --git a/drivers/net/wireless/ath9k/debug.c b/drivers/net/wireless/ath9k/debug.c
index 0c422c50e4f0..8d91422106d9 100644
--- a/drivers/net/wireless/ath9k/debug.c
+++ b/drivers/net/wireless/ath9k/debug.c
@@ -330,6 +330,163 @@ static const struct file_operations fops_rcstat = {
330 .owner = THIS_MODULE 330 .owner = THIS_MODULE
331}; 331};
332 332
333static const char * ath_wiphy_state_str(enum ath_wiphy_state state)
334{
335 switch (state) {
336 case ATH_WIPHY_INACTIVE:
337 return "INACTIVE";
338 case ATH_WIPHY_ACTIVE:
339 return "ACTIVE";
340 case ATH_WIPHY_PAUSING:
341 return "PAUSING";
342 case ATH_WIPHY_PAUSED:
343 return "PAUSED";
344 case ATH_WIPHY_SCAN:
345 return "SCAN";
346 }
347 return "?";
348}
349
350static ssize_t read_file_wiphy(struct file *file, char __user *user_buf,
351 size_t count, loff_t *ppos)
352{
353 struct ath_softc *sc = file->private_data;
354 char buf[512];
355 unsigned int len = 0;
356 int i;
357 u8 addr[ETH_ALEN];
358
359 len += snprintf(buf + len, sizeof(buf) - len,
360 "primary: %s (%s chan=%d ht=%d)\n",
361 wiphy_name(sc->pri_wiphy->hw->wiphy),
362 ath_wiphy_state_str(sc->pri_wiphy->state),
363 sc->pri_wiphy->chan_idx, sc->pri_wiphy->chan_is_ht);
364 for (i = 0; i < sc->num_sec_wiphy; i++) {
365 struct ath_wiphy *aphy = sc->sec_wiphy[i];
366 if (aphy == NULL)
367 continue;
368 len += snprintf(buf + len, sizeof(buf) - len,
369 "secondary: %s (%s chan=%d ht=%d)\n",
370 wiphy_name(aphy->hw->wiphy),
371 ath_wiphy_state_str(aphy->state),
372 aphy->chan_idx, aphy->chan_is_ht);
373 }
374
375 put_unaligned_le32(REG_READ(sc->sc_ah, AR_STA_ID0), addr);
376 put_unaligned_le16(REG_READ(sc->sc_ah, AR_STA_ID1) & 0xffff, addr + 4);
377 len += snprintf(buf + len, sizeof(buf) - len,
378 "addr: %pM\n", addr);
379 put_unaligned_le32(REG_READ(sc->sc_ah, AR_BSSMSKL), addr);
380 put_unaligned_le16(REG_READ(sc->sc_ah, AR_BSSMSKU) & 0xffff, addr + 4);
381 len += snprintf(buf + len, sizeof(buf) - len,
382 "addrmask: %pM\n", addr);
383
384 return simple_read_from_buffer(user_buf, count, ppos, buf, len);
385}
386
387static struct ath_wiphy * get_wiphy(struct ath_softc *sc, const char *name)
388{
389 int i;
390 if (strcmp(name, wiphy_name(sc->pri_wiphy->hw->wiphy)) == 0)
391 return sc->pri_wiphy;
392 for (i = 0; i < sc->num_sec_wiphy; i++) {
393 struct ath_wiphy *aphy = sc->sec_wiphy[i];
394 if (aphy && strcmp(name, wiphy_name(aphy->hw->wiphy)) == 0)
395 return aphy;
396 }
397 return NULL;
398}
399
400static int del_wiphy(struct ath_softc *sc, const char *name)
401{
402 struct ath_wiphy *aphy = get_wiphy(sc, name);
403 if (!aphy)
404 return -ENOENT;
405 return ath9k_wiphy_del(aphy);
406}
407
408static int pause_wiphy(struct ath_softc *sc, const char *name)
409{
410 struct ath_wiphy *aphy = get_wiphy(sc, name);
411 if (!aphy)
412 return -ENOENT;
413 return ath9k_wiphy_pause(aphy);
414}
415
416static int unpause_wiphy(struct ath_softc *sc, const char *name)
417{
418 struct ath_wiphy *aphy = get_wiphy(sc, name);
419 if (!aphy)
420 return -ENOENT;
421 return ath9k_wiphy_unpause(aphy);
422}
423
424static int select_wiphy(struct ath_softc *sc, const char *name)
425{
426 struct ath_wiphy *aphy = get_wiphy(sc, name);
427 if (!aphy)
428 return -ENOENT;
429 return ath9k_wiphy_select(aphy);
430}
431
432static int schedule_wiphy(struct ath_softc *sc, const char *msec)
433{
434 ath9k_wiphy_set_scheduler(sc, simple_strtoul(msec, NULL, 0));
435 return 0;
436}
437
438static ssize_t write_file_wiphy(struct file *file, const char __user *user_buf,
439 size_t count, loff_t *ppos)
440{
441 struct ath_softc *sc = file->private_data;
442 char buf[50];
443 size_t len;
444
445 len = min(count, sizeof(buf) - 1);
446 if (copy_from_user(buf, user_buf, len))
447 return -EFAULT;
448 buf[len] = '\0';
449 if (len > 0 && buf[len - 1] == '\n')
450 buf[len - 1] = '\0';
451
452 if (strncmp(buf, "add", 3) == 0) {
453 int res = ath9k_wiphy_add(sc);
454 if (res < 0)
455 return res;
456 } else if (strncmp(buf, "del=", 4) == 0) {
457 int res = del_wiphy(sc, buf + 4);
458 if (res < 0)
459 return res;
460 } else if (strncmp(buf, "pause=", 6) == 0) {
461 int res = pause_wiphy(sc, buf + 6);
462 if (res < 0)
463 return res;
464 } else if (strncmp(buf, "unpause=", 8) == 0) {
465 int res = unpause_wiphy(sc, buf + 8);
466 if (res < 0)
467 return res;
468 } else if (strncmp(buf, "select=", 7) == 0) {
469 int res = select_wiphy(sc, buf + 7);
470 if (res < 0)
471 return res;
472 } else if (strncmp(buf, "schedule=", 9) == 0) {
473 int res = schedule_wiphy(sc, buf + 9);
474 if (res < 0)
475 return res;
476 } else
477 return -EOPNOTSUPP;
478
479 return count;
480}
481
482static const struct file_operations fops_wiphy = {
483 .read = read_file_wiphy,
484 .write = write_file_wiphy,
485 .open = ath9k_debugfs_open,
486 .owner = THIS_MODULE
487};
488
489
333int ath9k_init_debug(struct ath_softc *sc) 490int ath9k_init_debug(struct ath_softc *sc)
334{ 491{
335 sc->debug.debug_mask = ath9k_debug; 492 sc->debug.debug_mask = ath9k_debug;
@@ -362,6 +519,12 @@ int ath9k_init_debug(struct ath_softc *sc)
362 if (!sc->debug.debugfs_rcstat) 519 if (!sc->debug.debugfs_rcstat)
363 goto err; 520 goto err;
364 521
522 sc->debug.debugfs_wiphy = debugfs_create_file(
523 "wiphy", S_IRUGO | S_IWUSR, sc->debug.debugfs_phy, sc,
524 &fops_wiphy);
525 if (!sc->debug.debugfs_wiphy)
526 goto err;
527
365 return 0; 528 return 0;
366err: 529err:
367 ath9k_exit_debug(sc); 530 ath9k_exit_debug(sc);
@@ -370,6 +533,7 @@ err:
370 533
371void ath9k_exit_debug(struct ath_softc *sc) 534void ath9k_exit_debug(struct ath_softc *sc)
372{ 535{
536 debugfs_remove(sc->debug.debugfs_wiphy);
373 debugfs_remove(sc->debug.debugfs_rcstat); 537 debugfs_remove(sc->debug.debugfs_rcstat);
374 debugfs_remove(sc->debug.debugfs_interrupt); 538 debugfs_remove(sc->debug.debugfs_interrupt);
375 debugfs_remove(sc->debug.debugfs_dma); 539 debugfs_remove(sc->debug.debugfs_dma);
diff --git a/drivers/net/wireless/ath9k/debug.h b/drivers/net/wireless/ath9k/debug.h
index 01681f2d0549..2a33d74fdbee 100644
--- a/drivers/net/wireless/ath9k/debug.h
+++ b/drivers/net/wireless/ath9k/debug.h
@@ -107,6 +107,7 @@ struct ath9k_debug {
107 struct dentry *debugfs_dma; 107 struct dentry *debugfs_dma;
108 struct dentry *debugfs_interrupt; 108 struct dentry *debugfs_interrupt;
109 struct dentry *debugfs_rcstat; 109 struct dentry *debugfs_rcstat;
110 struct dentry *debugfs_wiphy;
110 struct ath_stats stats; 111 struct ath_stats stats;
111}; 112};
112 113