builtin-top.c 34.4 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 30
#include "util/session.h"
#include "util/symbol.h"
31
#include "util/thread.h"
32
#include "util/thread_map.h"
33
#include "util/top.h"
34
#include "util/util.h"
35
#include <linux/rbtree.h>
36 37
#include "util/parse-options.h"
#include "util/parse-events.h"
38
#include "util/cpumap.h"
39
#include "util/xyarray.h"
40
#include "util/sort.h"
41
#include "util/intlist.h"
42

43 44
#include "util/debug.h"

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

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

54 55 56 57 58 59 60 61 62 63
#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>
64
#include <sys/utsname.h>
65 66 67 68 69
#include <sys/mman.h>

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

70
void get_term_dimensions(struct winsize *ws)
71
{
72 73 74 75 76 77 78 79 80 81
	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;
		}
82
	}
83 84 85 86
#ifdef TIOCGWINSZ
	if (ioctl(1, TIOCGWINSZ, ws) == 0 &&
	    ws->ws_row && ws->ws_col)
		return;
87
#endif
88 89
	ws->ws_row = 25;
	ws->ws_col = 80;
90 91
}

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

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

	get_term_dimensions(&top->winsize);
104 105 106 107 108 109 110
	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;
	}
111
	perf_top__update_print_entries(top);
112 113
}

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

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

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

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

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

143
	pthread_mutex_lock(&notes->lock);
144

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

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

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

163
static void __zero_source_counters(struct hist_entry *he)
164
{
165
	struct symbol *sym = he->ms.sym;
166
	symbol__annotate_zero_histograms(sym);
167 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
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;
}

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

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

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

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

215
	if (notes->src == NULL && symbol__alloc_hist(sym) < 0) {
216 217 218 219 220 221 222 223
		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);
224
	err = symbol__inc_addr_samples(sym, he->ms.map, counter, ip);
225

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

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

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

239
	if (!he)
240 241
		return;

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

	pthread_mutex_lock(&notes->lock);

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

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

253 254 255 256
	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);
257
	else
258
		symbol__annotate_decay_histogram(symbol, top->sym_evsel->idx);
259
	if (more != 0)
260
		printf("%d lines not displayed, maybe increase display entries [e]\n", more);
261 262
out_unlock:
	pthread_mutex_unlock(&notes->lock);
263
}
264 265 266

static const char		CONSOLE_CLEAR[] = "";

267 268 269
static struct hist_entry *perf_evsel__add_hist_entry(struct perf_evsel *evsel,
						     struct addr_location *al,
						     struct perf_sample *sample)
270
{
271 272 273 274 275 276 277 278
	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;
279
}
280

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

287
	puts(CONSOLE_CLEAR);
288

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

292
	perf_top__reset_sample_counters(top);
293

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

296 297 298 299
	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];
300 301
		color_fprintf(stdout, PERF_COLOR_RED,
			      "WARNING: LOST %d chunks, Check IO/CPU overload",
302
			      top->sym_evsel->hists.stats.nr_lost_warned);
303
		++printed;
304 305
	}

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

311 312 313 314 315 316 317
	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);
318
	putchar('\n');
319
	hists__fprintf(&top->sym_evsel->hists, false,
320
		       top->winsize.ws_row - 4 - printed, win_width, stdout);
321 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
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;
}

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

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

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

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

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

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

out_free:
	free(buf);
}

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

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

	fprintf(stdout, "\nMapped keys:\n");
409 410
	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);
411

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

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

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

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

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

	return 0;
453 454
}

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

461
		perf_top__print_mapped_keys(top);
462 463 464 465 466 467 468 469 470 471 472 473 474 475
		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);
476
		if (!perf_top__key_mapped(top, c))
477 478 479
			return;
	}

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

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

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

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

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

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

564 565 566 567 568 569 570 571 572 573
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);
574
	hists__decay_entries_threaded(&t->sym_evsel->hists,
575 576
				      t->hide_user_symbols,
				      t->hide_kernel_symbols);
577 578
}

