mirror of
https://github.com/torvalds/linux
synced 2024-11-05 18:23:50 +00:00
perf timechart: Add support for -P and -T in timechart recording
If we don't want either power or task events we may use -T or -P with the `perf timechart record` command to filter out events while recording to keep perf.data small. Signed-off-by: Stanislav Fomichev <stfomichev@yandex-team.ru> Acked-by: Namhyung Kim <namhyung@kernel.org> Cc: Ingo Molnar <mingo@redhat.com> Cc: Namhyung Kim <namhyung@kernel.org> Cc: Paul Mackerras <paulus@samba.org> Cc: Peter Zijlstra <a.p.zijlstra@chello.nl> Link: http://lkml.kernel.org/r/1383323151-19810-7-git-send-email-stfomichev@yandex-team.ru Signed-off-by: Arnaldo Carvalho de Melo <acme@redhat.com>
This commit is contained in:
parent
cbb2e81e52
commit
367b3152d7
2 changed files with 87 additions and 30 deletions
|
@ -8,8 +8,7 @@ perf-timechart - Tool to visualize total system behavior during a workload
|
||||||
SYNOPSIS
|
SYNOPSIS
|
||||||
--------
|
--------
|
||||||
[verse]
|
[verse]
|
||||||
'perf timechart' record <command>
|
'perf timechart' [<timechart options>] {record} [<record options>]
|
||||||
'perf timechart' [<options>]
|
|
||||||
|
|
||||||
DESCRIPTION
|
DESCRIPTION
|
||||||
-----------
|
-----------
|
||||||
|
@ -21,8 +20,8 @@ There are two variants of perf timechart:
|
||||||
'perf timechart' to turn a trace into a Scalable Vector Graphics file,
|
'perf timechart' to turn a trace into a Scalable Vector Graphics file,
|
||||||
that can be viewed with popular SVG viewers such as 'Inkscape'.
|
that can be viewed with popular SVG viewers such as 'Inkscape'.
|
||||||
|
|
||||||
OPTIONS
|
TIMECHART OPTIONS
|
||||||
-------
|
-----------------
|
||||||
-o::
|
-o::
|
||||||
--output=::
|
--output=::
|
||||||
Select the output file (default: output.svg)
|
Select the output file (default: output.svg)
|
||||||
|
@ -61,6 +60,15 @@ $ perf timechart
|
||||||
--proc-num::
|
--proc-num::
|
||||||
Print task info for at least given number of tasks.
|
Print task info for at least given number of tasks.
|
||||||
|
|
||||||
|
RECORD OPTIONS
|
||||||
|
--------------
|
||||||
|
-P::
|
||||||
|
--power-only::
|
||||||
|
Record only power-related events
|
||||||
|
-T::
|
||||||
|
--tasks-only::
|
||||||
|
Record only tasks-related events
|
||||||
|
|
||||||
SEE ALSO
|
SEE ALSO
|
||||||
--------
|
--------
|
||||||
linkperf:perf-record[1]
|
linkperf:perf-record[1]
|
||||||
|
|
|
@ -1040,50 +1040,81 @@ static int __cmd_timechart(const char *output_name)
|
||||||
|
|
||||||
static int __cmd_record(int argc, const char **argv)
|
static int __cmd_record(int argc, const char **argv)
|
||||||
{
|
{
|
||||||
#ifdef SUPPORT_OLD_POWER_EVENTS
|
unsigned int rec_argc, i, j;
|
||||||
const char * const record_old_args[] = {
|
const char **rec_argv;
|
||||||
|
const char **p;
|
||||||
|
unsigned int record_elems;
|
||||||
|
|
||||||
|
const char * const common_args[] = {
|
||||||
"record", "-a", "-R", "-c", "1",
|
"record", "-a", "-R", "-c", "1",
|
||||||
|
};
|
||||||
|
unsigned int common_args_nr = ARRAY_SIZE(common_args);
|
||||||
|
|
||||||
|
const char * const power_args[] = {
|
||||||
|
"-e", "power:cpu_frequency",
|
||||||
|
"-e", "power:cpu_idle",
|
||||||
|
};
|
||||||
|
unsigned int power_args_nr = ARRAY_SIZE(power_args);
|
||||||
|
|
||||||
|
const char * const old_power_args[] = {
|
||||||
|
#ifdef SUPPORT_OLD_POWER_EVENTS
|
||||||
"-e", "power:power_start",
|
"-e", "power:power_start",
|
||||||
"-e", "power:power_end",
|
"-e", "power:power_end",
|
||||||
"-e", "power:power_frequency",
|
"-e", "power:power_frequency",
|
||||||
"-e", "sched:sched_wakeup",
|
|
||||||
"-e", "sched:sched_switch",
|
|
||||||
};
|
|
||||||
#endif
|
#endif
|
||||||
const char * const record_new_args[] = {
|
};
|
||||||
"record", "-a", "-R", "-c", "1",
|
unsigned int old_power_args_nr = ARRAY_SIZE(old_power_args);
|
||||||
"-e", "power:cpu_frequency",
|
|
||||||
"-e", "power:cpu_idle",
|
const char * const tasks_args[] = {
|
||||||
"-e", "sched:sched_wakeup",
|
"-e", "sched:sched_wakeup",
|
||||||
"-e", "sched:sched_switch",
|
"-e", "sched:sched_switch",
|
||||||
};
|
};
|
||||||
unsigned int rec_argc, i, j;
|
unsigned int tasks_args_nr = ARRAY_SIZE(tasks_args);
|
||||||
const char **rec_argv;
|
|
||||||
const char * const *record_args = record_new_args;
|
|
||||||
unsigned int record_elems = ARRAY_SIZE(record_new_args);
|
|
||||||
|
|
||||||
#ifdef SUPPORT_OLD_POWER_EVENTS
|
#ifdef SUPPORT_OLD_POWER_EVENTS
|
||||||
if (!is_valid_tracepoint("power:cpu_idle") &&
|
if (!is_valid_tracepoint("power:cpu_idle") &&
|
||||||
is_valid_tracepoint("power:power_start")) {
|
is_valid_tracepoint("power:power_start")) {
|
||||||
use_old_power_events = 1;
|
use_old_power_events = 1;
|
||||||
record_args = record_old_args;
|
power_args_nr = 0;
|
||||||
record_elems = ARRAY_SIZE(record_old_args);
|
} else {
|
||||||
|
old_power_args_nr = 0;
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
rec_argc = record_elems + argc - 1;
|
if (power_only)
|
||||||
|
tasks_args_nr = 0;
|
||||||
|
|
||||||
|
if (tasks_only) {
|
||||||
|
power_args_nr = 0;
|
||||||
|
old_power_args_nr = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
record_elems = common_args_nr + tasks_args_nr +
|
||||||
|
power_args_nr + old_power_args_nr;
|
||||||
|
|
||||||
|
rec_argc = record_elems + argc;
|
||||||
rec_argv = calloc(rec_argc + 1, sizeof(char *));
|
rec_argv = calloc(rec_argc + 1, sizeof(char *));
|
||||||
|
|
||||||
if (rec_argv == NULL)
|
if (rec_argv == NULL)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
for (i = 0; i < record_elems; i++)
|
p = rec_argv;
|
||||||
rec_argv[i] = strdup(record_args[i]);
|
for (i = 0; i < common_args_nr; i++)
|
||||||
|
*p++ = strdup(common_args[i]);
|
||||||
|
|
||||||
for (j = 1; j < (unsigned int)argc; j++, i++)
|
for (i = 0; i < tasks_args_nr; i++)
|
||||||
rec_argv[i] = argv[j];
|
*p++ = strdup(tasks_args[i]);
|
||||||
|
|
||||||
return cmd_record(i, rec_argv, NULL);
|
for (i = 0; i < power_args_nr; i++)
|
||||||
|
*p++ = strdup(power_args[i]);
|
||||||
|
|
||||||
|
for (i = 0; i < old_power_args_nr; i++)
|
||||||
|
*p++ = strdup(old_power_args[i]);
|
||||||
|
|
||||||
|
for (j = 1; j < (unsigned int)argc; j++)
|
||||||
|
*p++ = argv[j];
|
||||||
|
|
||||||
|
return cmd_record(rec_argc, rec_argv, NULL);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int
|
static int
|
||||||
|
@ -1099,7 +1130,7 @@ int cmd_timechart(int argc, const char **argv,
|
||||||
const char *prefix __maybe_unused)
|
const char *prefix __maybe_unused)
|
||||||
{
|
{
|
||||||
const char *output_name = "output.svg";
|
const char *output_name = "output.svg";
|
||||||
const struct option options[] = {
|
const struct option timechart_options[] = {
|
||||||
OPT_STRING('i', "input", &input_name, "file", "input file name"),
|
OPT_STRING('i', "input", &input_name, "file", "input file name"),
|
||||||
OPT_STRING('o', "output", &output_name, "file", "output file name"),
|
OPT_STRING('o', "output", &output_name, "file", "output file name"),
|
||||||
OPT_INTEGER('w', "width", &svg_page_width, "page width"),
|
OPT_INTEGER('w', "width", &svg_page_width, "page width"),
|
||||||
|
@ -1120,7 +1151,17 @@ int cmd_timechart(int argc, const char **argv,
|
||||||
NULL
|
NULL
|
||||||
};
|
};
|
||||||
|
|
||||||
argc = parse_options(argc, argv, options, timechart_usage,
|
const struct option record_options[] = {
|
||||||
|
OPT_BOOLEAN('P', "power-only", &power_only, "output power data only"),
|
||||||
|
OPT_BOOLEAN('T', "tasks-only", &tasks_only,
|
||||||
|
"output processes data only"),
|
||||||
|
OPT_END()
|
||||||
|
};
|
||||||
|
const char * const record_usage[] = {
|
||||||
|
"perf timechart record [<options>]",
|
||||||
|
NULL
|
||||||
|
};
|
||||||
|
argc = parse_options(argc, argv, timechart_options, timechart_usage,
|
||||||
PARSE_OPT_STOP_AT_NON_OPTION);
|
PARSE_OPT_STOP_AT_NON_OPTION);
|
||||||
|
|
||||||
if (power_only && tasks_only) {
|
if (power_only && tasks_only) {
|
||||||
|
@ -1130,10 +1171,18 @@ int cmd_timechart(int argc, const char **argv,
|
||||||
|
|
||||||
symbol__init();
|
symbol__init();
|
||||||
|
|
||||||
if (argc && !strncmp(argv[0], "rec", 3))
|
if (argc && !strncmp(argv[0], "rec", 3)) {
|
||||||
|
argc = parse_options(argc, argv, record_options, record_usage,
|
||||||
|
PARSE_OPT_STOP_AT_NON_OPTION);
|
||||||
|
|
||||||
|
if (power_only && tasks_only) {
|
||||||
|
pr_err("-P and -T options cannot be used at the same time.\n");
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
return __cmd_record(argc, argv);
|
return __cmd_record(argc, argv);
|
||||||
else if (argc)
|
} else if (argc)
|
||||||
usage_with_options(timechart_usage, options);
|
usage_with_options(timechart_usage, timechart_options);
|
||||||
|
|
||||||
setup_pager();
|
setup_pager();
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue