blob: 15d89bede2fbd872499b4d3cffd30f9a2f3604f1 [file] [log] [blame]
Ingo Molnar07800602009-04-20 15:00:56 +02001/*
Ingo Molnarbf9e1872009-06-02 23:37:05 +02002 * builtin-top.c
3 *
4 * Builtin top command: Display a continuously updated profile of
5 * any workload, CPU or specific PID.
6 *
7 * Copyright (C) 2008, Red Hat Inc, Ingo Molnar <mingo@redhat.com>
8 *
9 * Improvements and fixes by:
10 *
11 * Arjan van de Ven <arjan@linux.intel.com>
12 * Yanmin Zhang <yanmin.zhang@intel.com>
13 * Wu Fengguang <fengguang.wu@intel.com>
14 * Mike Galbraith <efault@gmx.de>
15 * Paul Mackerras <paulus@samba.org>
16 *
17 * Released under the GPL v2. (and only v2, not any later version)
Ingo Molnar07800602009-04-20 15:00:56 +020018 */
Ingo Molnarbf9e1872009-06-02 23:37:05 +020019#include "builtin.h"
Ingo Molnar07800602009-04-20 15:00:56 +020020
Peter Zijlstra1a482f32009-05-23 18:28:58 +020021#include "perf.h"
Ingo Molnarbf9e1872009-06-02 23:37:05 +020022
Ingo Molnar8fc03212009-06-04 15:19:47 +020023#include "util/color.h"
Arnaldo Carvalho de Melo361c99a2011-01-11 20:56:53 -020024#include "util/evlist.h"
Arnaldo Carvalho de Melo69aad6f2011-01-03 16:39:04 -020025#include "util/evsel.h"
Arnaldo Carvalho de Melob3165f42009-12-13 19:50:28 -020026#include "util/session.h"
27#include "util/symbol.h"
Arnaldo Carvalho de Melo439d4732009-10-02 03:29:58 -030028#include "util/thread.h"
Ingo Molnar148be2c2009-04-27 08:02:14 +020029#include "util/util.h"
Arnaldo Carvalho de Melo43cbcd82009-07-01 12:28:37 -030030#include <linux/rbtree.h>
Ingo Molnarb456bae2009-05-26 09:17:18 +020031#include "util/parse-options.h"
32#include "util/parse-events.h"
Paul Mackerrasa12b51c2010-03-10 20:36:09 +110033#include "util/cpumap.h"
Arnaldo Carvalho de Melo69aad6f2011-01-03 16:39:04 -020034#include "util/xyarray.h"
Ingo Molnar07800602009-04-20 15:00:56 +020035
Frederic Weisbecker8f28827a2009-08-16 22:05:48 +020036#include "util/debug.h"
37
Ingo Molnar07800602009-04-20 15:00:56 +020038#include <assert.h>
39#include <fcntl.h>
Ingo Molnar0e9b20b2009-05-26 09:17:18 +020040
Ingo Molnar07800602009-04-20 15:00:56 +020041#include <stdio.h>
Mike Galbraith923c42c2009-07-22 20:36:03 +020042#include <termios.h>
43#include <unistd.h>
Arnaldo Carvalho de Melo9486aa32011-01-22 20:37:02 -020044#include <inttypes.h>
Ingo Molnar0e9b20b2009-05-26 09:17:18 +020045
Ingo Molnar07800602009-04-20 15:00:56 +020046#include <errno.h>
Ingo Molnar07800602009-04-20 15:00:56 +020047#include <time.h>
48#include <sched.h>
49#include <pthread.h>
50
51#include <sys/syscall.h>
52#include <sys/ioctl.h>
53#include <sys/poll.h>
54#include <sys/prctl.h>
55#include <sys/wait.h>
56#include <sys/uio.h>
57#include <sys/mman.h>
58
59#include <linux/unistd.h>
60#include <linux/types.h>
61
Arnaldo Carvalho de Melo69aad6f2011-01-03 16:39:04 -020062#define FD(e, x, y) (*(int *)xyarray__entry(e->fd, x, y))
Ingo Molnara21ca2c2009-06-06 09:58:57 +020063
Arnaldo Carvalho de Melo361c99a2011-01-11 20:56:53 -020064struct perf_evlist *evsel_list;
65
Ian Munsiec0555642010-04-13 18:37:33 +100066static bool system_wide = false;
Ingo Molnar07800602009-04-20 15:00:56 +020067
Mike Galbraith7e4ff9e2009-10-12 07:56:03 +020068static int default_interval = 0;
Ingo Molnar07800602009-04-20 15:00:56 +020069
Ingo Molnar42e59d72009-10-06 15:14:21 +020070static int count_filter = 5;
Arnaldo Carvalho de Melo3b6ed982009-11-16 19:30:27 -020071static int print_entries;
Ingo Molnar07800602009-04-20 15:00:56 +020072
Ingo Molnar42e59d72009-10-06 15:14:21 +020073static int target_pid = -1;
Zhang, Yanmind6d901c2010-03-18 11:36:05 -030074static int target_tid = -1;
Arnaldo Carvalho de Melo5c98d4662011-01-03 17:53:33 -020075static struct thread_map *threads;
Ian Munsiec0555642010-04-13 18:37:33 +100076static bool inherit = false;
Arnaldo Carvalho de Melo60d567e2011-01-03 17:49:48 -020077static struct cpu_map *cpus;
Arnaldo Carvalho de Melo19679362010-05-17 15:39:16 -030078static int realtime_prio = 0;
Ian Munsiec0555642010-04-13 18:37:33 +100079static bool group = false;
Ingo Molnar07800602009-04-20 15:00:56 +020080static unsigned int page_size;
Ingo Molnar42e59d72009-10-06 15:14:21 +020081static unsigned int mmap_pages = 16;
82static int freq = 1000; /* 1 KHz */
Ingo Molnar07800602009-04-20 15:00:56 +020083
Ingo Molnar42e59d72009-10-06 15:14:21 +020084static int delay_secs = 2;
Ian Munsiec0555642010-04-13 18:37:33 +100085static bool zero = false;
86static bool dump_symtab = false;
Ingo Molnar07800602009-04-20 15:00:56 +020087
Arnaldo Carvalho de Melo8ffcda12009-11-16 21:45:24 -020088static bool hide_kernel_symbols = false;
89static bool hide_user_symbols = false;
Arnaldo Carvalho de Melo13cc5072009-11-17 15:40:54 -020090static struct winsize winsize;
Arnaldo Carvalho de Melo8ffcda12009-11-16 21:45:24 -020091
Ingo Molnar07800602009-04-20 15:00:56 +020092/*
Mike Galbraith923c42c2009-07-22 20:36:03 +020093 * Source
94 */
95
96struct source_line {
97 u64 eip;
98 unsigned long count[MAX_COUNTERS];
99 char *line;
100 struct source_line *next;
101};
102
Arnaldo Carvalho de Meloedb7c602010-05-17 16:22:41 -0300103static const char *sym_filter = NULL;
Ingo Molnar42e59d72009-10-06 15:14:21 +0200104struct sym_entry *sym_filter_entry = NULL;
Kirill Smelkov6cff0e82010-02-03 16:52:08 -0200105struct sym_entry *sym_filter_entry_sched = NULL;
Ingo Molnar42e59d72009-10-06 15:14:21 +0200106static int sym_pcnt_filter = 5;
107static int sym_counter = 0;
Arnaldo Carvalho de Melo69aad6f2011-01-03 16:39:04 -0200108static struct perf_evsel *sym_evsel = NULL;
Ingo Molnar42e59d72009-10-06 15:14:21 +0200109static int display_weighted = -1;
Stephane Eranianc45c6ea2010-05-28 12:00:01 +0200110static const char *cpu_list;
Mike Galbraith923c42c2009-07-22 20:36:03 +0200111
112/*
Ingo Molnar07800602009-04-20 15:00:56 +0200113 * Symbols
114 */
115
Arnaldo Carvalho de Melob2698762009-11-17 18:38:02 -0200116struct sym_entry_source {
117 struct source_line *source;
118 struct source_line *lines;
119 struct source_line **lines_tail;
120 pthread_mutex_t lock;
121};
122
Ingo Molnar07800602009-04-20 15:00:56 +0200123struct sym_entry {
Arnaldo Carvalho de Melode046872009-05-28 14:55:41 -0300124 struct rb_node rb_node;
125 struct list_head node;
Arnaldo Carvalho de Meloc44613a2009-05-29 17:03:07 -0300126 unsigned long snap_count;
127 double weight;
Ingo Molnar07800602009-04-20 15:00:56 +0200128 int skip;
Arnaldo Carvalho de Melo13cc5072009-11-17 15:40:54 -0200129 u16 name_len;
Arnaldo Carvalho de Melo8ffcda12009-11-16 21:45:24 -0200130 u8 origin;
Arnaldo Carvalho de Melo439d4732009-10-02 03:29:58 -0300131 struct map *map;
Arnaldo Carvalho de Melob2698762009-11-17 18:38:02 -0200132 struct sym_entry_source *src;
Arnaldo Carvalho de Melo5a8e5a32009-11-17 18:38:01 -0200133 unsigned long count[0];
Ingo Molnar07800602009-04-20 15:00:56 +0200134};
135
Mike Galbraith923c42c2009-07-22 20:36:03 +0200136/*
137 * Source functions
138 */
139
Arnaldo Carvalho de Melo51a472d2009-11-17 18:38:00 -0200140static inline struct symbol *sym_entry__symbol(struct sym_entry *self)
141{
Arnaldo Carvalho de Melob32d1332009-11-24 12:05:15 -0200142 return ((void *)self) + symbol_conf.priv_size;
Arnaldo Carvalho de Melo51a472d2009-11-17 18:38:00 -0200143}
144
Arnaldo Carvalho de Melo895f0ed2010-03-11 20:12:41 -0300145void get_term_dimensions(struct winsize *ws)
Arnaldo Carvalho de Melo3b6ed982009-11-16 19:30:27 -0200146{
Arnaldo Carvalho de Melo13cc5072009-11-17 15:40:54 -0200147 char *s = getenv("LINES");
Arnaldo Carvalho de Melo3b6ed982009-11-16 19:30:27 -0200148
Arnaldo Carvalho de Melo13cc5072009-11-17 15:40:54 -0200149 if (s != NULL) {
150 ws->ws_row = atoi(s);
151 s = getenv("COLUMNS");
152 if (s != NULL) {
153 ws->ws_col = atoi(s);
154 if (ws->ws_row && ws->ws_col)
155 return;
156 }
Arnaldo Carvalho de Melo3b6ed982009-11-16 19:30:27 -0200157 }
Arnaldo Carvalho de Melo13cc5072009-11-17 15:40:54 -0200158#ifdef TIOCGWINSZ
159 if (ioctl(1, TIOCGWINSZ, ws) == 0 &&
160 ws->ws_row && ws->ws_col)
161 return;
Arnaldo Carvalho de Melo3b6ed982009-11-16 19:30:27 -0200162#endif
Arnaldo Carvalho de Melo13cc5072009-11-17 15:40:54 -0200163 ws->ws_row = 25;
164 ws->ws_col = 80;
Arnaldo Carvalho de Melo3b6ed982009-11-16 19:30:27 -0200165}
166
Arnaldo Carvalho de Melo13cc5072009-11-17 15:40:54 -0200167static void update_print_entries(struct winsize *ws)
Arnaldo Carvalho de Melo3b6ed982009-11-16 19:30:27 -0200168{
Arnaldo Carvalho de Melo13cc5072009-11-17 15:40:54 -0200169 print_entries = ws->ws_row;
170
Arnaldo Carvalho de Melo3b6ed982009-11-16 19:30:27 -0200171 if (print_entries > 9)
172 print_entries -= 9;
173}
174
175static void sig_winch_handler(int sig __used)
176{
Arnaldo Carvalho de Melo13cc5072009-11-17 15:40:54 -0200177 get_term_dimensions(&winsize);
178 update_print_entries(&winsize);
Arnaldo Carvalho de Melo3b6ed982009-11-16 19:30:27 -0200179}
180
Arnaldo Carvalho de Melob0a9ab62010-03-15 11:46:58 -0300181static int parse_source(struct sym_entry *syme)
Mike Galbraith923c42c2009-07-22 20:36:03 +0200182{
183 struct symbol *sym;
Arnaldo Carvalho de Melob2698762009-11-17 18:38:02 -0200184 struct sym_entry_source *source;
Arnaldo Carvalho de Melo439d4732009-10-02 03:29:58 -0300185 struct map *map;
Mike Galbraith923c42c2009-07-22 20:36:03 +0200186 FILE *file;
Ingo Molnar83a09442009-08-15 12:26:57 +0200187 char command[PATH_MAX*2];
Arnaldo Carvalho de Melo439d4732009-10-02 03:29:58 -0300188 const char *path;
189 u64 len;
Mike Galbraith923c42c2009-07-22 20:36:03 +0200190
191 if (!syme)
Arnaldo Carvalho de Melob0a9ab62010-03-15 11:46:58 -0300192 return -1;
193
194 sym = sym_entry__symbol(syme);
195 map = syme->map;
196
197 /*
198 * We can't annotate with just /proc/kallsyms
199 */
200 if (map->dso->origin == DSO__ORIG_KERNEL)
201 return -1;
Mike Galbraith923c42c2009-07-22 20:36:03 +0200202
Arnaldo Carvalho de Melob2698762009-11-17 18:38:02 -0200203 if (syme->src == NULL) {
Arnaldo Carvalho de Melo36479482009-11-24 12:05:16 -0200204 syme->src = zalloc(sizeof(*source));
Arnaldo Carvalho de Melob2698762009-11-17 18:38:02 -0200205 if (syme->src == NULL)
Arnaldo Carvalho de Melob0a9ab62010-03-15 11:46:58 -0300206 return -1;
Arnaldo Carvalho de Melob2698762009-11-17 18:38:02 -0200207 pthread_mutex_init(&syme->src->lock, NULL);
208 }
209
210 source = syme->src;
211
212 if (source->lines) {
213 pthread_mutex_lock(&source->lock);
Mike Galbraith923c42c2009-07-22 20:36:03 +0200214 goto out_assign;
215 }
Arnaldo Carvalho de Melo439d4732009-10-02 03:29:58 -0300216 path = map->dso->long_name;
Mike Galbraith923c42c2009-07-22 20:36:03 +0200217
Mike Galbraith923c42c2009-07-22 20:36:03 +0200218 len = sym->end - sym->start;
219
Arnaldo Carvalho de Melo439d4732009-10-02 03:29:58 -0300220 sprintf(command,
Arnaldo Carvalho de Melo9486aa32011-01-22 20:37:02 -0200221 "objdump --start-address=%#0*" PRIx64 " --stop-address=%#0*" PRIx64 " -dS %s",
Arnaldo Carvalho de Melo5f485362010-02-07 11:46:16 -0200222 BITS_PER_LONG / 4, map__rip_2objdump(map, sym->start),
223 BITS_PER_LONG / 4, map__rip_2objdump(map, sym->end), path);
Mike Galbraith923c42c2009-07-22 20:36:03 +0200224
225 file = popen(command, "r");
226 if (!file)
Arnaldo Carvalho de Melob0a9ab62010-03-15 11:46:58 -0300227 return -1;
Mike Galbraith923c42c2009-07-22 20:36:03 +0200228
Arnaldo Carvalho de Melob2698762009-11-17 18:38:02 -0200229 pthread_mutex_lock(&source->lock);
230 source->lines_tail = &source->lines;
Mike Galbraith923c42c2009-07-22 20:36:03 +0200231 while (!feof(file)) {
232 struct source_line *src;
233 size_t dummy = 0;
Kirill Smelkovee11b902010-02-07 11:46:15 -0200234 char *c, *sep;
Mike Galbraith923c42c2009-07-22 20:36:03 +0200235
236 src = malloc(sizeof(struct source_line));
237 assert(src != NULL);
238 memset(src, 0, sizeof(struct source_line));
239
240 if (getline(&src->line, &dummy, file) < 0)
241 break;
242 if (!src->line)
243 break;
244
245 c = strchr(src->line, '\n');
246 if (c)
247 *c = 0;
248
249 src->next = NULL;
Arnaldo Carvalho de Melob2698762009-11-17 18:38:02 -0200250 *source->lines_tail = src;
251 source->lines_tail = &src->next;
Mike Galbraith923c42c2009-07-22 20:36:03 +0200252
Kirill Smelkovee11b902010-02-07 11:46:15 -0200253 src->eip = strtoull(src->line, &sep, 16);
254 if (*sep == ':')
255 src->eip = map__objdump_2ip(map, src->eip);
256 else /* this line has no ip info (e.g. source line) */
257 src->eip = 0;
Mike Galbraith923c42c2009-07-22 20:36:03 +0200258 }
259 pclose(file);
260out_assign:
261 sym_filter_entry = syme;
Arnaldo Carvalho de Melob2698762009-11-17 18:38:02 -0200262 pthread_mutex_unlock(&source->lock);
Arnaldo Carvalho de Melob0a9ab62010-03-15 11:46:58 -0300263 return 0;
Mike Galbraith923c42c2009-07-22 20:36:03 +0200264}
265
266static void __zero_source_counters(struct sym_entry *syme)
267{
268 int i;
269 struct source_line *line;
270
Arnaldo Carvalho de Melob2698762009-11-17 18:38:02 -0200271 line = syme->src->lines;
Mike Galbraith923c42c2009-07-22 20:36:03 +0200272 while (line) {
Arnaldo Carvalho de Melo361c99a2011-01-11 20:56:53 -0200273 for (i = 0; i < evsel_list->nr_entries; i++)
Mike Galbraith923c42c2009-07-22 20:36:03 +0200274 line->count[i] = 0;
275 line = line->next;
276 }
277}
278
279static void record_precise_ip(struct sym_entry *syme, int counter, u64 ip)
280{
281 struct source_line *line;
282
283 if (syme != sym_filter_entry)
284 return;
285
Arnaldo Carvalho de Melob2698762009-11-17 18:38:02 -0200286 if (pthread_mutex_trylock(&syme->src->lock))
Mike Galbraith923c42c2009-07-22 20:36:03 +0200287 return;
288
Arnaldo Carvalho de Melob2698762009-11-17 18:38:02 -0200289 if (syme->src == NULL || syme->src->source == NULL)
Mike Galbraith923c42c2009-07-22 20:36:03 +0200290 goto out_unlock;
291
Arnaldo Carvalho de Melob2698762009-11-17 18:38:02 -0200292 for (line = syme->src->lines; line; line = line->next) {
Kirill Smelkovee11b902010-02-07 11:46:15 -0200293 /* skip lines without IP info */
294 if (line->eip == 0)
295 continue;
Mike Galbraith923c42c2009-07-22 20:36:03 +0200296 if (line->eip == ip) {
297 line->count[counter]++;
298 break;
299 }
300 if (line->eip > ip)
301 break;
302 }
303out_unlock:
Arnaldo Carvalho de Melob2698762009-11-17 18:38:02 -0200304 pthread_mutex_unlock(&syme->src->lock);
Mike Galbraith923c42c2009-07-22 20:36:03 +0200305}
306
Arnaldo Carvalho de Meloc7ad21a2010-02-22 16:14:22 -0300307#define PATTERN_LEN (BITS_PER_LONG / 4 + 2)
308
Mike Galbraith923c42c2009-07-22 20:36:03 +0200309static void lookup_sym_source(struct sym_entry *syme)
310{
Arnaldo Carvalho de Melo51a472d2009-11-17 18:38:00 -0200311 struct symbol *symbol = sym_entry__symbol(syme);
Mike Galbraith923c42c2009-07-22 20:36:03 +0200312 struct source_line *line;
Arnaldo Carvalho de Meloc7ad21a2010-02-22 16:14:22 -0300313 char pattern[PATTERN_LEN + 1];
Mike Galbraith923c42c2009-07-22 20:36:03 +0200314
Arnaldo Carvalho de Melo9486aa32011-01-22 20:37:02 -0200315 sprintf(pattern, "%0*" PRIx64 " <", BITS_PER_LONG / 4,
Arnaldo Carvalho de Melo5f485362010-02-07 11:46:16 -0200316 map__rip_2objdump(syme->map, symbol->start));
Mike Galbraith923c42c2009-07-22 20:36:03 +0200317
Arnaldo Carvalho de Melob2698762009-11-17 18:38:02 -0200318 pthread_mutex_lock(&syme->src->lock);
319 for (line = syme->src->lines; line; line = line->next) {
Arnaldo Carvalho de Meloc7ad21a2010-02-22 16:14:22 -0300320 if (memcmp(line->line, pattern, PATTERN_LEN) == 0) {
Arnaldo Carvalho de Melob2698762009-11-17 18:38:02 -0200321 syme->src->source = line;
Mike Galbraith923c42c2009-07-22 20:36:03 +0200322 break;
323 }
324 }
Arnaldo Carvalho de Melob2698762009-11-17 18:38:02 -0200325 pthread_mutex_unlock(&syme->src->lock);
Mike Galbraith923c42c2009-07-22 20:36:03 +0200326}
327
328static void show_lines(struct source_line *queue, int count, int total)
329{
330 int i;
331 struct source_line *line;
332
333 line = queue;
334 for (i = 0; i < count; i++) {
335 float pcnt = 100.0*(float)line->count[sym_counter]/(float)total;
336
337 printf("%8li %4.1f%%\t%s\n", line->count[sym_counter], pcnt, line->line);
338 line = line->next;
339 }
340}
341
342#define TRACE_COUNT 3
343
344static void show_details(struct sym_entry *syme)
345{
346 struct symbol *symbol;
347 struct source_line *line;
348 struct source_line *line_queue = NULL;
349 int displayed = 0;
350 int line_queue_count = 0, total = 0, more = 0;
351
352 if (!syme)
353 return;
354
Arnaldo Carvalho de Melob2698762009-11-17 18:38:02 -0200355 if (!syme->src->source)
Mike Galbraith923c42c2009-07-22 20:36:03 +0200356 lookup_sym_source(syme);
357
Arnaldo Carvalho de Melob2698762009-11-17 18:38:02 -0200358 if (!syme->src->source)
Mike Galbraith923c42c2009-07-22 20:36:03 +0200359 return;
360
Arnaldo Carvalho de Melo51a472d2009-11-17 18:38:00 -0200361 symbol = sym_entry__symbol(syme);
Arnaldo Carvalho de Melo69aad6f2011-01-03 16:39:04 -0200362 printf("Showing %s for %s\n", event_name(sym_evsel), symbol->name);
Mike Galbraith923c42c2009-07-22 20:36:03 +0200363 printf(" Events Pcnt (>=%d%%)\n", sym_pcnt_filter);
364
Arnaldo Carvalho de Melob2698762009-11-17 18:38:02 -0200365 pthread_mutex_lock(&syme->src->lock);
366 line = syme->src->source;
Mike Galbraith923c42c2009-07-22 20:36:03 +0200367 while (line) {
368 total += line->count[sym_counter];
369 line = line->next;
370 }
371
Arnaldo Carvalho de Melob2698762009-11-17 18:38:02 -0200372 line = syme->src->source;
Mike Galbraith923c42c2009-07-22 20:36:03 +0200373 while (line) {
374 float pcnt = 0.0;
375
376 if (!line_queue_count)
377 line_queue = line;
378 line_queue_count++;
379
380 if (line->count[sym_counter])
381 pcnt = 100.0 * line->count[sym_counter] / (float)total;
382 if (pcnt >= (float)sym_pcnt_filter) {
383 if (displayed <= print_entries)
384 show_lines(line_queue, line_queue_count, total);
385 else more++;
386 displayed += line_queue_count;
387 line_queue_count = 0;
388 line_queue = NULL;
389 } else if (line_queue_count > TRACE_COUNT) {
390 line_queue = line_queue->next;
391 line_queue_count--;
392 }
393
394 line->count[sym_counter] = zero ? 0 : line->count[sym_counter] * 7 / 8;
395 line = line->next;
396 }
Arnaldo Carvalho de Melob2698762009-11-17 18:38:02 -0200397 pthread_mutex_unlock(&syme->src->lock);
Mike Galbraith923c42c2009-07-22 20:36:03 +0200398 if (more)
399 printf("%d lines not displayed, maybe increase display entries [e]\n", more);
400}
Ingo Molnar07800602009-04-20 15:00:56 +0200401
Arnaldo Carvalho de Melode046872009-05-28 14:55:41 -0300402/*
Arnaldo Carvalho de Melo5b2bb752009-10-26 19:23:19 -0200403 * Symbols will be added here in event__process_sample and will get out
Arnaldo Carvalho de Melode046872009-05-28 14:55:41 -0300404 * after decayed.
405 */
406static LIST_HEAD(active_symbols);
Arnaldo Carvalho de Meloc44613a2009-05-29 17:03:07 -0300407static pthread_mutex_t active_symbols_lock = PTHREAD_MUTEX_INITIALIZER;
Ingo Molnar07800602009-04-20 15:00:56 +0200408
Ingo Molnar07800602009-04-20 15:00:56 +0200409/*
410 * Ordering weight: count-1 * count-2 * ... / count-n
411 */
412static double sym_weight(const struct sym_entry *sym)
413{
Arnaldo Carvalho de Meloc44613a2009-05-29 17:03:07 -0300414 double weight = sym->snap_count;
Ingo Molnar07800602009-04-20 15:00:56 +0200415 int counter;
416
Mike Galbraith46ab9762009-07-24 10:09:50 +0200417 if (!display_weighted)
418 return weight;
419
Arnaldo Carvalho de Melo361c99a2011-01-11 20:56:53 -0200420 for (counter = 1; counter < evsel_list->nr_entries - 1; counter++)
Ingo Molnar07800602009-04-20 15:00:56 +0200421 weight *= sym->count[counter];
422
423 weight /= (sym->count[counter] + 1);
424
425 return weight;
426}
427
Ingo Molnar2debbc82009-06-05 14:29:10 +0200428static long samples;
Zhang, Yanmina1645ce2010-04-19 13:32:50 +0800429static long kernel_samples, us_samples;
Peter Zijlstra1676b8a2010-03-04 14:19:36 +0100430static long exact_samples;
Zhang, Yanmina1645ce2010-04-19 13:32:50 +0800431static long guest_us_samples, guest_kernel_samples;
Ingo Molnar07800602009-04-20 15:00:56 +0200432static const char CONSOLE_CLEAR[] = "";
433
Arnaldo Carvalho de Meloc44613a2009-05-29 17:03:07 -0300434static void __list_insert_active_sym(struct sym_entry *syme)
Arnaldo Carvalho de Melode046872009-05-28 14:55:41 -0300435{
436 list_add(&syme->node, &active_symbols);
437}
438
Arnaldo Carvalho de Meloc44613a2009-05-29 17:03:07 -0300439static void list_remove_active_sym(struct sym_entry *syme)
440{
441 pthread_mutex_lock(&active_symbols_lock);
442 list_del_init(&syme->node);
443 pthread_mutex_unlock(&active_symbols_lock);
444}
445
Arnaldo Carvalho de Melode046872009-05-28 14:55:41 -0300446static void rb_insert_active_sym(struct rb_root *tree, struct sym_entry *se)
447{
448 struct rb_node **p = &tree->rb_node;
449 struct rb_node *parent = NULL;
450 struct sym_entry *iter;
451
452 while (*p != NULL) {
453 parent = *p;
454 iter = rb_entry(parent, struct sym_entry, rb_node);
455
Arnaldo Carvalho de Meloc44613a2009-05-29 17:03:07 -0300456 if (se->weight > iter->weight)
Arnaldo Carvalho de Melode046872009-05-28 14:55:41 -0300457 p = &(*p)->rb_left;
458 else
459 p = &(*p)->rb_right;
460 }
461
462 rb_link_node(&se->rb_node, parent, p);
463 rb_insert_color(&se->rb_node, tree);
464}
Ingo Molnar07800602009-04-20 15:00:56 +0200465
466static void print_sym_table(void)
467{
Ingo Molnar233f0b92009-06-03 21:48:40 +0200468 int printed = 0, j;
Arnaldo Carvalho de Melo69aad6f2011-01-03 16:39:04 -0200469 struct perf_evsel *counter;
470 int snap = !display_weighted ? sym_counter : 0;
Ingo Molnar2debbc82009-06-05 14:29:10 +0200471 float samples_per_sec = samples/delay_secs;
Zhang, Yanmina1645ce2010-04-19 13:32:50 +0800472 float ksamples_per_sec = kernel_samples/delay_secs;
473 float us_samples_per_sec = (us_samples)/delay_secs;
474 float guest_kernel_samples_per_sec = (guest_kernel_samples)/delay_secs;
475 float guest_us_samples_per_sec = (guest_us_samples)/delay_secs;
Peter Zijlstra1676b8a2010-03-04 14:19:36 +0100476 float esamples_percent = (100.0*exact_samples)/samples;
Ingo Molnar2debbc82009-06-05 14:29:10 +0200477 float sum_ksamples = 0.0;
Arnaldo Carvalho de Melode046872009-05-28 14:55:41 -0300478 struct sym_entry *syme, *n;
479 struct rb_root tmp = RB_ROOT;
480 struct rb_node *nd;
Arnaldo Carvalho de Melo00909e92010-03-16 18:28:46 -0300481 int sym_width = 0, dso_width = 0, dso_short_width = 0;
Arnaldo Carvalho de Melo13cc5072009-11-17 15:40:54 -0200482 const int win_width = winsize.ws_col - 1;
Ingo Molnar07800602009-04-20 15:00:56 +0200483
Zhang, Yanmina1645ce2010-04-19 13:32:50 +0800484 samples = us_samples = kernel_samples = exact_samples = 0;
485 guest_kernel_samples = guest_us_samples = 0;
Ingo Molnar07800602009-04-20 15:00:56 +0200486
Arnaldo Carvalho de Melode046872009-05-28 14:55:41 -0300487 /* Sort the active symbols */
Arnaldo Carvalho de Meloc44613a2009-05-29 17:03:07 -0300488 pthread_mutex_lock(&active_symbols_lock);
489 syme = list_entry(active_symbols.next, struct sym_entry, node);
490 pthread_mutex_unlock(&active_symbols_lock);
491
492 list_for_each_entry_safe_from(syme, n, &active_symbols, node) {
Mike Galbraith46ab9762009-07-24 10:09:50 +0200493 syme->snap_count = syme->count[snap];
Arnaldo Carvalho de Meloc44613a2009-05-29 17:03:07 -0300494 if (syme->snap_count != 0) {
Arnaldo Carvalho de Melo13cc5072009-11-17 15:40:54 -0200495
Arnaldo Carvalho de Melo8ffcda12009-11-16 21:45:24 -0200496 if ((hide_user_symbols &&
497 syme->origin == PERF_RECORD_MISC_USER) ||
498 (hide_kernel_symbols &&
499 syme->origin == PERF_RECORD_MISC_KERNEL)) {
500 list_remove_active_sym(syme);
501 continue;
502 }
Arnaldo Carvalho de Meloc44613a2009-05-29 17:03:07 -0300503 syme->weight = sym_weight(syme);
Arnaldo Carvalho de Melode046872009-05-28 14:55:41 -0300504 rb_insert_active_sym(&tmp, syme);
Ingo Molnar2debbc82009-06-05 14:29:10 +0200505 sum_ksamples += syme->snap_count;
Mike Galbraithd94b9432009-05-25 09:57:56 +0200506
Arnaldo Carvalho de Melo361c99a2011-01-11 20:56:53 -0200507 for (j = 0; j < evsel_list->nr_entries; j++)
Arnaldo Carvalho de Melode046872009-05-28 14:55:41 -0300508 syme->count[j] = zero ? 0 : syme->count[j] * 7 / 8;
509 } else
Arnaldo Carvalho de Meloc44613a2009-05-29 17:03:07 -0300510 list_remove_active_sym(syme);
Mike Galbraithd94b9432009-05-25 09:57:56 +0200511 }
512
Frederic Weisbecker0f5486b2009-06-04 20:48:04 +0200513 puts(CONSOLE_CLEAR);
Ingo Molnar07800602009-04-20 15:00:56 +0200514
Arnaldo Carvalho de Melo13cc5072009-11-17 15:40:54 -0200515 printf("%-*.*s\n", win_width, win_width, graph_dotted_line);
Zhang, Yanmina1645ce2010-04-19 13:32:50 +0800516 if (!perf_guest) {
517 printf(" PerfTop:%8.0f irqs/sec kernel:%4.1f%%"
518 " exact: %4.1f%% [",
519 samples_per_sec,
520 100.0 - (100.0 * ((samples_per_sec - ksamples_per_sec) /
521 samples_per_sec)),
522 esamples_percent);
523 } else {
524 printf(" PerfTop:%8.0f irqs/sec kernel:%4.1f%% us:%4.1f%%"
525 " guest kernel:%4.1f%% guest us:%4.1f%%"
526 " exact: %4.1f%% [",
527 samples_per_sec,
528 100.0 - (100.0 * ((samples_per_sec-ksamples_per_sec) /
529 samples_per_sec)),
530 100.0 - (100.0 * ((samples_per_sec-us_samples_per_sec) /
531 samples_per_sec)),
532 100.0 - (100.0 * ((samples_per_sec -
533 guest_kernel_samples_per_sec) /
534 samples_per_sec)),
535 100.0 - (100.0 * ((samples_per_sec -
536 guest_us_samples_per_sec) /
537 samples_per_sec)),
538 esamples_percent);
539 }
Ingo Molnar07800602009-04-20 15:00:56 +0200540
Arnaldo Carvalho de Melo361c99a2011-01-11 20:56:53 -0200541 if (evsel_list->nr_entries == 1 || !display_weighted) {
Arnaldo Carvalho de Melo69aad6f2011-01-03 16:39:04 -0200542 struct perf_evsel *first;
Arnaldo Carvalho de Melo361c99a2011-01-11 20:56:53 -0200543 first = list_entry(evsel_list->entries.next, struct perf_evsel, node);
Arnaldo Carvalho de Melo9486aa32011-01-22 20:37:02 -0200544 printf("%" PRIu64, (uint64_t)first->attr.sample_period);
Ingo Molnarcf1f4572009-06-05 13:27:02 +0200545 if (freq)
546 printf("Hz ");
547 else
548 printf(" ");
549 }
Ingo Molnar07800602009-04-20 15:00:56 +0200550
Mike Galbraith46ab9762009-07-24 10:09:50 +0200551 if (!display_weighted)
Arnaldo Carvalho de Melo69aad6f2011-01-03 16:39:04 -0200552 printf("%s", event_name(sym_evsel));
Arnaldo Carvalho de Melo361c99a2011-01-11 20:56:53 -0200553 else list_for_each_entry(counter, &evsel_list->entries, node) {
Arnaldo Carvalho de Melo69aad6f2011-01-03 16:39:04 -0200554 if (counter->idx)
Ingo Molnar07800602009-04-20 15:00:56 +0200555 printf("/");
556
557 printf("%s", event_name(counter));
558 }
559
560 printf( "], ");
561
Ingo Molnarb456bae2009-05-26 09:17:18 +0200562 if (target_pid != -1)
563 printf(" (target_pid: %d", target_pid);
Zhang, Yanmind6d901c2010-03-18 11:36:05 -0300564 else if (target_tid != -1)
565 printf(" (target_tid: %d", target_tid);
Ingo Molnar07800602009-04-20 15:00:56 +0200566 else
567 printf(" (all");
568
Corey Ashford01797c52010-11-08 18:20:45 -0800569 if (cpu_list)
Arnaldo Carvalho de Melo60d567e2011-01-03 17:49:48 -0200570 printf(", CPU%s: %s)\n", cpus->nr > 1 ? "s" : "", cpu_list);
Ingo Molnar07800602009-04-20 15:00:56 +0200571 else {
Zhang, Yanmind6d901c2010-03-18 11:36:05 -0300572 if (target_tid != -1)
Ingo Molnar07800602009-04-20 15:00:56 +0200573 printf(")\n");
574 else
Arnaldo Carvalho de Melo60d567e2011-01-03 17:49:48 -0200575 printf(", %d CPU%s)\n", cpus->nr, cpus->nr > 1 ? "s" : "");
Ingo Molnar07800602009-04-20 15:00:56 +0200576 }
577
Arnaldo Carvalho de Melo1a105f72009-11-17 15:40:55 -0200578 printf("%-*.*s\n", win_width, win_width, graph_dotted_line);
Ingo Molnar07800602009-04-20 15:00:56 +0200579
Mike Galbraith923c42c2009-07-22 20:36:03 +0200580 if (sym_filter_entry) {
581 show_details(sym_filter_entry);
582 return;
583 }
584
Arnaldo Carvalho de Melo13cc5072009-11-17 15:40:54 -0200585 /*
586 * Find the longest symbol name that will be displayed
587 */
588 for (nd = rb_first(&tmp); nd; nd = rb_next(nd)) {
589 syme = rb_entry(nd, struct sym_entry, rb_node);
590 if (++printed > print_entries ||
591 (int)syme->snap_count < count_filter)
592 continue;
593
Arnaldo Carvalho de Melo1a105f72009-11-17 15:40:55 -0200594 if (syme->map->dso->long_name_len > dso_width)
595 dso_width = syme->map->dso->long_name_len;
596
Arnaldo Carvalho de Melob63be8d2010-03-15 15:03:50 -0300597 if (syme->map->dso->short_name_len > dso_short_width)
598 dso_short_width = syme->map->dso->short_name_len;
599
Arnaldo Carvalho de Melo13cc5072009-11-17 15:40:54 -0200600 if (syme->name_len > sym_width)
601 sym_width = syme->name_len;
602 }
603
604 printed = 0;
605
Arnaldo Carvalho de Melob63be8d2010-03-15 15:03:50 -0300606 if (sym_width + dso_width > winsize.ws_col - 29) {
607 dso_width = dso_short_width;
608 if (sym_width + dso_width > winsize.ws_col - 29)
609 sym_width = winsize.ws_col - dso_width - 29;
610 }
Arnaldo Carvalho de Melo7cc017e2009-11-24 12:05:14 -0200611 putchar('\n');
Arnaldo Carvalho de Melo361c99a2011-01-11 20:56:53 -0200612 if (evsel_list->nr_entries == 1)
Arnaldo Carvalho de Melo5b2bb752009-10-26 19:23:19 -0200613 printf(" samples pcnt");
Ingo Molnar07800602009-04-20 15:00:56 +0200614 else
Arnaldo Carvalho de Melo5b2bb752009-10-26 19:23:19 -0200615 printf(" weight samples pcnt");
Ingo Molnar07800602009-04-20 15:00:56 +0200616
Arnaldo Carvalho de Melo7ced1562009-08-26 11:51:26 -0300617 if (verbose)
618 printf(" RIP ");
Arnaldo Carvalho de Melo7cc017e2009-11-24 12:05:14 -0200619 printf(" %-*.*s DSO\n", sym_width, sym_width, "function");
Arnaldo Carvalho de Melo5b2bb752009-10-26 19:23:19 -0200620 printf(" %s _______ _____",
Arnaldo Carvalho de Melo361c99a2011-01-11 20:56:53 -0200621 evsel_list->nr_entries == 1 ? " " : "______");
Arnaldo Carvalho de Melo7ced1562009-08-26 11:51:26 -0300622 if (verbose)
Arnaldo Carvalho de Melo5b2bb752009-10-26 19:23:19 -0200623 printf(" ________________");
Arnaldo Carvalho de Melo1a105f72009-11-17 15:40:55 -0200624 printf(" %-*.*s", sym_width, sym_width, graph_line);
Arnaldo Carvalho de Melo7cc017e2009-11-24 12:05:14 -0200625 printf(" %-*.*s", dso_width, dso_width, graph_line);
Arnaldo Carvalho de Melo1a105f72009-11-17 15:40:55 -0200626 puts("\n");
Ingo Molnar07800602009-04-20 15:00:56 +0200627
Arnaldo Carvalho de Melode046872009-05-28 14:55:41 -0300628 for (nd = rb_first(&tmp); nd; nd = rb_next(nd)) {
Ingo Molnar83a09442009-08-15 12:26:57 +0200629 struct symbol *sym;
Ingo Molnar8fc03212009-06-04 15:19:47 +0200630 double pcnt;
Ingo Molnar07800602009-04-20 15:00:56 +0200631
Ingo Molnar83a09442009-08-15 12:26:57 +0200632 syme = rb_entry(nd, struct sym_entry, rb_node);
Arnaldo Carvalho de Melo51a472d2009-11-17 18:38:00 -0200633 sym = sym_entry__symbol(syme);
Mike Galbraith923c42c2009-07-22 20:36:03 +0200634 if (++printed > print_entries || (int)syme->snap_count < count_filter)
Arnaldo Carvalho de Meloc44613a2009-05-29 17:03:07 -0300635 continue;
Ingo Molnar07800602009-04-20 15:00:56 +0200636
Ingo Molnar2debbc82009-06-05 14:29:10 +0200637 pcnt = 100.0 - (100.0 * ((sum_ksamples - syme->snap_count) /
638 sum_ksamples));
Mike Galbraithd94b9432009-05-25 09:57:56 +0200639
Arnaldo Carvalho de Melo361c99a2011-01-11 20:56:53 -0200640 if (evsel_list->nr_entries == 1 || !display_weighted)
Arnaldo Carvalho de Melo5b2bb752009-10-26 19:23:19 -0200641 printf("%20.2f ", syme->weight);
Mike Galbraithd94b9432009-05-25 09:57:56 +0200642 else
Arnaldo Carvalho de Melo5b2bb752009-10-26 19:23:19 -0200643 printf("%9.1f %10ld ", syme->weight, syme->snap_count);
Ingo Molnar8fc03212009-06-04 15:19:47 +0200644
Frederic Weisbecker1e11fd82009-07-02 20:14:34 +0200645 percent_color_fprintf(stdout, "%4.1f%%", pcnt);
Arnaldo Carvalho de Melo7ced1562009-08-26 11:51:26 -0300646 if (verbose)
Arnaldo Carvalho de Melo9486aa32011-01-22 20:37:02 -0200647 printf(" %016" PRIx64, sym->start);
Arnaldo Carvalho de Melo13cc5072009-11-17 15:40:54 -0200648 printf(" %-*.*s", sym_width, sym_width, sym->name);
Arnaldo Carvalho de Melo7cc017e2009-11-24 12:05:14 -0200649 printf(" %-*.*s\n", dso_width, dso_width,
650 dso_width >= syme->map->dso->long_name_len ?
651 syme->map->dso->long_name :
652 syme->map->dso->short_name);
Ingo Molnar07800602009-04-20 15:00:56 +0200653 }
Ingo Molnar07800602009-04-20 15:00:56 +0200654}
655
Mike Galbraith923c42c2009-07-22 20:36:03 +0200656static void prompt_integer(int *target, const char *msg)
657{
658 char *buf = malloc(0), *p;
659 size_t dummy = 0;
660 int tmp;
661
662 fprintf(stdout, "\n%s: ", msg);
663 if (getline(&buf, &dummy, stdin) < 0)
664 return;
665
666 p = strchr(buf, '\n');
667 if (p)
668 *p = 0;
669
670 p = buf;
671 while(*p) {
672 if (!isdigit(*p))
673 goto out_free;
674 p++;
675 }
676 tmp = strtoul(buf, NULL, 10);
677 *target = tmp;
678out_free:
679 free(buf);
680}
681
682static void prompt_percent(int *target, const char *msg)
683{
684 int tmp = 0;
685
686 prompt_integer(&tmp, msg);
687 if (tmp >= 0 && tmp <= 100)
688 *target = tmp;
689}
690
691static void prompt_symbol(struct sym_entry **target, const char *msg)
692{
693 char *buf = malloc(0), *p;
694 struct sym_entry *syme = *target, *n, *found = NULL;
695 size_t dummy = 0;
696
697 /* zero counters of active symbol */
698 if (syme) {
Arnaldo Carvalho de Melob2698762009-11-17 18:38:02 -0200699 pthread_mutex_lock(&syme->src->lock);
Mike Galbraith923c42c2009-07-22 20:36:03 +0200700 __zero_source_counters(syme);
701 *target = NULL;
Arnaldo Carvalho de Melob2698762009-11-17 18:38:02 -0200702 pthread_mutex_unlock(&syme->src->lock);
Mike Galbraith923c42c2009-07-22 20:36:03 +0200703 }
704
705 fprintf(stdout, "\n%s: ", msg);
706 if (getline(&buf, &dummy, stdin) < 0)
707 goto out_free;
708
709 p = strchr(buf, '\n');
710 if (p)
711 *p = 0;
712
713 pthread_mutex_lock(&active_symbols_lock);
714 syme = list_entry(active_symbols.next, struct sym_entry, node);
715 pthread_mutex_unlock(&active_symbols_lock);
716
717 list_for_each_entry_safe_from(syme, n, &active_symbols, node) {
Arnaldo Carvalho de Melo51a472d2009-11-17 18:38:00 -0200718 struct symbol *sym = sym_entry__symbol(syme);
Mike Galbraith923c42c2009-07-22 20:36:03 +0200719
720 if (!strcmp(buf, sym->name)) {
721 found = syme;
722 break;
723 }
724 }
725
726 if (!found) {
Kirill Smelkov66aeb6d2010-01-13 13:22:19 -0200727 fprintf(stderr, "Sorry, %s is not active.\n", buf);
Mike Galbraith923c42c2009-07-22 20:36:03 +0200728 sleep(1);
729 return;
730 } else
731 parse_source(found);
732
733out_free:
734 free(buf);
735}
736
Mike Galbraith091bd2e2009-08-04 10:21:23 +0200737static void print_mapped_keys(void)
Mike Galbraith923c42c2009-07-22 20:36:03 +0200738{
Mike Galbraith091bd2e2009-08-04 10:21:23 +0200739 char *name = NULL;
740
741 if (sym_filter_entry) {
Arnaldo Carvalho de Melo51a472d2009-11-17 18:38:00 -0200742 struct symbol *sym = sym_entry__symbol(sym_filter_entry);
Mike Galbraith091bd2e2009-08-04 10:21:23 +0200743 name = sym->name;
744 }
745
746 fprintf(stdout, "\nMapped keys:\n");
747 fprintf(stdout, "\t[d] display refresh delay. \t(%d)\n", delay_secs);
748 fprintf(stdout, "\t[e] display entries (lines). \t(%d)\n", print_entries);
749
Arnaldo Carvalho de Melo361c99a2011-01-11 20:56:53 -0200750 if (evsel_list->nr_entries > 1)
Arnaldo Carvalho de Melo69aad6f2011-01-03 16:39:04 -0200751 fprintf(stdout, "\t[E] active event counter. \t(%s)\n", event_name(sym_evsel));
Mike Galbraith091bd2e2009-08-04 10:21:23 +0200752
753 fprintf(stdout, "\t[f] profile display filter (count). \t(%d)\n", count_filter);
754
Kirill Smelkov6cff0e82010-02-03 16:52:08 -0200755 fprintf(stdout, "\t[F] annotate display filter (percent). \t(%d%%)\n", sym_pcnt_filter);
756 fprintf(stdout, "\t[s] annotate symbol. \t(%s)\n", name?: "NULL");
757 fprintf(stdout, "\t[S] stop annotation.\n");
Mike Galbraith091bd2e2009-08-04 10:21:23 +0200758
Arnaldo Carvalho de Melo361c99a2011-01-11 20:56:53 -0200759 if (evsel_list->nr_entries > 1)
Mike Galbraith091bd2e2009-08-04 10:21:23 +0200760 fprintf(stdout, "\t[w] toggle display weighted/count[E]r. \t(%d)\n", display_weighted ? 1 : 0);
761
Arnaldo Carvalho de Melo8ffcda12009-11-16 21:45:24 -0200762 fprintf(stdout,
Kirill Smelkov1a72cfa2010-02-12 19:20:59 +0300763 "\t[K] hide kernel_symbols symbols. \t(%s)\n",
Arnaldo Carvalho de Melo8ffcda12009-11-16 21:45:24 -0200764 hide_kernel_symbols ? "yes" : "no");
765 fprintf(stdout,
766 "\t[U] hide user symbols. \t(%s)\n",
767 hide_user_symbols ? "yes" : "no");
Mike Galbraith46ab9762009-07-24 10:09:50 +0200768 fprintf(stdout, "\t[z] toggle sample zeroing. \t(%d)\n", zero ? 1 : 0);
Mike Galbraith091bd2e2009-08-04 10:21:23 +0200769 fprintf(stdout, "\t[qQ] quit.\n");
770}
771
772static int key_mapped(int c)
773{
774 switch (c) {
775 case 'd':
776 case 'e':
777 case 'f':
778 case 'z':
779 case 'q':
780 case 'Q':
Arnaldo Carvalho de Melo8ffcda12009-11-16 21:45:24 -0200781 case 'K':
782 case 'U':
Kirill Smelkov6cff0e82010-02-03 16:52:08 -0200783 case 'F':
784 case 's':
785 case 'S':
Mike Galbraith091bd2e2009-08-04 10:21:23 +0200786 return 1;
787 case 'E':
788 case 'w':
Arnaldo Carvalho de Melo361c99a2011-01-11 20:56:53 -0200789 return evsel_list->nr_entries > 1 ? 1 : 0;
Ingo Molnar83a09442009-08-15 12:26:57 +0200790 default:
791 break;
Mike Galbraith091bd2e2009-08-04 10:21:23 +0200792 }
793
794 return 0;
Mike Galbraith923c42c2009-07-22 20:36:03 +0200795}
796
Zhang, Yanmina1645ce2010-04-19 13:32:50 +0800797static void handle_keypress(struct perf_session *session, int c)
Mike Galbraith923c42c2009-07-22 20:36:03 +0200798{
Mike Galbraith091bd2e2009-08-04 10:21:23 +0200799 if (!key_mapped(c)) {
800 struct pollfd stdin_poll = { .fd = 0, .events = POLLIN };
801 struct termios tc, save;
802
803 print_mapped_keys();
804 fprintf(stdout, "\nEnter selection, or unmapped key to continue: ");
805 fflush(stdout);
806
807 tcgetattr(0, &save);
808 tc = save;
809 tc.c_lflag &= ~(ICANON | ECHO);
810 tc.c_cc[VMIN] = 0;
811 tc.c_cc[VTIME] = 0;
812 tcsetattr(0, TCSANOW, &tc);
813
814 poll(&stdin_poll, 1, -1);
815 c = getc(stdin);
816
817 tcsetattr(0, TCSAFLUSH, &save);
818 if (!key_mapped(c))
819 return;
820 }
821
Mike Galbraith923c42c2009-07-22 20:36:03 +0200822 switch (c) {
823 case 'd':
824 prompt_integer(&delay_secs, "Enter display delay");
Tim Blechmanndc799592009-10-17 18:08:29 +0200825 if (delay_secs < 1)
826 delay_secs = 1;
Mike Galbraith923c42c2009-07-22 20:36:03 +0200827 break;
828 case 'e':
829 prompt_integer(&print_entries, "Enter display entries (lines)");
Arnaldo Carvalho de Melo3b6ed982009-11-16 19:30:27 -0200830 if (print_entries == 0) {
Arnaldo Carvalho de Melo13cc5072009-11-17 15:40:54 -0200831 sig_winch_handler(SIGWINCH);
Arnaldo Carvalho de Melo3b6ed982009-11-16 19:30:27 -0200832 signal(SIGWINCH, sig_winch_handler);
833 } else
834 signal(SIGWINCH, SIG_DFL);
Mike Galbraith923c42c2009-07-22 20:36:03 +0200835 break;
836 case 'E':
Arnaldo Carvalho de Melo361c99a2011-01-11 20:56:53 -0200837 if (evsel_list->nr_entries > 1) {
Mike Galbraith923c42c2009-07-22 20:36:03 +0200838 fprintf(stderr, "\nAvailable events:");
Arnaldo Carvalho de Melo69aad6f2011-01-03 16:39:04 -0200839
Arnaldo Carvalho de Melo361c99a2011-01-11 20:56:53 -0200840 list_for_each_entry(sym_evsel, &evsel_list->entries, node)
Arnaldo Carvalho de Melo69aad6f2011-01-03 16:39:04 -0200841 fprintf(stderr, "\n\t%d %s", sym_evsel->idx, event_name(sym_evsel));
Mike Galbraith923c42c2009-07-22 20:36:03 +0200842
843 prompt_integer(&sym_counter, "Enter details event counter");
844
Arnaldo Carvalho de Melo361c99a2011-01-11 20:56:53 -0200845 if (sym_counter >= evsel_list->nr_entries) {
846 sym_evsel = list_entry(evsel_list->entries.next, struct perf_evsel, node);
Mike Galbraith923c42c2009-07-22 20:36:03 +0200847 sym_counter = 0;
Arnaldo Carvalho de Melo69aad6f2011-01-03 16:39:04 -0200848 fprintf(stderr, "Sorry, no such event, using %s.\n", event_name(sym_evsel));
Mike Galbraith923c42c2009-07-22 20:36:03 +0200849 sleep(1);
Arnaldo Carvalho de Melo69aad6f2011-01-03 16:39:04 -0200850 break;
Mike Galbraith923c42c2009-07-22 20:36:03 +0200851 }
Arnaldo Carvalho de Melo361c99a2011-01-11 20:56:53 -0200852 list_for_each_entry(sym_evsel, &evsel_list->entries, node)
Arnaldo Carvalho de Melo69aad6f2011-01-03 16:39:04 -0200853 if (sym_evsel->idx == sym_counter)
854 break;
Mike Galbraith923c42c2009-07-22 20:36:03 +0200855 } else sym_counter = 0;
856 break;
857 case 'f':
858 prompt_integer(&count_filter, "Enter display event count filter");
859 break;
860 case 'F':
861 prompt_percent(&sym_pcnt_filter, "Enter details display event filter (percent)");
862 break;
Arnaldo Carvalho de Melo8ffcda12009-11-16 21:45:24 -0200863 case 'K':
864 hide_kernel_symbols = !hide_kernel_symbols;
865 break;
Mike Galbraith923c42c2009-07-22 20:36:03 +0200866 case 'q':
867 case 'Q':
868 printf("exiting.\n");
Arnaldo Carvalho de Meloc338aee2009-11-20 20:51:27 -0200869 if (dump_symtab)
Arnaldo Carvalho de Melocbf69682010-04-27 21:22:44 -0300870 perf_session__fprintf_dsos(session, stderr);
Mike Galbraith923c42c2009-07-22 20:36:03 +0200871 exit(0);
872 case 's':
873 prompt_symbol(&sym_filter_entry, "Enter details symbol");
874 break;
875 case 'S':
876 if (!sym_filter_entry)
877 break;
878 else {
879 struct sym_entry *syme = sym_filter_entry;
880
Arnaldo Carvalho de Melob2698762009-11-17 18:38:02 -0200881 pthread_mutex_lock(&syme->src->lock);
Mike Galbraith923c42c2009-07-22 20:36:03 +0200882 sym_filter_entry = NULL;
883 __zero_source_counters(syme);
Arnaldo Carvalho de Melob2698762009-11-17 18:38:02 -0200884 pthread_mutex_unlock(&syme->src->lock);
Mike Galbraith923c42c2009-07-22 20:36:03 +0200885 }
886 break;
Arnaldo Carvalho de Melo8ffcda12009-11-16 21:45:24 -0200887 case 'U':
888 hide_user_symbols = !hide_user_symbols;
889 break;
Mike Galbraith46ab9762009-07-24 10:09:50 +0200890 case 'w':
891 display_weighted = ~display_weighted;
892 break;
Mike Galbraith923c42c2009-07-22 20:36:03 +0200893 case 'z':
Ian Munsiec0555642010-04-13 18:37:33 +1000894 zero = !zero;
Mike Galbraith923c42c2009-07-22 20:36:03 +0200895 break;
Ingo Molnar83a09442009-08-15 12:26:57 +0200896 default:
897 break;
Mike Galbraith923c42c2009-07-22 20:36:03 +0200898 }
899}
900
Ingo Molnarf37a2912009-07-01 12:37:06 +0200901static void *display_thread(void *arg __used)
Ingo Molnar07800602009-04-20 15:00:56 +0200902{
Frederic Weisbecker0f5486b2009-06-04 20:48:04 +0200903 struct pollfd stdin_poll = { .fd = 0, .events = POLLIN };
Mike Galbraith923c42c2009-07-22 20:36:03 +0200904 struct termios tc, save;
905 int delay_msecs, c;
Zhang, Yanmina1645ce2010-04-19 13:32:50 +0800906 struct perf_session *session = (struct perf_session *) arg;
Frederic Weisbecker0f5486b2009-06-04 20:48:04 +0200907
Mike Galbraith923c42c2009-07-22 20:36:03 +0200908 tcgetattr(0, &save);
909 tc = save;
910 tc.c_lflag &= ~(ICANON | ECHO);
911 tc.c_cc[VMIN] = 0;
912 tc.c_cc[VTIME] = 0;
Mike Galbraith091bd2e2009-08-04 10:21:23 +0200913
Mike Galbraith923c42c2009-07-22 20:36:03 +0200914repeat:
915 delay_msecs = delay_secs * 1000;
916 tcsetattr(0, TCSANOW, &tc);
917 /* trash return*/
918 getc(stdin);
Ingo Molnar07800602009-04-20 15:00:56 +0200919
Frederic Weisbecker0f5486b2009-06-04 20:48:04 +0200920 do {
Ingo Molnar07800602009-04-20 15:00:56 +0200921 print_sym_table();
Frederic Weisbecker0f5486b2009-06-04 20:48:04 +0200922 } while (!poll(&stdin_poll, 1, delay_msecs) == 1);
923
Mike Galbraith923c42c2009-07-22 20:36:03 +0200924 c = getc(stdin);
925 tcsetattr(0, TCSAFLUSH, &save);
926
Zhang, Yanmina1645ce2010-04-19 13:32:50 +0800927 handle_keypress(session, c);
Mike Galbraith923c42c2009-07-22 20:36:03 +0200928 goto repeat;
Ingo Molnar07800602009-04-20 15:00:56 +0200929
930 return NULL;
931}
932
Anton Blanchard2ab52082009-07-01 09:00:46 +1000933/* Tag samples to be skipped. */
Ingo Molnarf37a2912009-07-01 12:37:06 +0200934static const char *skip_symbols[] = {
Anton Blanchard2ab52082009-07-01 09:00:46 +1000935 "default_idle",
936 "cpu_idle",
937 "enter_idle",
938 "exit_idle",
939 "mwait_idle",
Arnaldo Carvalho de Melo59b90052009-07-26 19:06:19 -0300940 "mwait_idle_with_hints",
Arnaldo Carvalho de Melo83572752009-09-25 15:02:39 -0700941 "poll_idle",
Anton Blanchard3a3393e2009-07-01 09:00:47 +1000942 "ppc64_runlatch_off",
943 "pseries_dedicated_idle_sleep",
Anton Blanchard2ab52082009-07-01 09:00:46 +1000944 NULL
945};
946
Arnaldo Carvalho de Melo439d4732009-10-02 03:29:58 -0300947static int symbol_filter(struct map *map, struct symbol *sym)
Ingo Molnar07800602009-04-20 15:00:56 +0200948{
Arnaldo Carvalho de Melode046872009-05-28 14:55:41 -0300949 struct sym_entry *syme;
950 const char *name = sym->name;
Anton Blanchard2ab52082009-07-01 09:00:46 +1000951 int i;
Ingo Molnar07800602009-04-20 15:00:56 +0200952
Anton Blanchard3a3393e2009-07-01 09:00:47 +1000953 /*
954 * ppc64 uses function descriptors and appends a '.' to the
955 * start of every instruction address. Remove it.
956 */
957 if (name[0] == '.')
958 name++;
959
Arnaldo Carvalho de Melode046872009-05-28 14:55:41 -0300960 if (!strcmp(name, "_text") ||
961 !strcmp(name, "_etext") ||
962 !strcmp(name, "_sinittext") ||
963 !strncmp("init_module", name, 11) ||
964 !strncmp("cleanup_module", name, 14) ||
965 strstr(name, "_text_start") ||
966 strstr(name, "_text_end"))
Ingo Molnar07800602009-04-20 15:00:56 +0200967 return 1;
968
Arnaldo Carvalho de Melo00a192b2009-10-30 16:28:24 -0200969 syme = symbol__priv(sym);
Arnaldo Carvalho de Melo439d4732009-10-02 03:29:58 -0300970 syme->map = map;
Arnaldo Carvalho de Melob2698762009-11-17 18:38:02 -0200971 syme->src = NULL;
Kirill Smelkov6cff0e82010-02-03 16:52:08 -0200972
973 if (!sym_filter_entry && sym_filter && !strcmp(name, sym_filter)) {
974 /* schedule initial sym_filter_entry setup */
975 sym_filter_entry_sched = syme;
976 sym_filter = NULL;
977 }
Mike Galbraith923c42c2009-07-22 20:36:03 +0200978
Anton Blanchard2ab52082009-07-01 09:00:46 +1000979 for (i = 0; skip_symbols[i]; i++) {
980 if (!strcmp(skip_symbols[i], name)) {
981 syme->skip = 1;
982 break;
983 }
984 }
Ingo Molnar07800602009-04-20 15:00:56 +0200985
Arnaldo Carvalho de Melo13cc5072009-11-17 15:40:54 -0200986 if (!syme->skip)
987 syme->name_len = strlen(sym->name);
988
Ingo Molnar07800602009-04-20 15:00:56 +0200989 return 0;
990}
991
Arnaldo Carvalho de Melob3165f42009-12-13 19:50:28 -0200992static void event__process_sample(const event_t *self,
Arnaldo Carvalho de Melo9c90a612010-12-02 10:25:28 -0200993 struct sample_data *sample,
Arnaldo Carvalho de Melo69aad6f2011-01-03 16:39:04 -0200994 struct perf_session *session,
995 struct perf_evsel *evsel)
Ingo Molnar07800602009-04-20 15:00:56 +0200996{
Arnaldo Carvalho de Melo5b2bb752009-10-26 19:23:19 -0200997 u64 ip = self->ip.ip;
Arnaldo Carvalho de Melo5b2bb752009-10-26 19:23:19 -0200998 struct sym_entry *syme;
Arnaldo Carvalho de Melo1ed091c2009-11-27 16:29:23 -0200999 struct addr_location al;
Arnaldo Carvalho de Melo23346f22010-04-27 21:17:50 -03001000 struct machine *machine;
Arnaldo Carvalho de Melo8ffcda12009-11-16 21:45:24 -02001001 u8 origin = self->header.misc & PERF_RECORD_MISC_CPUMODE_MASK;
Ingo Molnar07800602009-04-20 15:00:56 +02001002
Arnaldo Carvalho de Melo24bfef02010-01-21 13:04:43 -02001003 ++samples;
1004
Arnaldo Carvalho de Melo8ffcda12009-11-16 21:45:24 -02001005 switch (origin) {
Arnaldo Carvalho de Melo1ed091c2009-11-27 16:29:23 -02001006 case PERF_RECORD_MISC_USER:
Zhang, Yanmina1645ce2010-04-19 13:32:50 +08001007 ++us_samples;
Arnaldo Carvalho de Melo8ffcda12009-11-16 21:45:24 -02001008 if (hide_user_symbols)
1009 return;
Arnaldo Carvalho de Melo23346f22010-04-27 21:17:50 -03001010 machine = perf_session__find_host_machine(session);
Arnaldo Carvalho de Melo1ed091c2009-11-27 16:29:23 -02001011 break;
Arnaldo Carvalho de Melo5b2bb752009-10-26 19:23:19 -02001012 case PERF_RECORD_MISC_KERNEL:
Zhang, Yanmina1645ce2010-04-19 13:32:50 +08001013 ++kernel_samples;
Arnaldo Carvalho de Melo8ffcda12009-11-16 21:45:24 -02001014 if (hide_kernel_symbols)
1015 return;
Arnaldo Carvalho de Melo23346f22010-04-27 21:17:50 -03001016 machine = perf_session__find_host_machine(session);
Arnaldo Carvalho de Melo5b2bb752009-10-26 19:23:19 -02001017 break;
Zhang, Yanmina1645ce2010-04-19 13:32:50 +08001018 case PERF_RECORD_MISC_GUEST_KERNEL:
1019 ++guest_kernel_samples;
Arnaldo Carvalho de Melo23346f22010-04-27 21:17:50 -03001020 machine = perf_session__find_machine(session, self->ip.pid);
Zhang, Yanmina1645ce2010-04-19 13:32:50 +08001021 break;
1022 case PERF_RECORD_MISC_GUEST_USER:
1023 ++guest_us_samples;
1024 /*
1025 * TODO: we don't process guest user from host side
1026 * except simple counting.
1027 */
1028 return;
Arnaldo Carvalho de Melo5b2bb752009-10-26 19:23:19 -02001029 default:
Ingo Molnar07800602009-04-20 15:00:56 +02001030 return;
1031 }
1032
Arnaldo Carvalho de Melo23346f22010-04-27 21:17:50 -03001033 if (!machine && perf_guest) {
Zhang, Yanmina1645ce2010-04-19 13:32:50 +08001034 pr_err("Can't find guest [%d]'s kernel information\n",
1035 self->ip.pid);
1036 return;
1037 }
1038
Peter Zijlstraab608342010-04-08 23:03:20 +02001039 if (self->header.misc & PERF_RECORD_MISC_EXACT_IP)
Peter Zijlstra1676b8a2010-03-04 14:19:36 +01001040 exact_samples++;
1041
Arnaldo Carvalho de Melo9c90a612010-12-02 10:25:28 -02001042 if (event__preprocess_sample(self, session, &al, sample,
Arnaldo Carvalho de Melo41a37e22010-06-04 08:02:07 -03001043 symbol_filter) < 0 ||
Arnaldo Carvalho de Melo72b8fa12010-01-27 21:05:49 -02001044 al.filtered)
Arnaldo Carvalho de Melo1ed091c2009-11-27 16:29:23 -02001045 return;
Arnaldo Carvalho de Melo5b2bb752009-10-26 19:23:19 -02001046
Arnaldo Carvalho de Melo72b8fa12010-01-27 21:05:49 -02001047 if (al.sym == NULL) {
1048 /*
1049 * As we do lazy loading of symtabs we only will know if the
1050 * specified vmlinux file is invalid when we actually have a
1051 * hit in kernel space and then try to load it. So if we get
1052 * here and there are _no_ symbols in the DSO backing the
1053 * kernel map, bail out.
1054 *
1055 * We may never get here, for instance, if we use -K/
1056 * --hide-kernel-symbols, even if the user specifies an
1057 * invalid --vmlinux ;-)
1058 */
Arnaldo Carvalho de Melo23346f22010-04-27 21:17:50 -03001059 if (al.map == machine->vmlinux_maps[MAP__FUNCTION] &&
Arnaldo Carvalho de Melo72b8fa12010-01-27 21:05:49 -02001060 RB_EMPTY_ROOT(&al.map->dso->symbols[MAP__FUNCTION])) {
1061 pr_err("The %s file can't be used\n",
1062 symbol_conf.vmlinux_name);
1063 exit(1);
1064 }
1065
1066 return;
1067 }
1068
Kirill Smelkov6cff0e82010-02-03 16:52:08 -02001069 /* let's see, whether we need to install initial sym_filter_entry */
1070 if (sym_filter_entry_sched) {
1071 sym_filter_entry = sym_filter_entry_sched;
1072 sym_filter_entry_sched = NULL;
Arnaldo Carvalho de Melob0a9ab62010-03-15 11:46:58 -03001073 if (parse_source(sym_filter_entry) < 0) {
1074 struct symbol *sym = sym_entry__symbol(sym_filter_entry);
1075
1076 pr_err("Can't annotate %s", sym->name);
1077 if (sym_filter_entry->map->dso->origin == DSO__ORIG_KERNEL) {
1078 pr_err(": No vmlinux file was found in the path:\n");
Arnaldo Carvalho de Melo5ad90e42010-05-26 13:26:02 -03001079 machine__fprintf_vmlinux_path(machine, stderr);
Arnaldo Carvalho de Melob0a9ab62010-03-15 11:46:58 -03001080 } else
1081 pr_err(".\n");
1082 exit(1);
1083 }
Kirill Smelkov6cff0e82010-02-03 16:52:08 -02001084 }
1085
Arnaldo Carvalho de Melo1ed091c2009-11-27 16:29:23 -02001086 syme = symbol__priv(al.sym);
Arnaldo Carvalho de Melo5b2bb752009-10-26 19:23:19 -02001087 if (!syme->skip) {
Arnaldo Carvalho de Melo69aad6f2011-01-03 16:39:04 -02001088 syme->count[evsel->idx]++;
Arnaldo Carvalho de Melo8ffcda12009-11-16 21:45:24 -02001089 syme->origin = origin;
Arnaldo Carvalho de Melo69aad6f2011-01-03 16:39:04 -02001090 record_precise_ip(syme, evsel->idx, ip);
Arnaldo Carvalho de Melo5b2bb752009-10-26 19:23:19 -02001091 pthread_mutex_lock(&active_symbols_lock);
1092 if (list_empty(&syme->node) || !syme->node.next)
1093 __list_insert_active_sym(syme);
1094 pthread_mutex_unlock(&active_symbols_lock);
Arnaldo Carvalho de Melo5b2bb752009-10-26 19:23:19 -02001095 }
1096}
1097
Ingo Molnar07800602009-04-20 15:00:56 +02001098struct mmap_data {
Ingo Molnara21ca2c2009-06-06 09:58:57 +02001099 void *base;
Ingo Molnarf37a2912009-07-01 12:37:06 +02001100 int mask;
Ingo Molnara21ca2c2009-06-06 09:58:57 +02001101 unsigned int prev;
Ingo Molnar07800602009-04-20 15:00:56 +02001102};
1103
Arnaldo Carvalho de Melo69aad6f2011-01-03 16:39:04 -02001104static int perf_evsel__alloc_mmap_per_thread(struct perf_evsel *evsel,
1105 int ncpus, int nthreads)
1106{
1107 evsel->priv = xyarray__new(ncpus, nthreads, sizeof(struct mmap_data));
1108 return evsel->priv != NULL ? 0 : -ENOMEM;
1109}
1110
1111static void perf_evsel__free_mmap(struct perf_evsel *evsel)
1112{
1113 xyarray__delete(evsel->priv);
1114 evsel->priv = NULL;
1115}
1116
Ingo Molnar07800602009-04-20 15:00:56 +02001117static unsigned int mmap_read_head(struct mmap_data *md)
1118{
Ingo Molnarcdd6c482009-09-21 12:02:48 +02001119 struct perf_event_mmap_page *pc = md->base;
Ingo Molnar07800602009-04-20 15:00:56 +02001120 int head;
1121
1122 head = pc->data_head;
1123 rmb();
1124
1125 return head;
1126}
1127
Arnaldo Carvalho de Melod8f66242009-12-13 19:50:24 -02001128static void perf_session__mmap_read_counter(struct perf_session *self,
Arnaldo Carvalho de Melo69aad6f2011-01-03 16:39:04 -02001129 struct perf_evsel *evsel,
1130 int cpu, int thread_idx)
Ingo Molnar07800602009-04-20 15:00:56 +02001131{
Arnaldo Carvalho de Melo69aad6f2011-01-03 16:39:04 -02001132 struct xyarray *mmap_array = evsel->priv;
1133 struct mmap_data *md = xyarray__entry(mmap_array, cpu, thread_idx);
Ingo Molnar07800602009-04-20 15:00:56 +02001134 unsigned int head = mmap_read_head(md);
1135 unsigned int old = md->prev;
1136 unsigned char *data = md->base + page_size;
Arnaldo Carvalho de Melo9c90a612010-12-02 10:25:28 -02001137 struct sample_data sample;
Ingo Molnar07800602009-04-20 15:00:56 +02001138 int diff;
1139
Ingo Molnar07800602009-04-20 15:00:56 +02001140 /*
1141 * If we're further behind than half the buffer, there's a chance
Ingo Molnar2debbc82009-06-05 14:29:10 +02001142 * the writer will bite our tail and mess up the samples under us.
Ingo Molnar07800602009-04-20 15:00:56 +02001143 *
1144 * If we somehow ended up ahead of the head, we got messed up.
1145 *
1146 * In either case, truncate and restart at head.
1147 */
1148 diff = head - old;
1149 if (diff > md->mask / 2 || diff < 0) {
Mike Galbraithf4f0b412009-10-13 14:57:20 +02001150 fprintf(stderr, "WARNING: failed to keep up with mmap data.\n");
Ingo Molnar07800602009-04-20 15:00:56 +02001151
1152 /*
1153 * head points to a known good entry, start there.
1154 */
1155 old = head;
1156 }
1157
Ingo Molnar07800602009-04-20 15:00:56 +02001158 for (; old != head;) {
Ingo Molnar07800602009-04-20 15:00:56 +02001159 event_t *event = (event_t *)&data[old & md->mask];
1160
1161 event_t event_copy;
1162
Ingo Molnar6f06ccb2009-04-20 15:22:22 +02001163 size_t size = event->header.size;
Ingo Molnar07800602009-04-20 15:00:56 +02001164
1165 /*
1166 * Event straddles the mmap boundary -- header should always
1167 * be inside due to u64 alignment of output.
1168 */
1169 if ((old & md->mask) + size != ((old + size) & md->mask)) {
1170 unsigned int offset = old;
1171 unsigned int len = min(sizeof(*event), size), cpy;
1172 void *dst = &event_copy;
1173
1174 do {
1175 cpy = min(md->mask + 1 - (offset & md->mask), len);
1176 memcpy(dst, &data[offset & md->mask], cpy);
1177 offset += cpy;
1178 dst += cpy;
1179 len -= cpy;
1180 } while (len);
1181
1182 event = &event_copy;
1183 }
1184
Arnaldo Carvalho de Melo9c90a612010-12-02 10:25:28 -02001185 event__parse_sample(event, self, &sample);
Arnaldo Carvalho de Melo5b2bb752009-10-26 19:23:19 -02001186 if (event->header.type == PERF_RECORD_SAMPLE)
Arnaldo Carvalho de Melo69aad6f2011-01-03 16:39:04 -02001187 event__process_sample(event, &sample, self, evsel);
Arnaldo Carvalho de Melo5b2bb752009-10-26 19:23:19 -02001188 else
Arnaldo Carvalho de Melo9c90a612010-12-02 10:25:28 -02001189 event__process(event, &sample, self);
Ingo Molnar07800602009-04-20 15:00:56 +02001190 old += size;
Ingo Molnar07800602009-04-20 15:00:56 +02001191 }
1192
1193 md->prev = old;
1194}
1195
Arnaldo Carvalho de Melod8f66242009-12-13 19:50:24 -02001196static void perf_session__mmap_read(struct perf_session *self)
Frederic Weisbecker2f011902009-06-06 23:10:43 +02001197{
Arnaldo Carvalho de Melo69aad6f2011-01-03 16:39:04 -02001198 struct perf_evsel *counter;
1199 int i, thread_index;
Frederic Weisbecker2f011902009-06-06 23:10:43 +02001200
Arnaldo Carvalho de Melo60d567e2011-01-03 17:49:48 -02001201 for (i = 0; i < cpus->nr; i++) {
Arnaldo Carvalho de Melo361c99a2011-01-11 20:56:53 -02001202 list_for_each_entry(counter, &evsel_list->entries, node) {
Zhang, Yanmind6d901c2010-03-18 11:36:05 -03001203 for (thread_index = 0;
Arnaldo Carvalho de Melo5c98d4662011-01-03 17:53:33 -02001204 thread_index < threads->nr;
Zhang, Yanmind6d901c2010-03-18 11:36:05 -03001205 thread_index++) {
1206 perf_session__mmap_read_counter(self,
Arnaldo Carvalho de Melo69aad6f2011-01-03 16:39:04 -02001207 counter, i, thread_index);
Zhang, Yanmind6d901c2010-03-18 11:36:05 -03001208 }
Arnaldo Carvalho de Melo69aad6f2011-01-03 16:39:04 -02001209 }
Frederic Weisbecker2f011902009-06-06 23:10:43 +02001210 }
1211}
1212
Arnaldo Carvalho de Melo5c581042011-01-11 22:30:02 -02001213static void start_counter(int i, struct perf_evlist *evlist,
1214 struct perf_evsel *evsel)
Ingo Molnar07800602009-04-20 15:00:56 +02001215{
Arnaldo Carvalho de Melo69aad6f2011-01-03 16:39:04 -02001216 struct xyarray *mmap_array = evsel->priv;
1217 struct mmap_data *mm;
Zhang, Yanmind6d901c2010-03-18 11:36:05 -03001218 int thread_index;
Ingo Molnar716c69f2009-06-07 17:31:52 +02001219
Arnaldo Carvalho de Melo5c98d4662011-01-03 17:53:33 -02001220 for (thread_index = 0; thread_index < threads->nr; thread_index++) {
Arnaldo Carvalho de Melo69aad6f2011-01-03 16:39:04 -02001221 assert(FD(evsel, i, thread_index) >= 0);
1222 fcntl(FD(evsel, i, thread_index), F_SETFL, O_NONBLOCK);
Zhang, Yanmind6d901c2010-03-18 11:36:05 -03001223
Arnaldo Carvalho de Melo5c581042011-01-11 22:30:02 -02001224 evlist->pollfd[evlist->nr_fds].fd = FD(evsel, i, thread_index);
1225 evlist->pollfd[evlist->nr_fds].events = POLLIN;
1226 evlist->nr_fds++;
Zhang, Yanmind6d901c2010-03-18 11:36:05 -03001227
Arnaldo Carvalho de Melo69aad6f2011-01-03 16:39:04 -02001228 mm = xyarray__entry(mmap_array, i, thread_index);
1229 mm->prev = 0;
1230 mm->mask = mmap_pages*page_size - 1;
1231 mm->base = mmap(NULL, (mmap_pages+1)*page_size,
1232 PROT_READ, MAP_SHARED, FD(evsel, i, thread_index), 0);
1233 if (mm->base == MAP_FAILED)
Zhang, Yanmind6d901c2010-03-18 11:36:05 -03001234 die("failed to mmap with %d (%s)\n", errno, strerror(errno));
Ingo Molnar716c69f2009-06-07 17:31:52 +02001235 }
Ingo Molnar716c69f2009-06-07 17:31:52 +02001236}
1237
Arnaldo Carvalho de Melo72cb7012011-01-12 10:52:47 -02001238static void start_counters(struct perf_evlist *evlist)
1239{
1240 struct perf_evsel *counter;
1241 int i;
1242
1243 list_for_each_entry(counter, &evlist->entries, node) {
1244 struct perf_event_attr *attr = &counter->attr;
1245
1246 attr->sample_type = PERF_SAMPLE_IP | PERF_SAMPLE_TID;
1247
1248 if (freq) {
1249 attr->sample_type |= PERF_SAMPLE_PERIOD;
1250 attr->freq = 1;
1251 attr->sample_freq = freq;
1252 }
1253
1254 attr->mmap = 1;
1255try_again:
1256 if (perf_evsel__open(counter, cpus, threads, group, inherit) < 0) {
1257 int err = errno;
1258
1259 if (err == EPERM || err == EACCES)
1260 die("Permission error - are you root?\n"
1261 "\t Consider tweaking"
1262 " /proc/sys/kernel/perf_event_paranoid.\n");
1263 /*
1264 * If it's cycles then fall back to hrtimer
1265 * based cpu-clock-tick sw counter, which
1266 * is always available even if no PMU support:
1267 */
1268 if (attr->type == PERF_TYPE_HARDWARE &&
1269 attr->config == PERF_COUNT_HW_CPU_CYCLES) {
1270
1271 if (verbose)
1272 warning(" ... trying to fall back to cpu-clock-ticks\n");
1273
1274 attr->type = PERF_TYPE_SOFTWARE;
1275 attr->config = PERF_COUNT_SW_CPU_CLOCK;
1276 goto try_again;
1277 }
1278 printf("\n");
1279 error("sys_perf_event_open() syscall returned with %d "
1280 "(%s). /bin/dmesg may provide additional information.\n",
1281 err, strerror(err));
1282 die("No CONFIG_PERF_EVENTS=y kernel support configured?\n");
1283 exit(-1);
1284 }
1285 }
1286
1287 for (i = 0; i < cpus->nr; i++) {
1288 list_for_each_entry(counter, &evlist->entries, node)
1289 start_counter(i, evsel_list, counter);
1290 }
1291}
1292
Ingo Molnar716c69f2009-06-07 17:31:52 +02001293static int __cmd_top(void)
1294{
1295 pthread_t thread;
Arnaldo Carvalho de Melo72cb7012011-01-12 10:52:47 -02001296 int ret;
Arnaldo Carvalho de Melod8f66242009-12-13 19:50:24 -02001297 /*
Arnaldo Carvalho de Melob3165f42009-12-13 19:50:28 -02001298 * FIXME: perf_session__new should allow passing a O_MMAP, so that all this
1299 * mmap reading, etc is encapsulated in it. Use O_WRONLY for now.
Arnaldo Carvalho de Melod8f66242009-12-13 19:50:24 -02001300 */
Ian Munsie21ef97f2010-12-10 14:09:16 +11001301 struct perf_session *session = perf_session__new(NULL, O_WRONLY, false, false, NULL);
Arnaldo Carvalho de Melob3165f42009-12-13 19:50:28 -02001302 if (session == NULL)
1303 return -ENOMEM;
Ingo Molnar07800602009-04-20 15:00:56 +02001304
Zhang, Yanmind6d901c2010-03-18 11:36:05 -03001305 if (target_tid != -1)
1306 event__synthesize_thread(target_tid, event__process, session);
Arnaldo Carvalho de Melo5b2bb752009-10-26 19:23:19 -02001307 else
Arnaldo Carvalho de Melod8f66242009-12-13 19:50:24 -02001308 event__synthesize_threads(event__process, session);
Arnaldo Carvalho de Melo5b2bb752009-10-26 19:23:19 -02001309
Arnaldo Carvalho de Melo72cb7012011-01-12 10:52:47 -02001310 start_counters(evsel_list);
Ingo Molnar07800602009-04-20 15:00:56 +02001311
Frederic Weisbecker2f011902009-06-06 23:10:43 +02001312 /* Wait for a minimal set of events before starting the snapshot */
Arnaldo Carvalho de Melo5c581042011-01-11 22:30:02 -02001313 poll(evsel_list->pollfd, evsel_list->nr_fds, 100);
Frederic Weisbecker2f011902009-06-06 23:10:43 +02001314
Arnaldo Carvalho de Melod8f66242009-12-13 19:50:24 -02001315 perf_session__mmap_read(session);
Frederic Weisbecker2f011902009-06-06 23:10:43 +02001316
Zhang, Yanmina1645ce2010-04-19 13:32:50 +08001317 if (pthread_create(&thread, NULL, display_thread, session)) {
Ingo Molnar07800602009-04-20 15:00:56 +02001318 printf("Could not create display thread.\n");
1319 exit(-1);
1320 }
1321
1322 if (realtime_prio) {
1323 struct sched_param param;
1324
1325 param.sched_priority = realtime_prio;
1326 if (sched_setscheduler(0, SCHED_FIFO, &param)) {
1327 printf("Could not set realtime priority.\n");
1328 exit(-1);
1329 }
1330 }
1331
1332 while (1) {
Ingo Molnar2debbc82009-06-05 14:29:10 +02001333 int hits = samples;
Ingo Molnar07800602009-04-20 15:00:56 +02001334
Arnaldo Carvalho de Melod8f66242009-12-13 19:50:24 -02001335 perf_session__mmap_read(session);
Ingo Molnar07800602009-04-20 15:00:56 +02001336
Ingo Molnar2debbc82009-06-05 14:29:10 +02001337 if (hits == samples)
Arnaldo Carvalho de Melo5c581042011-01-11 22:30:02 -02001338 ret = poll(evsel_list->pollfd, evsel_list->nr_fds, 100);
Ingo Molnar07800602009-04-20 15:00:56 +02001339 }
1340
1341 return 0;
1342}
Ingo Molnarb456bae2009-05-26 09:17:18 +02001343
1344static const char * const top_usage[] = {
1345 "perf top [<options>]",
1346 NULL
1347};
1348
Ingo Molnarb456bae2009-05-26 09:17:18 +02001349static const struct option options[] = {
Arnaldo Carvalho de Melo361c99a2011-01-11 20:56:53 -02001350 OPT_CALLBACK('e', "event", &evsel_list, "event",
Thomas Gleixner86847b62009-06-06 12:24:17 +02001351 "event selector. use 'perf list' to list available events",
1352 parse_events),
Ingo Molnarb456bae2009-05-26 09:17:18 +02001353 OPT_INTEGER('c', "count", &default_interval,
1354 "event period to sample"),
1355 OPT_INTEGER('p', "pid", &target_pid,
Zhang, Yanmind6d901c2010-03-18 11:36:05 -03001356 "profile events on existing process id"),
1357 OPT_INTEGER('t', "tid", &target_tid,
1358 "profile events on existing thread id"),
Ingo Molnarb456bae2009-05-26 09:17:18 +02001359 OPT_BOOLEAN('a', "all-cpus", &system_wide,
1360 "system-wide collection from all CPUs"),
Stephane Eranianc45c6ea2010-05-28 12:00:01 +02001361 OPT_STRING('C', "cpu", &cpu_list, "cpu",
1362 "list of cpus to monitor"),
Arnaldo Carvalho de Melob32d1332009-11-24 12:05:15 -02001363 OPT_STRING('k', "vmlinux", &symbol_conf.vmlinux_name,
1364 "file", "vmlinux pathname"),
Arnaldo Carvalho de Melo8ffcda12009-11-16 21:45:24 -02001365 OPT_BOOLEAN('K', "hide_kernel_symbols", &hide_kernel_symbols,
1366 "hide kernel symbols"),
Arnaldo Carvalho de Melo19679362010-05-17 15:39:16 -03001367 OPT_UINTEGER('m', "mmap-pages", &mmap_pages, "number of mmap data pages"),
Ingo Molnarb456bae2009-05-26 09:17:18 +02001368 OPT_INTEGER('r', "realtime", &realtime_prio,
1369 "collect data with this RT SCHED_FIFO priority"),
Mike Galbraithdb20c002009-05-26 15:25:34 +02001370 OPT_INTEGER('d', "delay", &delay_secs,
Ingo Molnarb456bae2009-05-26 09:17:18 +02001371 "number of seconds to delay between refreshes"),
1372 OPT_BOOLEAN('D', "dump-symtab", &dump_symtab,
1373 "dump the symbol table used for profiling"),
Ingo Molnar6e53cdf2009-06-04 08:53:05 +02001374 OPT_INTEGER('f', "count-filter", &count_filter,
Ingo Molnarb456bae2009-05-26 09:17:18 +02001375 "only display functions with more events than this"),
1376 OPT_BOOLEAN('g', "group", &group,
1377 "put the counters into a counter group"),
Mike Galbraith0fdc7e62009-07-21 10:30:36 +02001378 OPT_BOOLEAN('i', "inherit", &inherit,
1379 "child tasks inherit counters"),
Mike Galbraith923c42c2009-07-22 20:36:03 +02001380 OPT_STRING('s', "sym-annotate", &sym_filter, "symbol name",
Kirill Smelkov6cff0e82010-02-03 16:52:08 -02001381 "symbol to annotate"),
Anton Blanchard1f208ea2009-07-01 09:00:44 +10001382 OPT_BOOLEAN('z', "zero", &zero,
Ingo Molnarb456bae2009-05-26 09:17:18 +02001383 "zero history across updates"),
Ingo Molnar6e53cdf2009-06-04 08:53:05 +02001384 OPT_INTEGER('F', "freq", &freq,
Ingo Molnarb456bae2009-05-26 09:17:18 +02001385 "profile at this frequency"),
Ingo Molnar6e53cdf2009-06-04 08:53:05 +02001386 OPT_INTEGER('E', "entries", &print_entries,
1387 "display this many functions"),
Arnaldo Carvalho de Melo8ffcda12009-11-16 21:45:24 -02001388 OPT_BOOLEAN('U', "hide_user_symbols", &hide_user_symbols,
1389 "hide user symbols"),
Ian Munsiec0555642010-04-13 18:37:33 +10001390 OPT_INCR('v', "verbose", &verbose,
Ingo Molnar3da297a2009-06-07 17:39:02 +02001391 "be more verbose (show counter open errors, etc)"),
Ingo Molnarb456bae2009-05-26 09:17:18 +02001392 OPT_END()
1393};
1394
Ingo Molnarf37a2912009-07-01 12:37:06 +02001395int cmd_top(int argc, const char **argv, const char *prefix __used)
Ingo Molnarb456bae2009-05-26 09:17:18 +02001396{
Arnaldo Carvalho de Melo69aad6f2011-01-03 16:39:04 -02001397 struct perf_evsel *pos;
1398 int status = -ENOMEM;
Ingo Molnarb456bae2009-05-26 09:17:18 +02001399
Arnaldo Carvalho de Melo361c99a2011-01-11 20:56:53 -02001400 evsel_list = perf_evlist__new();
1401 if (evsel_list == NULL)
1402 return -ENOMEM;
1403
Ingo Molnarb456bae2009-05-26 09:17:18 +02001404 page_size = sysconf(_SC_PAGE_SIZE);
1405
Ingo Molnarb456bae2009-05-26 09:17:18 +02001406 argc = parse_options(argc, argv, options, top_usage, 0);
1407 if (argc)
1408 usage_with_options(top_usage, options);
1409
Arnaldo Carvalho de Melo5c98d4662011-01-03 17:53:33 -02001410 if (target_pid != -1)
Zhang, Yanmind6d901c2010-03-18 11:36:05 -03001411 target_tid = target_pid;
Zhang, Yanmind6d901c2010-03-18 11:36:05 -03001412
Arnaldo Carvalho de Melo5c98d4662011-01-03 17:53:33 -02001413 threads = thread_map__new(target_pid, target_tid);
1414 if (threads == NULL) {
1415 pr_err("Problems finding threads of monitor\n");
1416 usage_with_options(top_usage, options);
Zhang, Yanmind6d901c2010-03-18 11:36:05 -03001417 }
1418
Ingo Molnarb456bae2009-05-26 09:17:18 +02001419 /* CPU and PID are mutually exclusive */
Stephane Eranianc45c6ea2010-05-28 12:00:01 +02001420 if (target_tid > 0 && cpu_list) {
Ingo Molnarb456bae2009-05-26 09:17:18 +02001421 printf("WARNING: PID switch overriding CPU\n");
1422 sleep(1);
Stephane Eranianc45c6ea2010-05-28 12:00:01 +02001423 cpu_list = NULL;
Ingo Molnarb456bae2009-05-26 09:17:18 +02001424 }
1425
Arnaldo Carvalho de Melo361c99a2011-01-11 20:56:53 -02001426 if (!evsel_list->nr_entries &&
1427 perf_evlist__add_default(evsel_list) < 0) {
Arnaldo Carvalho de Melo69aad6f2011-01-03 16:39:04 -02001428 pr_err("Not enough memory for event selector list\n");
1429 return -ENOMEM;
1430 }
Arnaldo Carvalho de Melo5a8e5a32009-11-17 18:38:01 -02001431
Frederic Weisbecker2f335a02009-06-05 19:31:01 +02001432 if (delay_secs < 1)
1433 delay_secs = 1;
1434
Mike Galbraith7e4ff9e2009-10-12 07:56:03 +02001435 /*
1436 * User specified count overrides default frequency.
1437 */
1438 if (default_interval)
1439 freq = 0;
1440 else if (freq) {
1441 default_interval = freq;
1442 } else {
1443 fprintf(stderr, "frequency and count are zero, aborting\n");
1444 exit(EXIT_FAILURE);
1445 }
1446
Stephane Eranianc45c6ea2010-05-28 12:00:01 +02001447 if (target_tid != -1)
Arnaldo Carvalho de Melo60d567e2011-01-03 17:49:48 -02001448 cpus = cpu_map__dummy_new();
Paul Mackerrasa12b51c2010-03-10 20:36:09 +11001449 else
Arnaldo Carvalho de Melo60d567e2011-01-03 17:49:48 -02001450 cpus = cpu_map__new(cpu_list);
Stephane Eranianc45c6ea2010-05-28 12:00:01 +02001451
Arnaldo Carvalho de Melo60d567e2011-01-03 17:49:48 -02001452 if (cpus == NULL)
Stephane Eranianc45c6ea2010-05-28 12:00:01 +02001453 usage_with_options(top_usage, options);
Ingo Molnarb456bae2009-05-26 09:17:18 +02001454
Arnaldo Carvalho de Melo361c99a2011-01-11 20:56:53 -02001455 list_for_each_entry(pos, &evsel_list->entries, node) {
Arnaldo Carvalho de Melo5c98d4662011-01-03 17:53:33 -02001456 if (perf_evsel__alloc_mmap_per_thread(pos, cpus->nr, threads->nr) < 0 ||
1457 perf_evsel__alloc_fd(pos, cpus->nr, threads->nr) < 0)
Arnaldo Carvalho de Melo69aad6f2011-01-03 16:39:04 -02001458 goto out_free_fd;
1459 /*
1460 * Fill in the ones not specifically initialized via -c:
1461 */
1462 if (pos->attr.sample_period)
1463 continue;
1464
1465 pos->attr.sample_period = default_interval;
1466 }
1467
Arnaldo Carvalho de Melo5c581042011-01-11 22:30:02 -02001468 if (perf_evlist__alloc_pollfd(evsel_list, cpus->nr, threads->nr) < 0)
1469 goto out_free_fd;
1470
Arnaldo Carvalho de Melo361c99a2011-01-11 20:56:53 -02001471 sym_evsel = list_entry(evsel_list->entries.next, struct perf_evsel, node);
Arnaldo Carvalho de Melocc841582011-01-11 15:16:52 -02001472
Arnaldo Carvalho de Melo69aad6f2011-01-03 16:39:04 -02001473 symbol_conf.priv_size = (sizeof(struct sym_entry) +
Arnaldo Carvalho de Melo361c99a2011-01-11 20:56:53 -02001474 (evsel_list->nr_entries + 1) * sizeof(unsigned long));
Arnaldo Carvalho de Melo69aad6f2011-01-03 16:39:04 -02001475
1476 symbol_conf.try_vmlinux_path = (symbol_conf.vmlinux_name == NULL);
1477 if (symbol__init() < 0)
1478 return -1;
1479
Arnaldo Carvalho de Melo13cc5072009-11-17 15:40:54 -02001480 get_term_dimensions(&winsize);
Arnaldo Carvalho de Melo3b6ed982009-11-16 19:30:27 -02001481 if (print_entries == 0) {
Arnaldo Carvalho de Melo13cc5072009-11-17 15:40:54 -02001482 update_print_entries(&winsize);
Arnaldo Carvalho de Melo3b6ed982009-11-16 19:30:27 -02001483 signal(SIGWINCH, sig_winch_handler);
1484 }
1485
Arnaldo Carvalho de Melo69aad6f2011-01-03 16:39:04 -02001486 status = __cmd_top();
1487out_free_fd:
Arnaldo Carvalho de Melo361c99a2011-01-11 20:56:53 -02001488 list_for_each_entry(pos, &evsel_list->entries, node)
Arnaldo Carvalho de Melo69aad6f2011-01-03 16:39:04 -02001489 perf_evsel__free_mmap(pos);
Arnaldo Carvalho de Melo361c99a2011-01-11 20:56:53 -02001490 perf_evlist__delete(evsel_list);
Arnaldo Carvalho de Melo69aad6f2011-01-03 16:39:04 -02001491
1492 return status;
Ingo Molnarb456bae2009-05-26 09:17:18 +02001493}