builtin-top.c 34.7 KB
Newer Older
1
/*
2 3 4 5 6 7
 * builtin-top.c
 *
 * Builtin top command: Display a continuously updated profile of
 * any workload, CPU or specific PID.
 *
 * Copyright (C) 2008, Red Hat Inc, Ingo Molnar <mingo@redhat.com>
8
 *		 2011, Red Hat Inc, Arnaldo Carvalho de Melo <acme@redhat.com>
9 10 11 12 13 14 15 16 17 18
 *
 * Improvements and fixes by:
 *
 *   Arjan van de Ven <arjan@linux.intel.com>
 *   Yanmin Zhang <yanmin.zhang@intel.com>
 *   Wu Fengguang <fengguang.wu@intel.com>
 *   Mike Galbraith <efault@gmx.de>
 *   Paul Mackerras <paulus@samba.org>
 *
 * Released under the GPL v2. (and only v2, not any later version)
19
 */
20
#include "builtin.h"
21

22
#include "perf.h"
23

24
#include "util/annotate.h"
25
#include "util/cache.h"
26
#include "util/color.h"
27
#include "util/evlist.h"
28
#include "util/evsel.h"
29
#include "util/machine.h"
30 31
#include "util/session.h"
#include "util/symbol.h"
32
#include "util/thread.h"
33
#include "util/thread_map.h"
34
#include "util/top.h"
35
#include "util/util.h"
36
#include <linux/rbtree.h>
37 38
#include "util/parse-options.h"
#include "util/parse-events.h"
39
#include "util/cpumap.h"
40
#include "util/xyarray.h"
41
#include "util/sort.h"
42
#include "util/intlist.h"
43

44 45
#include "util/debug.h"

46
#include <assert.h>
47
#include <elf.h>
48
#include <fcntl.h>
49

50
#include <stdio.h>
51 52
#include <termios.h>
#include <unistd.h>
53
#include <inttypes.h>
54

55 56 57 58 59 60 61 62 63 64
#include <errno.h>
#include <time.h>
#include <sched.h>

#include <sys/syscall.h>
#include <sys/ioctl.h>
#include <sys/poll.h>
#include <sys/prctl.h>
#include <sys/wait.h>
#include <sys/uio.h>
65
#include <sys/utsname.h>
66 67 68 69 70
#include <sys/mman.h>

#include <linux/unistd.h>
#include <linux/types.h>

71
void get_term_dimensions(struct winsize *ws)
72
{
73 74 75 76 77 78 79 80 81 82
	char *s = getenv("LINES");

	if (s != NULL) {
		ws->ws_row = atoi(s);
		s = getenv("COLUMNS");
		if (s != NULL) {
			ws->ws_col = atoi(s);
			if (ws->ws_row && ws->ws_col)
				return;
		}
83
	}
84 85 86 87
#ifdef TIOCGWINSZ
	if (ioctl(1, TIOCGWINSZ, ws) == 0 &&
	    ws->ws_row && ws->ws_col)
		return;
88
#endif
89 90
	ws->ws_row = 25;
	ws->ws_col = 80;
91 92
}

93
static void perf_top__update_print_entries(struct perf_top *top)
94
{
95 96
	if (top->print_entries > 9)
		top->print_entries -= 9;
97 98
}

99 100
static void perf_top__sig_winch(int sig __maybe_unused,
				siginfo_t *info __maybe_unused, void *arg)
101
{
102 103 104
	struct perf_top *top = arg;

	get_term_dimensions(&top->winsize);
105 106 107 108 109 110 111
	if (!top->print_entries
	    || (top->print_entries+4) > top->winsize.ws_row) {
		top->print_entries = top->winsize.ws_row;
	} else {
		top->print_entries += 4;
		top->winsize.ws_row = top->print_entries;
	}
112
	perf_top__update_print_entries(top);
113 114
}

115
static int perf_top__parse_source(struct perf_top *top, struct hist_entry *he)
116 117
{
	struct symbol *sym;
118
	struct annotation *notes;
119
	struct map *map;
120
	int err = -1;
121

122
	if (!he || !he->ms.sym)
123 124
		return -1;

125 126
	sym = he->ms.sym;
	map = he->ms.map;
127 128 129 130

	/*
	 * We can't annotate with just /proc/kallsyms
	 */
131
	if (map->dso->symtab_type == DSO_BINARY_TYPE__KALLSYMS) {
132 133 134
		pr_err("Can't annotate %s: No vmlinux file was found in the "
		       "path\n", sym->name);
		sleep(1);
135
		return -1;
136 137
	}

138 139 140
	notes = symbol__annotation(sym);
	if (notes->src != NULL) {
		pthread_mutex_lock(&notes->lock);
141 142 143
		goto out_assign;
	}

144
	pthread_mutex_lock(&notes->lock);
145

146
	if (symbol__alloc_hist(sym) < 0) {
147
		pthread_mutex_unlock(&notes->lock);
148 149
		pr_err("Not enough memory for annotating '%s' symbol!\n",
		       sym->name);
150
		sleep(1);
151
		return err;
152
	}
153

154
	err = symbol__annotate(sym, map, 0);
155
	if (err == 0) {
156
out_assign:
157
		top->sym_filter_entry = he;
158
	}
159

160
	pthread_mutex_unlock(&notes->lock);
161
	return err;
162 163
}

164
static void __zero_source_counters(struct hist_entry *he)
165
{
166
	struct symbol *sym = he->ms.sym;
167
	symbol__annotate_zero_histograms(sym);
168 169
}

170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196
static void ui__warn_map_erange(struct map *map, struct symbol *sym, u64 ip)
{
	struct utsname uts;
	int err = uname(&uts);

	ui__warning("Out of bounds address found:\n\n"
		    "Addr:   %" PRIx64 "\n"
		    "DSO:    %s %c\n"
		    "Map:    %" PRIx64 "-%" PRIx64 "\n"
		    "Symbol: %" PRIx64 "-%" PRIx64 " %c %s\n"
		    "Arch:   %s\n"
		    "Kernel: %s\n"
		    "Tools:  %s\n\n"
		    "Not all samples will be on the annotation output.\n\n"
		    "Please report to linux-kernel@vger.kernel.org\n",
		    ip, map->dso->long_name, dso__symtab_origin(map->dso),
		    map->start, map->end, sym->start, sym->end,
		    sym->binding == STB_GLOBAL ? 'g' :
		    sym->binding == STB_LOCAL  ? 'l' : 'w', sym->name,
		    err ? "[unknown]" : uts.machine,
		    err ? "[unknown]" : uts.release, perf_version_string);
	if (use_browser <= 0)
		sleep(5);
	
	map->erange_warned = true;
}