579
static void *display_thread_tui(void *arg)
580
{
581
	struct perf_evsel *pos;
582
	struct perf_top *top = arg;
583 584
	const char *help = "For a higher level overview, try: perf top --sort comm,dso";

585
	perf_top__sort_new_samples(top);
586 587 588 589 590 591 592

	/*
	 * 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)
593
		pos->hists.uid_filter_str = top->target.uid_str;
594

595
	perf_evlist__tui_browse_hists(top->evlist, help,
596
				      perf_top__sort_new_samples,
597
				      top, top->delay_secs);
598

599 600 601 602 603
	exit_browser(0);
	exit(0);
	return NULL;
}

604
static void *display_thread(void *arg)
605
{
606
	struct pollfd stdin_poll = { .fd = 0, .events = POLLIN };
607
	struct termios tc, save;
608
	struct perf_top *top = arg;
609 610 611 612 613 614 615
	int delay_msecs, c;

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

617
	pthread__unblock_sigwinch();
618
repeat:
619
	delay_msecs = top->delay_secs * 1000;
620 621 622
	tcsetattr(0, TCSANOW, &tc);
	/* trash return*/
	getc(stdin);
623

624
	while (1) {
625
		perf_top__print_sym_table(top);
626 627 628 629 630 631 632 633 634 635 636 637 638 639 640 641
		/*
		 * 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:
642 643 644
	c = getc(stdin);
	tcsetattr(0, TCSAFLUSH, &save);

645
	perf_top__handle_keypress(top, c);
646
	goto repeat;
647 648 649 650

	return NULL;
}

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

667
static int symbol_filter(struct map *map __maybe_unused, struct symbol *sym)
668
{
669
	const char *name = sym->name;
670
	int i;
671

672 673 674 675 676 677 678
	/*
	 * ppc64 uses function descriptors and appends a '.' to the
	 * start of every instruction address. Remove it.
	 */
	if (name[0] == '.')
		name++;

679 680 681 682 683 684 685
	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"))
686 687
		return 1;

688 689
	for (i = 0; skip_symbols[i]; i++) {
		if (!strcmp(skip_symbols[i], name)) {
690
			sym->ignore = true;
691 692 693
			break;
		}
	}
694 695 696 697

	return 0;
}

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

710
	if (!machine && perf_guest) {
711 712 713 714 715 716 717 718 719 720
		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);
		}
721 722 723
		return;
	}

724 725 726 727 728 729
	if (!machine) {
		pr_err("%u unprocessable samples recorded.",
		       top->session->hists.stats.nr_unprocessable_samples++);
		return;
	}

730
	if (event->header.misc & PERF_RECORD_MISC_EXACT_IP)
731
		top->exact_samples++;
732

733
	if (perf_event__preprocess_sample(event, machine, &al, sample,
734
					  symbol_filter) < 0 ||
735
	    al.filtered)
736
		return;
737

738
	if (!top->kptr_restrict_warned &&
739 740 741 742 743 744 745 746 747 748
	    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);
749
		top->kptr_restrict_warned = true;
750 751
	}

752
	if (al.sym == NULL) {
753
		const char *msg = "Kernel samples will not be resolved.\n";
754 755 756 757 758 759 760 761 762 763 764
		/*
		 * 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 ;-)
		 */
765
		if (!top->kptr_restrict_warned && !top->vmlinux_warned &&
766
		    al.map == machine->vmlinux_maps[MAP__FUNCTION] &&
767
		    RB_EMPTY_ROOT(&al.map->dso->symbols[MAP__FUNCTION])) {
768 769 770 771 772 773 774 775 776 777
			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);
778
			top->vmlinux_warned = true;
779
		}
780 781
	}

782 783
	if (al.sym == NULL || !al.sym->ignore) {
		struct hist_entry *he;
784

785 786
		if ((sort__has_parent || symbol_conf.use_callchain) &&
		    sample->callchain) {
787 788 789 790
			err = machine__resolve_callchain(machine, evsel,
							 al.thread, sample,
							 &parent);

791 792 793 794
			if (err)
				return;
		}

795
		he = perf_evsel__add_hist_entry(evsel, &al, sample);
796 797 798
		if (he == NULL) {
			pr_err("Problem incrementing symbol period, skipping event\n");
			return;
799
		}
800

801
		if (symbol_conf.use_callchain) {
802
			err = callchain_append(he->callchain, &callchain_cursor,
803 804 805 806 807
					       sample->period);
			if (err)
				return;
		}

808 809
		if (top->sort_has_symbols)
			perf_top__record_precise_ip(top, he, evsel->idx, ip);
810
	}
