hists.c 64.4 KB
Newer Older
1 2 3 4 5
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <linux/rbtree.h>

6 7 8 9 10 11
#include "../../util/evsel.h"
#include "../../util/evlist.h"
#include "../../util/hist.h"
#include "../../util/pstack.h"
#include "../../util/sort.h"
#include "../../util/util.h"
12
#include "../../util/top.h"
13
#include "../../arch/common.h"
14 15 16 17

#include "../browser.h"
#include "../helpline.h"
#include "../util.h"
18
#include "../ui.h"
19
#include "map.h"
20
#include "annotate.h"
21 22 23 24 25 26

struct hist_browser {
	struct ui_browser   b;
	struct hists	    *hists;
	struct hist_entry   *he_selection;
	struct map_symbol   *selection;
27
	struct hist_browser_timer *hbt;
28
	struct pstack	    *pstack;
29
	struct perf_env *env;
30
	int		     print_seq;
31
	bool		     show_dso;
32
	bool		     show_headers;
33
	float		     min_pcnt;
34
	u64		     nr_non_filtered_entries;
35
	u64		     nr_callchain_rows;
36 37
};

38 39
extern void hist_browser__init_hpp(void);

40 41 42
static int hists__browser_title(struct hists *hists,
				struct hist_browser_timer *hbt,
				char *bf, size_t size);
43
static void hist_browser__update_nr_entries(struct hist_browser *hb);
44

45 46 47
static struct rb_node *hists__filter_entries(struct rb_node *nd,
					     float min_pcnt);

48 49
static bool hist_browser__has_filter(struct hist_browser *hb)
{
50
	return hists__has_filter(hb->hists) || hb->min_pcnt || symbol_conf.has_filter;
51 52
}

53 54 55 56 57 58 59 60 61 62 63 64
static int hist_browser__get_folding(struct hist_browser *browser)
{
	struct rb_node *nd;
	struct hists *hists = browser->hists;
	int unfolded_rows = 0;

	for (nd = rb_first(&hists->entries);
	     (nd = hists__filter_entries(nd, browser->min_pcnt)) != NULL;
	     nd = rb_next(nd)) {
		struct hist_entry *he =
			rb_entry(nd, struct hist_entry, rb_node);

65
		if (he->unfolded)
66 67 68 69 70
			unfolded_rows += he->nr_rows;
	}
	return unfolded_rows;
}

71 72 73 74 75 76 77 78 79
static u32 hist_browser__nr_entries(struct hist_browser *hb)
{
	u32 nr_entries;

	if (hist_browser__has_filter(hb))
		nr_entries = hb->nr_non_filtered_entries;
	else
		nr_entries = hb->hists->nr_entries;

80
	hb->nr_callchain_rows = hist_browser__get_folding(hb);
81 82 83
	return nr_entries + hb->nr_callchain_rows;
}

84 85 86 87 88 89 90 91 92 93 94 95 96 97 98
static void hist_browser__update_rows(struct hist_browser *hb)
{
	struct ui_browser *browser = &hb->b;
	u16 header_offset = hb->show_headers ? 1 : 0, index_row;

	browser->rows = browser->height - header_offset;
	/*
	 * Verify if we were at the last line and that line isn't
	 * visibe because we now show the header line(s).
	 */
	index_row = browser->index - browser->top_idx;
	if (index_row >= browser->rows)
		browser->index -= index_row - browser->rows + 1;
}

99
static void hist_browser__refresh_dimensions(struct ui_browser *browser)
100
{
101 102
	struct hist_browser *hb = container_of(browser, struct hist_browser, b);

103
	/* 3 == +/- toggle symbol before actual hist_entry rendering */
104 105 106 107 108 109 110 111
	browser->width = 3 + (hists__sort_list_width(hb->hists) + sizeof("[k]"));
	/*
 	 * FIXME: Just keeping existing behaviour, but this really should be
 	 *	  before updating browser->width, as it will invalidate the
 	 *	  calculation above. Fix this and the fallout in another
 	 *	  changeset.
 	 */
	ui_browser__refresh_dimensions(browser);
112
	hist_browser__update_rows(hb);
113 114
}

115 116
static void hist_browser__gotorc(struct hist_browser *browser, int row, int column)
{
117 118 119
	u16 header_offset = browser->show_headers ? 1 : 0;

	ui_browser__gotorc(&browser->b, row + header_offset, column);
120 121
}

122
static void hist_browser__reset(struct hist_browser *browser)
123
{
124 125 126 127 128 129
	/*
	 * The hists__remove_entry_filter() already folds non-filtered
	 * entries so we can assume it has 0 callchain rows.
	 */
	browser->nr_callchain_rows = 0;

130
	hist_browser__update_nr_entries(browser);
131
	browser->b.nr_entries = hist_browser__nr_entries(browser);
132
	hist_browser__refresh_dimensions(&browser->b);
133
	ui_browser__reset_index(&browser->b);
134 135 136 137 138 139 140
}

static char tree__folded_sign(bool unfolded)
{
	return unfolded ? '-' : '+';
}

141
static char hist_entry__folded(const struct hist_entry *he)
142
{
143
	return he->has_children ? tree__folded_sign(he->unfolded) : ' ';
144 145
}

146
static char callchain_list__folded(const struct callchain_list *cl)
147
{
148
	return cl->has_children ? tree__folded_sign(cl->unfolded) : ' ';
149 150
}

151
static void callchain_list__set_folding(struct callchain_list *cl, bool unfold)
152
{
153
	cl->unfolded = unfold ? cl->has_children : false;
154 155
}

156
static int callchain_node__count_rows_rb_tree(struct callchain_node *node)
157 158 159 160
{
	int n = 0;
	struct rb_node *nd;

161
	for (nd = rb_first(&node->rb_root); nd; nd = rb_next(nd)) {
162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180
		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;
}

181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209
static int callchain_node__count_flat_rows(struct callchain_node *node)
{
	struct callchain_list *chain;
	char folded_sign = 0;
	int n = 0;

	list_for_each_entry(chain, &node->parent_val, list) {
		if (!folded_sign) {
			/* only check first chain list entry */
			folded_sign = callchain_list__folded(chain);
			if (folded_sign == '+')
				return 1;
		}
		n++;
	}

	list_for_each_entry(chain, &node->val, list) {
		if (!folded_sign) {
			/* node->parent_val list might be empty */
			folded_sign = callchain_list__folded(chain);
			if (folded_sign == '+')
				return 1;
		}
		n++;
	}

	return n;
}

210 211 212 213 214
static int callchain_node__count_folded_rows(struct callchain_node *node __maybe_unused)
{
	return 1;
}

215 216 217 218 219 220
static int callchain_node__count_rows(struct callchain_node *node)
{
	struct callchain_list *chain;
	bool unfolded = false;
	int n = 0;

221 222
	if (callchain_param.mode == CHAIN_FLAT)
		return callchain_node__count_flat_rows(node);
223 224
	else if (callchain_param.mode == CHAIN_FOLDED)
		return callchain_node__count_folded_rows(node);
225

226 227
	list_for_each_entry(chain, &node->val, list) {
		++n;
228
		unfolded = chain->unfolded;
229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249
	}

	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;
}

250
static bool hist_entry__toggle_fold(struct hist_entry *he)
251
{
252
	if (!he)
253 254
		return false;

255
	if (!he->has_children)
256 257
		return false;

258 259 260 261 262 263 264 265 266 267 268 269 270
	he->unfolded = !he->unfolded;
	return true;
}

static bool callchain_list__toggle_fold(struct callchain_list *cl)
{
	if (!cl)
		return false;

	if (!cl->has_children)
		return false;

	cl->unfolded = !cl->unfolded;
271 272 273
	return true;
}

274
static void callchain_node__init_have_children_rb_tree(struct callchain_node *node)
275
{
276
	struct rb_node *nd = rb_first(&node->rb_root);
277

278
	for (nd = rb_first(&node->rb_root); nd; nd = rb_next(nd)) {
279 280
		struct callchain_node *child = rb_entry(nd, struct callchain_node, rb_node);
		struct callchain_list *chain;
281
		bool first = true;
282 283 284 285

		list_for_each_entry(chain, &child->val, list) {
			if (first) {
				first = false;
286
				chain->has_children = chain->list.next != &child->val ||
287
							 !RB_EMPTY_ROOT(&child->rb_root);
288
			} else
289
				chain->has_children = chain->list.next == &child->val &&
290
							 !RB_EMPTY_ROOT(&child->rb_root);
291 292 293 294 295 296
		}

		callchain_node__init_have_children_rb_tree(child);
	}
}

297 298
static void callchain_node__init_have_children(struct callchain_node *node,
					       bool has_sibling)
299 300 301
{
	struct callchain_list *chain;

302
	chain = list_entry(node->val.next, struct callchain_list, list);
303
	chain->has_children = has_sibling;
304

305
	if (node->val.next != node->val.prev) {
306
		chain = list_entry(node->val.prev, struct callchain_list, list);
307
		chain->has_children = !RB_EMPTY_ROOT(&node->rb_root);
308
	}
309

310
	callchain_node__init_have_children_rb_tree(node);
311 312
}

313
static void callchain__init_have_children(struct rb_root *root)
314
{
315 316
	struct rb_node *nd = rb_first(root);
	bool has_sibling = nd && rb_next(nd);
317

318
	for (nd = rb_first(root); nd; nd = rb_next(nd)) {
319
		struct callchain_node *node = rb_entry(nd, struct callchain_node, rb_node);
320
		callchain_node__init_have_children(node, has_sibling);
321 322
		if (callchain_param.mode == CHAIN_FLAT ||
		    callchain_param.mode == CHAIN_FOLDED)
323
			callchain_node__make_parent_list(node);
324 325 326
	}
}

327
static void hist_entry__init_have_children(struct hist_entry *he)
328
{
329
	if (!he->init_have_children) {
330
		he->has_children = !RB_EMPTY_ROOT(&he->sorted_chain);
331 332
		callchain__init_have_children(&he->sorted_chain);
		he->init_have_children = true;
333 334 335
	}
}

336
static bool hist_browser__toggle_fold(struct hist_browser *browser)
337
{
338 339 340 341 342
	struct hist_entry *he = browser->he_selection;
	struct map_symbol *ms = browser->selection;
	struct callchain_list *cl = container_of(ms, struct callchain_list, ms);
	bool has_children;

343 344 345
	if (!he || !ms)
		return false;

346 347 348 349
	if (ms == &he->ms)
		has_children = hist_entry__toggle_fold(he);
	else
		has_children = callchain_list__toggle_fold(cl);
350

351
	if (has_children) {
352
		hist_entry__init_have_children(he);
353 354
		browser->b.nr_entries -= he->nr_rows;
		browser->nr_callchain_rows -= he->nr_rows;
355

356
		if (he->unfolded)
357 358 359
			he->nr_rows = callchain__count_rows(&he->sorted_chain);
		else
			he->nr_rows = 0;
360 361 362

		browser->b.nr_entries += he->nr_rows;
		browser->nr_callchain_rows += he->nr_rows;
363 364 365 366 367 368 369 370

		return true;
	}

	/* If it doesn't have children, no toggling performed */
	return false;
}

371
static int callchain_node__set_folding_rb_tree(struct callchain_node *node, bool unfold)
372 373 374 375
{
	int n = 0;
	struct rb_node *nd;

376
	for (nd = rb_first(&node->rb_root); nd; nd = rb_next(nd)) {
377 378 379 380 381 382
		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;
383 384
			callchain_list__set_folding(chain, unfold);
			has_children = chain->has_children;
385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401
		}

		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;
402 403
		callchain_list__set_folding(chain, unfold);
		has_children = chain->has_children;
404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 419 420 421 422 423 424
	}

	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;
}