197 198 199
static void perf_top__record_precise_ip(struct perf_top *top,
					struct hist_entry *he,
					int counter, u64 ip)
200
{
201 202
	struct annotation *notes;
	struct symbol *sym;
203
	int err;
204

205
	if (he == NULL || he->ms.sym == NULL ||
206 207
	    ((top->sym_filter_entry == NULL ||
	      top->sym_filter_entry->ms.sym != he->ms.sym) && use_browser != 1))
208 209
		return;

210
	sym = he->ms.sym;
211 212 213
	notes = symbol__annotation(sym);

	if (pthread_mutex_trylock(&notes->lock))
214 215
		return;

216
	if (notes->src == NULL && symbol__alloc_hist(sym) < 0) {
217 218 219 220 221 222 223 224
		pthread_mutex_unlock(&notes->lock);
		pr_err("Not enough memory for annotating '%s' symbol!\n",
		       sym->name);
		sleep(1);
		return;
	}

	ip = he->ms.map->map_ip(he->ms.map, ip);
225
	err = symbol__inc_addr_samples(sym, he->ms.map, counter, ip);
226

227
	pthread_mutex_unlock(&notes->lock);
228 229 230

	if (err == -ERANGE && !he->ms.map->erange_warned)
		ui__warn_map_erange(he->ms.map, sym, ip);
231 232
}

233
static void perf_top__show_details(struct perf_top *top)
234
{
235
	struct hist_entry *he = top->sym_filter_entry;
236
	struct annotation *notes;
237
	struct symbol *symbol;
238
	int more;
239

240
	if (!he)
241 242
		return;

243
	symbol = he->ms.sym;
244 245 246 247 248 249
	notes = symbol__annotation(symbol);

	pthread_mutex_lock(&notes->lock);

	if (notes->src == NULL)
		goto out_unlock;
250

251
	printf("Showing %s for %s\n", perf_evsel__name(top->sym_evsel), symbol->name);
252
	printf("  Events  Pcnt (>=%d%%)\n", top->sym_pcnt_filter);
253

254 255 256 257
	more = symbol__annotate_printf(symbol, he->ms.map, top->sym_evsel->idx,
				       0, top->sym_pcnt_filter, top->print_entries, 4);
	if (top->zero)
		symbol__annotate_zero_histogram(symbol, top->sym_evsel->idx);
258
	else
259
		symbol__annotate_decay_histogram(symbol, top->sym_evsel->idx);
260
	if (more != 0)
261
		printf("%d lines not displayed, maybe increase display entries [e]\n", more);
262 263
out_unlock:
	pthread_mutex_unlock(&notes->lock);
264
}
265 266 267

static const char		CONSOLE_CLEAR[] = "";

268 269 270
static struct hist_entry *perf_evsel__add_hist_entry(struct perf_evsel *evsel,
						     struct addr_location *al,
						     struct perf_sample *sample)
271
{
272 273 274 275 276 277 278 279
	struct hist_entry *he;

	he = __hists__add_entry(&evsel->hists, al, NULL, sample->period);
	if (he == NULL)
		return NULL;

	hists__inc_nr_events(&evsel->hists, PERF_RECORD_SAMPLE);
	return he;
280
}
281

282
static void perf_top__print_sym_table(struct perf_top *top)
283
{
284 285
	char bf[160];
	int printed = 0;
286
	const int win_width = top->winsize.ws_col - 1;
287

288
	puts(CONSOLE_CLEAR);
289

290
	perf_top__header_snprintf(top, bf, sizeof(bf));
291
	printf("%s\n", bf);
292

293
	perf_top__reset_sample_counters(top);
294

295
	printf("%-*.*s\n", win_width, win_width, graph_dotted_line);
296

297 298 299 300
	if (top->sym_evsel->hists.stats.nr_lost_warned !=
	    top->sym_evsel->hists.stats.nr_events[PERF_RECORD_LOST]) {
		top->sym_evsel->hists.stats.nr_lost_warned =
			top->sym_evsel->hists.stats.nr_events[PERF_RECORD_LOST];
301 302
		color_fprintf(stdout, PERF_COLOR_RED,
			      "WARNING: LOST %d chunks, Check IO/CPU overload",
303
			      top->sym_evsel->hists.stats.nr_lost_warned);
304
		++printed;
305 306
	}

307 308
	if (top->sym_filter_entry) {
		perf_top__show_details(top);
309 310 311
		return;
	}

312 313 314 315 316 317 318
	hists__collapse_resort_threaded(&top->sym_evsel->hists);
	hists__output_resort_threaded(&top->sym_evsel->hists);
	hists__decay_entries_threaded(&top->sym_evsel->hists,
				      top->hide_user_symbols,
				      top->hide_kernel_symbols);
	hists__output_recalc_col_len(&top->sym_evsel->hists,
				     top->winsize.ws_row - 3);
319
	putchar('\n');
320
	hists__fprintf(&top->sym_evsel->hists, false,
321
		       top->winsize.ws_row - 4 - printed, win_width, stdout);
322 323
}

324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358
static void prompt_integer(int *target, const char *msg)
{
	char *buf = malloc(0), *p;
	size_t dummy = 0;
	int tmp;

	fprintf(stdout, "\n%s: ", msg);
	if (getline(&buf, &dummy, stdin) < 0)
		return;

	p = strchr(buf, '\n');
	if (p)
		*p = 0;

	p = buf;
	while(*p) {
		if (!isdigit(*p))
			goto out_free;
		p++;
	}
	tmp = strtoul(buf, NULL, 10);
	*target = tmp;
out_free:
	free(buf);
}

static void prompt_percent(int *target, const char *msg)
{
	int tmp = 0;

	prompt_integer(&tmp, msg);
	if (tmp >= 0 && tmp <= 100)
		*target = tmp;
}