811 812

	return;
813 814
}

815
static void perf_top__mmap_read_idx(struct perf_top *top, int idx)
816
{
817
	struct perf_sample sample;
818
	struct perf_evsel *evsel;
819
	struct perf_session *session = top->session;
820
	union perf_event *event;
821 822
	struct machine *machine;
	u8 origin;
823
	int ret;
824

825
	while ((event = perf_evlist__mmap_read(top->evlist, idx)) != NULL) {
826
		ret = perf_evlist__parse_sample(top->evlist, event, &sample);
827 828 829 830
		if (ret) {
			pr_err("Can't parse sample, err = %d\n", ret);
			continue;
		}
831

832
		evsel = perf_evlist__id2evsel(session->evlist, sample.id);
833 834
		assert(evsel != NULL);

835 836
		origin = event->header.misc & PERF_RECORD_MISC_CPUMODE_MASK;

837
		if (event->header.type == PERF_RECORD_SAMPLE)
838
			++top->samples;
839 840 841

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


869 870 871 872
		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) {
873
			hists__inc_nr_events(&evsel->hists, event->header.type);
874
			perf_event__process(&top->tool, event, &sample, machine);
875
		} else
876
			++session->hists.stats.nr_unknown_events;
877 878 879
	}
}

880
static void perf_top__mmap_read(struct perf_top *top)
881
{
882 883
	int i;

884 885
	for (i = 0; i < top->evlist->nr_mmaps; i++)
		perf_top__mmap_read_idx(top, i);
886 887
}

888
static void perf_top__start_counters(struct perf_top *top)
889
{
890
	struct perf_evsel *counter;
891
	struct perf_evlist *evlist = top->evlist;
892

893
	if (top->group)
894
		perf_evlist__set_leader(evlist);
895

896 897
	list_for_each_entry(counter, &evlist->entries, node) {
		struct perf_event_attr *attr = &counter->attr;
898

899 900
		attr->sample_type = PERF_SAMPLE_IP | PERF_SAMPLE_TID;

901
		if (top->freq) {
902 903
			attr->sample_type |= PERF_SAMPLE_PERIOD;
			attr->freq	  = 1;
904
			attr->sample_freq = top->freq;
905
		}
906

907 908 909 910 911
		if (evlist->nr_entries > 1) {
			attr->sample_type |= PERF_SAMPLE_ID;
			attr->read_format |= PERF_FORMAT_ID;
		}

912 913 914
		if (perf_target__has_cpu(&top->target))
			attr->sample_type |= PERF_SAMPLE_CPU;

915 916 917
		if (symbol_conf.use_callchain)
			attr->sample_type |= PERF_SAMPLE_CALLCHAIN;

918
		attr->mmap = 1;
919
		attr->comm = 1;
920
		attr->inherit = top->inherit;
921 922 923
fallback_missing_features:
		if (top->exclude_guest_missing)
			attr->exclude_guest = attr->exclude_host = 0;
924
retry_sample_id:
925
		attr->sample_id_all = top->sample_id_all_missing ? 0 : 1;
926
try_again:
927
		if (perf_evsel__open(counter, top->evlist->cpus,
928
				     top->evlist->threads) < 0) {
929 930
			int err = errno;

931
			if (err == EPERM || err == EACCES) {
932
				ui__error_paranoid();
933
				goto out_err;
934 935 936 937 938 939 940
			} 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;
941
				} else if (!top->sample_id_all_missing) {
942 943 944
					/*
					 * Old kernel, no attr->sample_id_type_all field
					 */
945
					top->sample_id_all_missing = true;
946 947
					goto retry_sample_id;
				}
948
			}
949 950 951 952 953
			/*
			 * If it's cycles then fall back to hrtimer
			 * based cpu-clock-tick sw counter, which
			 * is always available even if no PMU support:
			 */
954 955 956 957
			if ((err == ENOENT || err == ENXIO) &&
			    (attr->type == PERF_TYPE_HARDWARE) &&
			    (attr->config == PERF_COUNT_HW_CPU_CYCLES)) {

958
				if (verbose)
959 960
					ui__warning("Cycles event not supported,\n"
						    "trying to fall back to cpu-clock-ticks\n");
961 962 963

				attr->type = PERF_TYPE_SOFTWARE;
				attr->config = PERF_COUNT_SW_CPU_CLOCK;
964 965
				if (counter->name) {
					free(counter->name);
966
					counter->name = NULL;
967
				}
968 969
				goto try_again;
			}
970

971
			if (err == ENOENT) {
972
				ui__error("The %s event is not supported.\n",
973
					  perf_evsel__name(counter));
974
				goto out_err;
975
			} else if (err == EMFILE) {
976
				ui__error("Too many events are opened.\n"
977 978
					    "Try again after reducing the number of events\n");
				goto out_err;
979 980
			}

981
			ui__error("The sys_perf_event_open() syscall "
982 983 984 985 986
				    "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;
987
		}
988
	}