425
static void hist_entry__set_folding(struct hist_entry *he, bool unfold)
426
{
427
	hist_entry__init_have_children(he);
428
	he->unfolded = unfold ? he->has_children : false;
429

430
	if (he->has_children) {
431 432
		int n = callchain__set_folding(&he->sorted_chain, unfold);
		he->nr_rows = unfold ? n : 0;
433
	} else
434
		he->nr_rows = 0;
435 436
}

437 438
static void
__hist_browser__set_folding(struct hist_browser *browser, bool unfold)
439 440
{
	struct rb_node *nd;
441
	struct hists *hists = browser->hists;
442

443
	for (nd = rb_first(&hists->entries);
444
	     (nd = hists__filter_entries(nd, browser->min_pcnt)) != NULL;
445
	     nd = rb_next(nd)) {
446 447
		struct hist_entry *he = rb_entry(nd, struct hist_entry, rb_node);
		hist_entry__set_folding(he, unfold);
448
		browser->nr_callchain_rows += he->nr_rows;
449 450 451
	}
}

452
static void hist_browser__set_folding(struct hist_browser *browser, bool unfold)
453
{
454 455 456 457
	browser->nr_callchain_rows = 0;
	__hist_browser__set_folding(browser, unfold);

	browser->b.nr_entries = hist_browser__nr_entries(browser);
458
	/* Go to the start, we may be way after valid entries after a collapse */
459
	ui_browser__reset_index(&browser->b);
460 461
}

462 463 464 465 466 467 468 469 470
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.");
}

471
static int hist_browser__run(struct hist_browser *browser, const char *help)
472
{
473
	int key;
474
	char title[160];
475
	struct hist_browser_timer *hbt = browser->hbt;
476
	int delay_secs = hbt ? hbt->refresh : 0;
477

478
	browser->b.entries = &browser->hists->entries;
479
	browser->b.nr_entries = hist_browser__nr_entries(browser);
480

481
	hists__browser_title(browser->hists, hbt, title, sizeof(title));
482

483
	if (ui_browser__show(&browser->b, title, "%s", help) < 0)
484 485 486
		return -1;

	while (1) {
487
		key = ui_browser__run(&browser->b, delay_secs);
488

489
		switch (key) {
490 491
		case K_TIMER: {
			u64 nr_entries;
492
			hbt->timer(hbt->arg);
493

494
			if (hist_browser__has_filter(browser))
495
				hist_browser__update_nr_entries(browser);
496

497
			nr_entries = hist_browser__nr_entries(browser);
498
			ui_browser__update_nr_entries(&browser->b, nr_entries);
499

500 501 502 503 504
			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);
505 506
			}

507 508
			hists__browser_title(browser->hists,
					     hbt, title, sizeof(title));
509
			ui_browser__show_title(&browser->b, title);
510
			continue;
511
		}
512
		case 'D': { /* Debug */
513
			static int seq;
514
			struct hist_entry *h = rb_entry(browser->b.top,
515 516
							struct hist_entry, rb_node);
			ui_helpline__pop();
517
			ui_helpline__fpush("%d: nr_ent=(%d,%d), rows=%d, idx=%d, fve: idx=%d, row_off=%d, nrows=%d",
518 519
					   seq++, browser->b.nr_entries,
					   browser->hists->nr_entries,
520
					   browser->b.rows,
521 522
					   browser->b.index,
					   browser->b.top_idx,
523 524
					   h->row_offset, h->nr_rows);
		}
525 526 527
			break;
		case 'C':
			/* Collapse the whole world. */
528
			hist_browser__set_folding(browser, false);
529 530 531
			break;
		case 'E':
			/* Expand the whole world. */
532
			hist_browser__set_folding(browser, true);
533
			break;
534 535 536 537
		case 'H':
			browser->show_headers = !browser->show_headers;
			hist_browser__update_rows(browser);
			break;
538
		case K_ENTER:
539
			if (hist_browser__toggle_fold(browser))
540 541 542
				break;
			/* fall thru */
		default:
543
			goto out;
544 545
		}
	}
546
out:
547
	ui_browser__hide(&browser->b);
548
	return key;
549 550
}

551 552 553 554 555 556 557 558 559 560 561 562 563 564 565 566
struct callchain_print_arg {
	/* for hists browser */
	off_t	row_offset;
	bool	is_current_entry;

	/* for file dump */
	FILE	*fp;
	int	printed;
};

typedef void (*print_callchain_entry_fn)(struct hist_browser *browser,
					 struct callchain_list *chain,
					 const char *str, int offset,
					 unsigned short row,
					 struct callchain_print_arg *arg);

567 568
static void hist_browser__show_callchain_entry(struct hist_browser *browser,
					       struct callchain_list *chain,
569 570 571
					       const char *str, int offset,
					       unsigned short row,
					       struct callchain_print_arg *arg)
572 573
{
	int color, width;
574
	char folded_sign = callchain_list__folded(chain);
575
	bool show_annotated = browser->show_dso && chain->ms.sym && symbol__annotation(chain->ms.sym)->src;
576 577 578 579 580 581

	color = HE_COLORSET_NORMAL;
	width = browser->b.width - (offset + 2);
	if (ui_browser__is_current_entry(&browser->b, row)) {
		browser->selection = &chain->ms;
		color = HE_COLORSET_SELECTED;
582
		arg->is_current_entry = true;
583 584 585 586
	}

	ui_browser__set_color(&browser->b, color);
	hist_browser__gotorc(browser, row, 0);
587
	ui_browser__write_nstring(&browser->b, " ", offset);
588
	ui_browser__printf(&browser->b, "%c", folded_sign);
589
	ui_browser__write_graph(&browser->b, show_annotated ? SLSMG_RARROW_CHAR : ' ');
590
	ui_browser__write_nstring(&browser->b, str, width);
591 592
}

593 594 595 596 597 598 599 600 601 602 603 604 605 606 607 608 609 610 611 612 613 614 615 616 617 618 619
static void hist_browser__fprintf_callchain_entry(struct hist_browser *b __maybe_unused,
						  struct callchain_list *chain,
						  const char *str, int offset,
						  unsigned short row __maybe_unused,
						  struct callchain_print_arg *arg)
{
	char folded_sign = callchain_list__folded(chain);

	arg->printed += fprintf(arg->fp, "%*s%c %s\n", offset, " ",
				folded_sign, str);
}

typedef bool (*check_output_full_fn)(struct hist_browser *browser,
				     unsigned short row);

static bool hist_browser__check_output_full(struct hist_browser *browser,
					    unsigned short row)
{
	return browser->b.rows == row;
}

static bool hist_browser__check_dump_full(struct hist_browser *browser __maybe_unused,
					  unsigned short row __maybe_unused)
{
	return false;
}

620 621
#define LEVEL_OFFSET_STEP 3

622 623 624 625 626 627 628 629 630 631 632 633 634 635 636 637 638 639 640 641 642 643 644 645 646 647 648 649 650 651 652 653 654 655 656 657 658 659
static int hist_browser__show_callchain_list(struct hist_browser *browser,
					     struct callchain_node *node,
					     struct callchain_list *chain,
					     unsigned short row, u64 total,
					     bool need_percent, int offset,
					     print_callchain_entry_fn print,
					     struct callchain_print_arg *arg)
{
	char bf[1024], *alloc_str;
	const char *str;

	if (arg->row_offset != 0) {
		arg->row_offset--;
		return 0;
	}

	alloc_str = NULL;
	str = callchain_list__sym_name(chain, bf, sizeof(bf),
				       browser->show_dso);

	if (need_percent) {
		char buf[64];

		callchain_node__scnprintf_value(node, buf, sizeof(buf),
						total);

		if (asprintf(&alloc_str, "%s %s", buf, str) < 0)
			str = "Not enough memory!";
		else
			str = alloc_str;
	}

	print(browser, chain, str, offset, row, arg);

	free(alloc_str);
	return 1;
}

660 661 662 663 664 665 666 667 668 669 670 671 672 673 674 675 676 677 678 679 680 681 682 683 684 685 686 687 688 689 690 691 692 693 694 695 696 697 698 699 700 701 702 703 704 705 706 707 708 709 710 711 712 713 714 715 716 717 718 719 720 721 722 723 724 725 726 727 728 729 730 731 732 733 734 735 736
static int hist_browser__show_callchain_flat(struct hist_browser *browser,
					     struct rb_root *root,
					     unsigned short row, u64 total,
					     print_callchain_entry_fn print,
					     struct callchain_print_arg *arg,
					     check_output_full_fn is_output_full)
{
	struct rb_node *node;
	int first_row = row, offset = LEVEL_OFFSET_STEP;
	bool need_percent;

	node = rb_first(root);
	need_percent = node && rb_next(node);

	while (node) {
		struct callchain_node *child = rb_entry(node, struct callchain_node, rb_node);
		struct rb_node *next = rb_next(node);
		struct callchain_list *chain;
		char folded_sign = ' ';
		int first = true;
		int extra_offset = 0;

		list_for_each_entry(chain, &child->parent_val, list) {
			bool was_first = first;

			if (first)
				first = false;
			else if (need_percent)
				extra_offset = LEVEL_OFFSET_STEP;

			folded_sign = callchain_list__folded(chain);

			row += hist_browser__show_callchain_list(browser, child,
							chain, row, total,
							was_first && need_percent,
							offset + extra_offset,
							print, arg);

			if (is_output_full(browser, row))
				goto out;

			if (folded_sign == '+')
				goto next;
		}

		list_for_each_entry(chain, &child->val, list) {
			bool was_first = first;

			if (first)
				first = false;
			else if (need_percent)
				extra_offset = LEVEL_OFFSET_STEP;

			folded_sign = callchain_list__folded(chain);

			row += hist_browser__show_callchain_list(browser, child,
							chain, row, total,
							was_first && need_percent,
							offset + extra_offset,
							print, arg);

			if (is_output_full(browser, row))
				goto out;

			if (folded_sign == '+')
				break;
		}

next:
		if (is_output_full(browser, row))
			break;
		node = next;
	}
out:
	return row - first_row;
}