359
static void perf_top__prompt_symbol(struct perf_top *top, const char *msg)
360 361
{
	char *buf = malloc(0), *p;
362
	struct hist_entry *syme = top->sym_filter_entry, *n, *found = NULL;
363
	struct rb_node *next;
364 365 366 367 368
	size_t dummy = 0;

	/* zero counters of active symbol */
	if (syme) {
		__zero_source_counters(syme);
369
		top->sym_filter_entry = NULL;
370 371 372 373 374 375 376 377 378 379
	}

	fprintf(stdout, "\n%s: ", msg);
	if (getline(&buf, &dummy, stdin) < 0)
		goto out_free;

	p = strchr(buf, '\n');
	if (p)
		*p = 0;

380
	next = rb_first(&top->sym_evsel->hists.entries);
381 382 383 384
	while (next) {
		n = rb_entry(next, struct hist_entry, rb_node);
		if (n->ms.sym && !strcmp(buf, n->ms.sym->name)) {
			found = n;
385 386
			break;
		}
387
		next = rb_next(&n->rb_node);
388 389 390
	}

	if (!found) {
391
		fprintf(stderr, "Sorry, %s is not active.\n", buf);
392 393
		sleep(1);
	} else
394
		perf_top__parse_source(top, found);
395 396 397 398 399

out_free:
	free(buf);
}

400
static void perf_top__print_mapped_keys(struct perf_top *top)
401
{
402 403
	char *name = NULL;

404 405
	if (top->sym_filter_entry) {
		struct symbol *sym = top->sym_filter_entry->ms.sym;
406 407 408 409
		name = sym->name;
	}

	fprintf(stdout, "\nMapped keys:\n");
410 411
	fprintf(stdout, "\t[d]     display refresh delay.             \t(%d)\n", top->delay_secs);
	fprintf(stdout, "\t[e]     display entries (lines).           \t(%d)\n", top->print_entries);
412

413
	if (top->evlist->nr_entries > 1)
414
		fprintf(stdout, "\t[E]     active event counter.              \t(%s)\n", perf_evsel__name(top->sym_evsel));
415

416
	fprintf(stdout, "\t[f]     profile display filter (count).    \t(%d)\n", top->count_filter);
417

418
	fprintf(stdout, "\t[F]     annotate display filter (percent). \t(%d%%)\n", top->sym_pcnt_filter);
419 420
	fprintf(stdout, "\t[s]     annotate symbol.                   \t(%s)\n", name?: "NULL");
	fprintf(stdout, "\t[S]     stop annotation.\n");
421

422
	fprintf(stdout,
423
		"\t[K]     hide kernel_symbols symbols.     \t(%s)\n",
424
		top->hide_kernel_symbols ? "yes" : "no");
425 426
	fprintf(stdout,
		"\t[U]     hide user symbols.               \t(%s)\n",
427 428
		top->hide_user_symbols ? "yes" : "no");
	fprintf(stdout, "\t[z]     toggle sample zeroing.             \t(%d)\n", top->zero ? 1 : 0);
429 430 431
	fprintf(stdout, "\t[qQ]    quit.\n");
}

432
static int perf_top__key_mapped(struct perf_top *top, int c)
433 434 435 436 437 438 439 440
{
	switch (c) {
		case 'd':
		case 'e':
		case 'f':
		case 'z':
		case 'q':
		case 'Q':
441 442
		case 'K':
		case 'U':
443 444 445
		case 'F':
		case 's':
		case 'S':
446 447
			return 1;
		case 'E':
448
			return top->evlist->nr_entries > 1 ? 1 : 0;
449 450
		default:
			break;
451 452 453
	}

	return 0;
454 455
}

456
static void perf_top__handle_keypress(struct perf_top *top, int c)
457
{
458
	if (!perf_top__key_mapped(top, c)) {
459 460 461
		struct pollfd stdin_poll = { .fd = 0, .events = POLLIN };
		struct termios tc, save;

462
		perf_top__print_mapped_keys(top);
463 464 465 466 467 468 469 470 471 472 473 474 475 476
		fprintf(stdout, "\nEnter selection, or unmapped key to continue: ");
		fflush(stdout);

		tcgetattr(0, &save);
		tc = save;
		tc.c_lflag &= ~(ICANON | ECHO);
		tc.c_cc[VMIN] = 0;
		tc.c_cc[VTIME] = 0;
		tcsetattr(0, TCSANOW, &tc);

		poll(&stdin_poll, 1, -1);
		c = getc(stdin);

		tcsetattr(0, TCSAFLUSH, &save);
477
		if (!perf_top__key_mapped(top, c))
478 479 480
			return;
	}

481 482
	switch (c) {
		case 'd':
483 484 485
			prompt_integer(&top->delay_secs, "Enter display delay");
			if (top->delay_secs < 1)
				top->delay_secs = 1;
486 487
			break;
		case 'e':
488 489 490 491 492 493 494 495
			prompt_integer(&top->print_entries, "Enter display entries (lines)");
			if (top->print_entries == 0) {
				struct sigaction act = {
					.sa_sigaction = perf_top__sig_winch,
					.sa_flags     = SA_SIGINFO,
				};
				perf_top__sig_winch(SIGWINCH, NULL, top);
				sigaction(SIGWINCH, &act, NULL);
496 497
			} else {
				perf_top__sig_winch(SIGWINCH, NULL, top);
498
				signal(SIGWINCH, SIG_DFL);
499
			}
500 501
			break;
		case 'E':
502
			if (top->evlist->nr_entries > 1) {
503 504 505
				/* Select 0 as the default event: */
				int counter = 0;

506
				fprintf(stderr, "\nAvailable events:");
507

508
				list_for_each_entry(top->sym_evsel, &top->evlist->entries, node)
509
					fprintf(stderr, "\n\t%d %s", top->sym_evsel->idx, perf_evsel__name(top->sym_evsel));
510

511
				prompt_integer(&counter, "Enter details event counter");
512

513
				if (counter >= top->evlist->nr_entries) {
514
					top->sym_evsel = perf_evlist__first(top->evlist);
515
					fprintf(stderr, "Sorry, no such event, using %s.\n", perf_evsel__name(top->sym_evsel));
516
					sleep(1);
517
					break;
518
				}
519 520
				list_for_each_entry(top->sym_evsel, &top->evlist->entries, node)
					if (top->sym_evsel->idx == counter)
521
						break;
522
			} else
523
				top->sym_evsel = perf_evlist__first(top->evlist);
524 525
			break;
		case 'f':
526
			prompt_integer(&top->count_filter, "Enter display event count filter");
527 528
			break;
		case 'F':
529 530
			prompt_percent(&top->sym_pcnt_filter,
				       "Enter details display event filter (percent)");
531
			break;
532
		case 'K':
533
			top->hide_kernel_symbols = !top->hide_kernel_symbols;
534
			break;
535 536 537
		case 'q':
		case 'Q':
			printf("exiting.\n");
538 539
			if (top->dump_symtab)
				perf_session__fprintf_dsos(top->session, stderr);
540 541
			exit(0);
		case 's':
542
			perf_top__prompt_symbol(top, "Enter details symbol");
543 544
			break;
		case 'S':
545
			if (!top->sym_filter_entry)
546 547
				break;
			else {
548
				struct hist_entry *syme = top->sym_filter_entry;
549

550
				top->sym_filter_entry = NULL;
551 552 553
				__zero_source_counters(syme);
			}
			break;
554
		case 'U':
555
			top->hide_user_symbols = !top->hide_user_symbols;
556
			break;
557
		case 'z':
558
			top->zero = !top->zero;
559
			break;
560 561
		default:
			break;
562 563 564
	}
}