989

990
	if (perf_evlist__mmap(evlist, top->mmap_pages, false) < 0) {
991
		ui__error("Failed to mmap with %d (%s)\n",
992 993 994 995 996 997 998 999 1000
			    errno, strerror(errno));
		goto out_err;
	}

	return;

out_err:
	exit_browser(0);
	exit(0);
1001 1002
}

1003
static int perf_top__setup_sample_type(struct perf_top *top)
1004
{
1005
	if (!top->sort_has_symbols) {
1006
		if (symbol_conf.use_callchain) {
1007
			ui__error("Selected -g but \"sym\" not present in --sort/-s.");
1008 1009
			return -EINVAL;
		}
1010
	} else if (!top->dont_use_callchains && callchain_param.mode != CHAIN_NONE) {
1011
		if (callchain_register_param(&callchain_param) < 0) {
1012
			ui__error("Can't register callchain params.\n");
1013 1014 1015 1016 1017 1018 1019
			return -EINVAL;
		}
	}

	return 0;
}

1020
static int __cmd_top(struct perf_top *top)
1021 1022
{
	pthread_t thread;
1023
	int ret;
1024
	/*
1025 1026
	 * 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.
1027
	 */
1028 1029
	top->session = perf_session__new(NULL, O_WRONLY, false, false, NULL);
	if (top->session == NULL)
1030
		return -ENOMEM;
1031

1032
	ret = perf_top__setup_sample_type(top);
1033 1034 1035
	if (ret)
		goto out_delete;

1036
	if (perf_target__has_task(&top->target))
1037
		perf_event__synthesize_thread_map(&top->tool, top->evlist->threads,
1038
						  perf_event__process,
1039
						  &top->session->host_machine);
1040
	else
1041 1042 1043 1044
		perf_event__synthesize_threads(&top->tool, perf_event__process,
					       &top->session->host_machine);
	perf_top__start_counters(top);
	top->session->evlist = top->evlist;
1045
	perf_session__set_id_hdr_size(top->session);
1046

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

1050
	perf_top__mmap_read(top);
1051

1052
	if (pthread_create(&thread, NULL, (use_browser > 0 ? display_thread_tui :
1053
							    display_thread), top)) {
1054
		ui__error("Could not create display thread.\n");
1055 1056 1057
		exit(-1);
	}

1058
	if (top->realtime_prio) {
1059 1060
		struct sched_param param;

1061
		param.sched_priority = top->realtime_prio;
1062
		if (sched_setscheduler(0, SCHED_FIFO, &param)) {
1063
			ui__error("Could not set realtime priority.\n");
1064 1065 1066 1067 1068
			exit(-1);
		}
	}

	while (1) {
1069
		u64 hits = top->samples;
1070

1071
		perf_top__mmap_read(top);
1072

1073 1074
		if (hits == top->samples)
			ret = poll(top->evlist->pollfd, top->evlist->nr_fds, 100);
1075 1076
	}

1077
out_delete:
1078 1079
	perf_session__delete(top->session);
	top->session = NULL;
1080 1081 1082 1083 1084

	return 0;
}