737 738 739 740 741 742 743 744 745 746 747 748 749 750 751 752 753 754 755 756 757 758 759 760 761 762 763 764 765 766 767 768 769 770 771 772 773 774 775 776 777 778 779 780 781 782 783 784 785 786 787 788 789 790 791 792 793 794 795 796 797 798 799 800 801 802 803 804 805 806 807 808 809 810 811 812 813 814 815 816 817 818 819 820 821 822 823 824 825 826 827 828 829 830 831 832 833 834 835 836 837 838 839 840 841 842 843 844 845 846
static char *hist_browser__folded_callchain_str(struct hist_browser *browser,
						struct callchain_list *chain,
						char *value_str, char *old_str)
{
	char bf[1024];
	const char *str;
	char *new;

	str = callchain_list__sym_name(chain, bf, sizeof(bf),
				       browser->show_dso);
	if (old_str) {
		if (asprintf(&new, "%s%s%s", old_str,
			     symbol_conf.field_sep ?: ";", str) < 0)
			new = NULL;
	} else {
		if (value_str) {
			if (asprintf(&new, "%s %s", value_str, str) < 0)
				new = NULL;
		} else {
			if (asprintf(&new, "%s", str) < 0)
				new = NULL;
		}
	}
	return new;
}

static int hist_browser__show_callchain_folded(struct hist_browser *browser,
					       struct rb_root *root,
					       unsigned short row, u64 total,
					       print_callchain_entry_fn print,
					       struct callchain_print_arg *arg,
					       check_output_full_fn is_output_full)
{
	struct rb_node *node;
	int first_row = row, offset = LEVEL_OFFSET_STEP;
	bool need_percent;

	node = rb_first(root);
	need_percent = node && rb_next(node);

	while (node) {
		struct callchain_node *child = rb_entry(node, struct callchain_node, rb_node);
		struct rb_node *next = rb_next(node);
		struct callchain_list *chain, *first_chain = NULL;
		int first = true;
		char *value_str = NULL, *value_str_alloc = NULL;
		char *chain_str = NULL, *chain_str_alloc = NULL;

		if (arg->row_offset != 0) {
			arg->row_offset--;
			goto next;
		}

		if (need_percent) {
			char buf[64];

			callchain_node__scnprintf_value(child, buf, sizeof(buf), total);
			if (asprintf(&value_str, "%s", buf) < 0) {
				value_str = (char *)"<...>";
				goto do_print;
			}
			value_str_alloc = value_str;
		}

		list_for_each_entry(chain, &child->parent_val, list) {
			chain_str = hist_browser__folded_callchain_str(browser,
						chain, value_str, chain_str);
			if (first) {
				first = false;
				first_chain = chain;
			}

			if (chain_str == NULL) {
				chain_str = (char *)"Not enough memory!";
				goto do_print;
			}

			chain_str_alloc = chain_str;
		}

		list_for_each_entry(chain, &child->val, list) {
			chain_str = hist_browser__folded_callchain_str(browser,
						chain, value_str, chain_str);
			if (first) {
				first = false;
				first_chain = chain;
			}

			if (chain_str == NULL) {
				chain_str = (char *)"Not enough memory!";
				goto do_print;
			}

			chain_str_alloc = chain_str;
		}

do_print:
		print(browser, first_chain, chain_str, offset, row++, arg);
		free(value_str_alloc);
		free(chain_str_alloc);

next:
		if (is_output_full(browser, row))
			break;
		node = next;
	}

	return row - first_row;
}

847
static int hist_browser__show_callchain_graph(struct hist_browser *browser,
848
					struct rb_root *root, int level,
849 850 851 852
					unsigned short row, u64 total,
					print_callchain_entry_fn print,
					struct callchain_print_arg *arg,
					check_output_full_fn is_output_full)
853 854
{
	struct rb_node *node;
855
	int first_row = row, offset = level * LEVEL_OFFSET_STEP;
856
	u64 new_total;
857
	bool need_percent;
858

859
	node = rb_first(root);
860
	need_percent = node && rb_next(node);
861

862 863 864 865 866 867 868 869 870 871 872
	while (node) {
		struct callchain_node *child = rb_entry(node, struct callchain_node, rb_node);
		struct rb_node *next = rb_next(node);
		struct callchain_list *chain;
		char folded_sign = ' ';
		int first = true;
		int extra_offset = 0;

		list_for_each_entry(chain, &child->val, list) {
			bool was_first = first;

873
			if (first)
874
				first = false;
875
			else if (need_percent)
876 877 878
				extra_offset = LEVEL_OFFSET_STEP;

			folded_sign = callchain_list__folded(chain);
879

880 881 882 883 884
			row += hist_browser__show_callchain_list(browser, child,
							chain, row, total,
							was_first && need_percent,
							offset + extra_offset,
							print, arg);
885

886
			if (is_output_full(browser, row))
887
				goto out;
888

889 890 891 892 893 894 895
			if (folded_sign == '+')
				break;
		}

		if (folded_sign == '-') {
			const int new_level = level + (extra_offset ? 2 : 1);

896 897 898 899
			if (callchain_param.mode == CHAIN_GRAPH_REL)
				new_total = child->children_hit;
			else
				new_total = total;
900

901
			row += hist_browser__show_callchain_graph(browser, &child->rb_root,
902 903
							    new_level, row, new_total,
							    print, arg, is_output_full);
904
		}
905
		if (is_output_full(browser, row))
906
			break;
907
		node = next;
908
	}
909
out:
910 911 912
	return row - first_row;
}

913 914 915 916 917 918 919 920 921 922 923 924 925 926 927 928 929 930 931 932 933 934 935 936 937 938 939 940 941 942 943 944 945 946 947 948 949
static int hist_browser__show_callchain(struct hist_browser *browser,
					struct hist_entry *entry, int level,
					unsigned short row,
					print_callchain_entry_fn print,
					struct callchain_print_arg *arg,
					check_output_full_fn is_output_full)
{
	u64 total = hists__total_period(entry->hists);
	int printed;

	if (callchain_param.mode == CHAIN_GRAPH_REL) {
		if (symbol_conf.cumulate_callchain)
			total = entry->stat_acc->period;
		else
			total = entry->stat.period;
	}

	if (callchain_param.mode == CHAIN_FLAT) {
		printed = hist_browser__show_callchain_flat(browser,
						&entry->sorted_chain, row, total,
						print, arg, is_output_full);
	} else if (callchain_param.mode == CHAIN_FOLDED) {
		printed = hist_browser__show_callchain_folded(browser,
						&entry->sorted_chain, row, total,
						print, arg, is_output_full);
	} else {
		printed = hist_browser__show_callchain_graph(browser,
						&entry->sorted_chain, level, row, total,
						print, arg, is_output_full);
	}

	if (arg->is_current_entry)
		browser->he_selection = entry;

	return printed;
}

950 951 952 953 954 955
struct hpp_arg {
	struct ui_browser *b;
	char folded_sign;
	bool current_entry;
};

956 957 958
static int __hpp__slsmg_color_printf(struct perf_hpp *hpp, const char *fmt, ...)
{
	struct hpp_arg *arg = hpp->ptr;
959
	int ret, len;
960 961
	va_list args;
	double percent;
962

963
	va_start(args, fmt);
964
	len = va_arg(args, int);
965 966
	percent = va_arg(args, double);
	va_end(args);
967

968
	ui_browser__set_percent_color(arg->b, percent, arg->current_entry);
969

970
	ret = scnprintf(hpp->buf, hpp->size, fmt, len, percent);
971
	ui_browser__printf(arg->b, "%s", hpp->buf);
972

973
	advance_hpp(hpp, ret);
974 975 976
	return ret;
}

977
#define __HPP_COLOR_PERCENT_FN(_type, _field)				\
978 979 980 981 982
static u64 __hpp_get_##_field(struct hist_entry *he)			\
{									\
	return he->stat._field;						\
}									\
									\
983
static int								\
984
hist_browser__hpp_color_##_type(struct perf_hpp_fmt *fmt,		\
985 986
				struct perf_hpp *hpp,			\
				struct hist_entry *he)			\
987
{									\
988 989
	return hpp__fmt(fmt, hpp, he, __hpp_get_##_field, " %*.2f%%",	\
			__hpp__slsmg_color_printf, true);		\
990 991
}

992 993 994 995 996 997 998
#define __HPP_COLOR_ACC_PERCENT_FN(_type, _field)			\
static u64 __hpp_get_acc_##_field(struct hist_entry *he)		\
{									\
	return he->stat_acc->_field;					\
}									\
									\
static int								\
999
hist_browser__hpp_color_##_type(struct perf_hpp_fmt *fmt,		\
1000 1001 1002 1003
				struct perf_hpp *hpp,			\
				struct hist_entry *he)			\
{									\
	if (!symbol_conf.cumulate_callchain) {				\
1004
		struct hpp_arg *arg = hpp->ptr;				\
1005
		int len = fmt->user_len ?: fmt->len;			\
1006
		int ret = scnprintf(hpp->buf, hpp->size,		\
1007
				    "%*s", len, "N/A");			\
1008
		ui_browser__printf(arg->b, "%s", hpp->buf);		\
1009 1010 1011
									\
		return ret;						\
	}								\
1012 1013
	return hpp__fmt(fmt, hpp, he, __hpp_get_acc_##_field,		\
			" %*.2f%%", __hpp__slsmg_color_printf, true);	\
1014 1015
}