565 566 567 568 569 570 571 572 573 574
static void perf_top__sort_new_samples(void *arg)
{
	struct perf_top *t = arg;
	perf_top__reset_sample_counters(t);

	if (t->evlist->selected != NULL)
		t->sym_evsel = t->evlist->selected;

	hists__collapse_resort_threaded(&t->sym_evsel->hists);
	hists__output_resort_threaded(&t->sym_evsel->hists);
575
	hists__decay_entries_threaded(&t->sym_evsel->hists,
576 577
				      t->hide_user_symbols,
				      t->hide_kernel_symbols);
578 579
}

580
static void *display_thread_tui(void *arg)
581
{
582
	struct perf_evsel *pos;
583
	struct perf_top *top = arg;
584
	const char *help = "For a higher level overview, try: perf top --sort comm,dso";
585 586 587 588 589
	struct hist_browser_timer hbt = {
		.timer		= perf_top__sort_new_samples,
		.arg		= top,
		.refresh	= top->delay_secs,
	};
590

591
	perf_top__sort_new_samples(top);
592 593 594 595 596 597 598

	/*
	 * Initialize the uid_filter_str, in the future the TUI will allow
	 * Zooming in/out UIDs. For now juse use whatever the user passed
	 * via --uid.
	 */
	list_for_each_entry(pos, &top->evlist->entries, node)
599
		pos->hists.uid_filter_str = top->target.uid_str;
600

601 602
	perf_evlist__tui_browse_hists(top->evlist, help, &hbt,
				      &top->session->header.env);
603

604 605 606 607 608
	exit_browser(0);
	exit(0);
	return NULL;
}

609
static void *display_thread(void *arg)
610
{
611
	struct pollfd stdin_poll = { .fd = 0, .events = POLLIN };
612
	struct termios tc, save;
613
	struct perf_top *top = arg;
614 615 616 617 618 619 620
	int delay_msecs, c;

	tcgetattr(0, &save);
	tc = save;
	tc.c_lflag &= ~(ICANON | ECHO);
	tc.c_cc[VMIN] = 0;
	tc.c_cc[VTIME] = 0;
621

622
	pthread__unblock_sigwinch();
623
repeat:
624
	delay_msecs = top->delay_secs * 1000;
625 626 627
	tcsetattr(0, TCSANOW, &tc);
	/* trash return*/
	getc(stdin);
628

629
	while (1) {
630
		perf_top__print_sym_table(top);
631 632 633 634 635 636 637 638 639 640 641 642 643 644 645 646
		/*
		 * Either timeout expired or we got an EINTR due to SIGWINCH,
		 * refresh screen in both cases.
		 */
		switch (poll(&stdin_poll, 1, delay_msecs)) {
		case 0:
			continue;
		case -1:
			if (errno == EINTR)
				continue;
			/* Fall trhu */
		default:
			goto process_hotkey;
		}
	}
process_hotkey:
647 648 649
	c = getc(stdin);
	tcsetattr(0, TCSAFLUSH, &save);

650
	perf_top__handle_keypress(top, c);
651
	goto repeat;
652 653 654 655

	return NULL;
}

656
/* Tag samples to be skipped. */
657
static const char *skip_symbols[] = {
658
	"intel_idle",
659
	"default_idle",
660
	"native_safe_halt",
661 662 663 664
	"cpu_idle",
	"enter_idle",
	"exit_idle",
	"mwait_idle",
665
	"mwait_idle_with_hints",
666
	"poll_idle",
667 668
	"ppc64_runlatch_off",
	"pseries_dedicated_idle_sleep",
669 670 671
	NULL
};

672
static int symbol_filter(struct map *map __maybe_unused, struct symbol *sym)
673
{
674
	const char *name = sym->name;
675
	int i;
676

677 678 679 680 681 682 683
	/*
	 * ppc64 uses function descriptors and appends a '.' to the
	 * start of every instruction address. Remove it.
	 */
	if (name[0] == '.')
		name++;

684 685 686 687 688 689 690
	if (!strcmp(name, "_text") ||
	    !strcmp(name, "_etext") ||
	    !strcmp(name, "_sinittext") ||
	    !strncmp("init_module", name, 11) ||
	    !strncmp("cleanup_module", name, 14) ||
	    strstr(name, "_text_start") ||
	    strstr(name, "_text_end"))
691 692
		return 1;

693 694
	for (i = 0; skip_symbols[i]; i++) {
		if (!strcmp(skip_symbols[i], name)) {
695
			sym->ignore = true;
696 697 698
			break;
		}
	}
699 700 701 702

	return 0;
}

703 704
static void perf_event__process_sample(struct perf_tool *tool,
				       const union perf_event *event,
705
				       struct perf_evsel *evsel,
706
				       struct perf_sample *sample,
707
				       struct machine *machine)
