diff options
author | Thomas Renninger <trenn@suse.de> | 2011-01-03 11:50:45 -0500 |
---|---|---|
committer | Ingo Molnar <mingo@elte.hu> | 2011-01-04 02:16:54 -0500 |
commit | 20c457b8587bee4644d998331d9e13be82e05b4c (patch) | |
tree | e1864952316ad24af3bea22d09753e043eb6ee75 /tools/perf/builtin-timechart.c | |
parent | 25e41933b58777f2d020c3b0186b430ea004ec28 (diff) |
perf timechart: Adjust perf timechart to the new power events
builtin-timechart must only pass -e power:xy events if they are supported by
the running kernel, otherwise try to fetch the old power:power{start,end}
events.
For this I added the tiny helper function:
int is_valid_tracepoint(const char *event_string)
to parse-events.[hc], which could be more generic as an interface and support
hardware/software/... events, not only tracepoints, but someone else could
extend that if needed...
Signed-off-by: Thomas Renninger <trenn@suse.de>
Acked-by: Arjan van de Ven <arjan@linux.intel.com>
Acked-by: Jean Pihet <j-pihet@ti.com>
LKML-Reference: <1294073445-14812-4-git-send-email-trenn@suse.de>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
Diffstat (limited to 'tools/perf/builtin-timechart.c')
-rw-r--r-- | tools/perf/builtin-timechart.c | 94 |
1 files changed, 76 insertions, 18 deletions
diff --git a/tools/perf/builtin-timechart.c b/tools/perf/builtin-timechart.c index d75084bccdb7..746cf03cb05d 100644 --- a/tools/perf/builtin-timechart.c +++ b/tools/perf/builtin-timechart.c | |||
@@ -32,6 +32,10 @@ | |||
32 | #include "util/session.h" | 32 | #include "util/session.h" |
33 | #include "util/svghelper.h" | 33 | #include "util/svghelper.h" |
34 | 34 | ||
35 | #define SUPPORT_OLD_POWER_EVENTS 1 | ||
36 | #define PWR_EVENT_EXIT -1 | ||
37 | |||
38 | |||
35 | static char const *input_name = "perf.data"; | 39 | static char const *input_name = "perf.data"; |
36 | static char const *output_name = "output.svg"; | 40 | static char const *output_name = "output.svg"; |
37 | 41 | ||
@@ -301,12 +305,21 @@ struct trace_entry { | |||
301 | int lock_depth; | 305 | int lock_depth; |
302 | }; | 306 | }; |
303 | 307 | ||
304 | struct power_entry { | 308 | #ifdef SUPPORT_OLD_POWER_EVENTS |
309 | static int use_old_power_events; | ||
310 | struct power_entry_old { | ||
305 | struct trace_entry te; | 311 | struct trace_entry te; |
306 | u64 type; | 312 | u64 type; |
307 | u64 value; | 313 | u64 value; |
308 | u64 cpu_id; | 314 | u64 cpu_id; |
309 | }; | 315 | }; |
316 | #endif | ||
317 | |||
318 | struct power_processor_entry { | ||
319 | struct trace_entry te; | ||
320 | u32 state; | ||
321 | u32 cpu_id; | ||
322 | }; | ||
310 | 323 | ||
311 | #define TASK_COMM_LEN 16 | 324 | #define TASK_COMM_LEN 16 |
312 | struct wakeup_entry { | 325 | struct wakeup_entry { |
@@ -489,29 +502,49 @@ static int process_sample_event(event_t *event __used, | |||
489 | te = (void *)sample->raw_data; | 502 | te = (void *)sample->raw_data; |
490 | if (session->sample_type & PERF_SAMPLE_RAW && sample->raw_size > 0) { | 503 | if (session->sample_type & PERF_SAMPLE_RAW && sample->raw_size > 0) { |
491 | char *event_str; | 504 | char *event_str; |
492 | struct power_entry *pe; | 505 | #ifdef SUPPORT_OLD_POWER_EVENTS |
493 | 506 | struct power_entry_old *peo; | |
494 | pe = (void *)te; | 507 | peo = (void *)te; |
495 | 508 | #endif | |
496 | event_str = perf_header__find_event(te->type); | 509 | event_str = perf_header__find_event(te->type); |
497 | 510 | ||
498 | if (!event_str) | 511 | if (!event_str) |
499 | return 0; | 512 | return 0; |
500 | 513 | ||
501 | if (strcmp(event_str, "power:power_start") == 0) | 514 | if (strcmp(event_str, "power:cpu_idle") == 0) { |
502 | c_state_start(pe->cpu_id, sample->time, pe->value); | 515 | struct power_processor_entry *ppe = (void *)te; |
503 | 516 | if (ppe->state == (u32)PWR_EVENT_EXIT) | |
504 | if (strcmp(event_str, "power:power_end") == 0) | 517 | c_state_end(ppe->cpu_id, sample->time); |
505 | c_state_end(pe->cpu_id, sample->time); | 518 | else |
506 | 519 | c_state_start(ppe->cpu_id, sample->time, | |
507 | if (strcmp(event_str, "power:power_frequency") == 0) | 520 | ppe->state); |
508 | p_state_change(pe->cpu_id, sample->time, pe->value); | 521 | } |
522 | else if (strcmp(event_str, "power:cpu_frequency") == 0) { | ||
523 | struct power_processor_entry *ppe = (void *)te; | ||
524 | p_state_change(ppe->cpu_id, sample->time, ppe->state); | ||
525 | } | ||
509 | 526 | ||
510 | if (strcmp(event_str, "sched:sched_wakeup") == 0) | 527 | else if (strcmp(event_str, "sched:sched_wakeup") == 0) |
511 | sched_wakeup(sample->cpu, sample->time, sample->pid, te); | 528 | sched_wakeup(sample->cpu, sample->time, sample->pid, te); |
512 | 529 | ||
513 | if (strcmp(event_str, "sched:sched_switch") == 0) | 530 | else if (strcmp(event_str, "sched:sched_switch") == 0) |
514 | sched_switch(sample->cpu, sample->time, te); | 531 | sched_switch(sample->cpu, sample->time, te); |
532 | |||
533 | #ifdef SUPPORT_OLD_POWER_EVENTS | ||
534 | if (use_old_power_events) { | ||
535 | if (strcmp(event_str, "power:power_start") == 0) | ||
536 | c_state_start(peo->cpu_id, sample->time, | ||
537 | peo->value); | ||
538 | |||
539 | else if (strcmp(event_str, "power:power_end") == 0) | ||
540 | c_state_end(sample->cpu, sample->time); | ||
541 | |||
542 | else if (strcmp(event_str, | ||
543 | "power:power_frequency") == 0) | ||
544 | p_state_change(peo->cpu_id, sample->time, | ||
545 | peo->value); | ||
546 | } | ||
547 | #endif | ||
515 | } | 548 | } |
516 | return 0; | 549 | return 0; |
517 | } | 550 | } |
@@ -969,7 +1002,8 @@ static const char * const timechart_usage[] = { | |||
969 | NULL | 1002 | NULL |
970 | }; | 1003 | }; |
971 | 1004 | ||
972 | static const char *record_args[] = { | 1005 | #ifdef SUPPORT_OLD_POWER_EVENTS |
1006 | static const char * const record_old_args[] = { | ||
973 | "record", | 1007 | "record", |
974 | "-a", | 1008 | "-a", |
975 | "-R", | 1009 | "-R", |
@@ -981,19 +1015,43 @@ static const char *record_args[] = { | |||
981 | "-e", "sched:sched_wakeup", | 1015 | "-e", "sched:sched_wakeup", |
982 | "-e", "sched:sched_switch", | 1016 | "-e", "sched:sched_switch", |
983 | }; | 1017 | }; |
1018 | #endif | ||
1019 | |||
1020 | static const char * const record_new_args[] = { | ||
1021 | "record", | ||
1022 | "-a", | ||
1023 | "-R", | ||
1024 | "-f", | ||
1025 | "-c", "1", | ||
1026 | "-e", "power:cpu_frequency", | ||
1027 | "-e", "power:cpu_idle", | ||
1028 | "-e", "sched:sched_wakeup", | ||
1029 | "-e", "sched:sched_switch", | ||
1030 | }; | ||
984 | 1031 | ||
985 | static int __cmd_record(int argc, const char **argv) | 1032 | static int __cmd_record(int argc, const char **argv) |
986 | { | 1033 | { |
987 | unsigned int rec_argc, i, j; | 1034 | unsigned int rec_argc, i, j; |
988 | const char **rec_argv; | 1035 | const char **rec_argv; |
1036 | const char * const *record_args = record_new_args; | ||
1037 | unsigned int record_elems = ARRAY_SIZE(record_new_args); | ||
1038 | |||
1039 | #ifdef SUPPORT_OLD_POWER_EVENTS | ||
1040 | if (!is_valid_tracepoint("power:cpu_idle") && | ||
1041 | is_valid_tracepoint("power:power_start")) { | ||
1042 | use_old_power_events = 1; | ||
1043 | record_args = record_old_args; | ||
1044 | record_elems = ARRAY_SIZE(record_old_args); | ||
1045 | } | ||
1046 | #endif | ||
989 | 1047 | ||
990 | rec_argc = ARRAY_SIZE(record_args) + argc - 1; | 1048 | rec_argc = record_elems + argc - 1; |
991 | rec_argv = calloc(rec_argc + 1, sizeof(char *)); | 1049 | rec_argv = calloc(rec_argc + 1, sizeof(char *)); |
992 | 1050 | ||
993 | if (rec_argv == NULL) | 1051 | if (rec_argv == NULL) |
994 | return -ENOMEM; | 1052 | return -ENOMEM; |
995 | 1053 | ||
996 | for (i = 0; i < ARRAY_SIZE(record_args); i++) | 1054 | for (i = 0; i < record_elems; i++) |
997 | rec_argv[i] = strdup(record_args[i]); | 1055 | rec_argv[i] = strdup(record_args[i]); |
998 | 1056 | ||
999 | for (j = 1; j < (unsigned int)argc; j++, i++) | 1057 | for (j = 1; j < (unsigned int)argc; j++, i++) |