1016 1017 1018 1019 1020
__HPP_COLOR_PERCENT_FN(overhead, period)
__HPP_COLOR_PERCENT_FN(overhead_sys, period_sys)
__HPP_COLOR_PERCENT_FN(overhead_us, period_us)
__HPP_COLOR_PERCENT_FN(overhead_guest_sys, period_guest_sys)
__HPP_COLOR_PERCENT_FN(overhead_guest_us, period_guest_us)
1021
__HPP_COLOR_ACC_PERCENT_FN(overhead_acc, period)
1022

1023
#undef __HPP_COLOR_PERCENT_FN
1024
#undef __HPP_COLOR_ACC_PERCENT_FN
1025 1026 1027 1028 1029 1030 1031 1032 1033 1034 1035 1036 1037

void hist_browser__init_hpp(void)
{
	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;
1038 1039
	perf_hpp__format[PERF_HPP__OVERHEAD_ACC].color =
				hist_browser__hpp_color_overhead_acc;
1040 1041
}

1042
static int hist_browser__show_entry(struct hist_browser *browser,
1043 1044 1045 1046
				    struct hist_entry *entry,
				    unsigned short row)
{
	char s[256];
1047
	int printed = 0;
1048
	int width = browser->b.width;
1049
	char folded_sign = ' ';
1050
	bool current_entry = ui_browser__is_current_entry(&browser->b, row);
1051
	off_t row_offset = entry->row_offset;
1052
	bool first = true;
1053
	struct perf_hpp_fmt *fmt;
1054 1055

	if (current_entry) {
1056 1057
		browser->he_selection = entry;
		browser->selection = &entry->ms;
1058 1059 1060
	}

	if (symbol_conf.use_callchain) {
1061
		hist_entry__init_have_children(entry);
1062 1063 1064 1065
		folded_sign = hist_entry__folded(entry);
	}

	if (row_offset == 0) {
1066
		struct hpp_arg arg = {
1067
			.b		= &browser->b,
1068 1069 1070
			.folded_sign	= folded_sign,
			.current_entry	= current_entry,
		};
1071 1072 1073
		struct perf_hpp hpp = {
			.buf		= s,
			.size		= sizeof(s),
1074
			.ptr		= &arg,
1075
		};
1076
		int column = 0;
1077

1078
		hist_browser__gotorc(browser, row, 0);
1079

1080
		perf_hpp__for_each_format(fmt) {
1081 1082
			if (perf_hpp__should_skip(fmt, entry->hists) ||
			    column++ < browser->b.horiz_scroll)
1083 1084
				continue;

1085 1086 1087 1088 1089 1090 1091 1092 1093 1094
			if (current_entry && browser->b.navkeypressed) {
				ui_browser__set_color(&browser->b,
						      HE_COLORSET_SELECTED);
			} else {
				ui_browser__set_color(&browser->b,
						      HE_COLORSET_NORMAL);
			}

			if (first) {
				if (symbol_conf.use_callchain) {
1095
					ui_browser__printf(&browser->b, "%c ", folded_sign);
1096 1097 1098 1099
					width -= 2;
				}
				first = false;
			} else {
1100
				ui_browser__printf(&browser->b, "  ");
1101 1102
				width -= 2;
			}
1103

1104
			if (fmt->color) {
1105
				width -= fmt->color(fmt, &hpp, entry);
1106
			} else {
1107
				width -= fmt->entry(fmt, &hpp, entry);
1108
				ui_browser__printf(&browser->b, "%s", s);
1109
			}
1110 1111
		}

1112 1113 1114 1115
		/* The scroll bar isn't being used */
		if (!browser->b.navkeypressed)
			width += 1;

1116
		ui_browser__write_nstring(&browser->b, "", width);
1117

1118 1119 1120 1121 1122
		++row;
		++printed;
	} else
		--row_offset;

1123
	if (folded_sign == '-' && row != browser->b.rows) {
1124 1125 1126 1127
		struct callchain_print_arg arg = {
			.row_offset = row_offset,
			.is_current_entry = current_entry,
		};
1128

1129
		printed += hist_browser__show_callchain(browser, entry, 1, row,
1130 1131
					hist_browser__show_callchain_entry, &arg,
					hist_browser__check_output_full);
1132 1133 1134 1135 1136
	}

	return printed;
}

1137 1138 1139 1140 1141 1142
static int advance_hpp_check(struct perf_hpp *hpp, int inc)
{
	advance_hpp(hpp, inc);
	return hpp->size <= 0;
}

1143
static int hists_browser__scnprintf_headers(struct hist_browser *browser, char *buf, size_t size)
1144
{
1145
	struct hists *hists = browser->hists;
1146 1147 1148 1149 1150 1151
	struct perf_hpp dummy_hpp = {
		.buf    = buf,
		.size   = size,
	};
	struct perf_hpp_fmt *fmt;
	size_t ret = 0;
1152
	int column = 0;
1153 1154 1155 1156 1157 1158 1159 1160

	if (symbol_conf.use_callchain) {
		ret = scnprintf(buf, size, "  ");
		if (advance_hpp_check(&dummy_hpp, ret))
			return ret;
	}

	perf_hpp__for_each_format(fmt) {
1161
		if (perf_hpp__should_skip(fmt, hists)  || column++ < browser->b.horiz_scroll)
1162 1163 1164 1165 1166 1167 1168 1169 1170 1171 1172 1173 1174 1175
			continue;

		ret = fmt->header(fmt, &dummy_hpp, hists_to_evsel(hists));
		if (advance_hpp_check(&dummy_hpp, ret))
			break;

		ret = scnprintf(dummy_hpp.buf, dummy_hpp.size, "  ");
		if (advance_hpp_check(&dummy_hpp, ret))
			break;
	}

	return ret;
}

1176 1177
static void hist_browser__show_headers(struct hist_browser *browser)
{
1178 1179
	char headers[1024];

1180
	hists_browser__scnprintf_headers(browser, headers, sizeof(headers));
1181 1182
	ui_browser__gotorc(&browser->b, 0, 0);
	ui_browser__set_color(&browser->b, HE_COLORSET_ROOT);
1183
	ui_browser__write_nstring(&browser->b, headers, browser->b.width + 1);
1184 1185
}

1186 1187 1188 1189 1190 1191 1192 1193 1194 1195
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);
	}
}

1196
static unsigned int hist_browser__refresh(struct ui_browser *browser)
1197 1198
{
	unsigned row = 0;
1199
	u16 header_offset = 0;
1200
	struct rb_node *nd;
1201
	struct hist_browser *hb = container_of(browser, struct hist_browser, b);
1202

1203 1204 1205 1206 1207
	if (hb->show_headers) {
		hist_browser__show_headers(hb);
		header_offset = 1;
	}

1208
	ui_browser__hists_init_top(browser);
1209 1210
	hb->he_selection = NULL;
	hb->selection = NULL;
1211

1212
	for (nd = browser->top; nd; nd = rb_next(nd)) {
1213
		struct hist_entry *h = rb_entry(nd, struct hist_entry, rb_node);
1214
		float percent;
1215 1216 1217 1218

		if (h->filtered)
			continue;

1219
		percent = hist_entry__get_percent_limit(h);
1220 1221 1222
		if (percent < hb->min_pcnt)
			continue;

1223
		row += hist_browser__show_entry(hb, h, row);
1224
		if (row == browser->rows)
1225 1226 1227
			break;
	}

1228
	return row + header_offset;
1229 1230
}

1231 1232
static struct rb_node *hists__filter_entries(struct rb_node *nd,
					     float min_pcnt)
1233 1234 1235
{
	while (nd != NULL) {
		struct hist_entry *h = rb_entry(nd, struct hist_entry, rb_node);
1236
		float percent = hist_entry__get_percent_limit(h);
1237

1238
		if (!h->filtered && percent >= min_pcnt)
1239 1240 1241 1242 1243 1244 1245 1246
			return nd;

		nd = rb_next(nd);
	}

	return NULL;
}

1247 1248
static struct rb_node *hists__filter_prev_entries(struct rb_node *nd,
						  float min_pcnt)
1249 1250 1251
{
	while (nd != NULL) {
		struct hist_entry *h = rb_entry(nd, struct hist_entry, rb_node);
1252
		float percent = hist_entry__get_percent_limit(h);
1253 1254

		if (!h->filtered && percent >= min_pcnt)
1255 1256 1257 1258 1259 1260 1261 1262
			return nd;

		nd = rb_prev(nd);
	}

	return NULL;
}

1263
static void ui_browser__hists_seek(struct ui_browser *browser,
1264 1265 1266 1267 1268
				   off_t offset, int whence)
{
	struct hist_entry *h;
	struct rb_node *nd;
	bool first = true;
1269 1270 1271
	struct hist_browser *hb;

	hb = container_of(browser, struct hist_browser, b);
1272

1273
	if (browser->nr_entries == 0)
1274 1275
		return;

1276
	ui_browser__hists_init_top(browser);
1277

1278 1279
	switch (whence) {
	case SEEK_SET:
1280
		nd = hists__filter_entries(rb_first(browser->entries),
1281
					   hb->min_pcnt);
1282 1283
		break;
	case SEEK_CUR:
1284
		nd = browser->top;
1285 1286
		goto do_offset;
	case SEEK_END:
1287
		nd = hists__filter_prev_entries(rb_last(browser->entries),
1288
						hb->min_pcnt);
1289 1290 1291 1292 1293 1294 1295 1296 1297 1298
		first = false;
		break;
	default:
		return;
	}

	/*
	 * Moves not relative to the first visible entry invalidates its
	 * row_offset:
	 */
1299
	h = rb_entry(browser->top, struct hist_entry, rb_node);
1300 1301 1302 1303 1304 1305 1306 1307 1308 1309 1310 1311 1312 1313 1314 1315
	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:
1316 1317 1318
	if (!nd)
		return;

1319 1320 1321
	if (offset > 0) {
		do {
			h = rb_entry(nd, struct hist_entry, rb_node);
1322
			if (h->unfolded) {
1323 1324 1325 1326 1327 1328 1329
				u16 remaining = h->nr_rows - h->row_offset;
				if (offset > remaining) {
					offset -= remaining;
					h->row_offset = 0;
				} else {
					h->row_offset += offset;
					offset = 0;
1330
					browser->top = nd;
1331 1332 1333
					break;
				}
			}
1334
			nd = hists__filter_entries(rb_next(nd), hb->min_pcnt);
1335 1336 1337
			if (nd == NULL)
				break;
			--offset;
1338
			browser->top = nd;
1339 1340 1341 1342
		} while (offset != 0);
	} else if (offset < 0) {
		while (1) {
			h = rb_entry(nd, struct hist_entry, rb_node);
1343
			if (h->unfolded) {
1344 1345 1346 1347 1348 1349 1350
				if (first) {
					if (-offset > h->row_offset) {
						offset += h->row_offset;
						h->row_offset = 0;
					} else {
						h->row_offset += offset;
						offset = 0;
1351
						browser->top = nd;
1352 1353 1354 1355 1356 1357 1358 1359 1360
						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;
1361
						browser->top = nd;
1362 1363 1364 1365 1366
						break;
					}
				}
			}

1367
			nd = hists__filter_prev_entries(rb_prev(nd),
1368
							hb->min_pcnt);
1369 1370 1371
			if (nd == NULL)
				break;
			++offset;
1372
			browser->top = nd;
1373 1374 1375 1376 1377 1378 1379
			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);
1380
				if (h->unfolded)
1381 1382 1383 1384 1385 1386
					h->row_offset = h->nr_rows;
				break;
			}
			first = false;
		}
	} else {
1387
		browser->top = nd;
1388 1389 1390 1391 1392
		h = rb_entry(nd, struct hist_entry, rb_node);
		h->row_offset = 0;
	}
}