708
{
709
	struct perf_top *top = container_of(tool, struct perf_top, tool);
710
	struct symbol *parent = NULL;
711
	u64 ip = event->ip.ip;
712
	struct addr_location al;
713
	int err;
714

715
	if (!machine && perf_guest) {
716 717 718 719 720 721 722 723 724 725
		static struct intlist *seen;

		if (!seen)
			seen = intlist__new();

		if (!intlist__has_entry(seen, event->ip.pid)) {
			pr_err("Can't find guest [%d]'s kernel information\n",
				event->ip.pid);
			intlist__add(seen, event->ip.pid);
		}
726 727 728
		return;
	}

729
	if (!machine) {
730
		pr_err("%u unprocessable samples recorded.\n",
731 732 733 734
		       top->session->hists.stats.nr_unprocessable_samples++);
		return;
	}

735
	if (event->header.misc & PERF_RECORD_MISC_EXACT_IP)
736
		top->exact_samples++;
737

738
	if (perf_event__preprocess_sample(event, machine, &al, sample,
739
					  symbol_filter) < 0 ||
740
	    al.filtered)
741
		return;
742

743
	if (!top->kptr_restrict_warned &&
744 745 746 747 748 749 750 751 752 753
	    symbol_conf.kptr_restrict &&
	    al.cpumode == PERF_RECORD_MISC_KERNEL) {
		ui__warning(
"Kernel address maps (/proc/{kallsyms,modules}) are restricted.\n\n"
"Check /proc/sys/kernel/kptr_restrict.\n\n"
"Kernel%s samples will not be resolved.\n",
			  !RB_EMPTY_ROOT(&al.map->dso->symbols[MAP__FUNCTION]) ?
			  " modules" : "");
		if (use_browser <= 0)
			sleep(5);
754
		top->kptr_restrict_warned = true;
755 756
	}

757
	if (al.sym == NULL) {
758
		const char *msg = "Kernel samples will not be resolved.\n";
759 760 761 762 763 764 765 766 767 768 769
		/*
		 * As we do lazy loading of symtabs we only will know if the
		 * specified vmlinux file is invalid when we actually have a
		 * hit in kernel space and then try to load it. So if we get
		 * here and there are _no_ symbols in the DSO backing the
		 * kernel map, bail out.
		 *
		 * We may never get here, for instance, if we use -K/
		 * --hide-kernel-symbols, even if the user specifies an
		 * invalid --vmlinux ;-)
		 */
770
		if (!top->kptr_restrict_warned && !top->vmlinux_warned &&
771
		    al.map == machine->vmlinux_maps[MAP__FUNCTION] &&
772
		    RB_EMPTY_ROOT(&al.map->dso->symbols[MAP__FUNCTION])) {
773 774 775 776 777 778 779 780 781 782
			if (symbol_conf.vmlinux_name) {
				ui__warning("The %s file can't be used.\n%s",
					    symbol_conf.vmlinux_name, msg);
			} else {
				ui__warning("A vmlinux file was not found.\n%s",
					    msg);
			}

			if (use_browser <= 0)
				sleep(5);
783
			top->vmlinux_warned = true;
784
		}
785 786
	}

787 788
	if (al.sym == NULL || !al.sym->ignore) {
		struct hist_entry *he;
789

790 791
		if ((sort__has_parent || symbol_conf.use_callchain) &&
		    sample->callchain) {
792 793 794 795
			err = machine__resolve_callchain(machine, evsel,
							 al.thread, sample,
							 &parent);

796 797 798 799
			if (err)
				return;
		}

800
		he = perf_evsel__add_hist_entry(evsel, &al, sample);
801 802 803
		if (he == NULL) {
			pr_err("Problem incrementing symbol period, skipping event\n");
			return;
804
		}
805

806
		if (symbol_conf.use_callchain) {
807
			err = callchain_append(he->callchain, &callchain_cursor,
808 809 810 811 812
					       sample->period);
			if (err)
				return;
		}

813 814
		if (top->sort_has_symbols)
			perf_top__record_precise_ip(top, he, evsel->idx, ip);
815
	}
816 817

	return;
818 819
}

820
static void perf_top__mmap_read_idx(struct perf_top *top, int idx)
821
{
822
	struct perf_sample sample;
823
	struct perf_evsel *evsel;
824
	struct perf_session *session = top->session;
825
	union perf_event *event;
826 827
	struct machine *machine;
	u8 origin;
828
	int ret;
829

830
	while ((event = perf_evlist__mmap_read(top->evlist, idx)) != NULL) {
831
		ret = perf_evlist__parse_sample(top->evlist, event, &sample);
832 833 834 835
		if (ret) {
			pr_err("Can't parse sample, err = %d\n", ret);
			continue;
		}
836

837
		evsel = perf_evlist__id2evsel(session->evlist, sample.id);
838 839
		assert(evsel != NULL);

840 841
		origin = event->header.misc & PERF_RECORD_MISC_CPUMODE_MASK;

842
		if (event->header.type == PERF_RECORD_SAMPLE)
843
			++top->samples;
844 845 846

		switch (origin) {
		case PERF_RECORD_MISC_USER:
847 848
			++top->us_samples;
			if (top->hide_user_symbols)
849
				continue;
850
			machine = perf_session__find_host_machine(session);
851 852
			break;
		case PERF_RECORD_MISC_KERNEL:
853 854
			++top->kernel_samples;
			if (top->hide_kernel_symbols)
855
				continue;
856
			machine = perf_session__find_host_machine(session);
857 858
			break;
		case PERF_RECORD_MISC_GUEST_KERNEL:
859 860
			++top->guest_kernel_samples;
			machine = perf_session__find_machine(session, event->ip.pid);
861 862
			break;
		case PERF_RECORD_MISC_GUEST_USER:
863
			++top->guest_us_samples;
864 865 866 867 868 869 870 871 872 873
			/*
			 * TODO: we don't process guest user from host side
			 * except simple counting.
			 */
			/* Fall thru */
		default:
			continue;
		}


874 875 876 877
		if (event->header.type == PERF_RECORD_SAMPLE) {
			perf_event__process_sample(&top->tool, event, evsel,
						   &sample, machine);
		} else if (event->header.type < PERF_RECORD_MAX) {
878
			hists__inc_nr_events(&evsel->hists, event->header.type);
879
			machine__process_event(machine, event);
880
		} else
881
			++session->hists.stats.nr_unknown_events;
882 883 884
	}
}

885
static void perf_top__mmap_read(struct perf_top *top)
886
{
887 888
	int i;

889 890
	for (i = 0; i < top->evlist->nr_mmaps; i++)
		perf_top__mmap_read_idx(top, i);
891 892
}

