diff options
Diffstat (limited to 'tools/perf/builtin-stat.c')
-rw-r--r-- | tools/perf/builtin-stat.c | 118 |
1 files changed, 73 insertions, 45 deletions
diff --git a/tools/perf/builtin-stat.c b/tools/perf/builtin-stat.c index a482a191a0c..21c02522249 100644 --- a/tools/perf/builtin-stat.c +++ b/tools/perf/builtin-stat.c | |||
@@ -43,11 +43,13 @@ | |||
43 | #include "util/parse-options.h" | 43 | #include "util/parse-options.h" |
44 | #include "util/parse-events.h" | 44 | #include "util/parse-events.h" |
45 | #include "util/event.h" | 45 | #include "util/event.h" |
46 | #include "util/evlist.h" | ||
46 | #include "util/evsel.h" | 47 | #include "util/evsel.h" |
47 | #include "util/debug.h" | 48 | #include "util/debug.h" |
48 | #include "util/header.h" | 49 | #include "util/header.h" |
49 | #include "util/cpumap.h" | 50 | #include "util/cpumap.h" |
50 | #include "util/thread.h" | 51 | #include "util/thread.h" |
52 | #include "util/thread_map.h" | ||
51 | 53 | ||
52 | #include <sys/prctl.h> | 54 | #include <sys/prctl.h> |
53 | #include <math.h> | 55 | #include <math.h> |
@@ -71,8 +73,9 @@ static struct perf_event_attr default_attrs[] = { | |||
71 | 73 | ||
72 | }; | 74 | }; |
73 | 75 | ||
76 | struct perf_evlist *evsel_list; | ||
77 | |||
74 | static bool system_wide = false; | 78 | static bool system_wide = false; |
75 | static struct cpu_map *cpus; | ||
76 | static int run_idx = 0; | 79 | static int run_idx = 0; |
77 | 80 | ||
78 | static int run_count = 1; | 81 | static int run_count = 1; |
@@ -81,7 +84,6 @@ static bool scale = true; | |||
81 | static bool no_aggr = false; | 84 | static bool no_aggr = false; |
82 | static pid_t target_pid = -1; | 85 | static pid_t target_pid = -1; |
83 | static pid_t target_tid = -1; | 86 | static pid_t target_tid = -1; |
84 | static struct thread_map *threads; | ||
85 | static pid_t child_pid = -1; | 87 | static pid_t child_pid = -1; |
86 | static bool null_run = false; | 88 | static bool null_run = false; |
87 | static bool big_num = true; | 89 | static bool big_num = true; |
@@ -166,7 +168,7 @@ static int create_perf_stat_counter(struct perf_evsel *evsel) | |||
166 | PERF_FORMAT_TOTAL_TIME_RUNNING; | 168 | PERF_FORMAT_TOTAL_TIME_RUNNING; |
167 | 169 | ||
168 | if (system_wide) | 170 | if (system_wide) |
169 | return perf_evsel__open_per_cpu(evsel, cpus); | 171 | return perf_evsel__open_per_cpu(evsel, evsel_list->cpus, false, false); |
170 | 172 | ||
171 | attr->inherit = !no_inherit; | 173 | attr->inherit = !no_inherit; |
172 | if (target_pid == -1 && target_tid == -1) { | 174 | if (target_pid == -1 && target_tid == -1) { |
@@ -174,7 +176,7 @@ static int create_perf_stat_counter(struct perf_evsel *evsel) | |||
174 | attr->enable_on_exec = 1; | 176 | attr->enable_on_exec = 1; |
175 | } | 177 | } |
176 | 178 | ||
177 | return perf_evsel__open_per_thread(evsel, threads); | 179 | return perf_evsel__open_per_thread(evsel, evsel_list->threads, false, false); |
178 | } | 180 | } |
179 | 181 | ||
180 | /* | 182 | /* |
@@ -199,7 +201,8 @@ static int read_counter_aggr(struct perf_evsel *counter) | |||
199 | u64 *count = counter->counts->aggr.values; | 201 | u64 *count = counter->counts->aggr.values; |
200 | int i; | 202 | int i; |
201 | 203 | ||
202 | if (__perf_evsel__read(counter, cpus->nr, threads->nr, scale) < 0) | 204 | if (__perf_evsel__read(counter, evsel_list->cpus->nr, |
205 | evsel_list->threads->nr, scale) < 0) | ||
203 | return -1; | 206 | return -1; |
204 | 207 | ||
205 | for (i = 0; i < 3; i++) | 208 | for (i = 0; i < 3; i++) |
@@ -232,7 +235,7 @@ static int read_counter(struct perf_evsel *counter) | |||
232 | u64 *count; | 235 | u64 *count; |
233 | int cpu; | 236 | int cpu; |
234 | 237 | ||
235 | for (cpu = 0; cpu < cpus->nr; cpu++) { | 238 | for (cpu = 0; cpu < evsel_list->cpus->nr; cpu++) { |
236 | if (__perf_evsel__read_on_cpu(counter, cpu, 0, scale) < 0) | 239 | if (__perf_evsel__read_on_cpu(counter, cpu, 0, scale) < 0) |
237 | return -1; | 240 | return -1; |
238 | 241 | ||
@@ -297,7 +300,7 @@ static int run_perf_stat(int argc __used, const char **argv) | |||
297 | } | 300 | } |
298 | 301 | ||
299 | if (target_tid == -1 && target_pid == -1 && !system_wide) | 302 | if (target_tid == -1 && target_pid == -1 && !system_wide) |
300 | threads->map[0] = child_pid; | 303 | evsel_list->threads->map[0] = child_pid; |
301 | 304 | ||
302 | /* | 305 | /* |
303 | * Wait for the child to be ready to exec. | 306 | * Wait for the child to be ready to exec. |
@@ -309,7 +312,7 @@ static int run_perf_stat(int argc __used, const char **argv) | |||
309 | close(child_ready_pipe[0]); | 312 | close(child_ready_pipe[0]); |
310 | } | 313 | } |
311 | 314 | ||
312 | list_for_each_entry(counter, &evsel_list, node) { | 315 | list_for_each_entry(counter, &evsel_list->entries, node) { |
313 | if (create_perf_stat_counter(counter) < 0) { | 316 | if (create_perf_stat_counter(counter) < 0) { |
314 | if (errno == -EPERM || errno == -EACCES) { | 317 | if (errno == -EPERM || errno == -EACCES) { |
315 | error("You may not have permission to collect %sstats.\n" | 318 | error("You may not have permission to collect %sstats.\n" |
@@ -347,14 +350,15 @@ static int run_perf_stat(int argc __used, const char **argv) | |||
347 | update_stats(&walltime_nsecs_stats, t1 - t0); | 350 | update_stats(&walltime_nsecs_stats, t1 - t0); |
348 | 351 | ||
349 | if (no_aggr) { | 352 | if (no_aggr) { |
350 | list_for_each_entry(counter, &evsel_list, node) { | 353 | list_for_each_entry(counter, &evsel_list->entries, node) { |
351 | read_counter(counter); | 354 | read_counter(counter); |
352 | perf_evsel__close_fd(counter, cpus->nr, 1); | 355 | perf_evsel__close_fd(counter, evsel_list->cpus->nr, 1); |
353 | } | 356 | } |
354 | } else { | 357 | } else { |
355 | list_for_each_entry(counter, &evsel_list, node) { | 358 | list_for_each_entry(counter, &evsel_list->entries, node) { |
356 | read_counter_aggr(counter); | 359 | read_counter_aggr(counter); |
357 | perf_evsel__close_fd(counter, cpus->nr, threads->nr); | 360 | perf_evsel__close_fd(counter, evsel_list->cpus->nr, |
361 | evsel_list->threads->nr); | ||
358 | } | 362 | } |
359 | } | 363 | } |
360 | 364 | ||
@@ -382,10 +386,13 @@ static void nsec_printout(int cpu, struct perf_evsel *evsel, double avg) | |||
382 | if (no_aggr) | 386 | if (no_aggr) |
383 | sprintf(cpustr, "CPU%*d%s", | 387 | sprintf(cpustr, "CPU%*d%s", |
384 | csv_output ? 0 : -4, | 388 | csv_output ? 0 : -4, |
385 | cpus->map[cpu], csv_sep); | 389 | evsel_list->cpus->map[cpu], csv_sep); |
386 | 390 | ||
387 | fprintf(stderr, fmt, cpustr, msecs, csv_sep, event_name(evsel)); | 391 | fprintf(stderr, fmt, cpustr, msecs, csv_sep, event_name(evsel)); |
388 | 392 | ||
393 | if (evsel->cgrp) | ||
394 | fprintf(stderr, "%s%s", csv_sep, evsel->cgrp->name); | ||
395 | |||
389 | if (csv_output) | 396 | if (csv_output) |
390 | return; | 397 | return; |
391 | 398 | ||
@@ -410,12 +417,15 @@ static void abs_printout(int cpu, struct perf_evsel *evsel, double avg) | |||
410 | if (no_aggr) | 417 | if (no_aggr) |
411 | sprintf(cpustr, "CPU%*d%s", | 418 | sprintf(cpustr, "CPU%*d%s", |
412 | csv_output ? 0 : -4, | 419 | csv_output ? 0 : -4, |
413 | cpus->map[cpu], csv_sep); | 420 | evsel_list->cpus->map[cpu], csv_sep); |
414 | else | 421 | else |
415 | cpu = 0; | 422 | cpu = 0; |
416 | 423 | ||
417 | fprintf(stderr, fmt, cpustr, avg, csv_sep, event_name(evsel)); | 424 | fprintf(stderr, fmt, cpustr, avg, csv_sep, event_name(evsel)); |
418 | 425 | ||
426 | if (evsel->cgrp) | ||
427 | fprintf(stderr, "%s%s", csv_sep, evsel->cgrp->name); | ||
428 | |||
419 | if (csv_output) | 429 | if (csv_output) |
420 | return; | 430 | return; |
421 | 431 | ||
@@ -456,9 +466,17 @@ static void print_counter_aggr(struct perf_evsel *counter) | |||
456 | int scaled = counter->counts->scaled; | 466 | int scaled = counter->counts->scaled; |
457 | 467 | ||
458 | if (scaled == -1) { | 468 | if (scaled == -1) { |
459 | fprintf(stderr, "%*s%s%-24s\n", | 469 | fprintf(stderr, "%*s%s%*s", |
460 | csv_output ? 0 : 18, | 470 | csv_output ? 0 : 18, |
461 | "<not counted>", csv_sep, event_name(counter)); | 471 | "<not counted>", |
472 | csv_sep, | ||
473 | csv_output ? 0 : -24, | ||
474 | event_name(counter)); | ||
475 | |||
476 | if (counter->cgrp) | ||
477 | fprintf(stderr, "%s%s", csv_sep, counter->cgrp->name); | ||
478 | |||
479 | fputc('\n', stderr); | ||
462 | return; | 480 | return; |
463 | } | 481 | } |
464 | 482 | ||
@@ -483,7 +501,6 @@ static void print_counter_aggr(struct perf_evsel *counter) | |||
483 | fprintf(stderr, " (scaled from %.2f%%)", | 501 | fprintf(stderr, " (scaled from %.2f%%)", |
484 | 100 * avg_running / avg_enabled); | 502 | 100 * avg_running / avg_enabled); |
485 | } | 503 | } |
486 | |||
487 | fprintf(stderr, "\n"); | 504 | fprintf(stderr, "\n"); |
488 | } | 505 | } |
489 | 506 | ||
@@ -496,19 +513,23 @@ static void print_counter(struct perf_evsel *counter) | |||
496 | u64 ena, run, val; | 513 | u64 ena, run, val; |
497 | int cpu; | 514 | int cpu; |
498 | 515 | ||
499 | for (cpu = 0; cpu < cpus->nr; cpu++) { | 516 | for (cpu = 0; cpu < evsel_list->cpus->nr; cpu++) { |
500 | val = counter->counts->cpu[cpu].val; | 517 | val = counter->counts->cpu[cpu].val; |
501 | ena = counter->counts->cpu[cpu].ena; | 518 | ena = counter->counts->cpu[cpu].ena; |
502 | run = counter->counts->cpu[cpu].run; | 519 | run = counter->counts->cpu[cpu].run; |
503 | if (run == 0 || ena == 0) { | 520 | if (run == 0 || ena == 0) { |
504 | fprintf(stderr, "CPU%*d%s%*s%s%-24s", | 521 | fprintf(stderr, "CPU%*d%s%*s%s%*s", |
505 | csv_output ? 0 : -4, | 522 | csv_output ? 0 : -4, |
506 | cpus->map[cpu], csv_sep, | 523 | evsel_list->cpus->map[cpu], csv_sep, |
507 | csv_output ? 0 : 18, | 524 | csv_output ? 0 : 18, |
508 | "<not counted>", csv_sep, | 525 | "<not counted>", csv_sep, |
526 | csv_output ? 0 : -24, | ||
509 | event_name(counter)); | 527 | event_name(counter)); |
510 | 528 | ||
511 | fprintf(stderr, "\n"); | 529 | if (counter->cgrp) |
530 | fprintf(stderr, "%s%s", csv_sep, counter->cgrp->name); | ||
531 | |||
532 | fputc('\n', stderr); | ||
512 | continue; | 533 | continue; |
513 | } | 534 | } |
514 | 535 | ||
@@ -525,7 +546,7 @@ static void print_counter(struct perf_evsel *counter) | |||
525 | 100.0 * run / ena); | 546 | 100.0 * run / ena); |
526 | } | 547 | } |
527 | } | 548 | } |
528 | fprintf(stderr, "\n"); | 549 | fputc('\n', stderr); |
529 | } | 550 | } |
530 | } | 551 | } |
531 | 552 | ||
@@ -555,10 +576,10 @@ static void print_stat(int argc, const char **argv) | |||
555 | } | 576 | } |
556 | 577 | ||
557 | if (no_aggr) { | 578 | if (no_aggr) { |
558 | list_for_each_entry(counter, &evsel_list, node) | 579 | list_for_each_entry(counter, &evsel_list->entries, node) |
559 | print_counter(counter); | 580 | print_counter(counter); |
560 | } else { | 581 | } else { |
561 | list_for_each_entry(counter, &evsel_list, node) | 582 | list_for_each_entry(counter, &evsel_list->entries, node) |
562 | print_counter_aggr(counter); | 583 | print_counter_aggr(counter); |
563 | } | 584 | } |
564 | 585 | ||
@@ -610,7 +631,7 @@ static int stat__set_big_num(const struct option *opt __used, | |||
610 | } | 631 | } |
611 | 632 | ||
612 | static const struct option options[] = { | 633 | static const struct option options[] = { |
613 | OPT_CALLBACK('e', "event", NULL, "event", | 634 | OPT_CALLBACK('e', "event", &evsel_list, "event", |
614 | "event selector. use 'perf list' to list available events", | 635 | "event selector. use 'perf list' to list available events", |
615 | parse_events), | 636 | parse_events), |
616 | OPT_BOOLEAN('i', "no-inherit", &no_inherit, | 637 | OPT_BOOLEAN('i', "no-inherit", &no_inherit, |
@@ -638,6 +659,9 @@ static const struct option options[] = { | |||
638 | "disable CPU count aggregation"), | 659 | "disable CPU count aggregation"), |
639 | OPT_STRING('x', "field-separator", &csv_sep, "separator", | 660 | OPT_STRING('x', "field-separator", &csv_sep, "separator", |
640 | "print counts with custom separator"), | 661 | "print counts with custom separator"), |
662 | OPT_CALLBACK('G', "cgroup", &evsel_list, "name", | ||
663 | "monitor event in cgroup name only", | ||
664 | parse_cgroups), | ||
641 | OPT_END() | 665 | OPT_END() |
642 | }; | 666 | }; |
643 | 667 | ||
@@ -648,6 +672,10 @@ int cmd_stat(int argc, const char **argv, const char *prefix __used) | |||
648 | 672 | ||
649 | setlocale(LC_ALL, ""); | 673 | setlocale(LC_ALL, ""); |
650 | 674 | ||
675 | evsel_list = perf_evlist__new(NULL, NULL); | ||
676 | if (evsel_list == NULL) | ||
677 | return -ENOMEM; | ||
678 | |||
651 | argc = parse_options(argc, argv, options, stat_usage, | 679 | argc = parse_options(argc, argv, options, stat_usage, |
652 | PARSE_OPT_STOP_AT_NON_OPTION); | 680 | PARSE_OPT_STOP_AT_NON_OPTION); |
653 | 681 | ||
@@ -674,49 +702,50 @@ int cmd_stat(int argc, const char **argv, const char *prefix __used) | |||
674 | if (run_count <= 0) | 702 | if (run_count <= 0) |
675 | usage_with_options(stat_usage, options); | 703 | usage_with_options(stat_usage, options); |
676 | 704 | ||
677 | /* no_aggr is for system-wide only */ | 705 | /* no_aggr, cgroup are for system-wide only */ |
678 | if (no_aggr && !system_wide) | 706 | if ((no_aggr || nr_cgroups) && !system_wide) { |
707 | fprintf(stderr, "both cgroup and no-aggregation " | ||
708 | "modes only available in system-wide mode\n"); | ||
709 | |||
679 | usage_with_options(stat_usage, options); | 710 | usage_with_options(stat_usage, options); |
711 | } | ||
680 | 712 | ||
681 | /* Set attrs and nr_counters if no event is selected and !null_run */ | 713 | /* Set attrs and nr_counters if no event is selected and !null_run */ |
682 | if (!null_run && !nr_counters) { | 714 | if (!null_run && !evsel_list->nr_entries) { |
683 | size_t c; | 715 | size_t c; |
684 | 716 | ||
685 | nr_counters = ARRAY_SIZE(default_attrs); | ||
686 | |||
687 | for (c = 0; c < ARRAY_SIZE(default_attrs); ++c) { | 717 | for (c = 0; c < ARRAY_SIZE(default_attrs); ++c) { |
688 | pos = perf_evsel__new(&default_attrs[c], | 718 | pos = perf_evsel__new(&default_attrs[c], c); |
689 | nr_counters); | ||
690 | if (pos == NULL) | 719 | if (pos == NULL) |
691 | goto out; | 720 | goto out; |
692 | list_add(&pos->node, &evsel_list); | 721 | perf_evlist__add(evsel_list, pos); |
693 | } | 722 | } |
694 | } | 723 | } |
695 | 724 | ||
696 | if (target_pid != -1) | 725 | if (target_pid != -1) |
697 | target_tid = target_pid; | 726 | target_tid = target_pid; |
698 | 727 | ||
699 | threads = thread_map__new(target_pid, target_tid); | 728 | evsel_list->threads = thread_map__new(target_pid, target_tid); |
700 | if (threads == NULL) { | 729 | if (evsel_list->threads == NULL) { |
701 | pr_err("Problems finding threads of monitor\n"); | 730 | pr_err("Problems finding threads of monitor\n"); |
702 | usage_with_options(stat_usage, options); | 731 | usage_with_options(stat_usage, options); |
703 | } | 732 | } |
704 | 733 | ||
705 | if (system_wide) | 734 | if (system_wide) |
706 | cpus = cpu_map__new(cpu_list); | 735 | evsel_list->cpus = cpu_map__new(cpu_list); |
707 | else | 736 | else |
708 | cpus = cpu_map__dummy_new(); | 737 | evsel_list->cpus = cpu_map__dummy_new(); |
709 | 738 | ||
710 | if (cpus == NULL) { | 739 | if (evsel_list->cpus == NULL) { |
711 | perror("failed to parse CPUs map"); | 740 | perror("failed to parse CPUs map"); |
712 | usage_with_options(stat_usage, options); | 741 | usage_with_options(stat_usage, options); |
713 | return -1; | 742 | return -1; |
714 | } | 743 | } |
715 | 744 | ||
716 | list_for_each_entry(pos, &evsel_list, node) { | 745 | list_for_each_entry(pos, &evsel_list->entries, node) { |
717 | if (perf_evsel__alloc_stat_priv(pos) < 0 || | 746 | if (perf_evsel__alloc_stat_priv(pos) < 0 || |
718 | perf_evsel__alloc_counts(pos, cpus->nr) < 0 || | 747 | perf_evsel__alloc_counts(pos, evsel_list->cpus->nr) < 0 || |
719 | perf_evsel__alloc_fd(pos, cpus->nr, threads->nr) < 0) | 748 | perf_evsel__alloc_fd(pos, evsel_list->cpus->nr, evsel_list->threads->nr) < 0) |
720 | goto out_free_fd; | 749 | goto out_free_fd; |
721 | } | 750 | } |
722 | 751 | ||
@@ -741,11 +770,10 @@ int cmd_stat(int argc, const char **argv, const char *prefix __used) | |||
741 | if (status != -1) | 770 | if (status != -1) |
742 | print_stat(argc, argv); | 771 | print_stat(argc, argv); |
743 | out_free_fd: | 772 | out_free_fd: |
744 | list_for_each_entry(pos, &evsel_list, node) | 773 | list_for_each_entry(pos, &evsel_list->entries, node) |
745 | perf_evsel__free_stat_priv(pos); | 774 | perf_evsel__free_stat_priv(pos); |
746 | perf_evsel_list__delete(); | 775 | perf_evlist__delete_maps(evsel_list); |
747 | out: | 776 | out: |
748 | thread_map__delete(threads); | 777 | perf_evlist__delete(evsel_list); |
749 | threads = NULL; | ||
750 | return status; | 778 | return status; |
751 | } | 779 | } |