static int
1085
parse_callchain_opt(const struct option *opt, const char *arg, int unset)
1086
{
1087
	struct perf_top *top = (struct perf_top *)opt->value;
1088 1089 1090 1091 1092 1093 1094
	char *tok, *tok2;
	char *endptr;

	/*
	 * --no-call-graph
	 */
	if (unset) {
1095
		top->dont_use_callchains = true;
1096 1097 1098 1099 1100 1101 1102 1103 1104 1105 1106 1107 1108 1109 1110 1111 1112 1113 1114 1115 1116 1117 1118 1119 1120 1121 1122
		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;
1123
	} else
1124 1125 1126 1127 1128 1129 1130 1131 1132 1133 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
		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;
	}
1159 1160
	return 0;
}
1161

1162
int cmd_top(int argc, const char **argv, const char *prefix __maybe_unused)
1163 1164
{
	struct perf_evsel *pos;
1165 1166
	int status;
	char errbuf[BUFSIZ];
1167 1168 1169
	struct perf_top top = {
		.count_filter	     = 5,
		.delay_secs	     = 2,
1170
		.freq		     = 4000, /* 4 KHz */
1171 1172
		.mmap_pages	     = 128,
		.sym_pcnt_filter     = 5,
N
Namhyung Kim 已提交
1173 1174 1175
		.target		     = {
			.uses_mmap   = true,
		},
1176 1177 1178
	};
	char callchain_default_opt[] = "fractal,0.5,callee";
	const struct option options[] = {
1179
	OPT_CALLBACK('e', "event", &top.evlist, "event",
1180
		     "event selector. use 'perf list' to list available events",
1181
		     parse_events_option),
1182
	OPT_INTEGER('c', "count", &top.default_interval,
1183
		    "event period to sample"),
1184
	OPT_STRING('p', "pid", &top.target.pid, "pid",
1185
		    "profile events on existing process id"),
1186
	OPT_STRING('t', "tid", &top.target.tid, "tid",
1187
		    "profile events on existing thread id"),
1188
	OPT_BOOLEAN('a', "all-cpus", &top.target.system_wide,
1189
			    "system-wide collection from all CPUs"),
1190
	OPT_STRING('C', "cpu", &top.target.cpu_list, "cpu",
1191
		    "list of cpus to monitor"),
1192 1193
	OPT_STRING('k', "vmlinux", &symbol_conf.vmlinux_name,
		   "file", "vmlinux pathname"),
1194
	OPT_BOOLEAN('K', "hide_kernel_symbols", &top.hide_kernel_symbols,
1195
		    "hide kernel symbols"),
1196 1197
	OPT_UINTEGER('m', "mmap-pages", &top.mmap_pages, "number of mmap data pages"),
	OPT_INTEGER('r', "realtime", &top.realtime_prio,
1198
		    "collect data with this RT SCHED_FIFO priority"),
1199
	OPT_INTEGER('d', "delay", &top.delay_secs,
1200
		    "number of seconds to delay between refreshes"),
1201
	OPT_BOOLEAN('D', "dump-symtab", &top.dump_symtab,
1202
			    "dump the symbol table used for profiling"),
1203
	OPT_INTEGER('f', "count-filter", &top.count_filter,
1204
		    "only display functions with more events than this"),
1205
	OPT_BOOLEAN('g', "group", &top.group,
1206
			    "put the counters into a counter group"),
1207
	OPT_BOOLEAN('i', "inherit", &top.inherit,
1208
		    "child tasks inherit counters"),
1209
	OPT_STRING(0, "sym-annotate", &top.sym_filter, "symbol name",
1210
		    "symbol to annotate"),
1211
	OPT_BOOLEAN('z', "zero", &top.zero,
1212
		    "zero history across updates"),
1213
	OPT_INTEGER('F', "freq", &top.freq,
1214
		    "profile at this frequency"),
1215
	OPT_INTEGER('E', "entries", &top.print_entries,
1216
		    "display this many functions"),
1217
	OPT_BOOLEAN('U', "hide_user_symbols", &top.hide_user_symbols,
1218
		    "hide user symbols"),
1219 1220
	OPT_BOOLEAN(0, "tui", &top.use_tui, "Use the TUI interface"),
	OPT_BOOLEAN(0, "stdio", &top.use_stdio, "Use the stdio interface"),
1221
	OPT_INCR('v', "verbose", &verbose,
1222
		    "be more verbose (show counter open errors, etc)"),
1223 1224 1225 1226
	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"),
1227
	OPT_CALLBACK_DEFAULT('G', "call-graph", &top, "output_type,min_percent, call_order",
1228 1229 1230
		     "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),
1231 1232 1233 1234 1235 1236 1237 1238
	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"),
1239 1240 1241 1242 1243 1244
	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)"),
1245
	OPT_STRING('u', "uid", &top.target.uid_str, "user", "user to profile"),
1246
	OPT_END()
1247
	};
1248 1249 1250 1251
	const char * const top_usage[] = {
		"perf top [<options>]",
		NULL
	};
1252

1253 1254
	top.evlist = perf_evlist__new(NULL, NULL);
	if (top.evlist == NULL)
1255 1256
		return -ENOMEM;

1257
	symbol_conf.exclude_other = false;
1258 1259 1260 1261 1262

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

1263 1264 1265 1266 1267
	if (sort_order == default_sort_order)
		sort_order = "dso,symbol";

	setup_sorting(top_usage, options);

1268
	if (top.use_stdio)
1269
		use_browser = 0;
1270
	else if (top.use_tui)
1271 1272 1273 1274
		use_browser = 1;

	setup_browser(false);

1275 1276 1277 1278 1279 1280 1281 1282 1283
	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;
1284

1285
		perf_target__strerror(&top.target, status, errbuf, BUFSIZ);
1286
		ui__error("%s", errbuf);
1287 1288

		status = -saved_errno;
1289
		goto out_delete_evlist;
1290
	}