893
static void perf_top__start_counters(struct perf_top *top)
894
{
895
	struct perf_evsel *counter;
896
	struct perf_evlist *evlist = top->evlist;
897

898
	if (top->group)
899
		perf_evlist__set_leader(evlist);
900

901 902
	list_for_each_entry(counter, &evlist->entries, node) {
		struct perf_event_attr *attr = &counter->attr;
903

904 905
		perf_evsel__set_sample_bit(counter, IP);
		perf_evsel__set_sample_bit(counter, TID);
906

907
		if (top->freq) {
908
			perf_evsel__set_sample_bit(counter, PERIOD);
909
			attr->freq	  = 1;
910
			attr->sample_freq = top->freq;
911
		}
912

913
		if (evlist->nr_entries > 1) {
914
			perf_evsel__set_sample_bit(counter, ID);
915 916 917
			attr->read_format |= PERF_FORMAT_ID;
		}

918
		if (perf_target__has_cpu(&top->target))
919
			perf_evsel__set_sample_bit(counter, CPU);
920

921
		if (symbol_conf.use_callchain)
922
			perf_evsel__set_sample_bit(counter, CALLCHAIN);
923

924
		attr->mmap = 1;
925
		attr->comm = 1;
926
		attr->inherit = top->inherit;
927 928 929
fallback_missing_features:
		if (top->exclude_guest_missing)
			attr->exclude_guest = attr->exclude_host = 0;
930
retry_sample_id:
931
		attr->sample_id_all = top->sample_id_all_missing ? 0 : 1;
932
try_again:
933
		if (perf_evsel__open(counter, top->evlist->cpus,
934
				     top->evlist->threads) < 0) {
935 936
			int err = errno;

937
			if (err == EPERM || err == EACCES) {
938
				ui__error_paranoid();
939
				goto out_err;
940 941 942 943 944 945 946
			} else if (err == EINVAL) {
				if (!top->exclude_guest_missing &&
				    (attr->exclude_guest || attr->exclude_host)) {
					pr_debug("Old kernel, cannot exclude "
						 "guest or host samples.\n");
					top->exclude_guest_missing = true;
					goto fallback_missing_features;
947
				} else if (!top->sample_id_all_missing) {
948 949 950
					/*
					 * Old kernel, no attr->sample_id_type_all field
					 */
951
					top->sample_id_all_missing = true;
952 953
					goto retry_sample_id;
				}
954
			}
955 956 957 958 959
			/*
			 * If it's cycles then fall back to hrtimer
			 * based cpu-clock-tick sw counter, which
			 * is always available even if no PMU support:
			 */
960 961 962 963
			if ((err == ENOENT || err == ENXIO) &&
			    (attr->type == PERF_TYPE_HARDWARE) &&
			    (attr->config == PERF_COUNT_HW_CPU_CYCLES)) {

964
				if (verbose)
965 966
					ui__warning("Cycles event not supported,\n"
						    "trying to fall back to cpu-clock-ticks\n");
967 968 969

				attr->type = PERF_TYPE_SOFTWARE;
				attr->config = PERF_COUNT_SW_CPU_CLOCK;
970 971
				if (counter->name) {
					free(counter->name);
972
					counter->name = NULL;
973
				}
974 975
				goto try_again;
			}
976

977
			if (err == ENOENT) {
978
				ui__error("The %s event is not supported.\n",
979
					  perf_evsel__name(counter));
980
				goto out_err;
981
			} else if (err == EMFILE) {
982
				ui__error("Too many events are opened.\n"
983 984
					    "Try again after reducing the number of events\n");
				goto out_err;
985 986 987 988
			} else if ((err == EOPNOTSUPP) && (attr->precise_ip)) {
				ui__error("\'precise\' request may not be supported. "
					  "Try removing 'p' modifier\n");
				goto out_err;
989 990
			}

991
			ui__error("The sys_perf_event_open() syscall "
992 993 994 995 996
				    "returned with %d (%s).  /bin/dmesg "
				    "may provide additional information.\n"
				    "No CONFIG_PERF_EVENTS=y kernel support "
				    "configured?\n", err, strerror(err));
			goto out_err;
997
		}
998
	}
999

1000
	if (perf_evlist__mmap(evlist, top->mmap_pages, false) < 0) {
1001
		ui__error("Failed to mmap with %d (%s)\n",
1002 1003 1004 1005 1006 1007 1008 1009 1010
			    errno, strerror(errno));
		goto out_err;
	}

	return;

out_err:
	exit_browser(0);
	exit(0);
1011 1012
}

1013
static int perf_top__setup_sample_type(struct perf_top *top)
1014
{
1015
	if (!top->sort_has_symbols) {
1016
		if (symbol_conf.use_callchain) {
1017
			ui__error("Selected -g but \"sym\" not present in --sort/-s.");
1018 1019
			return -EINVAL;
		}
1020
	} else if (!top->dont_use_callchains && callchain_param.mode != CHAIN_NONE) {
1021
		if (callchain_register_param(&callchain_param) < 0) {
1022
			ui__error("Can't register callchain params.\n");
1023 1024 1025 1026 1027 1028 1029
			return -EINVAL;
		}
	}

	return 0;
}

1030
static int __cmd_top(struct perf_top *top)
1031 1032
{
	pthread_t thread;
1033
	int ret;
1034
	/*
1035 1036
	 * FIXME: perf_session__new should allow passing a O_MMAP, so that all this
	 * mmap reading, etc is encapsulated in it. Use O_WRONLY for now.
1037
	 */
1038 1039
	top->session = perf_session__new(NULL, O_WRONLY, false, false, NULL);
	if (top->session == NULL)
1040
		return -ENOMEM;
1041

1042
	ret = perf_top__setup_sample_type(top);
1043 1044 1045
	if (ret)
		goto out_delete;

1046
	if (perf_target__has_task(&top->target))
1047
		perf_event__synthesize_thread_map(&top->tool, top->evlist->threads,
1048
						  perf_event__process,
1049
						  &top->session->host_machine);
1050
	else
1051 1052 1053 1054
		perf_event__synthesize_threads(&top->tool, perf_event__process,
					       &top->session->host_machine);
	perf_top__start_counters(top);
	top->session->evlist = top->evlist;
1055
	perf_session__set_id_hdr_size(top->session);
1056

1057
	/* Wait for a minimal set of events before starting the snapshot */
1058
	poll(top->evlist->pollfd, top->evlist->nr_fds, 100);
1059

1060
	perf_top__mmap_read(top);
1061

1062
	if (pthread_create(&thread, NULL, (use_browser > 0 ? display_thread_tui :
1063
							    display_thread), top)) {
1064
		ui__error("Could not create display thread.\n");
1065 1066 1067
		exit(-1);
	}

1068
	if (top->realtime_prio) {
1069 1070
		struct sched_param param;

1071
		param.sched_priority = top->realtime_prio;
1072
		if (sched_setscheduler(0, SCHED_FIFO, &param)) {
1073
			ui__error("Could not set realtime priority.\n");
1074 1075 1076 1077 1078
			exit(-1);
		}
	}

	while (1) {
1079
		u64 hits = top->samples;
1080

1081
		perf_top__mmap_read(top);
1082

1083 1084
		if (hits == top->samples)
			ret = poll(top->evlist->pollfd, top->evlist->nr_fds, 100);
1085 1086
	}

1087
out_delete:
1088 1089
	perf_session__delete(top->session);
	top->session = NULL;
1090 1091 1092 1093 1094

	return 0;
}