1393
static int hist_browser__fprintf_callchain(struct hist_browser *browser,
1394
					   struct hist_entry *he, FILE *fp)
1395
{
1396 1397 1398
	struct callchain_print_arg arg  = {
		.fp = fp,
	};
1399

1400
	hist_browser__show_callchain(browser, he, 1, 0,
1401 1402 1403
				     hist_browser__fprintf_callchain_entry, &arg,
				     hist_browser__check_dump_full);
	return arg.printed;
1404 1405 1406 1407 1408 1409 1410 1411
}

static int hist_browser__fprintf_entry(struct hist_browser *browser,
				       struct hist_entry *he, FILE *fp)
{
	char s[8192];
	int printed = 0;
	char folded_sign = ' ';
1412 1413 1414 1415 1416 1417 1418
	struct perf_hpp hpp = {
		.buf = s,
		.size = sizeof(s),
	};
	struct perf_hpp_fmt *fmt;
	bool first = true;
	int ret;
1419 1420 1421 1422 1423 1424 1425

	if (symbol_conf.use_callchain)
		folded_sign = hist_entry__folded(he);

	if (symbol_conf.use_callchain)
		printed += fprintf(fp, "%c ", folded_sign);

1426
	perf_hpp__for_each_format(fmt) {
1427
		if (perf_hpp__should_skip(fmt, he->hists))
1428 1429
			continue;

1430 1431 1432 1433 1434
		if (!first) {
			ret = scnprintf(hpp.buf, hpp.size, "  ");
			advance_hpp(&hpp, ret);
		} else
			first = false;
1435

1436 1437 1438
		ret = fmt->entry(fmt, &hpp, he);
		advance_hpp(&hpp, ret);
	}
1439 1440 1441
	printed += fprintf(fp, "%s\n", rtrim(s));

	if (folded_sign == '-')
1442
		printed += hist_browser__fprintf_callchain(browser, he, fp);
1443 1444 1445 1446 1447 1448

	return printed;
}

static int hist_browser__fprintf(struct hist_browser *browser, FILE *fp)
{
1449 1450
	struct rb_node *nd = hists__filter_entries(rb_first(browser->b.entries),
						   browser->min_pcnt);
1451 1452 1453 1454 1455 1456
	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);
1457
		nd = hists__filter_entries(rb_next(nd), browser->min_pcnt);
1458 1459 1460 1461 1462 1463 1464 1465 1466 1467 1468 1469 1470 1471 1472 1473 1474 1475 1476 1477 1478 1479 1480 1481 1482 1483
	}

	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];
1484 1485
		const char *err = strerror_r(errno, bf, sizeof(bf));
		ui_helpline__fpush("Couldn't write to %s: %s", filename, err);
1486 1487 1488 1489 1490 1491 1492 1493 1494 1495 1496
		return -1;
	}

	++browser->print_seq;
	hist_browser__fprintf(browser, fp);
	fclose(fp);
	ui_helpline__fpush("%s written!", filename);

	return 0;
}

1497
static struct hist_browser *hist_browser__new(struct hists *hists,
1498
					      struct hist_browser_timer *hbt,
1499
					      struct perf_env *env)
1500
{
1501
	struct hist_browser *browser = zalloc(sizeof(*browser));
1502

1503 1504 1505
	if (browser) {
		browser->hists = hists;
		browser->b.refresh = hist_browser__refresh;
1506
		browser->b.refresh_dimensions = hist_browser__refresh_dimensions;
1507 1508
		browser->b.seek = ui_browser__hists_seek;
		browser->b.use_navkeypressed = true;
1509
		browser->show_headers = symbol_conf.show_hist_headers;
1510
		browser->hbt = hbt;
1511
		browser->env = env;
1512 1513
	}

1514
	return browser;
1515 1516
}

1517
static void hist_browser__delete(struct hist_browser *browser)
1518
{
1519
	free(browser);
1520 1521
}

1522
static struct hist_entry *hist_browser__selected_entry(struct hist_browser *browser)
1523
{
1524
	return browser->he_selection;
1525 1526
}

1527
static struct thread *hist_browser__selected_thread(struct hist_browser *browser)
1528
{
1529
	return browser->he_selection->thread;
1530 1531
}

1532 1533 1534 1535 1536 1537 1538 1539 1540
/* Check whether the browser is for 'top' or 'report' */
static inline bool is_report_browser(void *timer)
{
	return timer == NULL;
}

static int hists__browser_title(struct hists *hists,
				struct hist_browser_timer *hbt,
				char *bf, size_t size)
1541
{
1542 1543
	char unit;
	int printed;
1544 1545
	const struct dso *dso = hists->dso_filter;
	const struct thread *thread = hists->thread_filter;
1546
	int socket_id = hists->socket_filter;
1547 1548
	unsigned long nr_samples = hists->stats.nr_events[PERF_RECORD_SAMPLE];
	u64 nr_events = hists->stats.total_period;
1549
	struct perf_evsel *evsel = hists_to_evsel(hists);
1550
	const char *ev_name = perf_evsel__name(evsel);
1551 1552
	char buf[512];
	size_t buflen = sizeof(buf);
1553 1554
	char ref[30] = " show reference callgraph, ";
	bool enable_ref = false;
1555

1556 1557 1558 1559 1560
	if (symbol_conf.filter_relative) {
		nr_samples = hists->stats.nr_non_filtered_samples;
		nr_events = hists->stats.total_non_filtered_period;
	}

1561
	if (perf_evsel__is_group_event(evsel)) {
1562 1563 1564 1565 1566 1567
		struct perf_evsel *pos;

		perf_evsel__group_desc(evsel, buf, buflen);
		ev_name = buf;

		for_each_group_member(pos, evsel) {
1568 1569
			struct hists *pos_hists = evsel__hists(pos);

1570
			if (symbol_conf.filter_relative) {
1571 1572
				nr_samples += pos_hists->stats.nr_non_filtered_samples;
				nr_events += pos_hists->stats.total_non_filtered_period;
1573
			} else {
1574 1575
				nr_samples += pos_hists->stats.nr_events[PERF_RECORD_SAMPLE];
				nr_events += pos_hists->stats.total_period;
1576
			}
1577 1578
		}
	}
1579

1580 1581 1582
	if (symbol_conf.show_ref_callgraph &&
	    strstr(ev_name, "call-graph=no"))
		enable_ref = true;
1583 1584
	nr_samples = convert_unit(nr_samples, &unit);
	printed = scnprintf(bf, size,
1585 1586
			   "Samples: %lu%c of event '%s',%sEvent count (approx.): %" PRIu64,
			   nr_samples, unit, ev_name, enable_ref ? ref : " ", nr_events);
1587

1588

1589
	if (hists->uid_filter_str)
1590
		printed += snprintf(bf + printed, size - printed,
1591
				    ", UID: %s", hists->uid_filter_str);
1592
	if (thread)
1593
		printed += scnprintf(bf + printed, size - printed,
1594
				    ", Thread: %s(%d)",
1595
				     (thread->comm_set ? thread__comm_str(thread) : ""),
1596
				    thread->tid);
1597
	if (dso)
1598
		printed += scnprintf(bf + printed, size - printed,
1599
				    ", DSO: %s", dso->short_name);
1600
	if (socket_id > -1)
1601
		printed += scnprintf(bf + printed, size - printed,
1602
				    ", Processor Socket: %d", socket_id);
1603 1604 1605 1606 1607 1608 1609
	if (!is_report_browser(hbt)) {
		struct perf_top *top = hbt->arg;

		if (top->zero)
			printed += scnprintf(bf + printed, size - printed, " [z]");
	}

1610
	return printed;
1611 1612
}

1613 1614 1615 1616
static inline void free_popup_options(char **options, int n)
{
	int i;

1617 1618
	for (i = 0; i < n; ++i)
		zfree(&options[i]);
1619 1620
}

1621 1622 1623 1624 1625 1626 1627 1628 1629 1630 1631 1632 1633 1634 1635 1636 1637 1638 1639 1640 1641 1642 1643 1644 1645 1646 1647 1648 1649 1650 1651 1652 1653 1654 1655 1656 1657 1658 1659 1660 1661 1662 1663 1664 1665 1666 1667 1668 1669 1670
/*
 * 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]) {
1671
				zfree(&options[nr_options]);
1672 1673 1674 1675 1676 1677 1678 1679 1680 1681 1682 1683 1684 1685 1686 1687 1688 1689 1690 1691 1692 1693 1694 1695 1696 1697 1698 1699 1700 1701 1702 1703 1704 1705 1706 1707 1708 1709
				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;
}

1710 1711 1712
struct popup_action {
	struct thread 		*thread;
	struct map_symbol 	ms;
1713
	int			socket;
1714 1715 1716 1717

	int (*fn)(struct hist_browser *browser, struct popup_action *act);
};

1718
static int
1719
do_annotate(struct hist_browser *browser, struct popup_action *act)
1720 1721 1722 1723 1724 1725
{
	struct perf_evsel *evsel;
	struct annotation *notes;
	struct hist_entry *he;
	int err;

1726
	if (!objdump_path && perf_env__lookup_objdump(browser->env))
1727 1728
		return 0;

1729
	notes = symbol__annotation(act->ms.sym);
1730 1731 1732 1733
	if (!notes->src)
		return 0;

	evsel = hists_to_evsel(browser->hists);
1734
	err = map_symbol__tui_annotate(&act->ms, evsel, browser->hbt);
1735 1736 1737 1738 1739 1740 1741 1742 1743 1744 1745 1746 1747 1748 1749
	he = hist_browser__selected_entry(browser);
	/*
	 * offer option to annotate the other branch source or target
	 * (if they exists) when returning from annotate
	 */
	if ((err == 'q' || err == CTRL('c')) && he->branch_info)
		return 1;

	ui_browser__update_nr_entries(&browser->b, browser->hists->nr_entries);
	if (err)
		ui_browser__handle_resize(&browser->b);
	return 0;
}

