Commit f2af0086 authored by Namhyung Kim's avatar Namhyung Kim Committed by Arnaldo Carvalho de Melo

perf report: Add callchain value option

Now -g/--call-graph option supports how to display callchain values.
Possible values are 'percent', 'period' and 'count'.  The percent is
same as before and it's the default behavior.  The period displays the
raw period value rather than the percentage.  The count displays the
number of occurrences.

  $ perf report --no-children --stdio -g percent
  ...
    39.93%  swapper  [kernel.vmlinux]  [k] intel_idel
            |
            ---intel_idle
               cpuidle_enter_state
               cpuidle_enter
               call_cpuidle
               cpu_startup_entry
               |
               |--28.63%-- start_secondary
               |
                --11.30%-- rest_init

  $ perf report --no-children --show-total-period --stdio -g period
  ...
    39.93%   13018705  swapper  [kernel.vmlinux]  [k] intel_idel
            |
            ---intel_idle
               cpuidle_enter_state
               cpuidle_enter
               call_cpuidle
               cpu_startup_entry
               |
               |--9334403-- start_secondary
               |
                --3684302-- rest_init

  $ perf report --no-children --show-nr-samples --stdio -g count
  ...
    39.93%     80  swapper  [kernel.vmlinux]  [k] intel_idel
            |
            ---intel_idle
               cpuidle_enter_state
               cpuidle_enter
               call_cpuidle
               cpu_startup_entry
               |
               |--57-- start_secondary
               |
                --23-- rest_init