static int
1095
parse_callchain_opt(const struct option *opt, const char *arg, int unset)
1096
{
1097
	struct perf_top *top = (struct perf_top *)opt->value;
1098 1099 1100 1101 1102 1103 1104
	char *tok, *tok2;
	char *endptr;

	/*
	 * --no-call-graph
	 */
	if (unset) {
1105
		top->dont_use_callchains = true;
1106 1107 1108 1109 1110 1111 1112 1113 1114 1115 1116 1117 1118 1119 1120 1121 1122 1123 1124 1125 1126 1127 1128 1129 1130 1131 1132
		return 0;
	}

	symbol_conf.use_callchain = true;

	if (!arg)
		return 0;

	tok = strtok((char *)arg, ",");
	if (!tok)
		return -1;

	/* get the output mode */
	if (!strncmp(tok, "graph", strlen(arg)))
		callchain_param.mode = CHAIN_GRAPH_ABS;

	else if (!strncmp(tok, "flat", strlen(arg)))
		callchain_param.mode = CHAIN_FLAT;

	else if (!strncmp(tok, "fractal", strlen(arg)))
		callchain_param.mode = CHAIN_GRAPH_REL;

	else if (!strncmp(tok, "none", strlen(arg))) {
		callchain_param.mode = CHAIN_NONE;
		symbol_conf.use_callchain = false;

		return 0;
1133
	} else
1134 1135 1136 1137 1138 1139 1140 1141 1142 1143 1144 1145 1146 1147 1148 1149 1150 1151 1152 1153 1154 1155 1156 1157 1158 1159 1160 1161 1162 1163 1164 1165 1166 1167 1168
		return -1;

	/* get the min percentage */
	tok = strtok(NULL, ",");
	if (!tok)
		goto setup;

	callchain_param.min_percent = strtod(tok, &endptr);
	if (tok == endptr)
		return -1;

	/* get the print limit */
	tok2 = strtok(NULL, ",");
	if (!tok2)
		goto setup;

	if (tok2[0] != 'c') {
		callchain_param.print_limit = strtod(tok2, &endptr);
		tok2 = strtok(NULL, ",");
		if (!tok2)
			goto setup;
	}

	/* get the call chain order */
	if (!strcmp(tok2, "caller"))
		callchain_param.order = ORDER_CALLER;
	else if (!strcmp(tok2, "callee"))
		callchain_param.order = ORDER_CALLEE;
	else
		return -1;
setup:
	if (callchain_register_param(&callchain_param) < 0) {
		fprintf(stderr, "Can't register callchain params\n");
		return -1;
	}
1169 1170
	return 0;
}
1171