static int
1750 1751 1752
add_annotate_opt(struct hist_browser *browser __maybe_unused,
		 struct popup_action *act, char **optstr,
		 struct map *map, struct symbol *sym)
1753
{
1754 1755 1756 1757 1758 1759 1760 1761 1762 1763 1764 1765 1766 1767 1768 1769 1770
	if (sym == NULL || map->dso->annotate_warned)
		return 0;

	if (asprintf(optstr, "Annotate %s", sym->name) < 0)
		return 0;

	act->ms.map = map;
	act->ms.sym = sym;
	act->fn = do_annotate;
	return 1;
}

static int
do_zoom_thread(struct hist_browser *browser, struct popup_action *act)
{
	struct thread *thread = act->thread;

1771 1772 1773 1774 1775 1776
	if (browser->hists->thread_filter) {
		pstack__remove(browser->pstack, &browser->hists->thread_filter);
		perf_hpp__set_elide(HISTC_THREAD, false);
		thread__zput(browser->hists->thread_filter);
		ui_helpline__pop();
	} else {
1777
		ui_helpline__fpush("To zoom out press ESC or ENTER + \"Zoom out of %s(%d) thread\"",
1778 1779 1780 1781 1782 1783 1784 1785 1786 1787 1788 1789 1790
				   thread->comm_set ? thread__comm_str(thread) : "",
				   thread->tid);
		browser->hists->thread_filter = thread__get(thread);
		perf_hpp__set_elide(HISTC_THREAD, false);
		pstack__push(browser->pstack, &browser->hists->thread_filter);
	}

	hists__filter_by_thread(browser->hists);
	hist_browser__reset(browser);
	return 0;
}

static int
1791 1792 1793
add_thread_opt(struct hist_browser *browser, struct popup_action *act,
	       char **optstr, struct thread *thread)
{
1794
	if (!sort__has_thread || thread == NULL)
1795 1796 1797 1798 1799 1800 1801 1802 1803 1804 1805 1806 1807 1808 1809
		return 0;

	if (asprintf(optstr, "Zoom %s %s(%d) thread",
		     browser->hists->thread_filter ? "out of" : "into",
		     thread->comm_set ? thread__comm_str(thread) : "",
		     thread->tid) < 0)
		return 0;

	act->thread = thread;
	act->fn = do_zoom_thread;
	return 1;
}

static int
do_zoom_dso(struct hist_browser *browser, struct popup_action *act)
1810
{
1811
	struct map *map = act->ms.map;
1812

1813 1814 1815 1816 1817 1818
	if (browser->hists->dso_filter) {
		pstack__remove(browser->pstack, &browser->hists->dso_filter);
		perf_hpp__set_elide(HISTC_DSO, false);
		browser->hists->dso_filter = NULL;
		ui_helpline__pop();
	} else {
1819
		if (map == NULL)
1820
			return 0;
1821
		ui_helpline__fpush("To zoom out press ESC or ENTER + \"Zoom out of %s DSO\"",
1822 1823
				   __map__is_kernel(map) ? "the Kernel" : map->dso->short_name);
		browser->hists->dso_filter = map->dso;
1824 1825 1826 1827 1828 1829 1830 1831 1832 1833
		perf_hpp__set_elide(HISTC_DSO, true);
		pstack__push(browser->pstack, &browser->hists->dso_filter);
	}

	hists__filter_by_dso(browser->hists);
	hist_browser__reset(browser);
	return 0;
}

static int
1834
add_dso_opt(struct hist_browser *browser, struct popup_action *act,
1835
	    char **optstr, struct map *map)
1836
{
1837
	if (!sort__has_dso || map == NULL)
1838 1839 1840 1841
		return 0;

	if (asprintf(optstr, "Zoom %s %s DSO",
		     browser->hists->dso_filter ? "out of" : "into",
1842
		     __map__is_kernel(map) ? "the Kernel" : map->dso->short_name) < 0)
1843 1844
		return 0;

1845
	act->ms.map = map;
1846 1847 1848 1849 1850 1851 1852 1853 1854
	act->fn = do_zoom_dso;
	return 1;
}

static int
do_browse_map(struct hist_browser *browser __maybe_unused,
	      struct popup_action *act)
{
	map__browse(act->ms.map);
1855 1856 1857
	return 0;
}

1858 1859 1860 1861
static int
add_map_opt(struct hist_browser *browser __maybe_unused,
	    struct popup_action *act, char **optstr, struct map *map)
{
1862
	if (!sort__has_dso || map == NULL)
1863 1864 1865 1866 1867 1868 1869 1870 1871 1872
		return 0;

	if (asprintf(optstr, "Browse map details") < 0)
		return 0;

	act->ms.map = map;
	act->fn = do_browse_map;
	return 1;
}

1873 1874
static int
do_run_script(struct hist_browser *browser __maybe_unused,
1875
	      struct popup_action *act)
1876 1877 1878 1879
{
	char script_opt[64];
	memset(script_opt, 0, sizeof(script_opt));

1880
	if (act->thread) {
1881
		scnprintf(script_opt, sizeof(script_opt), " -c %s ",
1882 1883
			  thread__comm_str(act->thread));
	} else if (act->ms.sym) {
1884
		scnprintf(script_opt, sizeof(script_opt), " -S %s ",
1885
			  act->ms.sym->name);
1886 1887 1888 1889 1890 1891 1892
	}

	script_browse(script_opt);
	return 0;
}

static int
1893 1894 1895 1896 1897 1898 1899 1900 1901 1902 1903 1904 1905 1906 1907 1908 1909 1910 1911 1912 1913 1914 1915 1916 1917 1918
add_script_opt(struct hist_browser *browser __maybe_unused,
	       struct popup_action *act, char **optstr,
	       struct thread *thread, struct symbol *sym)
{
	if (thread) {
		if (asprintf(optstr, "Run scripts for samples of thread [%s]",
			     thread__comm_str(thread)) < 0)
			return 0;
	} else if (sym) {
		if (asprintf(optstr, "Run scripts for samples of symbol [%s]",
			     sym->name) < 0)
			return 0;
	} else {
		if (asprintf(optstr, "Run scripts for all samples") < 0)
			return 0;
	}

	act->thread = thread;
	act->ms.sym = sym;
	act->fn = do_run_script;
	return 1;
}

static int
do_switch_data(struct hist_browser *browser __maybe_unused,
	       struct popup_action *act __maybe_unused)
1919 1920 1921 1922
{
	if (switch_data_file()) {
		ui__warning("Won't switch the data files due to\n"
			    "no valid data file get selected!\n");
1923
		return 0;
1924 1925 1926 1927 1928
	}

	return K_SWITCH_INPUT_DATA;
}

1929 1930 1931 1932 1933 1934 1935 1936 1937 1938 1939 1940 1941 1942 1943 1944 1945 1946 1947 1948 1949 1950 1951 1952 1953 1954 1955 1956 1957 1958 1959 1960
static int
add_switch_opt(struct hist_browser *browser,
	       struct popup_action *act, char **optstr)
{
	if (!is_report_browser(browser->hbt))
		return 0;

	if (asprintf(optstr, "Switch to another data file in PWD") < 0)
		return 0;

	act->fn = do_switch_data;
	return 1;
}

static int
do_exit_browser(struct hist_browser *browser __maybe_unused,
		struct popup_action *act __maybe_unused)
{
	return 0;
}

static int
add_exit_opt(struct hist_browser *browser __maybe_unused,
	     struct popup_action *act, char **optstr)
{
	if (asprintf(optstr, "Exit") < 0)
		return 0;

	act->fn = do_exit_browser;
	return 1;
}

1961 1962 1963 1964 1965 1966 1967 1968 1969 1970 1971 1972 1973 1974 1975 1976 1977 1978 1979 1980 1981 1982
static int
do_zoom_socket(struct hist_browser *browser, struct popup_action *act)
{
	if (browser->hists->socket_filter > -1) {
		pstack__remove(browser->pstack, &browser->hists->socket_filter);
		browser->hists->socket_filter = -1;
		perf_hpp__set_elide(HISTC_SOCKET, false);
	} else {
		browser->hists->socket_filter = act->socket;
		perf_hpp__set_elide(HISTC_SOCKET, true);
		pstack__push(browser->pstack, &browser->hists->socket_filter);
	}

	hists__filter_by_socket(browser->hists);
	hist_browser__reset(browser);
	return 0;
}

static int
add_socket_opt(struct hist_browser *browser, struct popup_action *act,
	       char **optstr, int socket_id)
{
1983
	if (!sort__has_socket || socket_id < 0)
1984 1985 1986 1987 1988 1989 1990 1991 1992 1993 1994 1995
		return 0;

	if (asprintf(optstr, "Zoom %s Processor Socket %d",
		     (browser->hists->socket_filter > -1) ? "out of" : "into",
		     socket_id) < 0)
		return 0;

	act->socket = socket_id;
	act->fn = do_zoom_socket;
	return 1;
}

1996
static void hist_browser__update_nr_entries(struct hist_browser *hb)
1997 1998 1999 2000
{
	u64 nr_entries = 0;
	struct rb_node *nd = rb_first(&hb->hists->entries);

2001 2002 2003 2004 2005
	if (hb->min_pcnt == 0) {
		hb->nr_non_filtered_entries = hb->hists->nr_non_filtered_entries;
		return;
	}

2006
	while ((nd = hists__filter_entries(nd, hb->min_pcnt)) != NULL) {
2007
		nr_entries++;
2008
		nd = rb_next(nd);
2009 2010
	}

2011
	hb->nr_non_filtered_entries = nr_entries;
2012
}
2013

