aboutsummaryrefslogtreecommitdiffstats
path: root/tools/perf/util/hist.c
diff options
context:
space:
mode:
authorNamhyung Kim <namhyung@kernel.org>2016-02-24 10:13:36 -0500
committerArnaldo Carvalho de Melo <acme@redhat.com>2016-02-24 14:55:17 -0500
commit8c01872fe3c17fde1ce74eecf523d6d7fce5ffec (patch)
treeaa7e0e5cd509ad83cb99639f6126417398b75534 /tools/perf/util/hist.c
parent1a3906a7e6b9cbfaf2a3d00c310aed8af8e10d92 (diff)
perf hists: Add helper functions for hierarchy mode
The rb_hierarchy_{next,prev,last} functions are to traverse all hist entries in a hierarchy. They will be used by various function which supports hierarchy output. As the rb_hierarchy_next() is used to traverse the whole hierarchy, it sometime needs to visit entries regardless of current folding state. So add enum hierarchy_move_dir and pass it to __rb_hierarchy_next() for those cases. Signed-off-by: Namhyung Kim <namhyung@kernel.org> Acked-by: Pekka Enberg <penberg@kernel.org> Cc: Andi Kleen <andi@firstfloor.org> Cc: David Ahern <dsahern@gmail.com> Cc: Jiri Olsa <jolsa@kernel.org> Cc: Peter Zijlstra <peterz@infradead.org> Cc: Stephane Eranian <eranian@google.com> Cc: Wang Nan <wangnan0@huawei.com> Link: http://lkml.kernel.org/r/1456326830-30456-5-git-send-email-namhyung@kernel.org Signed-off-by: Arnaldo Carvalho de Melo <acme@redhat.com>
Diffstat (limited to 'tools/perf/util/hist.c')
-rw-r--r--tools/perf/util/hist.c56
1 files changed, 56 insertions, 0 deletions
diff --git a/tools/perf/util/hist.c b/tools/perf/util/hist.c
index 6ddac2fb29b5..358af7ed07af 100644
--- a/tools/perf/util/hist.c
+++ b/tools/perf/util/hist.c
@@ -1500,6 +1500,62 @@ void hists__output_resort(struct hists *hists, struct ui_progress *prog)
1500 output_resort(hists, prog, symbol_conf.use_callchain); 1500 output_resort(hists, prog, symbol_conf.use_callchain);
1501} 1501}
1502 1502
1503static bool can_goto_child(struct hist_entry *he, enum hierarchy_move_dir hmd)
1504{
1505 if (he->leaf || hmd == HMD_FORCE_SIBLING)
1506 return false;
1507
1508 if (he->unfolded || hmd == HMD_FORCE_CHILD)
1509 return true;
1510
1511 return false;
1512}
1513
1514struct rb_node *rb_hierarchy_last(struct rb_node *node)
1515{
1516 struct hist_entry *he = rb_entry(node, struct hist_entry, rb_node);
1517
1518 while (can_goto_child(he, HMD_NORMAL)) {
1519 node = rb_last(&he->hroot_out);
1520 he = rb_entry(node, struct hist_entry, rb_node);
1521 }
1522 return node;
1523}
1524
1525struct rb_node *__rb_hierarchy_next(struct rb_node *node, enum hierarchy_move_dir hmd)
1526{
1527 struct hist_entry *he = rb_entry(node, struct hist_entry, rb_node);
1528
1529 if (can_goto_child(he, hmd))
1530 node = rb_first(&he->hroot_out);
1531 else
1532 node = rb_next(node);
1533
1534 while (node == NULL) {
1535 he = he->parent_he;
1536 if (he == NULL)
1537 break;
1538
1539 node = rb_next(&he->rb_node);
1540 }
1541 return node;
1542}
1543
1544struct rb_node *rb_hierarchy_prev(struct rb_node *node)
1545{
1546 struct hist_entry *he = rb_entry(node, struct hist_entry, rb_node);
1547
1548 node = rb_prev(node);
1549 if (node)
1550 return rb_hierarchy_last(node);
1551
1552 he = he->parent_he;
1553 if (he == NULL)
1554 return NULL;
1555
1556 return &he->rb_node;
1557}
1558
1503static void hists__remove_entry_filter(struct hists *hists, struct hist_entry *h, 1559static void hists__remove_entry_filter(struct hists *hists, struct hist_entry *h,
1504 enum hist_filter filter) 1560 enum hist_filter filter)
1505{ 1561{