perf_counter tools: Set the minimum percent for callchains to be displayed
Callchains output may become a burden on a trace because even rarely hit site are exposed. This can be too much information. Let the user set a threshold as a minimum percent of hits using the new pattern for the -c option: -c mode,min_percent Example: $ perf report -s sym -c flat,4 8.25% [k] copy_user_generic_string 4.19% copy_user_generic_string generic_file_aio_read do_sync_read vfs_read sys_pread64 system_call_fastpath pread64 5.39% [k] search_by_key 4.63% 0x00000000009e0a 2.36% [k] memcpy_c [...] $ perf report -s sym -c graph,2 8.25% [k] copy_user_generic_string | |--4.31%-- generic_file_aio_read | do_sync_read | vfs_read | | | --4.19%-- sys_pread64 | system_call_fastpath | pread64 | --3.24%-- generic_file_buffered_write __generic_file_aio_write_nolock generic_file_aio_write do_sync_write reiserfs_file_write vfs_write | --3.14%-- sys_pwrite64 system_call_fastpath __pwrite64 5.39% [k] search_by_key | --2.23%-- reiserfs_update_sd_size 4.63% 0x00000000009e0a 2.36% [k] memcpy_c [...] You can also omit it and it will default to 0. Signed-off-by: Frederic Weisbecker <fweisbec@gmail.com> Cc: Peter Zijlstra <a.p.zijlstra@chello.nl> Cc: Mike Galbraith <efault@gmx.de> Cc: Paul Mackerras <paulus@samba.org> Cc: Anton Blanchard <anton@samba.org> Cc: Arnaldo Carvalho de Melo <acme@redhat.com> LKML-Reference: <1246558475-10624-1-git-send-email-fweisbec@gmail.com> Signed-off-by: Ingo Molnar <mingo@elte.hu>
This commit is contained in:
parent
4eb3e4788b
commit
c20ab37ef3
|
@ -60,6 +60,7 @@ static regex_t parent_regex;
|
||||||
static int exclude_other = 1;
|
static int exclude_other = 1;
|
||||||
static int callchain;
|
static int callchain;
|
||||||
static enum chain_mode callchain_mode;
|
static enum chain_mode callchain_mode;
|
||||||
|
static double callchain_min_percent = 0.0;
|
||||||
|
|
||||||
static u64 sample_type;
|
static u64 sample_type;
|
||||||
|
|
||||||
|
@ -1224,7 +1225,7 @@ static void collapse__resort(void)
|
||||||
|
|
||||||
static struct rb_root output_hists;
|
static struct rb_root output_hists;
|
||||||
|
|
||||||
static void output__insert_entry(struct hist_entry *he)
|
static void output__insert_entry(struct hist_entry *he, u64 min_callchain_hits)
|
||||||
{
|
{
|
||||||
struct rb_node **p = &output_hists.rb_node;
|
struct rb_node **p = &output_hists.rb_node;
|
||||||
struct rb_node *parent = NULL;
|
struct rb_node *parent = NULL;
|
||||||
|
@ -1232,9 +1233,11 @@ static void output__insert_entry(struct hist_entry *he)
|
||||||
|
|
||||||
if (callchain) {
|
if (callchain) {
|
||||||
if (callchain_mode == FLAT)
|
if (callchain_mode == FLAT)
|
||||||
sort_chain_flat(&he->sorted_chain, &he->callchain);
|
sort_chain_flat(&he->sorted_chain, &he->callchain,
|
||||||
|
min_callchain_hits);
|
||||||
else if (callchain_mode == GRAPH)
|
else if (callchain_mode == GRAPH)
|
||||||
sort_chain_graph(&he->sorted_chain, &he->callchain);
|
sort_chain_graph(&he->sorted_chain, &he->callchain,
|
||||||
|
min_callchain_hits);
|
||||||
}
|
}
|
||||||
|
|
||||||
while (*p != NULL) {
|
while (*p != NULL) {
|
||||||
|
@ -1251,11 +1254,14 @@ static void output__insert_entry(struct hist_entry *he)
|
||||||
rb_insert_color(&he->rb_node, &output_hists);
|
rb_insert_color(&he->rb_node, &output_hists);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void output__resort(void)
|
static void output__resort(u64 total_samples)
|
||||||
{
|
{
|
||||||
struct rb_node *next;
|
struct rb_node *next;
|
||||||
struct hist_entry *n;
|
struct hist_entry *n;
|
||||||
struct rb_root *tree = &hist;
|
struct rb_root *tree = &hist;
|
||||||
|
u64 min_callchain_hits;
|
||||||
|
|
||||||
|
min_callchain_hits = total_samples * (callchain_min_percent / 100);
|
||||||
|
|
||||||
if (sort__need_collapse)
|
if (sort__need_collapse)
|
||||||
tree = &collapse_hists;
|
tree = &collapse_hists;
|
||||||
|
@ -1267,7 +1273,7 @@ static void output__resort(void)
|
||||||
next = rb_next(&n->rb_node);
|
next = rb_next(&n->rb_node);
|
||||||
|
|
||||||
rb_erase(&n->rb_node, tree);
|
rb_erase(&n->rb_node, tree);
|
||||||
output__insert_entry(n);
|
output__insert_entry(n, min_callchain_hits);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1801,7 +1807,7 @@ done:
|
||||||
dsos__fprintf(stdout);
|
dsos__fprintf(stdout);
|
||||||
|
|
||||||
collapse__resort();
|
collapse__resort();
|
||||||
output__resort();
|
output__resort(total);
|
||||||
output__fprintf(stdout, total);
|
output__fprintf(stdout, total);
|
||||||
|
|
||||||
return rc;
|
return rc;
|
||||||
|
@ -1811,19 +1817,36 @@ static int
|
||||||
parse_callchain_opt(const struct option *opt __used, const char *arg,
|
parse_callchain_opt(const struct option *opt __used, const char *arg,
|
||||||
int unset __used)
|
int unset __used)
|
||||||
{
|
{
|
||||||
|
char *tok;
|
||||||
|
char *endptr;
|
||||||
|
|
||||||
callchain = 1;
|
callchain = 1;
|
||||||
|
|
||||||
if (!arg)
|
if (!arg)
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
if (!strncmp(arg, "graph", strlen(arg)))
|
tok = strtok((char *)arg, ",");
|
||||||
|
if (!tok)
|
||||||
|
return -1;
|
||||||
|
|
||||||
|
/* get the output mode */
|
||||||
|
if (!strncmp(tok, "graph", strlen(arg)))
|
||||||
callchain_mode = GRAPH;
|
callchain_mode = GRAPH;
|
||||||
|
|
||||||
else if (!strncmp(arg, "flat", strlen(arg)))
|
else if (!strncmp(tok, "flat", strlen(arg)))
|
||||||
callchain_mode = FLAT;
|
callchain_mode = FLAT;
|
||||||
else
|
else
|
||||||
return -1;
|
return -1;
|
||||||
|
|
||||||
|
/* get the min percentage */
|
||||||
|
tok = strtok(NULL, ",");
|
||||||
|
if (!tok)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
callchain_min_percent = strtod(tok, &endptr);
|
||||||
|
if (tok == endptr)
|
||||||
|
return -1;
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1850,9 +1873,9 @@ static const struct option options[] = {
|
||||||
"regex filter to identify parent, see: '--sort parent'"),
|
"regex filter to identify parent, see: '--sort parent'"),
|
||||||
OPT_BOOLEAN('x', "exclude-other", &exclude_other,
|
OPT_BOOLEAN('x', "exclude-other", &exclude_other,
|
||||||
"Only display entries with parent-match"),
|
"Only display entries with parent-match"),
|
||||||
OPT_CALLBACK_DEFAULT('c', "callchain", NULL, "output_type",
|
OPT_CALLBACK_DEFAULT('c', "callchain", NULL, "output_type,min_percent",
|
||||||
"Display callchains with output_type: flat, graph. "
|
"Display callchains using output_type and min percent threshold. "
|
||||||
"Default to flat", &parse_callchain_opt, "flat"),
|
"Default: flat,0", &parse_callchain_opt, "flat,100"),
|
||||||
OPT_STRING('d', "dsos", &dso_list_str, "dso[,dso...]",
|
OPT_STRING('d', "dsos", &dso_list_str, "dso[,dso...]",
|
||||||
"only consider symbols in these dsos"),
|
"only consider symbols in these dsos"),
|
||||||
OPT_STRING('C', "comms", &comm_list_str, "comm[,comm...]",
|
OPT_STRING('C', "comms", &comm_list_str, "comm[,comm...]",
|
||||||
|
|
|
@ -57,18 +57,19 @@ rb_insert_callchain(struct rb_root *root, struct callchain_node *chain,
|
||||||
* Once we get every callchains from the stream, we can now
|
* Once we get every callchains from the stream, we can now
|
||||||
* sort them by hit
|
* sort them by hit
|
||||||
*/
|
*/
|
||||||
void sort_chain_flat(struct rb_root *rb_root, struct callchain_node *node)
|
void sort_chain_flat(struct rb_root *rb_root, struct callchain_node *node,
|
||||||
|
u64 min_hit)
|
||||||
{
|
{
|
||||||
struct callchain_node *child;
|
struct callchain_node *child;
|
||||||
|
|
||||||
chain_for_each_child(child, node)
|
chain_for_each_child(child, node)
|
||||||
sort_chain_flat(rb_root, child);
|
sort_chain_flat(rb_root, child, min_hit);
|
||||||
|
|
||||||
if (node->hit)
|
if (node->hit && node->hit >= min_hit)
|
||||||
rb_insert_callchain(rb_root, node, FLAT);
|
rb_insert_callchain(rb_root, node, FLAT);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void __sort_chain_graph(struct callchain_node *node)
|
static void __sort_chain_graph(struct callchain_node *node, u64 min_hit)
|
||||||
{
|
{
|
||||||
struct callchain_node *child;
|
struct callchain_node *child;
|
||||||
|
|
||||||
|
@ -76,16 +77,18 @@ static void __sort_chain_graph(struct callchain_node *node)
|
||||||
node->cumul_hit = node->hit;
|
node->cumul_hit = node->hit;
|
||||||
|
|
||||||
chain_for_each_child(child, node) {
|
chain_for_each_child(child, node) {
|
||||||
__sort_chain_graph(child);
|
__sort_chain_graph(child, min_hit);
|
||||||
rb_insert_callchain(&node->rb_root, child, GRAPH);
|
if (child->cumul_hit >= min_hit)
|
||||||
|
rb_insert_callchain(&node->rb_root, child, GRAPH);
|
||||||
node->cumul_hit += child->cumul_hit;
|
node->cumul_hit += child->cumul_hit;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
sort_chain_graph(struct rb_root *rb_root, struct callchain_node *chain_root)
|
sort_chain_graph(struct rb_root *rb_root, struct callchain_node *chain_root,
|
||||||
|
u64 min_hit)
|
||||||
{
|
{
|
||||||
__sort_chain_graph(chain_root);
|
__sort_chain_graph(chain_root, min_hit);
|
||||||
rb_root->rb_node = chain_root->rb_root.rb_node;
|
rb_root->rb_node = chain_root->rb_root.rb_node;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -38,6 +38,8 @@ static inline void callchain_init(struct callchain_node *node)
|
||||||
|
|
||||||
void append_chain(struct callchain_node *root, struct ip_callchain *chain,
|
void append_chain(struct callchain_node *root, struct ip_callchain *chain,
|
||||||
struct symbol **syms);
|
struct symbol **syms);
|
||||||
void sort_chain_flat(struct rb_root *rb_root, struct callchain_node *node);
|
void sort_chain_flat(struct rb_root *rb_root, struct callchain_node *node,
|
||||||
void sort_chain_graph(struct rb_root *rb_root, struct callchain_node *node);
|
u64 min_hit);
|
||||||
|
void sort_chain_graph(struct rb_root *rb_root, struct callchain_node *node,
|
||||||
|
u64 min_hit);
|
||||||
#endif
|
#endif
|
||||||
|
|
Loading…
Reference in New Issue