2014
static int perf_evsel__hists_browse(struct perf_evsel *evsel, int nr_events,
2015
				    const char *helpline,
2016
				    bool left_exits,
2017
				    struct hist_browser_timer *hbt,
2018
				    float min_pcnt,
2019
				    struct perf_env *env)
2020
{
2021
	struct hists *hists = evsel__hists(evsel);
2022
	struct hist_browser *browser = hist_browser__new(hists, hbt, env);
2023
	struct branch_info *bi;
2024 2025
#define MAX_OPTIONS  16
	char *options[MAX_OPTIONS];
2026
	struct popup_action actions[MAX_OPTIONS];
2027
	int nr_options = 0;
2028
	int key = -1;
2029
	char buf[64];
2030
	int delay_secs = hbt ? hbt->refresh : 0;
2031
	struct perf_hpp_fmt *fmt;
2032

2033 2034 2035 2036 2037 2038 2039 2040
#define HIST_BROWSER_HELP_COMMON					\
	"h/?/F1        Show this window\n"				\
	"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"				\
	"For symbolic views (--sort has sym):\n\n"			\
2041 2042
	"ENTER         Zoom into DSO/Threads & Annotate current symbol\n" \
	"ESC           Zoom out\n"					\
2043 2044 2045 2046
	"a             Annotate current symbol\n"			\
	"C             Collapse all callchains\n"			\
	"d             Zoom into current DSO\n"				\
	"E             Expand all callchains\n"				\
2047
	"F             Toggle percentage of filtered entries\n"		\
2048
	"H             Display column headers\n"			\
2049
	"m             Display context menu\n"				\
2050
	"S             Zoom into current Processor Socket\n"		\
2051 2052 2053

	/* help messages are sorted by lexical order of the hotkey */
	const char report_help[] = HIST_BROWSER_HELP_COMMON
2054
	"i             Show header information\n"
2055 2056 2057 2058 2059 2060 2061 2062 2063 2064
	"P             Print histograms to perf.hist.N\n"
	"r             Run available scripts\n"
	"s             Switch to another data file in PWD\n"
	"t             Zoom into current Thread\n"
	"V             Verbose (DSO names in callchains, etc)\n"
	"/             Filter symbol by name";
	const char top_help[] = HIST_BROWSER_HELP_COMMON
	"P             Print histograms to perf.hist.N\n"
	"t             Zoom into current Thread\n"
	"V             Verbose (DSO names in callchains, etc)\n"
2065
	"z             Toggle zeroing of samples\n"
2066
	"f             Enable/Disable events\n"
2067 2068
	"/             Filter symbol by name";

2069 2070 2071
	if (browser == NULL)
		return -1;

2072 2073 2074 2075
	/* reset abort key so that it can get Ctrl-C as a key */
	SLang_reset_tty();
	SLang_init_tty(0, 0, 0);

2076
	if (min_pcnt)
2077
		browser->min_pcnt = min_pcnt;
2078
	hist_browser__update_nr_entries(browser);
2079

2080
	browser->pstack = pstack__new(3);
2081
	if (browser->pstack == NULL)
2082 2083 2084 2085
		goto out;

	ui_helpline__push(helpline);

2086
	memset(options, 0, sizeof(options));
2087
	memset(actions, 0, sizeof(actions));
2088

2089
	perf_hpp__for_each_format(fmt) {
2090
		perf_hpp__reset_width(fmt, hists);
2091 2092 2093 2094 2095 2096 2097 2098 2099
		/*
		 * This is done just once, and activates the horizontal scrolling
		 * code in the ui_browser code, it would be better to have a the
		 * counter in the perf_hpp code, but I couldn't find doing it here
		 * works, FIXME by setting this in hist_browser__new, for now, be
		 * clever 8-)
		 */
		++browser->b.columns;
	}
2100

2101 2102 2103
	if (symbol_conf.col_width_list_str)
		perf_hpp__set_user_width(symbol_conf.col_width_list_str);

2104
	while (1) {
2105
		struct thread *thread = NULL;
2106
		struct map *map = NULL;
2107
		int choice = 0;
2108
		int socked_id = -1;
2109

2110 2111
		nr_options = 0;

2112
		key = hist_browser__run(browser, helpline);
2113

2114 2115
		if (browser->he_selection != NULL) {
			thread = hist_browser__selected_thread(browser);
2116
			map = browser->selection->map;
2117
			socked_id = browser->he_selection->socket;
2118
		}
2119
		switch (key) {
2120 2121
		case K_TAB:
		case K_UNTAB:
2122 2123
			if (nr_events == 1)
				continue;
2124 2125 2126 2127 2128 2129
			/*
			 * Exit the browser, let hists__browser_tree
			 * go to the next or previous
			 */
			goto out_free_stack;
		case 'a':
2130
			if (!sort__has_sym) {
2131
				ui_browser__warning(&browser->b, delay_secs * 2,
2132
			"Annotation is only available for symbolic views, "
2133
			"include \"sym*\" in --sort to use it.");
2134 2135 2136
				continue;
			}

2137
			if (browser->selection == NULL ||
2138
			    browser->selection->sym == NULL ||
2139
			    browser->selection->map->dso->annotate_warned)
2140
				continue;
2141

2142 2143 2144
			actions->ms.map = browser->selection->map;
			actions->ms.sym = browser->selection->sym;
			do_annotate(browser, actions);
2145
			continue;
2146 2147 2148
		case 'P':
			hist_browser__dump(browser);
			continue;
2149
		case 'd':
2150
			actions->ms.map = map;
2151
			do_zoom_dso(browser, actions);
2152
			continue;
2153 2154 2155
		case 'V':
			browser->show_dso = !browser->show_dso;
			continue;
2156
		case 't':
2157 2158
			actions->thread = thread;
			do_zoom_thread(browser, actions);
2159
			continue;
2160 2161 2162 2163
		case 'S':
			actions->socket = socked_id;
			do_zoom_socket(browser, actions);
			continue;
2164
		case '/':
2165
			if (ui_browser__input_window("Symbol to show",
2166 2167
					"Please enter the name of symbol you want to see.\n"
					"To remove the filter later, press / + ENTER.",
2168 2169
					buf, "ENTER: OK, ESC: Cancel",
					delay_secs * 2) == K_ENTER) {
2170 2171
				hists->symbol_filter_str = *buf ? buf : NULL;
				hists__filter_by_symbol(hists);
2172 2173 2174
				hist_browser__reset(browser);
			}
			continue;
2175
		case 'r':
2176 2177 2178 2179 2180
			if (is_report_browser(hbt)) {
				actions->thread = NULL;
				actions->ms.sym = NULL;
				do_run_script(browser, actions);
			}
2181
			continue;
2182
		case 's':
2183
			if (is_report_browser(hbt)) {
2184
				key = do_switch_data(browser, actions);
2185 2186 2187
				if (key == K_SWITCH_INPUT_DATA)
					goto out_free_stack;
			}
2188
			continue;
2189 2190 2191 2192 2193
		case 'i':
			/* env->arch is NULL for live-mode (i.e. perf top) */
			if (env->arch)
				tui__header_window(env);
			continue;
2194 2195 2196
		case 'F':
			symbol_conf.filter_relative ^= 1;
			continue;
2197 2198 2199 2200 2201 2202 2203
		case 'z':
			if (!is_report_browser(hbt)) {
				struct perf_top *top = hbt->arg;

				top->zero = !top->zero;
			}
			continue;
2204
		case K_F1:
2205 2206
		case 'h':
		case '?':
2207
			ui_browser__help_window(&browser->b,
2208
				is_report_browser(hbt) ? report_help : top_help);
2209
			continue;
2210 2211
		case K_ENTER:
		case K_RIGHT:
2212
		case 'm':
2213 2214
			/* menu */
			break;
2215
		case K_ESC:
2216
		case K_LEFT: {
2217
			const void *top;
2218

2219
			if (pstack__empty(browser->pstack)) {
2220 2221 2222 2223 2224
				/*
				 * Go back to the perf_evsel_menu__run or other user
				 */
				if (left_exits)
					goto out_free_stack;
2225 2226 2227 2228 2229 2230

				if (key == K_ESC &&
				    ui_browser__dialog_yesno(&browser->b,
							     "Do you really want to exit?"))
					goto out_free_stack;

2231
				continue;
2232
			}
2233
			top = pstack__peek(browser->pstack);
2234
			if (top == &browser->hists->dso_filter) {
2235 2236 2237 2238 2239 2240
				/*
				 * No need to set actions->dso here since
				 * it's just to remove the current filter.
				 * Ditto for thread below.
				 */
				do_zoom_dso(browser, actions);
2241
			} else if (top == &browser->hists->thread_filter) {
2242
				do_zoom_thread(browser, actions);
2243 2244 2245
			} else if (top == &browser->hists->socket_filter) {
				do_zoom_socket(browser, actions);
			}
2246 2247
			continue;
		}
2248 2249
		case 'q':
		case CTRL('c'):
2250
			goto out_free_stack;
2251
		case 'f':
2252 2253 2254 2255 2256 2257 2258 2259 2260 2261 2262 2263 2264 2265 2266 2267 2268
			if (!is_report_browser(hbt)) {
				struct perf_top *top = hbt->arg;

				perf_evlist__toggle_enable(top->evlist);
				/*
				 * No need to refresh, resort/decay histogram
				 * entries if we are not collecting samples:
				 */
				if (top->evlist->enabled) {
					helpline = "Press 'f' to disable the events or 'h' to see other hotkeys";
					hbt->refresh = delay_secs;
				} else {
					helpline = "Press 'f' again to re-enable the events";
					hbt->refresh = 0;
				}
				continue;
			}
2269
			/* Fall thru */
2270
		default:
2271
			helpline = "Press '?' for help on key bindings";
2272
			continue;
2273 2274
		}

2275
		if (!sort__has_sym || browser->selection == NULL)
2276 2277
			goto skip_annotation;

2278
		if (sort__mode == SORT_MODE__BRANCH) {
2279
			bi = browser->he_selection->branch_info;
2280 2281 2282 2283

			if (bi == NULL)
				goto skip_annotation;

2284 2285 2286 2287 2288 2289 2290 2291 2292 2293 2294
			nr_options += add_annotate_opt(browser,
						       &actions[nr_options],
						       &options[nr_options],
						       bi->from.map,
						       bi->from.sym);
			if (bi->to.sym != bi->from.sym)
				nr_options += add_annotate_opt(browser,
							&actions[nr_options],
							&options[nr_options],
							bi->to.map,
							bi->to.sym);
2295
		} else {
2296 2297 2298 2299 2300
			nr_options += add_annotate_opt(browser,
						       &actions[nr_options],
						       &options[nr_options],
						       browser->selection->map,
						       browser->selection->sym);
2301
		}
2302
skip_annotation:
2303 2304 2305
		nr_options += add_thread_opt(browser, &actions[nr_options],
					     &options[nr_options], thread);
		nr_options += add_dso_opt(browser, &actions[nr_options],
2306
					  &options[nr_options], map);
2307 2308
		nr_options += add_map_opt(browser, &actions[nr_options],
					  &options[nr_options],
2309 2310
					  browser->selection ?
						browser->selection->map : NULL);
2311 2312 2313
		nr_options += add_socket_opt(browser, &actions[nr_options],
					     &options[nr_options],
					     socked_id);
2314
		/* perf script support */
2315 2316 2317
		if (!is_report_browser(hbt))
			goto skip_scripting;

2318
		if (browser->he_selection) {
2319 2320 2321 2322 2323 2324
			if (sort__has_thread && thread) {
				nr_options += add_script_opt(browser,
							     &actions[nr_options],
							     &options[nr_options],
							     thread, NULL);
			}
2325 2326 2327 2328 2329 2330 2331 2332 2333
			/*
			 * Note that browser->selection != NULL
			 * when browser->he_selection is not NULL,
			 * so we don't need to check browser->selection
			 * before fetching browser->selection->sym like what
			 * we do before fetching browser->selection->map.
			 *
			 * See hist_browser__show_entry.
			 */
2334 2335 2336 2337 2338 2339
			if (sort__has_sym && browser->selection->sym) {
				nr_options += add_script_opt(browser,
							     &actions[nr_options],
							     &options[nr_options],
							     NULL, browser->selection->sym);
			}
2340
		}
2341 2342 2343 2344
		nr_options += add_script_opt(browser, &actions[nr_options],
					     &options[nr_options], NULL, NULL);
		nr_options += add_switch_opt(browser, &actions[nr_options],
					     &options[nr_options]);
2345
skip_scripting:
2346 2347
		nr_options += add_exit_opt(browser, &actions[nr_options],
					   &options[nr_options]);
2348

2349 2350
		do {
			struct popup_action *act;
2351

2352 2353 2354
			choice = ui__popup_menu(nr_options, options);
			if (choice == -1 || choice >= nr_options)
				break;
2355

2356 2357 2358
			act = &actions[choice];
			key = act->fn(browser, act);
		} while (key == 1);
2359

2360 2361
		if (key == K_SWITCH_INPUT_DATA)
			break;
2362 2363
	}
out_free_stack:
2364
	pstack__delete(browser->pstack);
2365 2366
out:
	hist_browser__delete(browser);
2367
	free_popup_options(options, MAX_OPTIONS);
2368 2369 2370
	return key;
}