1291

1292
	if (perf_target__none(&top.target))
1293 1294
		top.target.system_wide = true;

1295
	if (perf_evlist__create_maps(top.evlist, &top.target) < 0)
1296 1297
		usage_with_options(top_usage, options);

1298 1299
	if (!top.evlist->nr_entries &&
	    perf_evlist__add_default(top.evlist) < 0) {
1300
		ui__error("Not enough memory for event selector list\n");
1301 1302
		return -ENOMEM;
	}
1303

1304 1305
	symbol_conf.nr_events = top.evlist->nr_entries;

1306 1307
	if (top.delay_secs < 1)
		top.delay_secs = 1;
1308

1309 1310 1311
	/*
	 * User specified count overrides default frequency.
	 */
1312
	if (top.default_interval)
1313 1314
		top.freq = 0;
	else if (top.freq) {
1315
		top.default_interval = top.freq;
1316
	} else {
1317
		ui__error("frequency and count are zero, aborting\n");
1318 1319 1320
		exit(EXIT_FAILURE);
	}

1321
	list_for_each_entry(pos, &top.evlist->entries, node) {
1322 1323 1324
		/*
		 * Fill in the ones not specifically initialized via -c:
		 */
1325 1326
		if (!pos->attr.sample_period)
			pos->attr.sample_period = top.default_interval;
1327 1328
	}

1329
	top.sym_evsel = perf_evlist__first(top.evlist);
1330

1331
	symbol_conf.priv_size = sizeof(struct annotation);
1332 1333 1334 1335 1336

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

1337 1338 1339 1340
	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);

1341 1342 1343 1344
	/*
	 * Avoid annotation data structures overhead when symbols aren't on the
	 * sort list.
	 */
1345
	top.sort_has_symbols = sort_sym.list.next != NULL;
1346

1347
	get_term_dimensions(&top.winsize);
1348
	if (top.print_entries == 0) {
1349 1350 1351 1352 1353 1354
		struct sigaction act = {
			.sa_sigaction = perf_top__sig_winch,
			.sa_flags     = SA_SIGINFO,
		};
		perf_top__update_print_entries(&top);
		sigaction(SIGWINCH, &act, NULL);
1355 1356
	}

1357
	status = __cmd_top(&top);
1358

1359
out_delete_evlist:
1360
	perf_evlist__delete(top.evlist);
1361 1362

	return status;
1363
}