diff options
author | Matti Gottlieb <matti.gottlieb@intel.com> | 2013-11-18 12:06:45 -0500 |
---|---|---|
committer | Johannes Berg <johannes.berg@intel.com> | 2013-12-02 05:51:50 -0500 |
commit | ad38bfc916da6aee9160bfa5335aed8d6c190e39 (patch) | |
tree | bc00170187f5e2036a8c804f546c32d94bdee349 /net/mac80211/debugfs_sta.c | |
parent | 1d940aaab881b0ee62557ffbaad877ac5a1b51db (diff) |
mac80211: Tx frame latency statistics
Measure TX latency and jitter statistics per station per TID.
These Measurements are disabled by default and can be enabled
via debugfs.
Features included for each station's TID:
1. Keep count of the maximum and average latency of Tx frames.
2. Keep track of many frames arrived in a specific time range
(need to enable through debugfs and configure the bins ranges)
Signed-off-by: Matti Gottlieb <matti.gottlieb@intel.com>
Signed-off-by: Johannes Berg <johannes.berg@intel.com>
Diffstat (limited to 'net/mac80211/debugfs_sta.c')
-rw-r--r-- | net/mac80211/debugfs_sta.c | 134 |
1 files changed, 134 insertions, 0 deletions
diff --git a/net/mac80211/debugfs_sta.c b/net/mac80211/debugfs_sta.c index 19c54a44ed47..80194b557a0c 100644 --- a/net/mac80211/debugfs_sta.c +++ b/net/mac80211/debugfs_sta.c | |||
@@ -38,6 +38,13 @@ static const struct file_operations sta_ ##name## _ops = { \ | |||
38 | .llseek = generic_file_llseek, \ | 38 | .llseek = generic_file_llseek, \ |
39 | } | 39 | } |
40 | 40 | ||
41 | #define STA_OPS_W(name) \ | ||
42 | static const struct file_operations sta_ ##name## _ops = { \ | ||
43 | .write = sta_##name##_write, \ | ||
44 | .open = simple_open, \ | ||
45 | .llseek = generic_file_llseek, \ | ||
46 | } | ||
47 | |||
41 | #define STA_OPS_RW(name) \ | 48 | #define STA_OPS_RW(name) \ |
42 | static const struct file_operations sta_ ##name## _ops = { \ | 49 | static const struct file_operations sta_ ##name## _ops = { \ |
43 | .read = sta_##name##_read, \ | 50 | .read = sta_##name##_read, \ |
@@ -388,6 +395,131 @@ static ssize_t sta_last_rx_rate_read(struct file *file, char __user *userbuf, | |||
388 | } | 395 | } |
389 | STA_OPS(last_rx_rate); | 396 | STA_OPS(last_rx_rate); |
390 | 397 | ||
398 | static int | ||
399 | sta_tx_latency_stat_header(struct ieee80211_tx_latency_bin_ranges *tx_latency, | ||
400 | char *buf, int pos, int bufsz) | ||
401 | { | ||
402 | int i; | ||
403 | int range_count = tx_latency->n_ranges; | ||
404 | u32 *bin_ranges = tx_latency->ranges; | ||
405 | |||
406 | pos += scnprintf(buf + pos, bufsz - pos, | ||
407 | "Station\t\t\tTID\tMax\tAvg"); | ||
408 | if (range_count) { | ||
409 | pos += scnprintf(buf + pos, bufsz - pos, | ||
410 | "\t<=%d", bin_ranges[0]); | ||
411 | for (i = 0; i < range_count - 1; i++) | ||
412 | pos += scnprintf(buf + pos, bufsz - pos, "\t%d-%d", | ||
413 | bin_ranges[i], bin_ranges[i+1]); | ||
414 | pos += scnprintf(buf + pos, bufsz - pos, | ||
415 | "\t%d<", bin_ranges[range_count - 1]); | ||
416 | } | ||
417 | |||
418 | pos += scnprintf(buf + pos, bufsz - pos, "\n"); | ||
419 | |||
420 | return pos; | ||
421 | } | ||
422 | |||
423 | static int | ||
424 | sta_tx_latency_stat_table(struct ieee80211_tx_latency_bin_ranges *tx_lat_range, | ||
425 | struct ieee80211_tx_latency_stat *tx_lat, | ||
426 | char *buf, int pos, int bufsz, int tid) | ||
427 | { | ||
428 | u32 avg = 0; | ||
429 | int j; | ||
430 | int bin_count = tx_lat->bin_count; | ||
431 | |||
432 | pos += scnprintf(buf + pos, bufsz - pos, "\t\t\t%d", tid); | ||
433 | /* make sure you don't divide in 0 */ | ||
434 | if (tx_lat->counter) | ||
435 | avg = tx_lat->sum / tx_lat->counter; | ||
436 | |||
437 | pos += scnprintf(buf + pos, bufsz - pos, "\t%d\t%d", | ||
438 | tx_lat->max, avg); | ||
439 | |||
440 | if (tx_lat_range->n_ranges && tx_lat->bins) | ||
441 | for (j = 0; j < bin_count; j++) | ||
442 | pos += scnprintf(buf + pos, bufsz - pos, | ||
443 | "\t%d", tx_lat->bins[j]); | ||
444 | pos += scnprintf(buf + pos, bufsz - pos, "\n"); | ||
445 | |||
446 | return pos; | ||
447 | } | ||
448 | |||
449 | /* | ||
450 | * Output Tx latency statistics station && restart all statistics information | ||
451 | */ | ||
452 | static ssize_t sta_tx_latency_stat_read(struct file *file, | ||
453 | char __user *userbuf, | ||
454 | size_t count, loff_t *ppos) | ||
455 | { | ||
456 | struct sta_info *sta = file->private_data; | ||
457 | struct ieee80211_local *local = sta->local; | ||
458 | struct ieee80211_tx_latency_bin_ranges *tx_latency; | ||
459 | char *buf; | ||
460 | int bufsz, ret, i; | ||
461 | int pos = 0; | ||
462 | |||
463 | bufsz = 20 * IEEE80211_NUM_TIDS * | ||
464 | sizeof(struct ieee80211_tx_latency_stat); | ||
465 | buf = kzalloc(bufsz, GFP_KERNEL); | ||
466 | if (!buf) | ||
467 | return -ENOMEM; | ||
468 | |||
469 | rcu_read_lock(); | ||
470 | |||
471 | tx_latency = rcu_dereference(local->tx_latency); | ||
472 | |||
473 | if (!sta->tx_lat) { | ||
474 | pos += scnprintf(buf + pos, bufsz - pos, | ||
475 | "Tx latency statistics are not enabled\n"); | ||
476 | goto unlock; | ||
477 | } | ||
478 | |||
479 | pos = sta_tx_latency_stat_header(tx_latency, buf, pos, bufsz); | ||
480 | |||
481 | pos += scnprintf(buf + pos, bufsz - pos, "%pM\n", sta->sta.addr); | ||
482 | for (i = 0; i < IEEE80211_NUM_TIDS; i++) | ||
483 | pos = sta_tx_latency_stat_table(tx_latency, &sta->tx_lat[i], | ||
484 | buf, pos, bufsz, i); | ||
485 | unlock: | ||
486 | rcu_read_unlock(); | ||
487 | |||
488 | ret = simple_read_from_buffer(userbuf, count, ppos, buf, pos); | ||
489 | kfree(buf); | ||
490 | |||
491 | return ret; | ||
492 | } | ||
493 | STA_OPS(tx_latency_stat); | ||
494 | |||
495 | static ssize_t sta_tx_latency_stat_reset_write(struct file *file, | ||
496 | const char __user *userbuf, | ||
497 | size_t count, loff_t *ppos) | ||
498 | { | ||
499 | u32 *bins; | ||
500 | int bin_count; | ||
501 | struct sta_info *sta = file->private_data; | ||
502 | int i; | ||
503 | |||
504 | if (!sta->tx_lat) | ||
505 | return -EINVAL; | ||
506 | |||
507 | for (i = 0; i < IEEE80211_NUM_TIDS; i++) { | ||
508 | bins = sta->tx_lat[i].bins; | ||
509 | bin_count = sta->tx_lat[i].bin_count; | ||
510 | |||
511 | sta->tx_lat[i].max = 0; | ||
512 | sta->tx_lat[i].sum = 0; | ||
513 | sta->tx_lat[i].counter = 0; | ||
514 | |||
515 | if (bin_count) | ||
516 | memset(bins, 0, bin_count * sizeof(u32)); | ||
517 | } | ||
518 | |||
519 | return count; | ||
520 | } | ||
521 | STA_OPS_W(tx_latency_stat_reset); | ||
522 | |||
391 | #define DEBUGFS_ADD(name) \ | 523 | #define DEBUGFS_ADD(name) \ |
392 | debugfs_create_file(#name, 0400, \ | 524 | debugfs_create_file(#name, 0400, \ |
393 | sta->debugfs.dir, sta, &sta_ ##name## _ops); | 525 | sta->debugfs.dir, sta, &sta_ ##name## _ops); |
@@ -441,6 +573,8 @@ void ieee80211_sta_debugfs_add(struct sta_info *sta) | |||
441 | DEBUGFS_ADD(last_ack_signal); | 573 | DEBUGFS_ADD(last_ack_signal); |
442 | DEBUGFS_ADD(current_tx_rate); | 574 | DEBUGFS_ADD(current_tx_rate); |
443 | DEBUGFS_ADD(last_rx_rate); | 575 | DEBUGFS_ADD(last_rx_rate); |
576 | DEBUGFS_ADD(tx_latency_stat); | ||
577 | DEBUGFS_ADD(tx_latency_stat_reset); | ||
444 | 578 | ||
445 | DEBUGFS_ADD_COUNTER(rx_packets, rx_packets); | 579 | DEBUGFS_ADD_COUNTER(rx_packets, rx_packets); |
446 | DEBUGFS_ADD_COUNTER(tx_packets, tx_packets); | 580 | DEBUGFS_ADD_COUNTER(tx_packets, tx_packets); |