1172
int cmd_top(int argc, const char **argv, const char *prefix __maybe_unused)
1173 1174
{
	struct perf_evsel *pos;
1175 1176
	int status;
	char errbuf[BUFSIZ];
1177 1178 1179
	struct perf_top top = {
		.count_filter	     = 5,
		.delay_secs	     = 2,
1180
		.freq		     = 4000, /* 4 KHz */
1181 1182
		.mmap_pages	     = 128,
		.sym_pcnt_filter     = 5,
N
Namhyung Kim 已提交
1183 1184 1185
		.target		     = {
			.uses_mmap   = true,
		},
1186 1187 1188
	};
	char callchain_default_opt[] = "fractal,0.5,callee";
	const struct option options[] = {
1189
	OPT_CALLBACK('e', "event", &top.evlist, "event",
1190
		     "event selector. use 'perf list' to list available events",
1191
		     parse_events_option),
1192
	OPT_INTEGER('c', "count", &top.default_interval,
1193
		    "event period to sample"),
1194
	OPT_STRING('p', "pid", &top.target.pid, "pid",
1195
		    "profile events on existing process id"),
1196
	OPT_STRING('t', "tid", &top.target.tid, "tid",
1197
		    "profile events on existing thread id"),
1198
	OPT_BOOLEAN('a', "all-cpus", &top.target.system_wide,
1199
			    "system-wide collection from all CPUs"),
1200
	OPT_STRING('C', "cpu", &top.target.cpu_list, "cpu",
1201
		    "list of cpus to monitor"),
1202 1203
	OPT_STRING('k', "vmlinux", &symbol_conf.vmlinux_name,
		   "file", "vmlinux pathname"),
1204
	OPT_BOOLEAN('K', "hide_kernel_symbols", &top.hide_kernel_symbols,
1205
		    "hide kernel symbols"),
1206 1207
	OPT_UINTEGER('m', "mmap-pages", &top.mmap_pages, "number of mmap data pages"),
	OPT_INTEGER('r', "realtime", &top.realtime_prio,
1208
		    "collect data with this RT SCHED_FIFO priority"),
1209
	OPT_INTEGER('d', "delay", &top.delay_secs,
1210
		    "number of seconds to delay between refreshes"),
1211
	OPT_BOOLEAN('D', "dump-symtab", &top.dump_symtab,
1212
			    "dump the symbol table used for profiling"),
1213
	OPT_INTEGER('f', "count-filter", &top.count_filter,
1214
		    "only display functions with more events than this"),
1215
	OPT_BOOLEAN('g', "group", &top.group,
1216
			    "put the counters into a counter group"),
1217
	OPT_BOOLEAN('i', "inherit", &top.inherit,
1218
		    "child tasks inherit counters"),
1219
	OPT_STRING(0, "sym-annotate", &top.sym_filter, "symbol name",
1220
		    "symbol to annotate"),
1221
	OPT_BOOLEAN('z', "zero", &top.zero,
1222
		    "zero history across updates"),
1223
	OPT_INTEGER('F', "freq", &top.freq,
1224
		    "profile at this frequency"),
1225
	OPT_INTEGER('E', "entries", &top.print_entries,
1226
		    "display this many functions"),
1227
	OPT_BOOLEAN('U', "hide_user_symbols", &top.hide_user_symbols,
1228
		    "hide user symbols"),
1229 1230
	OPT_BOOLEAN(0, "tui", &top.use_tui, "Use the TUI interface"),
	OPT_BOOLEAN(0, "stdio", &top.use_stdio, "Use the stdio interface"),
1231
	OPT_INCR('v', "verbose", &verbose,
1232
		    "be more verbose (show counter open errors, etc)"),
1233 1234 1235 1236
	OPT_STRING('s', "sort", &sort_order, "key[,key2...]",
		   "sort by key(s): pid, comm, dso, symbol, parent"),
	OPT_BOOLEAN('n', "show-nr-samples", &symbol_conf.show_nr_samples,
		    "Show a column with the number of samples"),
1237
	OPT_CALLBACK_DEFAULT('G', "call-graph", &top, "output_type,min_percent, call_order",
1238 1239 1240
		     "Display callchains using output_type (graph, flat, fractal, or none), min percent threshold and callchain order. "
		     "Default: fractal,0.5,callee", &parse_callchain_opt,
		     callchain_default_opt),
1241 1242 1243 1244 1245 1246 1247 1248
	OPT_BOOLEAN(0, "show-total-period", &symbol_conf.show_total_period,
		    "Show a column with the sum of periods"),
	OPT_STRING(0, "dsos", &symbol_conf.dso_list_str, "dso[,dso...]",
		   "only consider symbols in these dsos"),
	OPT_STRING(0, "comms", &symbol_conf.comm_list_str, "comm[,comm...]",
		   "only consider symbols in these comms"),
	OPT_STRING(0, "symbols", &symbol_conf.sym_list_str, "symbol[,symbol...]",
		   "only consider these symbols"),
1249 1250 1251 1252 1253 1254
	OPT_BOOLEAN(0, "source", &symbol_conf.annotate_src,
		    "Interleave source code with assembly code (default)"),
	OPT_BOOLEAN(0, "asm-raw", &symbol_conf.annotate_asm_raw,
		    "Display raw encoding of assembly instructions (default)"),
	OPT_STRING('M', "disassembler-style", &disassembler_style, "disassembler style",
		   "Specify disassembler style (e.g. -M intel for intel syntax)"),
1255
	OPT_STRING('u', "uid", &top.target.uid_str, "user", "user to profile"),
1256
	OPT_END()
1257
	};
1258 1259 1260 1261
	const char * const top_usage[] = {
		"perf top [<options>]",
		NULL
	};
1262

1263 1264
	top.evlist = perf_evlist__new(NULL, NULL);
	if (top.evlist == NULL)
1265 1266
		return -ENOMEM;

1267
	symbol_conf.exclude_other = false;
1268 1269 1270 1271 1272

	argc = parse_options(argc, argv, options, top_usage, 0);
	if (argc)
		usage_with_options(top_usage, options);

1273 1274 1275 1276 1277
	if (sort_order == default_sort_order)
		sort_order = "dso,symbol";

	setup_sorting(top_usage, options);

1278
	if (top.use_stdio)
1279
		use_browser = 0;
1280
	else if (top.use_tui)
1281 1282 1283 1284
		use_browser = 1;

	setup_browser(false);

1285 1286 1287 1288 1289 1290 1291 1292 1293
	status = perf_target__validate(&top.target);
	if (status) {
		perf_target__strerror(&top.target, status, errbuf, BUFSIZ);
		ui__warning("%s", errbuf);
	}

	status = perf_target__parse_uid(&top.target);
	if (status) {
		int saved_errno = errno;
1294

1295
		perf_target__strerror(&top.target, status, errbuf, BUFSIZ);
1296
		ui__error("%s", errbuf);
1297 1298

		status = -saved_errno;
1299
		goto out_delete_evlist;
1300
	}
1301

1302
	if (perf_target__none(&top.target))
1303 1304
		top.target.system_wide = true;

1305
	if (perf_evlist__create_maps(top.evlist, &top.target) < 0)
1306 1307
		usage_with_options(top_usage, options);

1308 1309
	if (!top.evlist->nr_entries &&
	    perf_evlist__add_default(top.evlist) < 0) {
1310
		ui__error("Not enough memory for event selector list\n");
1311 1312
		return -ENOMEM;
	}
1313

1314 1315
	symbol_conf.nr_events = top.evlist->nr_entries;

1316 1317
	if (top.delay_secs < 1)
		top.delay_secs = 1;
1318

1319 1320 1321
	/*
	 * User specified count overrides default frequency.
	 */
1322
	if (top.default_interval)
1323 1324
		top.freq = 0;
	else if (top.freq) {
1325
		top.default_interval = top.freq;
1326
	} else {
1327
		ui__error("frequency and count are zero, aborting\n");
1328 1329 1330
		exit(EXIT_FAILURE);
	}

1331
	list_for_each_entry(pos, &top.evlist->entries, node) {
1332 1333 1334
		/*
		 * Fill in the ones not specifically initialized via -c:
		 */
1335 1336
		if (!pos->attr.sample_period)
			pos->attr.sample_period = top.default_interval;
1337 1338
	}

1339
	top.sym_evsel = perf_evlist__first(top.evlist);
1340

1341
	symbol_conf.priv_size = sizeof(struct annotation);
1342 1343 1344 1345 1346

	symbol_conf.try_vmlinux_path = (symbol_conf.vmlinux_name == NULL);
	if (symbol__init() < 0)
		return -1;

1347 1348 1349 1350
	sort_entry__setup_elide(&sort_dso, symbol_conf.dso_list, "dso", stdout);
	sort_entry__setup_elide(&sort_comm, symbol_conf.comm_list, "comm", stdout);
	sort_entry__setup_elide(&sort_sym, symbol_conf.sym_list, "symbol", stdout);

1351 1352 1353 1354
	/*
	 * Avoid annotation data structures overhead when symbols aren't on the
	 * sort list.
	 */
1355
	top.sort_has_symbols = sort_sym.list.next != NULL;
1356

1357
	get_term_dimensions(&top.winsize);
1358
	if (top.print_entries == 0) {
1359 1360 1361 1362 1363 1364
		struct sigaction act = {
			.sa_sigaction = perf_top__sig_winch,
			.sa_flags     = SA_SIGINFO,
		};
		perf_top__update_print_entries(&top);
		sigaction(SIGWINCH, &act, NULL);
1365 1366
	}

1367
	status = __cmd_top(&top);
1368

1369
out_delete_evlist:
1370
	perf_evlist__delete(top.evlist);
1371 1372

	return status;
1373
}