| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | #include <stdio.h>
 | 
					
						
							|  |  |  | #include "../libslang.h"
 | 
					
						
							|  |  |  | #include <stdlib.h>
 | 
					
						
							|  |  |  | #include <string.h>
 | 
					
						
							|  |  |  | #include <newt.h>
 | 
					
						
							|  |  |  | #include <linux/rbtree.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-04 00:14:26 -07:00
										 |  |  | #include "../../util/evsel.h"
 | 
					
						
							|  |  |  | #include "../../util/evlist.h"
 | 
					
						
							|  |  |  | #include "../../util/hist.h"
 | 
					
						
							|  |  |  | #include "../../util/pstack.h"
 | 
					
						
							|  |  |  | #include "../../util/sort.h"
 | 
					
						
							|  |  |  | #include "../../util/util.h"
 | 
					
						
							| 
									
										
										
										
											2012-11-02 14:50:06 +09:00
										 |  |  | #include "../../arch/common.h"
 | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include "../browser.h"
 | 
					
						
							|  |  |  | #include "../helpline.h"
 | 
					
						
							|  |  |  | #include "../util.h"
 | 
					
						
							| 
									
										
										
										
											2011-10-26 12:04:37 -02:00
										 |  |  | #include "../ui.h"
 | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | #include "map.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct hist_browser { | 
					
						
							|  |  |  | 	struct ui_browser   b; | 
					
						
							|  |  |  | 	struct hists	    *hists; | 
					
						
							|  |  |  | 	struct hist_entry   *he_selection; | 
					
						
							|  |  |  | 	struct map_symbol   *selection; | 
					
						
							| 
									
										
										
										
											2012-06-07 19:31:28 -03:00
										 |  |  | 	int		     print_seq; | 
					
						
							| 
									
										
										
										
											2012-08-03 13:53:40 -03:00
										 |  |  | 	bool		     show_dso; | 
					
						
							| 
									
										
										
										
											2011-10-06 10:36:35 -03:00
										 |  |  | 	bool		     has_symbols; | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-09-03 11:53:09 +09:00
										 |  |  | extern void hist_browser__init_hpp(void); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | static int hists__browser_title(struct hists *hists, char *bf, size_t size, | 
					
						
							| 
									
										
										
										
											2011-10-18 19:07:34 -02:00
										 |  |  | 				const char *ev_name); | 
					
						
							| 
									
										
										
										
											2011-10-05 19:11:32 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | static void hist_browser__refresh_dimensions(struct hist_browser *browser) | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | { | 
					
						
							|  |  |  | 	/* 3 == +/- toggle symbol before actual hist_entry rendering */ | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 	browser->b.width = 3 + (hists__sort_list_width(browser->hists) + | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 			     sizeof("[k]")); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | static void hist_browser__reset(struct hist_browser *browser) | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 	browser->b.nr_entries = browser->hists->nr_entries; | 
					
						
							|  |  |  | 	hist_browser__refresh_dimensions(browser); | 
					
						
							|  |  |  | 	ui_browser__reset_index(&browser->b); | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static char tree__folded_sign(bool unfolded) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return unfolded ? '-' : '+'; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | static char map_symbol__folded(const struct map_symbol *ms) | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 	return ms->has_children ? tree__folded_sign(ms->unfolded) : ' '; | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | static char hist_entry__folded(const struct hist_entry *he) | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 	return map_symbol__folded(&he->ms); | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | static char callchain_list__folded(const struct callchain_list *cl) | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 	return map_symbol__folded(&cl->ms); | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | static void map_symbol__set_folding(struct map_symbol *ms, bool unfold) | 
					
						
							| 
									
										
										
										
											2010-08-25 17:18:35 -03:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 	ms->unfolded = unfold ? ms->has_children : false; | 
					
						
							| 
									
										
										
										
											2010-08-25 17:18:35 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | static int callchain_node__count_rows_rb_tree(struct callchain_node *node) | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | { | 
					
						
							|  |  |  | 	int n = 0; | 
					
						
							|  |  |  | 	struct rb_node *nd; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 	for (nd = rb_first(&node->rb_root); nd; nd = rb_next(nd)) { | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 		struct callchain_node *child = rb_entry(nd, struct callchain_node, rb_node); | 
					
						
							|  |  |  | 		struct callchain_list *chain; | 
					
						
							|  |  |  | 		char folded_sign = ' '; /* No children */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		list_for_each_entry(chain, &child->val, list) { | 
					
						
							|  |  |  | 			++n; | 
					
						
							|  |  |  | 			/* We need this because we may not have children */ | 
					
						
							|  |  |  | 			folded_sign = callchain_list__folded(chain); | 
					
						
							|  |  |  | 			if (folded_sign == '+') | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (folded_sign == '-') /* Have children and they're unfolded */ | 
					
						
							|  |  |  | 			n += callchain_node__count_rows_rb_tree(child); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return n; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int callchain_node__count_rows(struct callchain_node *node) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct callchain_list *chain; | 
					
						
							|  |  |  | 	bool unfolded = false; | 
					
						
							|  |  |  | 	int n = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	list_for_each_entry(chain, &node->val, list) { | 
					
						
							|  |  |  | 		++n; | 
					
						
							|  |  |  | 		unfolded = chain->ms.unfolded; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (unfolded) | 
					
						
							|  |  |  | 		n += callchain_node__count_rows_rb_tree(node); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return n; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int callchain__count_rows(struct rb_root *chain) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct rb_node *nd; | 
					
						
							|  |  |  | 	int n = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for (nd = rb_first(chain); nd; nd = rb_next(nd)) { | 
					
						
							|  |  |  | 		struct callchain_node *node = rb_entry(nd, struct callchain_node, rb_node); | 
					
						
							|  |  |  | 		n += callchain_node__count_rows(node); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return n; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | static bool map_symbol__toggle_fold(struct map_symbol *ms) | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 	if (!ms) | 
					
						
							| 
									
										
										
										
											2012-04-04 22:21:31 +02:00
										 |  |  | 		return false; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 	if (!ms->has_children) | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 		return false; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 	ms->unfolded = !ms->unfolded; | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 	return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | static void callchain_node__init_have_children_rb_tree(struct callchain_node *node) | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 	struct rb_node *nd = rb_first(&node->rb_root); | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 	for (nd = rb_first(&node->rb_root); nd; nd = rb_next(nd)) { | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 		struct callchain_node *child = rb_entry(nd, struct callchain_node, rb_node); | 
					
						
							|  |  |  | 		struct callchain_list *chain; | 
					
						
							| 
									
										
										
										
											2010-08-25 16:05:36 -03:00
										 |  |  | 		bool first = true; | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		list_for_each_entry(chain, &child->val, list) { | 
					
						
							|  |  |  | 			if (first) { | 
					
						
							|  |  |  | 				first = false; | 
					
						
							|  |  |  | 				chain->ms.has_children = chain->list.next != &child->val || | 
					
						
							| 
									
										
										
										
											2010-08-25 16:05:36 -03:00
										 |  |  | 							 !RB_EMPTY_ROOT(&child->rb_root); | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 			} else | 
					
						
							|  |  |  | 				chain->ms.has_children = chain->list.next == &child->val && | 
					
						
							| 
									
										
										
										
											2010-08-25 16:05:36 -03:00
										 |  |  | 							 !RB_EMPTY_ROOT(&child->rb_root); | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		callchain_node__init_have_children_rb_tree(child); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | static void callchain_node__init_have_children(struct callchain_node *node) | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | { | 
					
						
							|  |  |  | 	struct callchain_list *chain; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 	list_for_each_entry(chain, &node->val, list) | 
					
						
							|  |  |  | 		chain->ms.has_children = !RB_EMPTY_ROOT(&node->rb_root); | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 	callchain_node__init_have_children_rb_tree(node); | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | static void callchain__init_have_children(struct rb_root *root) | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | { | 
					
						
							|  |  |  | 	struct rb_node *nd; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 	for (nd = rb_first(root); nd; nd = rb_next(nd)) { | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 		struct callchain_node *node = rb_entry(nd, struct callchain_node, rb_node); | 
					
						
							|  |  |  | 		callchain_node__init_have_children(node); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | static void hist_entry__init_have_children(struct hist_entry *he) | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 	if (!he->init_have_children) { | 
					
						
							|  |  |  | 		he->ms.has_children = !RB_EMPTY_ROOT(&he->sorted_chain); | 
					
						
							|  |  |  | 		callchain__init_have_children(&he->sorted_chain); | 
					
						
							|  |  |  | 		he->init_have_children = true; | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | static bool hist_browser__toggle_fold(struct hist_browser *browser) | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 	if (map_symbol__toggle_fold(browser->selection)) { | 
					
						
							|  |  |  | 		struct hist_entry *he = browser->he_selection; | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		hist_entry__init_have_children(he); | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 		browser->hists->nr_entries -= he->nr_rows; | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		if (he->ms.unfolded) | 
					
						
							|  |  |  | 			he->nr_rows = callchain__count_rows(&he->sorted_chain); | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 			he->nr_rows = 0; | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 		browser->hists->nr_entries += he->nr_rows; | 
					
						
							|  |  |  | 		browser->b.nr_entries = browser->hists->nr_entries; | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		return true; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* If it doesn't have children, no toggling performed */ | 
					
						
							|  |  |  | 	return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | static int callchain_node__set_folding_rb_tree(struct callchain_node *node, bool unfold) | 
					
						
							| 
									
										
										
										
											2010-08-25 17:18:35 -03:00
										 |  |  | { | 
					
						
							|  |  |  | 	int n = 0; | 
					
						
							|  |  |  | 	struct rb_node *nd; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 	for (nd = rb_first(&node->rb_root); nd; nd = rb_next(nd)) { | 
					
						
							| 
									
										
										
										
											2010-08-25 17:18:35 -03:00
										 |  |  | 		struct callchain_node *child = rb_entry(nd, struct callchain_node, rb_node); | 
					
						
							|  |  |  | 		struct callchain_list *chain; | 
					
						
							|  |  |  | 		bool has_children = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		list_for_each_entry(chain, &child->val, list) { | 
					
						
							|  |  |  | 			++n; | 
					
						
							|  |  |  | 			map_symbol__set_folding(&chain->ms, unfold); | 
					
						
							|  |  |  | 			has_children = chain->ms.has_children; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (has_children) | 
					
						
							|  |  |  | 			n += callchain_node__set_folding_rb_tree(child, unfold); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return n; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int callchain_node__set_folding(struct callchain_node *node, bool unfold) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct callchain_list *chain; | 
					
						
							|  |  |  | 	bool has_children = false; | 
					
						
							|  |  |  | 	int n = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	list_for_each_entry(chain, &node->val, list) { | 
					
						
							|  |  |  | 		++n; | 
					
						
							|  |  |  | 		map_symbol__set_folding(&chain->ms, unfold); | 
					
						
							|  |  |  | 		has_children = chain->ms.has_children; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (has_children) | 
					
						
							|  |  |  | 		n += callchain_node__set_folding_rb_tree(node, unfold); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return n; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int callchain__set_folding(struct rb_root *chain, bool unfold) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct rb_node *nd; | 
					
						
							|  |  |  | 	int n = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for (nd = rb_first(chain); nd; nd = rb_next(nd)) { | 
					
						
							|  |  |  | 		struct callchain_node *node = rb_entry(nd, struct callchain_node, rb_node); | 
					
						
							|  |  |  | 		n += callchain_node__set_folding(node, unfold); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return n; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | static void hist_entry__set_folding(struct hist_entry *he, bool unfold) | 
					
						
							| 
									
										
										
										
											2010-08-25 17:18:35 -03:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 	hist_entry__init_have_children(he); | 
					
						
							|  |  |  | 	map_symbol__set_folding(&he->ms, unfold); | 
					
						
							| 
									
										
										
										
											2010-08-25 17:18:35 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 	if (he->ms.has_children) { | 
					
						
							|  |  |  | 		int n = callchain__set_folding(&he->sorted_chain, unfold); | 
					
						
							|  |  |  | 		he->nr_rows = unfold ? n : 0; | 
					
						
							| 
									
										
										
										
											2010-08-25 17:18:35 -03:00
										 |  |  | 	} else | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 		he->nr_rows = 0; | 
					
						
							| 
									
										
										
										
											2010-08-25 17:18:35 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | static void hists__set_folding(struct hists *hists, bool unfold) | 
					
						
							| 
									
										
										
										
											2010-08-25 17:18:35 -03:00
										 |  |  | { | 
					
						
							|  |  |  | 	struct rb_node *nd; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 	hists->nr_entries = 0; | 
					
						
							| 
									
										
										
										
											2010-08-25 17:18:35 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 	for (nd = rb_first(&hists->entries); nd; nd = rb_next(nd)) { | 
					
						
							| 
									
										
										
										
											2010-08-25 17:18:35 -03:00
										 |  |  | 		struct hist_entry *he = rb_entry(nd, struct hist_entry, rb_node); | 
					
						
							|  |  |  | 		hist_entry__set_folding(he, unfold); | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 		hists->nr_entries += 1 + he->nr_rows; | 
					
						
							| 
									
										
										
										
											2010-08-25 17:18:35 -03:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | static void hist_browser__set_folding(struct hist_browser *browser, bool unfold) | 
					
						
							| 
									
										
										
										
											2010-08-25 17:18:35 -03:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 	hists__set_folding(browser->hists, unfold); | 
					
						
							|  |  |  | 	browser->b.nr_entries = browser->hists->nr_entries; | 
					
						
							| 
									
										
										
										
											2010-08-25 17:18:35 -03:00
										 |  |  | 	/* Go to the start, we may be way after valid entries after a collapse */ | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 	ui_browser__reset_index(&browser->b); | 
					
						
							| 
									
										
										
										
											2010-08-25 17:18:35 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-10-29 12:15:04 -02:00
										 |  |  | static void ui_browser__warn_lost_events(struct ui_browser *browser) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	ui_browser__warning(browser, 4, | 
					
						
							|  |  |  | 		"Events are being lost, check IO/CPU overload!\n\n" | 
					
						
							|  |  |  | 		"You may want to run 'perf' using a RT scheduler policy:\n\n" | 
					
						
							|  |  |  | 		" perf top -r 80\n\n" | 
					
						
							|  |  |  | 		"Or reduce the sampling frequency."); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | static int hist_browser__run(struct hist_browser *browser, const char *ev_name, | 
					
						
							| 
									
										
										
										
											2012-11-02 14:50:05 +09:00
										 |  |  | 			     struct hist_browser_timer *hbt) | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-08-11 10:07:43 -03:00
										 |  |  | 	int key; | 
					
						
							| 
									
										
										
										
											2011-10-05 19:11:32 -03:00
										 |  |  | 	char title[160]; | 
					
						
							| 
									
										
										
										
											2012-11-02 14:50:05 +09:00
										 |  |  | 	int delay_secs = hbt ? hbt->refresh : 0; | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 	browser->b.entries = &browser->hists->entries; | 
					
						
							|  |  |  | 	browser->b.nr_entries = browser->hists->nr_entries; | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 	hist_browser__refresh_dimensions(browser); | 
					
						
							|  |  |  | 	hists__browser_title(browser->hists, title, sizeof(title), ev_name); | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 	if (ui_browser__show(&browser->b, title, | 
					
						
							| 
									
										
										
										
											2010-08-10 15:44:20 -03:00
										 |  |  | 			     "Press '?' for help on key bindings") < 0) | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 		return -1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	while (1) { | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 		key = ui_browser__run(&browser->b, delay_secs); | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-11 10:07:43 -03:00
										 |  |  | 		switch (key) { | 
					
						
							| 
									
										
										
										
											2011-10-26 08:05:34 -02:00
										 |  |  | 		case K_TIMER: | 
					
						
							| 
									
										
										
										
											2012-11-02 14:50:05 +09:00
										 |  |  | 			hbt->timer(hbt->arg); | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 			ui_browser__update_nr_entries(&browser->b, browser->hists->nr_entries); | 
					
						
							| 
									
										
										
										
											2011-10-29 12:15:04 -02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 			if (browser->hists->stats.nr_lost_warned != | 
					
						
							|  |  |  | 			    browser->hists->stats.nr_events[PERF_RECORD_LOST]) { | 
					
						
							|  |  |  | 				browser->hists->stats.nr_lost_warned = | 
					
						
							|  |  |  | 					browser->hists->stats.nr_events[PERF_RECORD_LOST]; | 
					
						
							|  |  |  | 				ui_browser__warn_lost_events(&browser->b); | 
					
						
							| 
									
										
										
										
											2011-10-29 12:15:04 -02:00
										 |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 			hists__browser_title(browser->hists, title, sizeof(title), ev_name); | 
					
						
							|  |  |  | 			ui_browser__show_title(&browser->b, title); | 
					
						
							| 
									
										
										
										
											2011-10-05 19:11:32 -03:00
										 |  |  | 			continue; | 
					
						
							| 
									
										
										
										
											2010-08-10 15:50:07 -03:00
										 |  |  | 		case 'D': { /* Debug */ | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 			static int seq; | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 			struct hist_entry *h = rb_entry(browser->b.top, | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 							struct hist_entry, rb_node); | 
					
						
							|  |  |  | 			ui_helpline__pop(); | 
					
						
							|  |  |  | 			ui_helpline__fpush("%d: nr_ent=(%d,%d), height=%d, idx=%d, fve: idx=%d, row_off=%d, nrows=%d", | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 					   seq++, browser->b.nr_entries, | 
					
						
							|  |  |  | 					   browser->hists->nr_entries, | 
					
						
							|  |  |  | 					   browser->b.height, | 
					
						
							|  |  |  | 					   browser->b.index, | 
					
						
							|  |  |  | 					   browser->b.top_idx, | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 					   h->row_offset, h->nr_rows); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2010-08-25 17:18:35 -03:00
										 |  |  | 			break; | 
					
						
							|  |  |  | 		case 'C': | 
					
						
							|  |  |  | 			/* Collapse the whole world. */ | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 			hist_browser__set_folding(browser, false); | 
					
						
							| 
									
										
										
										
											2010-08-25 17:18:35 -03:00
										 |  |  | 			break; | 
					
						
							|  |  |  | 		case 'E': | 
					
						
							|  |  |  | 			/* Expand the whole world. */ | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 			hist_browser__set_folding(browser, true); | 
					
						
							| 
									
										
										
										
											2010-08-25 17:18:35 -03:00
										 |  |  | 			break; | 
					
						
							| 
									
										
										
										
											2011-10-20 16:59:15 -02:00
										 |  |  | 		case K_ENTER: | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 			if (hist_browser__toggle_fold(browser)) | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 				break; | 
					
						
							|  |  |  | 			/* fall thru */ | 
					
						
							|  |  |  | 		default: | 
					
						
							| 
									
										
										
										
											2010-08-11 10:07:43 -03:00
										 |  |  | 			goto out; | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2010-08-11 10:07:43 -03:00
										 |  |  | out: | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 	ui_browser__hide(&browser->b); | 
					
						
							| 
									
										
										
										
											2010-08-11 10:07:43 -03:00
										 |  |  | 	return key; | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | static char *callchain_list__sym_name(struct callchain_list *cl, | 
					
						
							| 
									
										
										
										
											2012-08-03 13:53:40 -03:00
										 |  |  | 				      char *bf, size_t bfsize, bool show_dso) | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-08-03 13:53:40 -03:00
										 |  |  | 	int printed; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 	if (cl->ms.sym) | 
					
						
							| 
									
										
										
										
											2012-08-03 13:53:40 -03:00
										 |  |  | 		printed = scnprintf(bf, bfsize, "%s", cl->ms.sym->name); | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		printed = scnprintf(bf, bfsize, "%#" PRIx64, cl->ip); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (show_dso) | 
					
						
							|  |  |  | 		scnprintf(bf + printed, bfsize - printed, " %s", | 
					
						
							|  |  |  | 			  cl->ms.map ? cl->ms.map->dso->short_name : "unknown"); | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return bf; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define LEVEL_OFFSET_STEP 3
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | static int hist_browser__show_callchain_node_rb_tree(struct hist_browser *browser, | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 						     struct callchain_node *chain_node, | 
					
						
							|  |  |  | 						     u64 total, int level, | 
					
						
							|  |  |  | 						     unsigned short row, | 
					
						
							|  |  |  | 						     off_t *row_offset, | 
					
						
							|  |  |  | 						     bool *is_current_entry) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct rb_node *node; | 
					
						
							|  |  |  | 	int first_row = row, width, offset = level * LEVEL_OFFSET_STEP; | 
					
						
							|  |  |  | 	u64 new_total, remaining; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (callchain_param.mode == CHAIN_GRAPH_REL) | 
					
						
							|  |  |  | 		new_total = chain_node->children_hit; | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		new_total = total; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	remaining = new_total; | 
					
						
							|  |  |  | 	node = rb_first(&chain_node->rb_root); | 
					
						
							|  |  |  | 	while (node) { | 
					
						
							|  |  |  | 		struct callchain_node *child = rb_entry(node, struct callchain_node, rb_node); | 
					
						
							|  |  |  | 		struct rb_node *next = rb_next(node); | 
					
						
							| 
									
										
										
										
											2011-01-14 04:51:59 +01:00
										 |  |  | 		u64 cumul = callchain_cumul_hits(child); | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 		struct callchain_list *chain; | 
					
						
							|  |  |  | 		char folded_sign = ' '; | 
					
						
							|  |  |  | 		int first = true; | 
					
						
							|  |  |  | 		int extra_offset = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		remaining -= cumul; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		list_for_each_entry(chain, &child->val, list) { | 
					
						
							| 
									
										
										
										
											2012-08-03 13:53:40 -03:00
										 |  |  | 			char bf[1024], *alloc_str; | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 			const char *str; | 
					
						
							|  |  |  | 			int color; | 
					
						
							|  |  |  | 			bool was_first = first; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-25 16:30:03 -03:00
										 |  |  | 			if (first) | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 				first = false; | 
					
						
							| 
									
										
										
										
											2010-08-25 16:30:03 -03:00
										 |  |  | 			else | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 				extra_offset = LEVEL_OFFSET_STEP; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			folded_sign = callchain_list__folded(chain); | 
					
						
							|  |  |  | 			if (*row_offset != 0) { | 
					
						
							|  |  |  | 				--*row_offset; | 
					
						
							|  |  |  | 				goto do_next; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			alloc_str = NULL; | 
					
						
							| 
									
										
										
										
											2012-08-03 13:53:40 -03:00
										 |  |  | 			str = callchain_list__sym_name(chain, bf, sizeof(bf), | 
					
						
							|  |  |  | 						       browser->show_dso); | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 			if (was_first) { | 
					
						
							|  |  |  | 				double percent = cumul * 100.0 / new_total; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				if (asprintf(&alloc_str, "%2.2f%% %s", percent, str) < 0) | 
					
						
							|  |  |  | 					str = "Not enough memory!"; | 
					
						
							|  |  |  | 				else | 
					
						
							|  |  |  | 					str = alloc_str; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			color = HE_COLORSET_NORMAL; | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 			width = browser->b.width - (offset + extra_offset + 2); | 
					
						
							|  |  |  | 			if (ui_browser__is_current_entry(&browser->b, row)) { | 
					
						
							|  |  |  | 				browser->selection = &chain->ms; | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 				color = HE_COLORSET_SELECTED; | 
					
						
							|  |  |  | 				*is_current_entry = true; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 			ui_browser__set_color(&browser->b, color); | 
					
						
							|  |  |  | 			ui_browser__gotorc(&browser->b, row, 0); | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 			slsmg_write_nstring(" ", offset + extra_offset); | 
					
						
							|  |  |  | 			slsmg_printf("%c ", folded_sign); | 
					
						
							|  |  |  | 			slsmg_write_nstring(str, width); | 
					
						
							|  |  |  | 			free(alloc_str); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 			if (++row == browser->b.height) | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 				goto out; | 
					
						
							|  |  |  | do_next: | 
					
						
							|  |  |  | 			if (folded_sign == '+') | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (folded_sign == '-') { | 
					
						
							|  |  |  | 			const int new_level = level + (extra_offset ? 2 : 1); | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 			row += hist_browser__show_callchain_node_rb_tree(browser, child, new_total, | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 									 new_level, row, row_offset, | 
					
						
							|  |  |  | 									 is_current_entry); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 		if (row == browser->b.height) | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 			goto out; | 
					
						
							|  |  |  | 		node = next; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | out: | 
					
						
							|  |  |  | 	return row - first_row; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | static int hist_browser__show_callchain_node(struct hist_browser *browser, | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 					     struct callchain_node *node, | 
					
						
							|  |  |  | 					     int level, unsigned short row, | 
					
						
							|  |  |  | 					     off_t *row_offset, | 
					
						
							|  |  |  | 					     bool *is_current_entry) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct callchain_list *chain; | 
					
						
							|  |  |  | 	int first_row = row, | 
					
						
							|  |  |  | 	     offset = level * LEVEL_OFFSET_STEP, | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 	     width = browser->b.width - offset; | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 	char folded_sign = ' '; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	list_for_each_entry(chain, &node->val, list) { | 
					
						
							| 
									
										
										
										
											2012-08-03 13:53:40 -03:00
										 |  |  | 		char bf[1024], *s; | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 		int color; | 
					
						
							| 
									
										
										
										
											2010-08-25 16:30:03 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 		folded_sign = callchain_list__folded(chain); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (*row_offset != 0) { | 
					
						
							|  |  |  | 			--*row_offset; | 
					
						
							|  |  |  | 			continue; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		color = HE_COLORSET_NORMAL; | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 		if (ui_browser__is_current_entry(&browser->b, row)) { | 
					
						
							|  |  |  | 			browser->selection = &chain->ms; | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 			color = HE_COLORSET_SELECTED; | 
					
						
							|  |  |  | 			*is_current_entry = true; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-08-03 13:53:40 -03:00
										 |  |  | 		s = callchain_list__sym_name(chain, bf, sizeof(bf), | 
					
						
							|  |  |  | 					     browser->show_dso); | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 		ui_browser__gotorc(&browser->b, row, 0); | 
					
						
							|  |  |  | 		ui_browser__set_color(&browser->b, color); | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 		slsmg_write_nstring(" ", offset); | 
					
						
							|  |  |  | 		slsmg_printf("%c ", folded_sign); | 
					
						
							|  |  |  | 		slsmg_write_nstring(s, width - 2); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 		if (++row == browser->b.height) | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 			goto out; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (folded_sign == '-') | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 		row += hist_browser__show_callchain_node_rb_tree(browser, node, | 
					
						
							|  |  |  | 								 browser->hists->stats.total_period, | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 								 level + 1, row, | 
					
						
							|  |  |  | 								 row_offset, | 
					
						
							|  |  |  | 								 is_current_entry); | 
					
						
							|  |  |  | out: | 
					
						
							|  |  |  | 	return row - first_row; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | static int hist_browser__show_callchain(struct hist_browser *browser, | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 					struct rb_root *chain, | 
					
						
							|  |  |  | 					int level, unsigned short row, | 
					
						
							|  |  |  | 					off_t *row_offset, | 
					
						
							|  |  |  | 					bool *is_current_entry) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct rb_node *nd; | 
					
						
							|  |  |  | 	int first_row = row; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for (nd = rb_first(chain); nd; nd = rb_next(nd)) { | 
					
						
							|  |  |  | 		struct callchain_node *node = rb_entry(nd, struct callchain_node, rb_node); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 		row += hist_browser__show_callchain_node(browser, node, level, | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 							 row, row_offset, | 
					
						
							|  |  |  | 							 is_current_entry); | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 		if (row == browser->b.height) | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 			break; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return row - first_row; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-22 18:09:38 +09:00
										 |  |  | struct hpp_arg { | 
					
						
							|  |  |  | 	struct ui_browser *b; | 
					
						
							|  |  |  | 	char folded_sign; | 
					
						
							|  |  |  | 	bool current_entry; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int __hpp__color_callchain(struct hpp_arg *arg) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (!symbol_conf.use_callchain) | 
					
						
							|  |  |  | 		return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	slsmg_printf("%c ", arg->folded_sign); | 
					
						
							|  |  |  | 	return 2; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-22 18:09:35 +09:00
										 |  |  | static int __hpp__color_fmt(struct perf_hpp *hpp, struct hist_entry *he, | 
					
						
							| 
									
										
										
										
											2013-01-22 18:09:38 +09:00
										 |  |  | 			    u64 (*get_field)(struct hist_entry *), | 
					
						
							|  |  |  | 			    int (*callchain_cb)(struct hpp_arg *)) | 
					
						
							| 
									
										
										
										
											2013-01-22 18:09:35 +09:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-01-22 18:09:38 +09:00
										 |  |  | 	int ret = 0; | 
					
						
							| 
									
										
										
										
											2013-01-22 18:09:35 +09:00
										 |  |  | 	double percent = 0.0; | 
					
						
							|  |  |  | 	struct hists *hists = he->hists; | 
					
						
							| 
									
										
										
										
											2013-01-22 18:09:38 +09:00
										 |  |  | 	struct hpp_arg *arg = hpp->ptr; | 
					
						
							| 
									
										
										
										
											2013-01-22 18:09:35 +09:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (hists->stats.total_period) | 
					
						
							|  |  |  | 		percent = 100.0 * get_field(he) / hists->stats.total_period; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-22 18:09:38 +09:00
										 |  |  | 	ui_browser__set_percent_color(arg->b, percent, arg->current_entry); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (callchain_cb) | 
					
						
							|  |  |  | 		ret += callchain_cb(arg); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ret += scnprintf(hpp->buf, hpp->size, "%6.2f%%", percent); | 
					
						
							|  |  |  | 	slsmg_printf("%s", hpp->buf); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-22 18:09:39 +09:00
										 |  |  | 	if (symbol_conf.event_group) { | 
					
						
							|  |  |  | 		int prev_idx, idx_delta; | 
					
						
							|  |  |  | 		struct perf_evsel *evsel = hists_to_evsel(hists); | 
					
						
							|  |  |  | 		struct hist_entry *pair; | 
					
						
							|  |  |  | 		int nr_members = evsel->nr_members; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (nr_members <= 1) | 
					
						
							|  |  |  | 			goto out; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		prev_idx = perf_evsel__group_idx(evsel); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		list_for_each_entry(pair, &he->pairs.head, pairs.node) { | 
					
						
							|  |  |  | 			u64 period = get_field(pair); | 
					
						
							|  |  |  | 			u64 total = pair->hists->stats.total_period; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if (!total) | 
					
						
							|  |  |  | 				continue; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			evsel = hists_to_evsel(pair->hists); | 
					
						
							|  |  |  | 			idx_delta = perf_evsel__group_idx(evsel) - prev_idx - 1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			while (idx_delta--) { | 
					
						
							|  |  |  | 				/*
 | 
					
						
							|  |  |  | 				 * zero-fill group members in the middle which | 
					
						
							|  |  |  | 				 * have no sample | 
					
						
							|  |  |  | 				 */ | 
					
						
							|  |  |  | 				ui_browser__set_percent_color(arg->b, 0.0, | 
					
						
							|  |  |  | 							arg->current_entry); | 
					
						
							|  |  |  | 				ret += scnprintf(hpp->buf, hpp->size, | 
					
						
							|  |  |  | 						 " %6.2f%%", 0.0); | 
					
						
							|  |  |  | 				slsmg_printf("%s", hpp->buf); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			percent = 100.0 * period / total; | 
					
						
							|  |  |  | 			ui_browser__set_percent_color(arg->b, percent, | 
					
						
							|  |  |  | 						      arg->current_entry); | 
					
						
							|  |  |  | 			ret += scnprintf(hpp->buf, hpp->size, | 
					
						
							|  |  |  | 					 " %6.2f%%", percent); | 
					
						
							|  |  |  | 			slsmg_printf("%s", hpp->buf); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			prev_idx = perf_evsel__group_idx(evsel); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		idx_delta = nr_members - prev_idx - 1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		while (idx_delta--) { | 
					
						
							|  |  |  | 			/*
 | 
					
						
							|  |  |  | 			 * zero-fill group members at last which have no sample | 
					
						
							|  |  |  | 			 */ | 
					
						
							|  |  |  | 			ui_browser__set_percent_color(arg->b, 0.0, | 
					
						
							|  |  |  | 						      arg->current_entry); | 
					
						
							|  |  |  | 			ret += scnprintf(hpp->buf, hpp->size, | 
					
						
							|  |  |  | 					 " %6.2f%%", 0.0); | 
					
						
							|  |  |  | 			slsmg_printf("%s", hpp->buf); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | out: | 
					
						
							| 
									
										
										
										
											2013-01-22 18:09:38 +09:00
										 |  |  | 	if (!arg->current_entry || !arg->b->navkeypressed) | 
					
						
							|  |  |  | 		ui_browser__set_color(arg->b, HE_COLORSET_NORMAL); | 
					
						
							| 
									
										
										
										
											2013-01-22 18:09:35 +09:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return ret; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-22 18:09:38 +09:00
										 |  |  | #define __HPP_COLOR_PERCENT_FN(_type, _field, _cb)			\
 | 
					
						
							| 
									
										
										
										
											2013-01-22 18:09:35 +09:00
										 |  |  | static u64 __hpp_get_##_field(struct hist_entry *he)			\ | 
					
						
							|  |  |  | {									\ | 
					
						
							|  |  |  | 	return he->stat._field;						\ | 
					
						
							|  |  |  | }									\ | 
					
						
							|  |  |  | 									\ | 
					
						
							|  |  |  | static int hist_browser__hpp_color_##_type(struct perf_hpp *hpp,	\ | 
					
						
							|  |  |  | 					   struct hist_entry *he)	\ | 
					
						
							| 
									
										
										
										
											2012-09-03 11:53:09 +09:00
										 |  |  | {									\ | 
					
						
							| 
									
										
										
										
											2013-01-22 18:09:38 +09:00
										 |  |  | 	return __hpp__color_fmt(hpp, he, __hpp_get_##_field, _cb);	\ | 
					
						
							| 
									
										
										
										
											2012-09-03 11:53:09 +09:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-22 18:09:38 +09:00
										 |  |  | __HPP_COLOR_PERCENT_FN(overhead, period, __hpp__color_callchain) | 
					
						
							|  |  |  | __HPP_COLOR_PERCENT_FN(overhead_sys, period_sys, NULL) | 
					
						
							|  |  |  | __HPP_COLOR_PERCENT_FN(overhead_us, period_us, NULL) | 
					
						
							|  |  |  | __HPP_COLOR_PERCENT_FN(overhead_guest_sys, period_guest_sys, NULL) | 
					
						
							|  |  |  | __HPP_COLOR_PERCENT_FN(overhead_guest_us, period_guest_us, NULL) | 
					
						
							| 
									
										
										
										
											2012-09-03 11:53:09 +09:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-22 18:09:35 +09:00
										 |  |  | #undef __HPP_COLOR_PERCENT_FN
 | 
					
						
							| 
									
										
										
										
											2012-09-03 11:53:09 +09:00
										 |  |  | 
 | 
					
						
							|  |  |  | void hist_browser__init_hpp(void) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2012-10-13 00:06:16 +02:00
										 |  |  | 	perf_hpp__column_enable(PERF_HPP__OVERHEAD); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-10-04 21:49:39 +09:00
										 |  |  | 	perf_hpp__init(); | 
					
						
							| 
									
										
										
										
											2012-09-03 11:53:09 +09:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	perf_hpp__format[PERF_HPP__OVERHEAD].color = | 
					
						
							|  |  |  | 				hist_browser__hpp_color_overhead; | 
					
						
							|  |  |  | 	perf_hpp__format[PERF_HPP__OVERHEAD_SYS].color = | 
					
						
							|  |  |  | 				hist_browser__hpp_color_overhead_sys; | 
					
						
							|  |  |  | 	perf_hpp__format[PERF_HPP__OVERHEAD_US].color = | 
					
						
							|  |  |  | 				hist_browser__hpp_color_overhead_us; | 
					
						
							|  |  |  | 	perf_hpp__format[PERF_HPP__OVERHEAD_GUEST_SYS].color = | 
					
						
							|  |  |  | 				hist_browser__hpp_color_overhead_guest_sys; | 
					
						
							|  |  |  | 	perf_hpp__format[PERF_HPP__OVERHEAD_GUEST_US].color = | 
					
						
							|  |  |  | 				hist_browser__hpp_color_overhead_guest_us; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | static int hist_browser__show_entry(struct hist_browser *browser, | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 				    struct hist_entry *entry, | 
					
						
							|  |  |  | 				    unsigned short row) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	char s[256]; | 
					
						
							| 
									
										
										
										
											2012-10-13 00:06:16 +02:00
										 |  |  | 	int printed = 0; | 
					
						
							| 
									
										
										
										
											2012-09-12 15:35:06 +09:00
										 |  |  | 	int width = browser->b.width; | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 	char folded_sign = ' '; | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 	bool current_entry = ui_browser__is_current_entry(&browser->b, row); | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 	off_t row_offset = entry->row_offset; | 
					
						
							| 
									
										
										
										
											2012-10-15 18:14:35 -03:00
										 |  |  | 	bool first = true; | 
					
						
							| 
									
										
										
										
											2012-10-13 00:06:16 +02:00
										 |  |  | 	struct perf_hpp_fmt *fmt; | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (current_entry) { | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 		browser->he_selection = entry; | 
					
						
							|  |  |  | 		browser->selection = &entry->ms; | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (symbol_conf.use_callchain) { | 
					
						
							| 
									
										
										
										
											2010-08-25 16:30:03 -03:00
										 |  |  | 		hist_entry__init_have_children(entry); | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 		folded_sign = hist_entry__folded(entry); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (row_offset == 0) { | 
					
						
							| 
									
										
										
										
											2013-01-22 18:09:38 +09:00
										 |  |  | 		struct hpp_arg arg = { | 
					
						
							|  |  |  | 			.b 		= &browser->b, | 
					
						
							|  |  |  | 			.folded_sign	= folded_sign, | 
					
						
							|  |  |  | 			.current_entry	= current_entry, | 
					
						
							|  |  |  | 		}; | 
					
						
							| 
									
										
										
										
											2012-09-03 11:53:09 +09:00
										 |  |  | 		struct perf_hpp hpp = { | 
					
						
							|  |  |  | 			.buf		= s, | 
					
						
							|  |  |  | 			.size		= sizeof(s), | 
					
						
							| 
									
										
										
										
											2013-01-22 18:09:38 +09:00
										 |  |  | 			.ptr		= &arg, | 
					
						
							| 
									
										
										
										
											2012-09-03 11:53:09 +09:00
										 |  |  | 		}; | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-09-12 15:35:06 +09:00
										 |  |  | 		ui_browser__gotorc(&browser->b, row, 0); | 
					
						
							| 
									
										
										
										
											2011-10-18 14:31:35 -02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-10-13 00:06:16 +02:00
										 |  |  | 		perf_hpp__for_each_format(fmt) { | 
					
						
							| 
									
										
										
										
											2012-10-15 18:14:35 -03:00
										 |  |  | 			if (!first) { | 
					
						
							| 
									
										
										
										
											2012-09-03 11:53:09 +09:00
										 |  |  | 				slsmg_printf("  "); | 
					
						
							|  |  |  | 				width -= 2; | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2012-10-15 18:14:35 -03:00
										 |  |  | 			first = false; | 
					
						
							| 
									
										
										
										
											2011-10-18 14:31:35 -02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-10-13 00:06:16 +02:00
										 |  |  | 			if (fmt->color) { | 
					
						
							|  |  |  | 				width -= fmt->color(&hpp, entry); | 
					
						
							| 
									
										
										
										
											2012-09-03 11:53:09 +09:00
										 |  |  | 			} else { | 
					
						
							| 
									
										
										
										
											2012-10-13 00:06:16 +02:00
										 |  |  | 				width -= fmt->entry(&hpp, entry); | 
					
						
							| 
									
										
										
										
											2012-09-03 11:53:09 +09:00
										 |  |  | 				slsmg_printf("%s", s); | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2011-10-19 14:37:59 -02:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-09-03 11:53:09 +09:00
										 |  |  | 		/* The scroll bar isn't being used */ | 
					
						
							|  |  |  | 		if (!browser->b.navkeypressed) | 
					
						
							|  |  |  | 			width += 1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		hist_entry__sort_snprintf(entry, s, sizeof(s), browser->hists); | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 		slsmg_write_nstring(s, width); | 
					
						
							|  |  |  | 		++row; | 
					
						
							|  |  |  | 		++printed; | 
					
						
							|  |  |  | 	} else | 
					
						
							|  |  |  | 		--row_offset; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 	if (folded_sign == '-' && row != browser->b.height) { | 
					
						
							|  |  |  | 		printed += hist_browser__show_callchain(browser, &entry->sorted_chain, | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 							1, row, &row_offset, | 
					
						
							|  |  |  | 							¤t_entry); | 
					
						
							|  |  |  | 		if (current_entry) | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 			browser->he_selection = entry; | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return printed; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-10-14 09:31:53 -03:00
										 |  |  | static void ui_browser__hists_init_top(struct ui_browser *browser) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (browser->top == NULL) { | 
					
						
							|  |  |  | 		struct hist_browser *hb; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		hb = container_of(browser, struct hist_browser, b); | 
					
						
							|  |  |  | 		browser->top = rb_first(&hb->hists->entries); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | static unsigned int hist_browser__refresh(struct ui_browser *browser) | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | { | 
					
						
							|  |  |  | 	unsigned row = 0; | 
					
						
							|  |  |  | 	struct rb_node *nd; | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 	struct hist_browser *hb = container_of(browser, struct hist_browser, b); | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 	ui_browser__hists_init_top(browser); | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 	for (nd = browser->top; nd; nd = rb_next(nd)) { | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 		struct hist_entry *h = rb_entry(nd, struct hist_entry, rb_node); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (h->filtered) | 
					
						
							|  |  |  | 			continue; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		row += hist_browser__show_entry(hb, h, row); | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 		if (row == browser->height) | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 			break; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return row; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static struct rb_node *hists__filter_entries(struct rb_node *nd) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	while (nd != NULL) { | 
					
						
							|  |  |  | 		struct hist_entry *h = rb_entry(nd, struct hist_entry, rb_node); | 
					
						
							|  |  |  | 		if (!h->filtered) | 
					
						
							|  |  |  | 			return nd; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		nd = rb_next(nd); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static struct rb_node *hists__filter_prev_entries(struct rb_node *nd) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	while (nd != NULL) { | 
					
						
							|  |  |  | 		struct hist_entry *h = rb_entry(nd, struct hist_entry, rb_node); | 
					
						
							|  |  |  | 		if (!h->filtered) | 
					
						
							|  |  |  | 			return nd; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		nd = rb_prev(nd); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | static void ui_browser__hists_seek(struct ui_browser *browser, | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 				   off_t offset, int whence) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct hist_entry *h; | 
					
						
							|  |  |  | 	struct rb_node *nd; | 
					
						
							|  |  |  | 	bool first = true; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 	if (browser->nr_entries == 0) | 
					
						
							| 
									
										
										
										
											2011-03-04 21:19:21 -03:00
										 |  |  | 		return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 	ui_browser__hists_init_top(browser); | 
					
						
							| 
									
										
										
										
											2011-10-14 09:31:53 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 	switch (whence) { | 
					
						
							|  |  |  | 	case SEEK_SET: | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 		nd = hists__filter_entries(rb_first(browser->entries)); | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 		break; | 
					
						
							|  |  |  | 	case SEEK_CUR: | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 		nd = browser->top; | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 		goto do_offset; | 
					
						
							|  |  |  | 	case SEEK_END: | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 		nd = hists__filter_prev_entries(rb_last(browser->entries)); | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 		first = false; | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	default: | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/*
 | 
					
						
							|  |  |  | 	 * Moves not relative to the first visible entry invalidates its | 
					
						
							|  |  |  | 	 * row_offset: | 
					
						
							|  |  |  | 	 */ | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 	h = rb_entry(browser->top, struct hist_entry, rb_node); | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 	h->row_offset = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/*
 | 
					
						
							|  |  |  | 	 * Here we have to check if nd is expanded (+), if it is we can't go | 
					
						
							|  |  |  | 	 * the next top level hist_entry, instead we must compute an offset of | 
					
						
							|  |  |  | 	 * what _not_ to show and not change the first visible entry. | 
					
						
							|  |  |  | 	 * | 
					
						
							|  |  |  | 	 * This offset increments when we are going from top to bottom and | 
					
						
							|  |  |  | 	 * decreases when we're going from bottom to top. | 
					
						
							|  |  |  | 	 * | 
					
						
							|  |  |  | 	 * As we don't have backpointers to the top level in the callchains | 
					
						
							|  |  |  | 	 * structure, we need to always print the whole hist_entry callchain, | 
					
						
							|  |  |  | 	 * skipping the first ones that are before the first visible entry | 
					
						
							|  |  |  | 	 * and stop when we printed enough lines to fill the screen. | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | do_offset: | 
					
						
							|  |  |  | 	if (offset > 0) { | 
					
						
							|  |  |  | 		do { | 
					
						
							|  |  |  | 			h = rb_entry(nd, struct hist_entry, rb_node); | 
					
						
							|  |  |  | 			if (h->ms.unfolded) { | 
					
						
							|  |  |  | 				u16 remaining = h->nr_rows - h->row_offset; | 
					
						
							|  |  |  | 				if (offset > remaining) { | 
					
						
							|  |  |  | 					offset -= remaining; | 
					
						
							|  |  |  | 					h->row_offset = 0; | 
					
						
							|  |  |  | 				} else { | 
					
						
							|  |  |  | 					h->row_offset += offset; | 
					
						
							|  |  |  | 					offset = 0; | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 					browser->top = nd; | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 					break; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			nd = hists__filter_entries(rb_next(nd)); | 
					
						
							|  |  |  | 			if (nd == NULL) | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 			--offset; | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 			browser->top = nd; | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 		} while (offset != 0); | 
					
						
							|  |  |  | 	} else if (offset < 0) { | 
					
						
							|  |  |  | 		while (1) { | 
					
						
							|  |  |  | 			h = rb_entry(nd, struct hist_entry, rb_node); | 
					
						
							|  |  |  | 			if (h->ms.unfolded) { | 
					
						
							|  |  |  | 				if (first) { | 
					
						
							|  |  |  | 					if (-offset > h->row_offset) { | 
					
						
							|  |  |  | 						offset += h->row_offset; | 
					
						
							|  |  |  | 						h->row_offset = 0; | 
					
						
							|  |  |  | 					} else { | 
					
						
							|  |  |  | 						h->row_offset += offset; | 
					
						
							|  |  |  | 						offset = 0; | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 						browser->top = nd; | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 						break; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} else { | 
					
						
							|  |  |  | 					if (-offset > h->nr_rows) { | 
					
						
							|  |  |  | 						offset += h->nr_rows; | 
					
						
							|  |  |  | 						h->row_offset = 0; | 
					
						
							|  |  |  | 					} else { | 
					
						
							|  |  |  | 						h->row_offset = h->nr_rows + offset; | 
					
						
							|  |  |  | 						offset = 0; | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 						browser->top = nd; | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 						break; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			nd = hists__filter_prev_entries(rb_prev(nd)); | 
					
						
							|  |  |  | 			if (nd == NULL) | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 			++offset; | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 			browser->top = nd; | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 			if (offset == 0) { | 
					
						
							|  |  |  | 				/*
 | 
					
						
							|  |  |  | 				 * Last unfiltered hist_entry, check if it is | 
					
						
							|  |  |  | 				 * unfolded, if it is then we should have | 
					
						
							|  |  |  | 				 * row_offset at its last entry. | 
					
						
							|  |  |  | 				 */ | 
					
						
							|  |  |  | 				h = rb_entry(nd, struct hist_entry, rb_node); | 
					
						
							|  |  |  | 				if (h->ms.unfolded) | 
					
						
							|  |  |  | 					h->row_offset = h->nr_rows; | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			first = false; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} else { | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 		browser->top = nd; | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 		h = rb_entry(nd, struct hist_entry, rb_node); | 
					
						
							|  |  |  | 		h->row_offset = 0; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-07 19:31:28 -03:00
										 |  |  | static int hist_browser__fprintf_callchain_node_rb_tree(struct hist_browser *browser, | 
					
						
							|  |  |  | 							struct callchain_node *chain_node, | 
					
						
							|  |  |  | 							u64 total, int level, | 
					
						
							|  |  |  | 							FILE *fp) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct rb_node *node; | 
					
						
							|  |  |  | 	int offset = level * LEVEL_OFFSET_STEP; | 
					
						
							|  |  |  | 	u64 new_total, remaining; | 
					
						
							|  |  |  | 	int printed = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (callchain_param.mode == CHAIN_GRAPH_REL) | 
					
						
							|  |  |  | 		new_total = chain_node->children_hit; | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		new_total = total; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	remaining = new_total; | 
					
						
							|  |  |  | 	node = rb_first(&chain_node->rb_root); | 
					
						
							|  |  |  | 	while (node) { | 
					
						
							|  |  |  | 		struct callchain_node *child = rb_entry(node, struct callchain_node, rb_node); | 
					
						
							|  |  |  | 		struct rb_node *next = rb_next(node); | 
					
						
							|  |  |  | 		u64 cumul = callchain_cumul_hits(child); | 
					
						
							|  |  |  | 		struct callchain_list *chain; | 
					
						
							|  |  |  | 		char folded_sign = ' '; | 
					
						
							|  |  |  | 		int first = true; | 
					
						
							|  |  |  | 		int extra_offset = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		remaining -= cumul; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		list_for_each_entry(chain, &child->val, list) { | 
					
						
							| 
									
										
										
										
											2012-08-03 13:53:40 -03:00
										 |  |  | 			char bf[1024], *alloc_str; | 
					
						
							| 
									
										
										
										
											2012-06-07 19:31:28 -03:00
										 |  |  | 			const char *str; | 
					
						
							|  |  |  | 			bool was_first = first; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if (first) | 
					
						
							|  |  |  | 				first = false; | 
					
						
							|  |  |  | 			else | 
					
						
							|  |  |  | 				extra_offset = LEVEL_OFFSET_STEP; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			folded_sign = callchain_list__folded(chain); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			alloc_str = NULL; | 
					
						
							| 
									
										
										
										
											2012-08-03 13:53:40 -03:00
										 |  |  | 			str = callchain_list__sym_name(chain, bf, sizeof(bf), | 
					
						
							|  |  |  | 						       browser->show_dso); | 
					
						
							| 
									
										
										
										
											2012-06-07 19:31:28 -03:00
										 |  |  | 			if (was_first) { | 
					
						
							|  |  |  | 				double percent = cumul * 100.0 / new_total; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				if (asprintf(&alloc_str, "%2.2f%% %s", percent, str) < 0) | 
					
						
							|  |  |  | 					str = "Not enough memory!"; | 
					
						
							|  |  |  | 				else | 
					
						
							|  |  |  | 					str = alloc_str; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			printed += fprintf(fp, "%*s%c %s\n", offset + extra_offset, " ", folded_sign, str); | 
					
						
							|  |  |  | 			free(alloc_str); | 
					
						
							|  |  |  | 			if (folded_sign == '+') | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (folded_sign == '-') { | 
					
						
							|  |  |  | 			const int new_level = level + (extra_offset ? 2 : 1); | 
					
						
							|  |  |  | 			printed += hist_browser__fprintf_callchain_node_rb_tree(browser, child, new_total, | 
					
						
							|  |  |  | 										new_level, fp); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		node = next; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return printed; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int hist_browser__fprintf_callchain_node(struct hist_browser *browser, | 
					
						
							|  |  |  | 						struct callchain_node *node, | 
					
						
							|  |  |  | 						int level, FILE *fp) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct callchain_list *chain; | 
					
						
							|  |  |  | 	int offset = level * LEVEL_OFFSET_STEP; | 
					
						
							|  |  |  | 	char folded_sign = ' '; | 
					
						
							|  |  |  | 	int printed = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	list_for_each_entry(chain, &node->val, list) { | 
					
						
							| 
									
										
										
										
											2012-08-03 13:53:40 -03:00
										 |  |  | 		char bf[1024], *s; | 
					
						
							| 
									
										
										
										
											2012-06-07 19:31:28 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		folded_sign = callchain_list__folded(chain); | 
					
						
							| 
									
										
										
										
											2012-08-03 13:53:40 -03:00
										 |  |  | 		s = callchain_list__sym_name(chain, bf, sizeof(bf), browser->show_dso); | 
					
						
							| 
									
										
										
										
											2012-06-07 19:31:28 -03:00
										 |  |  | 		printed += fprintf(fp, "%*s%c %s\n", offset, " ", folded_sign, s); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (folded_sign == '-') | 
					
						
							|  |  |  | 		printed += hist_browser__fprintf_callchain_node_rb_tree(browser, node, | 
					
						
							|  |  |  | 									browser->hists->stats.total_period, | 
					
						
							|  |  |  | 									level + 1,  fp); | 
					
						
							|  |  |  | 	return printed; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int hist_browser__fprintf_callchain(struct hist_browser *browser, | 
					
						
							|  |  |  | 					   struct rb_root *chain, int level, FILE *fp) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct rb_node *nd; | 
					
						
							|  |  |  | 	int printed = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for (nd = rb_first(chain); nd; nd = rb_next(nd)) { | 
					
						
							|  |  |  | 		struct callchain_node *node = rb_entry(nd, struct callchain_node, rb_node); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		printed += hist_browser__fprintf_callchain_node(browser, node, level, fp); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return printed; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int hist_browser__fprintf_entry(struct hist_browser *browser, | 
					
						
							|  |  |  | 				       struct hist_entry *he, FILE *fp) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	char s[8192]; | 
					
						
							|  |  |  | 	double percent; | 
					
						
							|  |  |  | 	int printed = 0; | 
					
						
							|  |  |  | 	char folded_sign = ' '; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (symbol_conf.use_callchain) | 
					
						
							|  |  |  | 		folded_sign = hist_entry__folded(he); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-08-20 13:52:06 +09:00
										 |  |  | 	hist_entry__sort_snprintf(he, s, sizeof(s), browser->hists); | 
					
						
							| 
									
										
										
										
											2012-10-04 21:49:41 +09:00
										 |  |  | 	percent = (he->stat.period * 100.0) / browser->hists->stats.total_period; | 
					
						
							| 
									
										
										
										
											2012-06-07 19:31:28 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (symbol_conf.use_callchain) | 
					
						
							|  |  |  | 		printed += fprintf(fp, "%c ", folded_sign); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	printed += fprintf(fp, " %5.2f%%", percent); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (symbol_conf.show_nr_samples) | 
					
						
							| 
									
										
										
										
											2012-10-04 21:49:41 +09:00
										 |  |  | 		printed += fprintf(fp, " %11u", he->stat.nr_events); | 
					
						
							| 
									
										
										
										
											2012-06-07 19:31:28 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (symbol_conf.show_total_period) | 
					
						
							| 
									
										
										
										
											2012-10-04 21:49:41 +09:00
										 |  |  | 		printed += fprintf(fp, " %12" PRIu64, he->stat.period); | 
					
						
							| 
									
										
										
										
											2012-06-07 19:31:28 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	printed += fprintf(fp, "%s\n", rtrim(s)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (folded_sign == '-') | 
					
						
							|  |  |  | 		printed += hist_browser__fprintf_callchain(browser, &he->sorted_chain, 1, fp); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return printed; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int hist_browser__fprintf(struct hist_browser *browser, FILE *fp) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct rb_node *nd = hists__filter_entries(rb_first(browser->b.entries)); | 
					
						
							|  |  |  | 	int printed = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	while (nd) { | 
					
						
							|  |  |  | 		struct hist_entry *h = rb_entry(nd, struct hist_entry, rb_node); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		printed += hist_browser__fprintf_entry(browser, h, fp); | 
					
						
							|  |  |  | 		nd = hists__filter_entries(rb_next(nd)); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return printed; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int hist_browser__dump(struct hist_browser *browser) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	char filename[64]; | 
					
						
							|  |  |  | 	FILE *fp; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	while (1) { | 
					
						
							|  |  |  | 		scnprintf(filename, sizeof(filename), "perf.hist.%d", browser->print_seq); | 
					
						
							|  |  |  | 		if (access(filename, F_OK)) | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		/*
 | 
					
						
							|  |  |  |  		 * XXX: Just an arbitrary lazy upper limit | 
					
						
							|  |  |  |  		 */ | 
					
						
							|  |  |  | 		if (++browser->print_seq == 8192) { | 
					
						
							|  |  |  | 			ui_helpline__fpush("Too many perf.hist.N files, nothing written!"); | 
					
						
							|  |  |  | 			return -1; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	fp = fopen(filename, "w"); | 
					
						
							|  |  |  | 	if (fp == NULL) { | 
					
						
							|  |  |  | 		char bf[64]; | 
					
						
							| 
									
										
										
										
											2012-07-24 00:06:54 +03:00
										 |  |  | 		const char *err = strerror_r(errno, bf, sizeof(bf)); | 
					
						
							|  |  |  | 		ui_helpline__fpush("Couldn't write to %s: %s", filename, err); | 
					
						
							| 
									
										
										
										
											2012-06-07 19:31:28 -03:00
										 |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	++browser->print_seq; | 
					
						
							|  |  |  | 	hist_browser__fprintf(browser, fp); | 
					
						
							|  |  |  | 	fclose(fp); | 
					
						
							|  |  |  | 	ui_helpline__fpush("%s written!", filename); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | static struct hist_browser *hist_browser__new(struct hists *hists) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 	struct hist_browser *browser = zalloc(sizeof(*browser)); | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 	if (browser) { | 
					
						
							|  |  |  | 		browser->hists = hists; | 
					
						
							|  |  |  | 		browser->b.refresh = hist_browser__refresh; | 
					
						
							|  |  |  | 		browser->b.seek = ui_browser__hists_seek; | 
					
						
							|  |  |  | 		browser->b.use_navkeypressed = true; | 
					
						
							| 
									
										
										
										
											2012-03-08 23:47:48 +01:00
										 |  |  | 		if (sort__branch_mode == 1) | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 			browser->has_symbols = sort_sym_from.list.next != NULL; | 
					
						
							| 
									
										
										
										
											2012-03-08 23:47:48 +01:00
										 |  |  | 		else | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 			browser->has_symbols = sort_sym.list.next != NULL; | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 	return browser; | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | static void hist_browser__delete(struct hist_browser *browser) | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 	free(browser); | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | static struct hist_entry *hist_browser__selected_entry(struct hist_browser *browser) | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 	return browser->he_selection; | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | static struct thread *hist_browser__selected_thread(struct hist_browser *browser) | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 	return browser->he_selection->thread; | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | static int hists__browser_title(struct hists *hists, char *bf, size_t size, | 
					
						
							| 
									
										
										
										
											2011-10-18 19:07:34 -02:00
										 |  |  | 				const char *ev_name) | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-09-13 10:25:04 -03:00
										 |  |  | 	char unit; | 
					
						
							|  |  |  | 	int printed; | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 	const struct dso *dso = hists->dso_filter; | 
					
						
							|  |  |  | 	const struct thread *thread = hists->thread_filter; | 
					
						
							|  |  |  | 	unsigned long nr_samples = hists->stats.nr_events[PERF_RECORD_SAMPLE]; | 
					
						
							|  |  |  | 	u64 nr_events = hists->stats.total_period; | 
					
						
							| 
									
										
										
										
											2013-01-22 18:09:44 +09:00
										 |  |  | 	struct perf_evsel *evsel = hists_to_evsel(hists); | 
					
						
							|  |  |  | 	char buf[512]; | 
					
						
							|  |  |  | 	size_t buflen = sizeof(buf); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (symbol_conf.event_group && evsel->nr_members > 1) { | 
					
						
							|  |  |  | 		struct perf_evsel *pos; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		perf_evsel__group_desc(evsel, buf, buflen); | 
					
						
							|  |  |  | 		ev_name = buf; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		for_each_group_member(pos, evsel) { | 
					
						
							|  |  |  | 			nr_samples += pos->hists.stats.nr_events[PERF_RECORD_SAMPLE]; | 
					
						
							|  |  |  | 			nr_events += pos->hists.stats.total_period; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2012-04-05 21:01:01 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	nr_samples = convert_unit(nr_samples, &unit); | 
					
						
							|  |  |  | 	printed = scnprintf(bf, size, | 
					
						
							|  |  |  | 			   "Samples: %lu%c of event '%s', Event count (approx.): %lu", | 
					
						
							|  |  |  | 			   nr_samples, unit, ev_name, nr_events); | 
					
						
							| 
									
										
										
										
											2010-09-13 10:25:04 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 	if (hists->uid_filter_str) | 
					
						
							| 
									
										
										
										
											2012-01-19 14:08:15 -02:00
										 |  |  | 		printed += snprintf(bf + printed, size - printed, | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 				    ", UID: %s", hists->uid_filter_str); | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 	if (thread) | 
					
						
							| 
									
										
										
										
											2012-03-14 12:29:29 -03:00
										 |  |  | 		printed += scnprintf(bf + printed, size - printed, | 
					
						
							| 
									
										
										
										
											2010-09-13 10:25:04 -03:00
										 |  |  | 				    ", Thread: %s(%d)", | 
					
						
							|  |  |  | 				    (thread->comm_set ? thread->comm : ""), | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 				    thread->pid); | 
					
						
							|  |  |  | 	if (dso) | 
					
						
							| 
									
										
										
										
											2012-03-14 12:29:29 -03:00
										 |  |  | 		printed += scnprintf(bf + printed, size - printed, | 
					
						
							| 
									
										
										
										
											2010-09-13 10:25:04 -03:00
										 |  |  | 				    ", DSO: %s", dso->short_name); | 
					
						
							|  |  |  | 	return printed; | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-12 16:13:30 +01:00
										 |  |  | static inline void free_popup_options(char **options, int n) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for (i = 0; i < n; ++i) { | 
					
						
							|  |  |  | 		free(options[i]); | 
					
						
							|  |  |  | 		options[i] = NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-01 00:00:55 +08:00
										 |  |  | /* Check whether the browser is for 'top' or 'report' */ | 
					
						
							|  |  |  | static inline bool is_report_browser(void *timer) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return timer == NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-03 14:38:20 +08:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * Only runtime switching of perf data file will make "input_name" point | 
					
						
							|  |  |  |  * to a malloced buffer. So add "is_input_name_malloced" flag to decide | 
					
						
							|  |  |  |  * whether we need to call free() for current "input_name" during the switch. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static bool is_input_name_malloced = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int switch_data_file(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	char *pwd, *options[32], *abs_path[32], *tmp; | 
					
						
							|  |  |  | 	DIR *pwd_dir; | 
					
						
							|  |  |  | 	int nr_options = 0, choice = -1, ret = -1; | 
					
						
							|  |  |  | 	struct dirent *dent; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	pwd = getenv("PWD"); | 
					
						
							|  |  |  | 	if (!pwd) | 
					
						
							|  |  |  | 		return ret; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	pwd_dir = opendir(pwd); | 
					
						
							|  |  |  | 	if (!pwd_dir) | 
					
						
							|  |  |  | 		return ret; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	memset(options, 0, sizeof(options)); | 
					
						
							|  |  |  | 	memset(options, 0, sizeof(abs_path)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	while ((dent = readdir(pwd_dir))) { | 
					
						
							|  |  |  | 		char path[PATH_MAX]; | 
					
						
							|  |  |  | 		u64 magic; | 
					
						
							|  |  |  | 		char *name = dent->d_name; | 
					
						
							|  |  |  | 		FILE *file; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (!(dent->d_type == DT_REG)) | 
					
						
							|  |  |  | 			continue; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		snprintf(path, sizeof(path), "%s/%s", pwd, name); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		file = fopen(path, "r"); | 
					
						
							|  |  |  | 		if (!file) | 
					
						
							|  |  |  | 			continue; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (fread(&magic, 1, 8, file) < 8) | 
					
						
							|  |  |  | 			goto close_file_and_continue; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (is_perf_magic(magic)) { | 
					
						
							|  |  |  | 			options[nr_options] = strdup(name); | 
					
						
							|  |  |  | 			if (!options[nr_options]) | 
					
						
							|  |  |  | 				goto close_file_and_continue; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			abs_path[nr_options] = strdup(path); | 
					
						
							|  |  |  | 			if (!abs_path[nr_options]) { | 
					
						
							|  |  |  | 				free(options[nr_options]); | 
					
						
							|  |  |  | 				ui__warning("Can't search all data files due to memory shortage.\n"); | 
					
						
							|  |  |  | 				fclose(file); | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			nr_options++; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | close_file_and_continue: | 
					
						
							|  |  |  | 		fclose(file); | 
					
						
							|  |  |  | 		if (nr_options >= 32) { | 
					
						
							|  |  |  | 			ui__warning("Too many perf data files in PWD!\n" | 
					
						
							|  |  |  | 				    "Only the first 32 files will be listed.\n"); | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	closedir(pwd_dir); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (nr_options) { | 
					
						
							|  |  |  | 		choice = ui__popup_menu(nr_options, options); | 
					
						
							|  |  |  | 		if (choice < nr_options && choice >= 0) { | 
					
						
							|  |  |  | 			tmp = strdup(abs_path[choice]); | 
					
						
							|  |  |  | 			if (tmp) { | 
					
						
							|  |  |  | 				if (is_input_name_malloced) | 
					
						
							|  |  |  | 					free((void *)input_name); | 
					
						
							|  |  |  | 				input_name = tmp; | 
					
						
							|  |  |  | 				is_input_name_malloced = true; | 
					
						
							|  |  |  | 				ret = 0; | 
					
						
							|  |  |  | 			} else | 
					
						
							|  |  |  | 				ui__warning("Data switch failed due to memory shortage!\n"); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	free_popup_options(options, nr_options); | 
					
						
							|  |  |  | 	free_popup_options(abs_path, nr_options); | 
					
						
							|  |  |  | 	return ret; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-10-05 19:35:54 -03:00
										 |  |  | static int perf_evsel__hists_browse(struct perf_evsel *evsel, int nr_events, | 
					
						
							| 
									
										
										
										
											2011-03-06 13:07:30 -03:00
										 |  |  | 				    const char *helpline, const char *ev_name, | 
					
						
							| 
									
										
										
										
											2011-10-05 19:11:32 -03:00
										 |  |  | 				    bool left_exits, | 
					
						
							| 
									
										
										
										
											2012-11-02 14:50:06 +09:00
										 |  |  | 				    struct hist_browser_timer *hbt, | 
					
						
							|  |  |  | 				    struct perf_session_env *env) | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 	struct hists *hists = &evsel->hists; | 
					
						
							|  |  |  | 	struct hist_browser *browser = hist_browser__new(hists); | 
					
						
							| 
									
										
										
										
											2012-03-08 23:47:48 +01:00
										 |  |  | 	struct branch_info *bi; | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 	struct pstack *fstack; | 
					
						
							| 
									
										
										
										
											2012-03-12 16:13:30 +01:00
										 |  |  | 	char *options[16]; | 
					
						
							|  |  |  | 	int nr_options = 0; | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 	int key = -1; | 
					
						
							| 
									
										
										
										
											2012-03-16 17:50:53 +09:00
										 |  |  | 	char buf[64]; | 
					
						
							| 
									
										
										
										
											2012-10-30 11:56:06 +08:00
										 |  |  | 	char script_opt[64]; | 
					
						
							| 
									
										
										
										
											2012-11-02 14:50:05 +09:00
										 |  |  | 	int delay_secs = hbt ? hbt->refresh : 0; | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (browser == NULL) | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	fstack = pstack__new(2); | 
					
						
							|  |  |  | 	if (fstack == NULL) | 
					
						
							|  |  |  | 		goto out; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ui_helpline__push(helpline); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-12 16:13:30 +01:00
										 |  |  | 	memset(options, 0, sizeof(options)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 	while (1) { | 
					
						
							| 
									
										
										
										
											2011-03-04 21:19:21 -03:00
										 |  |  | 		const struct thread *thread = NULL; | 
					
						
							|  |  |  | 		const struct dso *dso = NULL; | 
					
						
							| 
									
										
										
										
											2012-03-12 16:13:30 +01:00
										 |  |  | 		int choice = 0, | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 		    annotate = -2, zoom_dso = -2, zoom_thread = -2, | 
					
						
							| 
									
										
										
										
											2012-03-08 23:47:48 +01:00
										 |  |  | 		    annotate_f = -2, annotate_t = -2, browse_map = -2; | 
					
						
							| 
									
										
										
										
											2013-02-03 14:38:20 +08:00
										 |  |  | 		int scripts_comm = -2, scripts_symbol = -2, | 
					
						
							|  |  |  | 		    scripts_all = -2, switch_data = -2; | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-12 16:13:30 +01:00
										 |  |  | 		nr_options = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-02 14:50:05 +09:00
										 |  |  | 		key = hist_browser__run(browser, ev_name, hbt); | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-03-04 21:19:21 -03:00
										 |  |  | 		if (browser->he_selection != NULL) { | 
					
						
							|  |  |  | 			thread = hist_browser__selected_thread(browser); | 
					
						
							|  |  |  | 			dso = browser->selection->map ? browser->selection->map->dso : NULL; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2010-08-11 10:07:43 -03:00
										 |  |  | 		switch (key) { | 
					
						
							| 
									
										
										
										
											2011-10-20 16:59:15 -02:00
										 |  |  | 		case K_TAB: | 
					
						
							|  |  |  | 		case K_UNTAB: | 
					
						
							| 
									
										
										
										
											2011-10-19 11:37:47 -06:00
										 |  |  | 			if (nr_events == 1) | 
					
						
							|  |  |  | 				continue; | 
					
						
							| 
									
										
										
										
											2010-08-11 10:07:43 -03:00
										 |  |  | 			/*
 | 
					
						
							|  |  |  | 			 * Exit the browser, let hists__browser_tree | 
					
						
							|  |  |  | 			 * go to the next or previous | 
					
						
							|  |  |  | 			 */ | 
					
						
							|  |  |  | 			goto out_free_stack; | 
					
						
							|  |  |  | 		case 'a': | 
					
						
							| 
									
										
										
										
											2011-10-21 10:58:24 -02:00
										 |  |  | 			if (!browser->has_symbols) { | 
					
						
							| 
									
										
										
										
											2011-10-29 12:15:04 -02:00
										 |  |  | 				ui_browser__warning(&browser->b, delay_secs * 2, | 
					
						
							| 
									
										
										
										
											2011-10-21 10:58:24 -02:00
										 |  |  | 			"Annotation is only available for symbolic views, " | 
					
						
							| 
									
										
										
										
											2012-03-08 23:47:48 +01:00
										 |  |  | 			"include \"sym*\" in --sort to use it."); | 
					
						
							| 
									
										
										
										
											2011-10-21 10:58:24 -02:00
										 |  |  | 				continue; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-03-04 21:19:21 -03:00
										 |  |  | 			if (browser->selection == NULL || | 
					
						
							| 
									
										
										
										
											2011-04-08 14:31:26 +08:00
										 |  |  | 			    browser->selection->sym == NULL || | 
					
						
							| 
									
										
										
										
											2010-08-11 10:07:43 -03:00
										 |  |  | 			    browser->selection->map->dso->annotate_warned) | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 				continue; | 
					
						
							| 
									
										
										
										
											2010-08-11 10:07:43 -03:00
										 |  |  | 			goto do_annotate; | 
					
						
							| 
									
										
										
										
											2012-06-07 19:31:28 -03:00
										 |  |  | 		case 'P': | 
					
						
							|  |  |  | 			hist_browser__dump(browser); | 
					
						
							|  |  |  | 			continue; | 
					
						
							| 
									
										
										
										
											2010-08-11 10:07:43 -03:00
										 |  |  | 		case 'd': | 
					
						
							|  |  |  | 			goto zoom_dso; | 
					
						
							| 
									
										
										
										
											2012-08-03 13:53:40 -03:00
										 |  |  | 		case 'V': | 
					
						
							|  |  |  | 			browser->show_dso = !browser->show_dso; | 
					
						
							|  |  |  | 			continue; | 
					
						
							| 
									
										
										
										
											2010-08-11 10:07:43 -03:00
										 |  |  | 		case 't': | 
					
						
							|  |  |  | 			goto zoom_thread; | 
					
						
							| 
									
										
										
										
											2012-05-09 12:21:22 -03:00
										 |  |  | 		case '/': | 
					
						
							| 
									
										
										
										
											2012-03-16 17:50:53 +09:00
										 |  |  | 			if (ui_browser__input_window("Symbol to show", | 
					
						
							|  |  |  | 					"Please enter the name of symbol you want to see", | 
					
						
							|  |  |  | 					buf, "ENTER: OK, ESC: Cancel", | 
					
						
							|  |  |  | 					delay_secs * 2) == K_ENTER) { | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 				hists->symbol_filter_str = *buf ? buf : NULL; | 
					
						
							|  |  |  | 				hists__filter_by_symbol(hists); | 
					
						
							| 
									
										
										
										
											2012-03-16 17:50:53 +09:00
										 |  |  | 				hist_browser__reset(browser); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			continue; | 
					
						
							| 
									
										
										
										
											2012-10-30 11:56:06 +08:00
										 |  |  | 		case 'r': | 
					
						
							| 
									
										
										
										
											2012-11-02 14:50:05 +09:00
										 |  |  | 			if (is_report_browser(hbt)) | 
					
						
							| 
									
										
										
										
											2012-11-01 00:00:55 +08:00
										 |  |  | 				goto do_scripts; | 
					
						
							|  |  |  | 			continue; | 
					
						
							| 
									
										
										
										
											2013-02-03 14:38:20 +08:00
										 |  |  | 		case 's': | 
					
						
							|  |  |  | 			if (is_report_browser(hbt)) | 
					
						
							|  |  |  | 				goto do_data_switch; | 
					
						
							|  |  |  | 			continue; | 
					
						
							| 
									
										
										
										
											2011-10-20 16:59:15 -02:00
										 |  |  | 		case K_F1: | 
					
						
							| 
									
										
										
										
											2010-08-11 10:07:43 -03:00
										 |  |  | 		case 'h': | 
					
						
							|  |  |  | 		case '?': | 
					
						
							| 
									
										
										
										
											2011-10-26 12:04:37 -02:00
										 |  |  | 			ui_browser__help_window(&browser->b, | 
					
						
							|  |  |  | 					"h/?/F1        Show this window\n" | 
					
						
							| 
									
										
										
										
											2011-10-18 13:02:52 -02:00
										 |  |  | 					"UP/DOWN/PGUP\n" | 
					
						
							|  |  |  | 					"PGDN/SPACE    Navigate\n" | 
					
						
							|  |  |  | 					"q/ESC/CTRL+C  Exit browser\n\n" | 
					
						
							|  |  |  | 					"For multiple event sessions:\n\n" | 
					
						
							|  |  |  | 					"TAB/UNTAB Switch events\n\n" | 
					
						
							| 
									
										
										
										
											2011-10-06 10:36:35 -03:00
										 |  |  | 					"For symbolic views (--sort has sym):\n\n" | 
					
						
							| 
									
										
										
										
											2011-10-18 13:02:52 -02:00
										 |  |  | 					"->            Zoom into DSO/Threads & Annotate current symbol\n" | 
					
						
							|  |  |  | 					"<-            Zoom out\n" | 
					
						
							|  |  |  | 					"a             Annotate current symbol\n" | 
					
						
							|  |  |  | 					"C             Collapse all callchains\n" | 
					
						
							|  |  |  | 					"E             Expand all callchains\n" | 
					
						
							|  |  |  | 					"d             Zoom into current DSO\n" | 
					
						
							| 
									
										
										
										
											2012-03-16 17:50:53 +09:00
										 |  |  | 					"t             Zoom into current Thread\n" | 
					
						
							| 
									
										
										
										
											2012-11-01 00:00:55 +08:00
										 |  |  | 					"r             Run available scripts('perf report' only)\n" | 
					
						
							| 
									
										
										
										
											2013-02-03 14:38:20 +08:00
										 |  |  | 					"s             Switch to another data file in PWD ('perf report' only)\n" | 
					
						
							| 
									
										
										
										
											2012-06-07 19:31:28 -03:00
										 |  |  | 					"P             Print histograms to perf.hist.N\n" | 
					
						
							| 
									
										
										
										
											2012-08-03 13:53:40 -03:00
										 |  |  | 					"V             Verbose (DSO names in callchains, etc)\n" | 
					
						
							| 
									
										
										
										
											2012-05-09 12:21:22 -03:00
										 |  |  | 					"/             Filter symbol by name"); | 
					
						
							| 
									
										
										
										
											2010-08-11 10:07:43 -03:00
										 |  |  | 			continue; | 
					
						
							| 
									
										
										
										
											2011-10-20 16:59:15 -02:00
										 |  |  | 		case K_ENTER: | 
					
						
							|  |  |  | 		case K_RIGHT: | 
					
						
							| 
									
										
										
										
											2010-08-11 10:07:43 -03:00
										 |  |  | 			/* menu */ | 
					
						
							|  |  |  | 			break; | 
					
						
							| 
									
										
										
										
											2011-10-20 16:59:15 -02:00
										 |  |  | 		case K_LEFT: { | 
					
						
							| 
									
										
										
										
											2010-08-11 10:07:43 -03:00
										 |  |  | 			const void *top; | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-03-06 13:07:30 -03:00
										 |  |  | 			if (pstack__empty(fstack)) { | 
					
						
							|  |  |  | 				/*
 | 
					
						
							|  |  |  | 				 * Go back to the perf_evsel_menu__run or other user | 
					
						
							|  |  |  | 				 */ | 
					
						
							|  |  |  | 				if (left_exits) | 
					
						
							|  |  |  | 					goto out_free_stack; | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 				continue; | 
					
						
							| 
									
										
										
										
											2011-03-06 13:07:30 -03:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2010-08-11 10:07:43 -03:00
										 |  |  | 			top = pstack__pop(fstack); | 
					
						
							| 
									
										
										
										
											2011-10-18 19:07:34 -02:00
										 |  |  | 			if (top == &browser->hists->dso_filter) | 
					
						
							| 
									
										
										
										
											2010-08-11 10:07:43 -03:00
										 |  |  | 				goto zoom_out_dso; | 
					
						
							| 
									
										
										
										
											2011-10-18 19:07:34 -02:00
										 |  |  | 			if (top == &browser->hists->thread_filter) | 
					
						
							| 
									
										
										
										
											2010-08-11 10:07:43 -03:00
										 |  |  | 				goto zoom_out_thread; | 
					
						
							|  |  |  | 			continue; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2011-10-20 16:59:15 -02:00
										 |  |  | 		case K_ESC: | 
					
						
							| 
									
										
										
										
											2011-03-06 13:07:30 -03:00
										 |  |  | 			if (!left_exits && | 
					
						
							| 
									
										
										
										
											2011-10-26 12:04:37 -02:00
										 |  |  | 			    !ui_browser__dialog_yesno(&browser->b, | 
					
						
							|  |  |  | 					       "Do you really want to exit?")) | 
					
						
							| 
									
										
										
										
											2010-08-11 10:07:43 -03:00
										 |  |  | 				continue; | 
					
						
							|  |  |  | 			/* Fall thru */ | 
					
						
							| 
									
										
										
										
											2011-10-13 08:31:22 -03:00
										 |  |  | 		case 'q': | 
					
						
							|  |  |  | 		case CTRL('c'): | 
					
						
							| 
									
										
										
										
											2010-08-11 10:07:43 -03:00
										 |  |  | 			goto out_free_stack; | 
					
						
							| 
									
										
										
										
											2011-10-13 08:31:22 -03:00
										 |  |  | 		default: | 
					
						
							|  |  |  | 			continue; | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-10-06 10:36:35 -03:00
										 |  |  | 		if (!browser->has_symbols) | 
					
						
							|  |  |  | 			goto add_exit_option; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-08 23:47:48 +01:00
										 |  |  | 		if (sort__branch_mode == 1) { | 
					
						
							|  |  |  | 			bi = browser->he_selection->branch_info; | 
					
						
							|  |  |  | 			if (browser->selection != NULL && | 
					
						
							|  |  |  | 			    bi && | 
					
						
							|  |  |  | 			    bi->from.sym != NULL && | 
					
						
							|  |  |  | 			    !bi->from.map->dso->annotate_warned && | 
					
						
							|  |  |  | 				asprintf(&options[nr_options], "Annotate %s", | 
					
						
							|  |  |  | 					 bi->from.sym->name) > 0) | 
					
						
							|  |  |  | 				annotate_f = nr_options++; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if (browser->selection != NULL && | 
					
						
							|  |  |  | 			    bi && | 
					
						
							|  |  |  | 			    bi->to.sym != NULL && | 
					
						
							|  |  |  | 			    !bi->to.map->dso->annotate_warned && | 
					
						
							| 
									
										
										
										
											2012-03-12 16:13:29 +01:00
										 |  |  | 			    (bi->to.sym != bi->from.sym || | 
					
						
							|  |  |  | 			     bi->to.map->dso != bi->from.map->dso) && | 
					
						
							| 
									
										
										
										
											2012-03-08 23:47:48 +01:00
										 |  |  | 				asprintf(&options[nr_options], "Annotate %s", | 
					
						
							|  |  |  | 					 bi->to.sym->name) > 0) | 
					
						
							|  |  |  | 				annotate_t = nr_options++; | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if (browser->selection != NULL && | 
					
						
							|  |  |  | 			    browser->selection->sym != NULL && | 
					
						
							|  |  |  | 			    !browser->selection->map->dso->annotate_warned && | 
					
						
							|  |  |  | 				asprintf(&options[nr_options], "Annotate %s", | 
					
						
							|  |  |  | 					 browser->selection->sym->name) > 0) | 
					
						
							|  |  |  | 				annotate = nr_options++; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		if (thread != NULL && | 
					
						
							|  |  |  | 		    asprintf(&options[nr_options], "Zoom %s %s(%d) thread", | 
					
						
							| 
									
										
										
										
											2011-10-18 19:07:34 -02:00
										 |  |  | 			     (browser->hists->thread_filter ? "out of" : "into"), | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 			     (thread->comm_set ? thread->comm : ""), | 
					
						
							|  |  |  | 			     thread->pid) > 0) | 
					
						
							|  |  |  | 			zoom_thread = nr_options++; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (dso != NULL && | 
					
						
							|  |  |  | 		    asprintf(&options[nr_options], "Zoom %s %s DSO", | 
					
						
							| 
									
										
										
										
											2011-10-18 19:07:34 -02:00
										 |  |  | 			     (browser->hists->dso_filter ? "out of" : "into"), | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 			     (dso->kernel ? "the Kernel" : dso->short_name)) > 0) | 
					
						
							|  |  |  | 			zoom_dso = nr_options++; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-03-04 21:19:21 -03:00
										 |  |  | 		if (browser->selection != NULL && | 
					
						
							|  |  |  | 		    browser->selection->map != NULL && | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 		    asprintf(&options[nr_options], "Browse map details") > 0) | 
					
						
							|  |  |  | 			browse_map = nr_options++; | 
					
						
							| 
									
										
										
										
											2012-10-30 11:56:06 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		/* perf script support */ | 
					
						
							|  |  |  | 		if (browser->he_selection) { | 
					
						
							|  |  |  | 			struct symbol *sym; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if (asprintf(&options[nr_options], "Run scripts for samples of thread [%s]", | 
					
						
							|  |  |  | 				browser->he_selection->thread->comm) > 0) | 
					
						
							|  |  |  | 				scripts_comm = nr_options++; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			sym = browser->he_selection->ms.sym; | 
					
						
							|  |  |  | 			if (sym && sym->namelen && | 
					
						
							|  |  |  | 				asprintf(&options[nr_options], "Run scripts for samples of symbol [%s]", | 
					
						
							|  |  |  | 						sym->name) > 0) | 
					
						
							|  |  |  | 				scripts_symbol = nr_options++; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (asprintf(&options[nr_options], "Run scripts for all samples") > 0) | 
					
						
							|  |  |  | 			scripts_all = nr_options++; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-03 14:38:20 +08:00
										 |  |  | 		if (is_report_browser(hbt) && asprintf(&options[nr_options], | 
					
						
							|  |  |  | 				"Switch to another data file in PWD") > 0) | 
					
						
							|  |  |  | 			switch_data = nr_options++; | 
					
						
							| 
									
										
										
										
											2011-10-06 10:36:35 -03:00
										 |  |  | add_exit_option: | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 		options[nr_options++] = (char *)"Exit"; | 
					
						
							| 
									
										
										
										
											2012-03-12 16:13:30 +01:00
										 |  |  | retry_popup_menu: | 
					
						
							| 
									
										
										
										
											2010-08-10 15:58:50 -03:00
										 |  |  | 		choice = ui__popup_menu(nr_options, options); | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		if (choice == nr_options - 1) | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-12 16:13:30 +01:00
										 |  |  | 		if (choice == -1) { | 
					
						
							|  |  |  | 			free_popup_options(options, nr_options - 1); | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 			continue; | 
					
						
							| 
									
										
										
										
											2012-03-12 16:13:30 +01:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-08 23:47:48 +01:00
										 |  |  | 		if (choice == annotate || choice == annotate_t || choice == annotate_f) { | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 			struct hist_entry *he; | 
					
						
							| 
									
										
										
										
											2011-10-26 12:04:37 -02:00
										 |  |  | 			int err; | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | do_annotate: | 
					
						
							| 
									
										
										
										
											2012-11-02 14:50:06 +09:00
										 |  |  | 			if (!objdump_path && perf_session_env__lookup_objdump(env)) | 
					
						
							|  |  |  | 				continue; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 			he = hist_browser__selected_entry(browser); | 
					
						
							|  |  |  | 			if (he == NULL) | 
					
						
							|  |  |  | 				continue; | 
					
						
							| 
									
										
										
										
											2012-03-08 23:47:48 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			/*
 | 
					
						
							|  |  |  | 			 * we stash the branch_info symbol + map into the | 
					
						
							|  |  |  | 			 * the ms so we don't have to rewrite all the annotation | 
					
						
							|  |  |  | 			 * code to use branch_info. | 
					
						
							|  |  |  | 			 * in branch mode, the ms struct is not used | 
					
						
							|  |  |  | 			 */ | 
					
						
							|  |  |  | 			if (choice == annotate_f) { | 
					
						
							|  |  |  | 				he->ms.sym = he->branch_info->from.sym; | 
					
						
							|  |  |  | 				he->ms.map = he->branch_info->from.map; | 
					
						
							|  |  |  | 			}  else if (choice == annotate_t) { | 
					
						
							|  |  |  | 				he->ms.sym = he->branch_info->to.sym; | 
					
						
							|  |  |  | 				he->ms.map = he->branch_info->to.map; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-10-13 08:01:33 -03:00
										 |  |  | 			/*
 | 
					
						
							|  |  |  | 			 * Don't let this be freed, say, by hists__decay_entry. | 
					
						
							|  |  |  | 			 */ | 
					
						
							|  |  |  | 			he->used = true; | 
					
						
							| 
									
										
										
										
											2012-11-02 14:50:05 +09:00
										 |  |  | 			err = hist_entry__tui_annotate(he, evsel->idx, hbt); | 
					
						
							| 
									
										
										
										
											2011-10-13 08:01:33 -03:00
										 |  |  | 			he->used = false; | 
					
						
							| 
									
										
										
										
											2012-03-12 16:13:30 +01:00
										 |  |  | 			/*
 | 
					
						
							|  |  |  | 			 * offer option to annotate the other branch source or target | 
					
						
							|  |  |  | 			 * (if they exists) when returning from annotate | 
					
						
							|  |  |  | 			 */ | 
					
						
							|  |  |  | 			if ((err == 'q' || err == CTRL('c')) | 
					
						
							|  |  |  | 			    && annotate_t != -2 && annotate_f != -2) | 
					
						
							|  |  |  | 				goto retry_popup_menu; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-10-11 16:15:39 -03:00
										 |  |  | 			ui_browser__update_nr_entries(&browser->b, browser->hists->nr_entries); | 
					
						
							| 
									
										
										
										
											2011-10-26 12:04:37 -02:00
										 |  |  | 			if (err) | 
					
						
							|  |  |  | 				ui_browser__handle_resize(&browser->b); | 
					
						
							| 
									
										
										
										
											2012-03-12 16:13:30 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 		} else if (choice == browse_map) | 
					
						
							|  |  |  | 			map__browse(browser->selection->map); | 
					
						
							|  |  |  | 		else if (choice == zoom_dso) { | 
					
						
							|  |  |  | zoom_dso: | 
					
						
							| 
									
										
										
										
											2011-10-18 19:07:34 -02:00
										 |  |  | 			if (browser->hists->dso_filter) { | 
					
						
							|  |  |  | 				pstack__remove(fstack, &browser->hists->dso_filter); | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | zoom_out_dso: | 
					
						
							|  |  |  | 				ui_helpline__pop(); | 
					
						
							| 
									
										
										
										
											2011-10-18 19:07:34 -02:00
										 |  |  | 				browser->hists->dso_filter = NULL; | 
					
						
							| 
									
										
										
										
											2011-10-20 08:02:30 -02:00
										 |  |  | 				sort_dso.elide = false; | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 			} else { | 
					
						
							|  |  |  | 				if (dso == NULL) | 
					
						
							|  |  |  | 					continue; | 
					
						
							|  |  |  | 				ui_helpline__fpush("To zoom out press <- or -> + \"Zoom out of %s DSO\"", | 
					
						
							|  |  |  | 						   dso->kernel ? "the Kernel" : dso->short_name); | 
					
						
							| 
									
										
										
										
											2011-10-18 19:07:34 -02:00
										 |  |  | 				browser->hists->dso_filter = dso; | 
					
						
							| 
									
										
										
										
											2011-10-20 08:02:30 -02:00
										 |  |  | 				sort_dso.elide = true; | 
					
						
							| 
									
										
										
										
											2011-10-18 19:07:34 -02:00
										 |  |  | 				pstack__push(fstack, &browser->hists->dso_filter); | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 			hists__filter_by_dso(hists); | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 			hist_browser__reset(browser); | 
					
						
							|  |  |  | 		} else if (choice == zoom_thread) { | 
					
						
							|  |  |  | zoom_thread: | 
					
						
							| 
									
										
										
										
											2011-10-18 19:07:34 -02:00
										 |  |  | 			if (browser->hists->thread_filter) { | 
					
						
							|  |  |  | 				pstack__remove(fstack, &browser->hists->thread_filter); | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | zoom_out_thread: | 
					
						
							|  |  |  | 				ui_helpline__pop(); | 
					
						
							| 
									
										
										
										
											2011-10-18 19:07:34 -02:00
										 |  |  | 				browser->hists->thread_filter = NULL; | 
					
						
							| 
									
										
										
										
											2011-10-20 08:02:30 -02:00
										 |  |  | 				sort_thread.elide = false; | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 			} else { | 
					
						
							|  |  |  | 				ui_helpline__fpush("To zoom out press <- or -> + \"Zoom out of %s(%d) thread\"", | 
					
						
							|  |  |  | 						   thread->comm_set ? thread->comm : "", | 
					
						
							|  |  |  | 						   thread->pid); | 
					
						
							| 
									
										
										
										
											2011-10-18 19:07:34 -02:00
										 |  |  | 				browser->hists->thread_filter = thread; | 
					
						
							| 
									
										
										
										
											2011-10-20 08:02:30 -02:00
										 |  |  | 				sort_thread.elide = true; | 
					
						
							| 
									
										
										
										
											2011-10-18 19:07:34 -02:00
										 |  |  | 				pstack__push(fstack, &browser->hists->thread_filter); | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2012-05-29 22:42:18 -03:00
										 |  |  | 			hists__filter_by_thread(hists); | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 			hist_browser__reset(browser); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2012-10-30 11:56:06 +08:00
										 |  |  | 		/* perf scripts support */ | 
					
						
							|  |  |  | 		else if (choice == scripts_all || choice == scripts_comm || | 
					
						
							|  |  |  | 				choice == scripts_symbol) { | 
					
						
							|  |  |  | do_scripts: | 
					
						
							|  |  |  | 			memset(script_opt, 0, 64); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if (choice == scripts_comm) | 
					
						
							|  |  |  | 				sprintf(script_opt, " -c %s ", browser->he_selection->thread->comm); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if (choice == scripts_symbol) | 
					
						
							|  |  |  | 				sprintf(script_opt, " -S %s ", browser->he_selection->ms.sym->name); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			script_browse(script_opt); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2013-02-03 14:38:20 +08:00
										 |  |  | 		/* Switch to another data file */ | 
					
						
							|  |  |  | 		else if (choice == switch_data) { | 
					
						
							|  |  |  | do_data_switch: | 
					
						
							|  |  |  | 			if (!switch_data_file()) { | 
					
						
							|  |  |  | 				key = K_SWITCH_INPUT_DATA; | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 			} else | 
					
						
							|  |  |  | 				ui__warning("Won't switch the data files due to\n" | 
					
						
							|  |  |  | 					"no valid data file get selected!\n"); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 	} | 
					
						
							|  |  |  | out_free_stack: | 
					
						
							|  |  |  | 	pstack__delete(fstack); | 
					
						
							|  |  |  | out: | 
					
						
							|  |  |  | 	hist_browser__delete(browser); | 
					
						
							| 
									
										
										
										
											2012-03-12 16:13:30 +01:00
										 |  |  | 	free_popup_options(options, nr_options - 1); | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 	return key; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-03-06 13:07:30 -03:00
										 |  |  | struct perf_evsel_menu { | 
					
						
							|  |  |  | 	struct ui_browser b; | 
					
						
							|  |  |  | 	struct perf_evsel *selection; | 
					
						
							| 
									
										
										
										
											2011-10-29 12:15:04 -02:00
										 |  |  | 	bool lost_events, lost_events_warned; | 
					
						
							| 
									
										
										
										
											2012-11-02 14:50:06 +09:00
										 |  |  | 	struct perf_session_env *env; | 
					
						
							| 
									
										
										
										
											2011-03-06 13:07:30 -03:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void perf_evsel_menu__write(struct ui_browser *browser, | 
					
						
							|  |  |  | 				   void *entry, int row) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct perf_evsel_menu *menu = container_of(browser, | 
					
						
							|  |  |  | 						    struct perf_evsel_menu, b); | 
					
						
							|  |  |  | 	struct perf_evsel *evsel = list_entry(entry, struct perf_evsel, node); | 
					
						
							|  |  |  | 	bool current_entry = ui_browser__is_current_entry(browser, row); | 
					
						
							|  |  |  | 	unsigned long nr_events = evsel->hists.stats.nr_events[PERF_RECORD_SAMPLE]; | 
					
						
							| 
									
										
										
										
											2012-06-12 12:34:58 -03:00
										 |  |  | 	const char *ev_name = perf_evsel__name(evsel); | 
					
						
							| 
									
										
										
										
											2011-03-06 13:07:30 -03:00
										 |  |  | 	char bf[256], unit; | 
					
						
							| 
									
										
										
										
											2011-10-29 12:15:04 -02:00
										 |  |  | 	const char *warn = " "; | 
					
						
							|  |  |  | 	size_t printed; | 
					
						
							| 
									
										
										
										
											2011-03-06 13:07:30 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	ui_browser__set_color(browser, current_entry ? HE_COLORSET_SELECTED : | 
					
						
							|  |  |  | 						       HE_COLORSET_NORMAL); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-22 18:09:44 +09:00
										 |  |  | 	if (symbol_conf.event_group && evsel->nr_members > 1) { | 
					
						
							|  |  |  | 		struct perf_evsel *pos; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		ev_name = perf_evsel__group_name(evsel); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		for_each_group_member(pos, evsel) { | 
					
						
							|  |  |  | 			nr_events += pos->hists.stats.nr_events[PERF_RECORD_SAMPLE]; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-03-06 13:07:30 -03:00
										 |  |  | 	nr_events = convert_unit(nr_events, &unit); | 
					
						
							| 
									
										
										
										
											2012-03-14 12:29:29 -03:00
										 |  |  | 	printed = scnprintf(bf, sizeof(bf), "%lu%c%s%s", nr_events, | 
					
						
							| 
									
										
										
										
											2011-10-29 12:15:04 -02:00
										 |  |  | 			   unit, unit == ' ' ? "" : " ", ev_name); | 
					
						
							|  |  |  | 	slsmg_printf("%s", bf); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	nr_events = evsel->hists.stats.nr_events[PERF_RECORD_LOST]; | 
					
						
							|  |  |  | 	if (nr_events != 0) { | 
					
						
							|  |  |  | 		menu->lost_events = true; | 
					
						
							|  |  |  | 		if (!current_entry) | 
					
						
							|  |  |  | 			ui_browser__set_color(browser, HE_COLORSET_TOP); | 
					
						
							|  |  |  | 		nr_events = convert_unit(nr_events, &unit); | 
					
						
							| 
									
										
										
										
											2012-03-14 12:29:29 -03:00
										 |  |  | 		printed += scnprintf(bf, sizeof(bf), ": %ld%c%schunks LOST!", | 
					
						
							|  |  |  | 				     nr_events, unit, unit == ' ' ? "" : " "); | 
					
						
							| 
									
										
										
										
											2011-10-29 12:15:04 -02:00
										 |  |  | 		warn = bf; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	slsmg_write_nstring(warn, browser->width - printed); | 
					
						
							| 
									
										
										
										
											2011-03-06 13:07:30 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (current_entry) | 
					
						
							|  |  |  | 		menu->selection = evsel; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-10-05 19:35:54 -03:00
										 |  |  | static int perf_evsel_menu__run(struct perf_evsel_menu *menu, | 
					
						
							|  |  |  | 				int nr_events, const char *help, | 
					
						
							| 
									
										
										
										
											2012-11-02 14:50:05 +09:00
										 |  |  | 				struct hist_browser_timer *hbt) | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-03-06 13:07:30 -03:00
										 |  |  | 	struct perf_evlist *evlist = menu->b.priv; | 
					
						
							| 
									
										
										
										
											2011-03-05 21:40:06 -03:00
										 |  |  | 	struct perf_evsel *pos; | 
					
						
							| 
									
										
										
										
											2011-03-06 13:07:30 -03:00
										 |  |  | 	const char *ev_name, *title = "Available samples"; | 
					
						
							| 
									
										
										
										
											2012-11-02 14:50:05 +09:00
										 |  |  | 	int delay_secs = hbt ? hbt->refresh : 0; | 
					
						
							| 
									
										
										
										
											2011-03-06 13:07:30 -03:00
										 |  |  | 	int key; | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-03-06 13:07:30 -03:00
										 |  |  | 	if (ui_browser__show(&menu->b, title, | 
					
						
							|  |  |  | 			     "ESC: exit, ENTER|->: Browse histograms") < 0) | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	while (1) { | 
					
						
							| 
									
										
										
										
											2011-10-13 08:52:46 -03:00
										 |  |  | 		key = ui_browser__run(&menu->b, delay_secs); | 
					
						
							| 
									
										
										
										
											2011-03-06 13:07:30 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		switch (key) { | 
					
						
							| 
									
										
										
										
											2011-10-20 16:59:15 -02:00
										 |  |  | 		case K_TIMER: | 
					
						
							| 
									
										
										
										
											2012-11-02 14:50:05 +09:00
										 |  |  | 			hbt->timer(hbt->arg); | 
					
						
							| 
									
										
										
										
											2011-10-29 12:15:04 -02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			if (!menu->lost_events_warned && menu->lost_events) { | 
					
						
							|  |  |  | 				ui_browser__warn_lost_events(&menu->b); | 
					
						
							|  |  |  | 				menu->lost_events_warned = true; | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2011-10-05 19:11:32 -03:00
										 |  |  | 			continue; | 
					
						
							| 
									
										
										
										
											2011-10-20 16:59:15 -02:00
										 |  |  | 		case K_RIGHT: | 
					
						
							|  |  |  | 		case K_ENTER: | 
					
						
							| 
									
										
										
										
											2011-03-06 13:07:30 -03:00
										 |  |  | 			if (!menu->selection) | 
					
						
							|  |  |  | 				continue; | 
					
						
							|  |  |  | 			pos = menu->selection; | 
					
						
							|  |  |  | browse_hists: | 
					
						
							| 
									
										
										
										
											2011-10-13 12:22:28 -03:00
										 |  |  | 			perf_evlist__set_selected(evlist, pos); | 
					
						
							|  |  |  | 			/*
 | 
					
						
							|  |  |  | 			 * Give the calling tool a chance to populate the non | 
					
						
							|  |  |  | 			 * default evsel resorted hists tree. | 
					
						
							|  |  |  | 			 */ | 
					
						
							| 
									
										
										
										
											2012-11-02 14:50:05 +09:00
										 |  |  | 			if (hbt) | 
					
						
							|  |  |  | 				hbt->timer(hbt->arg); | 
					
						
							| 
									
										
										
										
											2012-06-12 12:34:58 -03:00
										 |  |  | 			ev_name = perf_evsel__name(pos); | 
					
						
							| 
									
										
										
										
											2011-10-05 19:35:54 -03:00
										 |  |  | 			key = perf_evsel__hists_browse(pos, nr_events, help, | 
					
						
							| 
									
										
										
										
											2012-11-02 14:50:06 +09:00
										 |  |  | 						       ev_name, true, hbt, | 
					
						
							|  |  |  | 						       menu->env); | 
					
						
							| 
									
										
										
										
											2011-03-06 13:07:30 -03:00
										 |  |  | 			ui_browser__show_title(&menu->b, title); | 
					
						
							| 
									
										
										
										
											2011-10-13 12:22:28 -03:00
										 |  |  | 			switch (key) { | 
					
						
							| 
									
										
										
										
											2011-10-20 16:59:15 -02:00
										 |  |  | 			case K_TAB: | 
					
						
							| 
									
										
										
										
											2011-10-13 12:22:28 -03:00
										 |  |  | 				if (pos->node.next == &evlist->entries) | 
					
						
							|  |  |  | 					pos = list_entry(evlist->entries.next, struct perf_evsel, node); | 
					
						
							|  |  |  | 				else | 
					
						
							|  |  |  | 					pos = list_entry(pos->node.next, struct perf_evsel, node); | 
					
						
							|  |  |  | 				goto browse_hists; | 
					
						
							| 
									
										
										
										
											2011-10-20 16:59:15 -02:00
										 |  |  | 			case K_UNTAB: | 
					
						
							| 
									
										
										
										
											2011-10-13 12:22:28 -03:00
										 |  |  | 				if (pos->node.prev == &evlist->entries) | 
					
						
							|  |  |  | 					pos = list_entry(evlist->entries.prev, struct perf_evsel, node); | 
					
						
							|  |  |  | 				else | 
					
						
							|  |  |  | 					pos = list_entry(pos->node.prev, struct perf_evsel, node); | 
					
						
							|  |  |  | 				goto browse_hists; | 
					
						
							| 
									
										
										
										
											2011-10-20 16:59:15 -02:00
										 |  |  | 			case K_ESC: | 
					
						
							| 
									
										
										
										
											2011-10-26 12:04:37 -02:00
										 |  |  | 				if (!ui_browser__dialog_yesno(&menu->b, | 
					
						
							|  |  |  | 						"Do you really want to exit?")) | 
					
						
							| 
									
										
										
										
											2011-10-13 12:22:28 -03:00
										 |  |  | 					continue; | 
					
						
							|  |  |  | 				/* Fall thru */ | 
					
						
							| 
									
										
										
										
											2013-02-03 14:38:20 +08:00
										 |  |  | 			case K_SWITCH_INPUT_DATA: | 
					
						
							| 
									
										
										
										
											2011-10-13 12:22:28 -03:00
										 |  |  | 			case 'q': | 
					
						
							|  |  |  | 			case CTRL('c'): | 
					
						
							|  |  |  | 				goto out; | 
					
						
							|  |  |  | 			default: | 
					
						
							|  |  |  | 				continue; | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2011-10-20 16:59:15 -02:00
										 |  |  | 		case K_LEFT: | 
					
						
							| 
									
										
										
										
											2011-03-06 13:07:30 -03:00
										 |  |  | 			continue; | 
					
						
							| 
									
										
										
										
											2011-10-20 16:59:15 -02:00
										 |  |  | 		case K_ESC: | 
					
						
							| 
									
										
										
										
											2011-10-26 12:04:37 -02:00
										 |  |  | 			if (!ui_browser__dialog_yesno(&menu->b, | 
					
						
							|  |  |  | 					       "Do you really want to exit?")) | 
					
						
							| 
									
										
										
										
											2011-10-13 08:31:22 -03:00
										 |  |  | 				continue; | 
					
						
							|  |  |  | 			/* Fall thru */ | 
					
						
							| 
									
										
										
										
											2011-03-06 13:07:30 -03:00
										 |  |  | 		case 'q': | 
					
						
							|  |  |  | 		case CTRL('c'): | 
					
						
							|  |  |  | 			goto out; | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 		default: | 
					
						
							| 
									
										
										
										
											2011-10-13 12:22:28 -03:00
										 |  |  | 			continue; | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-03-06 13:07:30 -03:00
										 |  |  | out: | 
					
						
							|  |  |  | 	ui_browser__hide(&menu->b); | 
					
						
							|  |  |  | 	return key; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-22 18:09:43 +09:00
										 |  |  | static bool filter_group_entries(struct ui_browser *self __maybe_unused, | 
					
						
							|  |  |  | 				 void *entry) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct perf_evsel *evsel = list_entry(entry, struct perf_evsel, node); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (symbol_conf.event_group && !perf_evsel__is_group_leader(evsel)) | 
					
						
							|  |  |  | 		return true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-03-06 13:07:30 -03:00
										 |  |  | static int __perf_evlist__tui_browse_hists(struct perf_evlist *evlist, | 
					
						
							| 
									
										
										
										
											2013-01-22 18:09:43 +09:00
										 |  |  | 					   int nr_entries, const char *help, | 
					
						
							| 
									
										
										
										
											2012-11-02 14:50:06 +09:00
										 |  |  | 					   struct hist_browser_timer *hbt, | 
					
						
							|  |  |  | 					   struct perf_session_env *env) | 
					
						
							| 
									
										
										
										
											2011-03-06 13:07:30 -03:00
										 |  |  | { | 
					
						
							|  |  |  | 	struct perf_evsel *pos; | 
					
						
							|  |  |  | 	struct perf_evsel_menu menu = { | 
					
						
							|  |  |  | 		.b = { | 
					
						
							|  |  |  | 			.entries    = &evlist->entries, | 
					
						
							|  |  |  | 			.refresh    = ui_browser__list_head_refresh, | 
					
						
							|  |  |  | 			.seek	    = ui_browser__list_head_seek, | 
					
						
							|  |  |  | 			.write	    = perf_evsel_menu__write, | 
					
						
							| 
									
										
										
										
											2013-01-22 18:09:43 +09:00
										 |  |  | 			.filter	    = filter_group_entries, | 
					
						
							|  |  |  | 			.nr_entries = nr_entries, | 
					
						
							| 
									
										
										
										
											2011-03-06 13:07:30 -03:00
										 |  |  | 			.priv	    = evlist, | 
					
						
							|  |  |  | 		}, | 
					
						
							| 
									
										
										
										
											2012-11-02 14:50:06 +09:00
										 |  |  | 		.env = env, | 
					
						
							| 
									
										
										
										
											2011-03-06 13:07:30 -03:00
										 |  |  | 	}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ui_helpline__push("Press ESC to exit"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	list_for_each_entry(pos, &evlist->entries, node) { | 
					
						
							| 
									
										
										
										
											2012-06-12 12:34:58 -03:00
										 |  |  | 		const char *ev_name = perf_evsel__name(pos); | 
					
						
							| 
									
										
										
										
											2011-03-06 13:07:30 -03:00
										 |  |  | 		size_t line_len = strlen(ev_name) + 7; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (menu.b.width < line_len) | 
					
						
							|  |  |  | 			menu.b.width = line_len; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-22 18:09:43 +09:00
										 |  |  | 	return perf_evsel_menu__run(&menu, nr_entries, help, hbt); | 
					
						
							| 
									
										
										
										
											2011-03-06 13:07:30 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-10-05 19:11:32 -03:00
										 |  |  | int perf_evlist__tui_browse_hists(struct perf_evlist *evlist, const char *help, | 
					
						
							| 
									
										
										
										
											2012-11-02 14:50:06 +09:00
										 |  |  | 				  struct hist_browser_timer *hbt, | 
					
						
							|  |  |  | 				  struct perf_session_env *env) | 
					
						
							| 
									
										
										
										
											2011-03-06 13:07:30 -03:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-01-22 18:09:43 +09:00
										 |  |  | 	int nr_entries = evlist->nr_entries; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | single_entry: | 
					
						
							|  |  |  | 	if (nr_entries == 1) { | 
					
						
							| 
									
										
										
										
											2011-03-06 13:07:30 -03:00
										 |  |  | 		struct perf_evsel *first = list_entry(evlist->entries.next, | 
					
						
							|  |  |  | 						      struct perf_evsel, node); | 
					
						
							| 
									
										
										
										
											2012-06-12 12:34:58 -03:00
										 |  |  | 		const char *ev_name = perf_evsel__name(first); | 
					
						
							| 
									
										
										
										
											2013-01-22 18:09:43 +09:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		return perf_evsel__hists_browse(first, nr_entries, help, | 
					
						
							| 
									
										
										
										
											2012-11-02 14:50:06 +09:00
										 |  |  | 						ev_name, false, hbt, env); | 
					
						
							| 
									
										
										
										
											2011-03-06 13:07:30 -03:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-22 18:09:43 +09:00
										 |  |  | 	if (symbol_conf.event_group) { | 
					
						
							|  |  |  | 		struct perf_evsel *pos; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		nr_entries = 0; | 
					
						
							|  |  |  | 		list_for_each_entry(pos, &evlist->entries, node) | 
					
						
							|  |  |  | 			if (perf_evsel__is_group_leader(pos)) | 
					
						
							|  |  |  | 				nr_entries++; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (nr_entries == 1) | 
					
						
							|  |  |  | 			goto single_entry; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return __perf_evlist__tui_browse_hists(evlist, nr_entries, help, | 
					
						
							|  |  |  | 					       hbt, env); | 
					
						
							| 
									
										
										
										
											2010-08-10 15:49:07 -03:00
										 |  |  | } |