2371 2372 2373
struct perf_evsel_menu {
	struct ui_browser b;
	struct perf_evsel *selection;
2374
	bool lost_events, lost_events_warned;
2375
	float min_pcnt;
2376
	struct perf_env *env;
2377 2378 2379 2380 2381 2382 2383 2384
};

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);
2385
	struct hists *hists = evsel__hists(evsel);
2386
	bool current_entry = ui_browser__is_current_entry(browser, row);
2387
	unsigned long nr_events = hists->stats.nr_events[PERF_RECORD_SAMPLE];
2388
	const char *ev_name = perf_evsel__name(evsel);
2389
	char bf[256], unit;
2390 2391
	const char *warn = " ";
	size_t printed;
2392 2393 2394 2395

	ui_browser__set_color(browser, current_entry ? HE_COLORSET_SELECTED :
						       HE_COLORSET_NORMAL);

2396
	if (perf_evsel__is_group_event(evsel)) {
2397 2398 2399 2400 2401
		struct perf_evsel *pos;

		ev_name = perf_evsel__group_name(evsel);

		for_each_group_member(pos, evsel) {
2402 2403
			struct hists *pos_hists = evsel__hists(pos);
			nr_events += pos_hists->stats.nr_events[PERF_RECORD_SAMPLE];
2404 2405 2406
		}
	}

2407
	nr_events = convert_unit(nr_events, &unit);
2408
	printed = scnprintf(bf, sizeof(bf), "%lu%c%s%s", nr_events,
2409
			   unit, unit == ' ' ? "" : " ", ev_name);
2410
	ui_browser__printf(browser, "%s", bf);
2411

2412
	nr_events = hists->stats.nr_events[PERF_RECORD_LOST];
2413 2414 2415 2416 2417
	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);
2418 2419
		printed += scnprintf(bf, sizeof(bf), ": %ld%c%schunks LOST!",
				     nr_events, unit, unit == ' ' ? "" : " ");
2420 2421 2422
		warn = bf;
	}

2423
	ui_browser__write_nstring(browser, warn, browser->width - printed);
2424 2425 2426 2427 2428

	if (current_entry)
		menu->selection = evsel;
}

2429 2430
static int perf_evsel_menu__run(struct perf_evsel_menu *menu,
				int nr_events, const char *help,
2431
				struct hist_browser_timer *hbt)
2432
{
2433
	struct perf_evlist *evlist = menu->b.priv;
2434
	struct perf_evsel *pos;
2435
	const char *title = "Available samples";
2436
	int delay_secs = hbt ? hbt->refresh : 0;
2437
	int key;
2438

2439 2440 2441 2442 2443
	if (ui_browser__show(&menu->b, title,
			     "ESC: exit, ENTER|->: Browse histograms") < 0)
		return -1;

	while (1) {
2444
		key = ui_browser__run(&menu->b, delay_secs);
2445 2446

		switch (key) {
2447
		case K_TIMER:
2448
			hbt->timer(hbt->arg);
2449 2450 2451 2452 2453

			if (!menu->lost_events_warned && menu->lost_events) {
				ui_browser__warn_lost_events(&menu->b);
				menu->lost_events_warned = true;
			}
2454
			continue;
2455 2456
		case K_RIGHT:
		case K_ENTER:
2457 2458 2459 2460
			if (!menu->selection)
				continue;
			pos = menu->selection;
browse_hists:
2461 2462 2463 2464 2465
			perf_evlist__set_selected(evlist, pos);
			/*
			 * Give the calling tool a chance to populate the non
			 * default evsel resorted hists tree.
			 */
2466 2467
			if (hbt)
				hbt->timer(hbt->arg);
2468
			key = perf_evsel__hists_browse(pos, nr_events, help,
2469
						       true, hbt,
2470
						       menu->min_pcnt,
2471
						       menu->env);
2472
			ui_browser__show_title(&menu->b, title);
2473
			switch (key) {
2474
			case K_TAB:
2475
				if (pos->node.next == &evlist->entries)
2476
					pos = perf_evlist__first(evlist);
2477
				else
2478
					pos = perf_evsel__next(pos);
2479
				goto browse_hists;
2480
			case K_UNTAB:
2481
				if (pos->node.prev == &evlist->entries)
2482
					pos = perf_evlist__last(evlist);
2483
				else
2484
					pos = perf_evsel__prev(pos);
2485
				goto browse_hists;
2486
			case K_SWITCH_INPUT_DATA:
2487 2488 2489
			case 'q':
			case CTRL('c'):
				goto out;
2490
			case K_ESC:
2491 2492 2493
			default:
				continue;
			}
2494
		case K_LEFT:
2495
			continue;
2496
		case K_ESC:
2497 2498
			if (!ui_browser__dialog_yesno(&menu->b,
					       "Do you really want to exit?"))
2499 2500
				continue;
			/* Fall thru */
2501 2502 2503
		case 'q':
		case CTRL('c'):
			goto out;
2504
		default:
2505
			continue;
2506 2507 2508
		}
	}

2509 2510 2511 2512 2513
out:
	ui_browser__hide(&menu->b);
	return key;
}

2514
static bool filter_group_entries(struct ui_browser *browser __maybe_unused,
2515 2516 2517 2518 2519 2520 2521 2522 2523 2524
				 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;
}

2525
static int __perf_evlist__tui_browse_hists(struct perf_evlist *evlist,
2526
					   int nr_entries, const char *help,
2527
					   struct hist_browser_timer *hbt,
2528
					   float min_pcnt,
2529
					   struct perf_env *env)
2530 2531 2532 2533 2534 2535 2536 2537
{
	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,
2538 2539
			.filter	    = filter_group_entries,
			.nr_entries = nr_entries,
2540 2541
			.priv	    = evlist,
		},
2542
		.min_pcnt = min_pcnt,
2543
		.env = env,
2544 2545 2546 2547
	};

	ui_helpline__push("Press ESC to exit");

2548
	evlist__for_each(evlist, pos) {
2549
		const char *ev_name = perf_evsel__name(pos);
2550 2551 2552 2553 2554 2555
		size_t line_len = strlen(ev_name) + 7;

		if (menu.b.width < line_len)
			menu.b.width = line_len;
	}

2556
	return perf_evsel_menu__run(&menu, nr_entries, help, hbt);
2557 2558
}

2559
int perf_evlist__tui_browse_hists(struct perf_evlist *evlist, const char *help,
2560
				  struct hist_browser_timer *hbt,
2561
				  float min_pcnt,
2562
				  struct perf_env *env)
2563
{
2564 2565 2566 2567
	int nr_entries = evlist->nr_entries;

single_entry:
	if (nr_entries == 1) {
2568
		struct perf_evsel *first = perf_evlist__first(evlist);
2569 2570

		return perf_evsel__hists_browse(first, nr_entries, help,
2571
						false, hbt, min_pcnt,
2572
						env);
2573 2574
	}

2575 2576 2577 2578
	if (symbol_conf.event_group) {
		struct perf_evsel *pos;

		nr_entries = 0;
2579
		evlist__for_each(evlist, pos) {
2580 2581
			if (perf_evsel__is_group_leader(pos))
				nr_entries++;
2582
		}
2583 2584 2585 2586 2587 2588

		if (nr_entries == 1)
			goto single_entry;
	}

	return __perf_evlist__tui_browse_hists(evlist, nr_entries, help,
2589
					       hbt, min_pcnt, env);
2590
}