Signed-off-by: default avatarNamhyung Kim <namhyung@kernel.org>
Acked-by: default avatarBrendan Gregg <brendan.d.gregg@gmail.com>
Cc: Andi Kleen <andi@firstfloor.org>
Cc: David Ahern <dsahern@gmail.com>
Cc: Frederic Weisbecker <fweisbec@gmail.com>
Cc: Jiri Olsa <jolsa@redhat.com>
Cc: Kan Liang <kan.liang@intel.com>
Cc: Peter Zijlstra <a.p.zijlstra@chello.nl>
Link: http://lkml.kernel.org/r/1447047946-1691-6-git-send-email-namhyung@kernel.orgSigned-off-by: default avatarArnaldo Carvalho de Melo <acme@redhat.com>
parent 5e47f8ff
...@@ -170,11 +170,11 @@ OPTIONS ...@@ -170,11 +170,11 @@ OPTIONS
Dump raw trace in ASCII. Dump raw trace in ASCII.
-g:: -g::
--call-graph=<print_type,threshold[,print_limit],order,sort_key,branch>:: --call-graph=<print_type,threshold[,print_limit],order,sort_key[,branch],value>::
Display call chains using type, min percent threshold, print limit, Display call chains using type, min percent threshold, print limit,
call order, sort key and branch. Note that ordering of parameters is not call order, sort key, optional branch and value. Note that ordering of
fixed so any parement can be given in an arbitraty order. One exception parameters is not fixed so any parement can be given in an arbitraty order.
is the print_limit which should be preceded by threshold. One exception is the print_limit which should be preceded by threshold.
print_type can be either: print_type can be either:
- flat: single column, linear exposure of call chains. - flat: single column, linear exposure of call chains.
...@@ -205,6 +205,11 @@ OPTIONS ...@@ -205,6 +205,11 @@ OPTIONS
- branch: include last branch information in callgraph when available. - branch: include last branch information in callgraph when available.
Usually more convenient to use --branch-history for this. Usually more convenient to use --branch-history for this.
value can be:
- percent: diplay overhead percent (default)
- period: display event period
- count: display event count
--children:: --children::
Accumulate callchain of children to parent entry so that then can Accumulate callchain of children to parent entry so that then can
show up in the output. The output will have a new "Children" column show up in the output. The output will have a new "Children" column
......
...@@ -625,7 +625,7 @@ parse_percent_limit(const struct option *opt, const char *str, ...@@ -625,7 +625,7 @@ parse_percent_limit(const struct option *opt, const char *str,
return 0; return 0;
} }
#define CALLCHAIN_DEFAULT_OPT "graph,0.5,caller,function" #define CALLCHAIN_DEFAULT_OPT "graph,0.5,caller,function,percent"
const char report_callchain_help[] = "Display call graph (stack chain/backtrace):\n\n" const char report_callchain_help[] = "Display call graph (stack chain/backtrace):\n\n"
CALLCHAIN_REPORT_HELP CALLCHAIN_REPORT_HELP
...@@ -708,7 +708,7 @@ int cmd_report(int argc, const char **argv, const char *prefix __maybe_unused) ...@@ -708,7 +708,7 @@ int cmd_report(int argc, const char **argv, const char *prefix __maybe_unused)
OPT_BOOLEAN('x', "exclude-other", &symbol_conf.exclude_other, OPT_BOOLEAN('x', "exclude-other", &symbol_conf.exclude_other,
"Only display entries with parent-match"), "Only display entries with parent-match"),
OPT_CALLBACK_DEFAULT('g', "call-graph", &report, OPT_CALLBACK_DEFAULT('g', "call-graph", &report,
"print_type,threshold[,print_limit],order,sort_key[,branch]", "print_type,threshold[,print_limit],order,sort_key[,branch],value",
report_callchain_help, &report_parse_callchain_opt, report_callchain_help, &report_parse_callchain_opt,
callchain_default_opt), callchain_default_opt),
OPT_BOOLEAN(0, "children", &symbol_conf.cumulate_callchain, OPT_BOOLEAN(0, "children", &symbol_conf.cumulate_callchain,
......
...@@ -81,13 +81,14 @@ static size_t __callchain__fprintf_graph(FILE *fp, struct rb_root *root, ...@@ -81,13 +81,14 @@ static size_t __callchain__fprintf_graph(FILE *fp, struct rb_root *root,
int depth_mask, int left_margin) int depth_mask, int left_margin)
{ {
struct rb_node *node, *next; struct rb_node *node, *next;
struct callchain_node *child; struct callchain_node *child = NULL;
struct callchain_list *chain; struct callchain_list *chain;
int new_depth_mask = depth_mask; int new_depth_mask = depth_mask;
u64 remaining; u64 remaining;
size_t ret = 0; size_t ret = 0;
int i; int i;
uint entries_printed = 0; uint entries_printed = 0;
int cumul_count = 0;
remaining = total_samples; remaining = total_samples;
...@@ -99,6 +100,7 @@ static size_t __callchain__fprintf_graph(FILE *fp, struct rb_root *root, ...@@ -99,6 +100,7 @@ static size_t __callchain__fprintf_graph(FILE *fp, struct rb_root *root,
child = rb_entry(node, struct callchain_node, rb_node); child = rb_entry(node, struct callchain_node, rb_node);
cumul = callchain_cumul_hits(child); cumul = callchain_cumul_hits(child);
remaining -= cumul; remaining -= cumul;
cumul_count += callchain_cumul_counts(child);
/* /*
* The depth mask manages the output of pipes that show * The depth mask manages the output of pipes that show
...@@ -148,6 +150,12 @@ static size_t __callchain__fprintf_graph(FILE *fp, struct rb_root *root, ...@@ -148,6 +150,12 @@ static size_t __callchain__fprintf_graph(FILE *fp, struct rb_root *root,
if (!rem_sq_bracket) if (!rem_sq_bracket)
return ret; return ret;
if (callchain_param.value == CCVAL_COUNT && child && child->parent) {
rem_node.count = child->parent->children_count - cumul_count;
if (rem_node.count <= 0)
return ret;
}
new_depth_mask &= ~(1 << (depth - 1)); new_depth_mask &= ~(1 << (depth - 1));
ret += ipchain__fprintf_graph(fp, &rem_node, &rem_hits, depth, ret += ipchain__fprintf_graph(fp, &rem_node, &rem_hits, depth,
new_depth_mask, 0, total_samples, new_depth_mask, 0, total_samples,
......
...@@ -83,6 +83,23 @@ static int parse_callchain_sort_key(const char *value) ...@@ -83,6 +83,23 @@ static int parse_callchain_sort_key(const char *value)
return -1; return -1;
} }
static int parse_callchain_value(const char *value)
{
if (!strncmp(value, "percent", strlen(value))) {
callchain_param.value = CCVAL_PERCENT;
return 0;
}
if (!strncmp(value, "period", strlen(value))) {
callchain_param.value = CCVAL_PERIOD;
return 0;
}
if (!strncmp(value, "count", strlen(value))) {
callchain_param.value = CCVAL_COUNT;
return 0;
}
return -1;
}
static int static int
__parse_callchain_report_opt(const char *arg, bool allow_record_opt) __parse_callchain_report_opt(const char *arg, bool allow_record_opt)
{ {
...@@ -106,7 +123,8 @@ __parse_callchain_report_opt(const char *arg, bool allow_record_opt) ...@@ -106,7 +123,8 @@ __parse_callchain_report_opt(const char *arg, bool allow_record_opt)
if (!parse_callchain_mode(tok) || if (!parse_callchain_mode(tok) ||
!parse_callchain_order(tok) || !parse_callchain_order(tok) ||
!parse_callchain_sort_key(tok)) { !parse_callchain_sort_key(tok) ||
!parse_callchain_value(tok)) {
/* parsing ok - move on to the next */ /* parsing ok - move on to the next */
try_stack_size = false; try_stack_size = false;
goto next; goto next;
...@@ -820,13 +838,27 @@ char *callchain_node__scnprintf_value(struct callchain_node *node, ...@@ -820,13 +838,27 @@ char *callchain_node__scnprintf_value(struct callchain_node *node,
{ {
double percent = 0.0; double percent = 0.0;
u64 period = callchain_cumul_hits(node); u64 period = callchain_cumul_hits(node);
unsigned count = callchain_cumul_counts(node);
if (callchain_param.mode == CHAIN_FOLDED) if (callchain_param.mode == CHAIN_FOLDED) {
period = node->hit; period = node->hit;
if (total) count = node->count;
percent = period * 100.0 / total; }
scnprintf(bf, bfsize, "%.2f%%", percent); switch (callchain_param.value) {
case CCVAL_PERIOD:
scnprintf(bf, bfsize, "%"PRIu64, period);
break;
case CCVAL_COUNT:
scnprintf(bf, bfsize, "%u", count);
break;
case CCVAL_PERCENT:
default:
if (total)
percent = period * 100.0 / total;
scnprintf(bf, bfsize, "%.2f%%", percent);
break;
}
return bf; return bf;
} }
...@@ -835,13 +867,25 @@ int callchain_node__fprintf_value(struct callchain_node *node, ...@@ -835,13 +867,25 @@ int callchain_node__fprintf_value(struct callchain_node *node,
{ {
double percent = 0.0; double percent = 0.0;
u64 period = callchain_cumul_hits(node); u64 period = callchain_cumul_hits(node);
unsigned count = callchain_cumul_counts(node);
if (callchain_param.mode == CHAIN_FOLDED) if (callchain_param.mode == CHAIN_FOLDED) {
period = node->hit; period = node->hit;
if (total) count = node->count;
percent = period * 100.0 / total; }
return percent_color_fprintf(fp, "%.2f%%", percent); switch (callchain_param.value) {
case CCVAL_PERIOD:
return fprintf(fp, "%"PRIu64, period);
case CCVAL_COUNT:
return fprintf(fp, "%u", count);
case CCVAL_PERCENT:
default:
if (total)
percent = period * 100.0 / total;
return percent_color_fprintf(fp, "%.2f%%", percent);
}
return 0;
} }
static void free_callchain_node(struct callchain_node *node) static void free_callchain_node(struct callchain_node *node)
......
...@@ -29,7 +29,8 @@ ...@@ -29,7 +29,8 @@
HELP_PAD "print_limit:\tmaximum number of call graph entry (<number>)\n" \ HELP_PAD "print_limit:\tmaximum number of call graph entry (<number>)\n" \
HELP_PAD "order:\t\tcall graph order (caller|callee)\n" \ HELP_PAD "order:\t\tcall graph order (caller|callee)\n" \
HELP_PAD "sort_key:\tcall graph sort key (function|address)\n" \ HELP_PAD "sort_key:\tcall graph sort key (function|address)\n" \
HELP_PAD "branch:\t\tinclude last branch info to call graph (branch)\n" HELP_PAD "branch:\t\tinclude last branch info to call graph (branch)\n" \
HELP_PAD "value:\t\tcall graph value (percent|period|count)\n"
enum perf_call_graph_mode { enum perf_call_graph_mode {
CALLCHAIN_NONE, CALLCHAIN_NONE,
...@@ -81,6 +82,12 @@ enum chain_key { ...@@ -81,6 +82,12 @@ enum chain_key {
CCKEY_ADDRESS CCKEY_ADDRESS
}; };
enum chain_value {
CCVAL_PERCENT,
CCVAL_PERIOD,
CCVAL_COUNT,
};
struct callchain_param { struct callchain_param {
bool enabled; bool enabled;
enum perf_call_graph_mode record_mode; enum perf_call_graph_mode record_mode;
...@@ -93,6 +100,7 @@ struct callchain_param { ...@@ -93,6 +100,7 @@ struct callchain_param {
bool order_set; bool order_set;
enum chain_key key; enum chain_key key;
bool branch_callstack; bool branch_callstack;
enum chain_value value;
}; };
extern struct callchain_param callchain_param; extern struct callchain_param callchain_param;
......
...@@ -21,7 +21,8 @@ struct callchain_param callchain_param = { ...@@ -21,7 +21,8 @@ struct callchain_param callchain_param = {
.mode = CHAIN_GRAPH_ABS, .mode = CHAIN_GRAPH_ABS,
.min_percent = 0.5, .min_percent = 0.5,
.order = ORDER_CALLEE, .order = ORDER_CALLEE,
.key = CCKEY_FUNCTION .key = CCKEY_FUNCTION,
.value = CCVAL_PERCENT,
}; };
/* /*
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment