blob: a97bceeac0e78fe1c210466f1df474ecd3085ab8 [file] [log] [blame]
Don Zickus9b32ba72014-06-01 15:38:29 +02001#include <sys/mman.h>
John Kacurdd68ada2009-09-24 18:02:49 +02002#include "sort.h"
Arnaldo Carvalho de Melo8a6c5b22010-07-20 14:42:52 -03003#include "hist.h"
Namhyung Kim4dfced32013-09-13 16:28:57 +09004#include "comm.h"
Namhyung Kim08e71542013-04-03 21:26:19 +09005#include "symbol.h"
Namhyung Kim8b536992014-03-03 11:46:55 +09006#include "evsel.h"
John Kacurdd68ada2009-09-24 18:02:49 +02007
8regex_t parent_regex;
Arnaldo Carvalho de Meloedb7c602010-05-17 16:22:41 -03009const char default_parent_pattern[] = "^sys_|^do_page_fault";
10const char *parent_pattern = default_parent_pattern;
11const char default_sort_order[] = "comm,dso,symbol";
Andi Kleen40997d62015-07-18 08:24:53 -070012const char default_branch_sort_order[] = "comm,dso_from,symbol_from,symbol_to,cycles";
Namhyung Kim512ae1b2014-03-18 11:31:39 +090013const char default_mem_sort_order[] = "local_weight,mem,sym,dso,symbol_daddr,dso_daddr,snoop,tlb,locked";
14const char default_top_sort_order[] = "dso,symbol";
15const char default_diff_sort_order[] = "dso,symbol";
16const char *sort_order;
Namhyung Kima7d945b2014-03-04 10:46:34 +090017const char *field_order;
Greg Priceb21484f2012-12-06 21:48:05 -080018regex_t ignore_callees_regex;
19int have_ignore_callees = 0;
Frederic Weisbeckeraf0a6fa2009-10-22 23:23:22 +020020int sort__need_collapse = 0;
21int sort__has_parent = 0;
Namhyung Kim1af556402012-09-14 17:35:27 +090022int sort__has_sym = 0;
Namhyung Kim68f6d022013-12-18 14:21:10 +090023int sort__has_dso = 0;
Namhyung Kim55369fc2013-04-01 20:35:20 +090024enum sort_mode sort__mode = SORT_MODE__NORMAL;
Frederic Weisbeckera4fb5812009-10-22 23:23:23 +020025
John Kacurdd68ada2009-09-24 18:02:49 +020026
Arnaldo Carvalho de Meloa4e3b952010-03-31 11:33:40 -030027static int repsep_snprintf(char *bf, size_t size, const char *fmt, ...)
John Kacurdd68ada2009-09-24 18:02:49 +020028{
29 int n;
30 va_list ap;
31
32 va_start(ap, fmt);
Arnaldo Carvalho de Meloa4e3b952010-03-31 11:33:40 -030033 n = vsnprintf(bf, size, fmt, ap);
Jiri Olsa0ca0c132012-09-06 17:46:56 +020034 if (symbol_conf.field_sep && n > 0) {
Arnaldo Carvalho de Meloa4e3b952010-03-31 11:33:40 -030035 char *sep = bf;
John Kacurdd68ada2009-09-24 18:02:49 +020036
Arnaldo Carvalho de Meloa4e3b952010-03-31 11:33:40 -030037 while (1) {
Jiri Olsa0ca0c132012-09-06 17:46:56 +020038 sep = strchr(sep, *symbol_conf.field_sep);
Arnaldo Carvalho de Meloa4e3b952010-03-31 11:33:40 -030039 if (sep == NULL)
40 break;
41 *sep = '.';
John Kacurdd68ada2009-09-24 18:02:49 +020042 }
John Kacurdd68ada2009-09-24 18:02:49 +020043 }
44 va_end(ap);
Anton Blanchardb8327962012-03-07 11:42:49 +110045
46 if (n >= (int)size)
47 return size - 1;
John Kacurdd68ada2009-09-24 18:02:49 +020048 return n;
49}
50
Frederic Weisbeckerb9c51432013-09-11 14:46:56 +020051static int64_t cmp_null(const void *l, const void *r)
Frederic Weisbecker872a8782011-06-29 03:14:52 +020052{
53 if (!l && !r)
54 return 0;
55 else if (!l)
56 return -1;
57 else
58 return 1;
59}
60
61/* --sort pid */
62
63static int64_t
64sort__thread_cmp(struct hist_entry *left, struct hist_entry *right)
65{
Adrian Hunter38051232013-07-04 16:20:31 +030066 return right->thread->tid - left->thread->tid;
Frederic Weisbecker872a8782011-06-29 03:14:52 +020067}
68
Arnaldo Carvalho de Meloc824c432013-10-22 19:01:31 -030069static int hist_entry__thread_snprintf(struct hist_entry *he, char *bf,
Arnaldo Carvalho de Meloa4e3b952010-03-31 11:33:40 -030070 size_t size, unsigned int width)
John Kacurdd68ada2009-09-24 18:02:49 +020071{
Frederic Weisbeckerb9c51432013-09-11 14:46:56 +020072 const char *comm = thread__comm_str(he->thread);
Namhyung Kim5b591662014-07-31 14:47:38 +090073
74 width = max(7U, width) - 6;
75 return repsep_snprintf(bf, size, "%5d:%-*.*s", he->thread->tid,
76 width, width, comm ?: "");
John Kacurdd68ada2009-09-24 18:02:49 +020077}
78
Frederic Weisbecker872a8782011-06-29 03:14:52 +020079struct sort_entry sort_thread = {
Namhyung Kim8246de82014-07-31 14:47:35 +090080 .se_header = " Pid:Command",
Frederic Weisbecker872a8782011-06-29 03:14:52 +020081 .se_cmp = sort__thread_cmp,
82 .se_snprintf = hist_entry__thread_snprintf,
83 .se_width_idx = HISTC_THREAD,
84};
85
86/* --sort comm */
87
88static int64_t
89sort__comm_cmp(struct hist_entry *left, struct hist_entry *right)
90{
Frederic Weisbeckerfedd63d2013-09-11 17:18:09 +020091 /* Compare the addr that should be unique among comm */
Jiri Olsa2f15bd82015-05-15 17:54:28 +020092 return strcmp(comm__str(right->comm), comm__str(left->comm));
Frederic Weisbecker872a8782011-06-29 03:14:52 +020093}
94
95static int64_t
96sort__comm_collapse(struct hist_entry *left, struct hist_entry *right)
97{
Namhyung Kim4dfced32013-09-13 16:28:57 +090098 /* Compare the addr that should be unique among comm */
Jiri Olsa2f15bd82015-05-15 17:54:28 +020099 return strcmp(comm__str(right->comm), comm__str(left->comm));
Frederic Weisbecker872a8782011-06-29 03:14:52 +0200100}
101
Namhyung Kim202e7a62014-03-04 11:01:41 +0900102static int64_t
103sort__comm_sort(struct hist_entry *left, struct hist_entry *right)
104{
105 return strcmp(comm__str(right->comm), comm__str(left->comm));
106}
107
Arnaldo Carvalho de Meloc824c432013-10-22 19:01:31 -0300108static int hist_entry__comm_snprintf(struct hist_entry *he, char *bf,
Arnaldo Carvalho de Meloa4e3b952010-03-31 11:33:40 -0300109 size_t size, unsigned int width)
John Kacurdd68ada2009-09-24 18:02:49 +0200110{
Namhyung Kim5b591662014-07-31 14:47:38 +0900111 return repsep_snprintf(bf, size, "%-*.*s", width, width, comm__str(he->comm));
John Kacurdd68ada2009-09-24 18:02:49 +0200112}
113
Namhyung Kim14d1ac72012-12-27 18:11:38 +0900114struct sort_entry sort_comm = {
115 .se_header = "Command",
116 .se_cmp = sort__comm_cmp,
117 .se_collapse = sort__comm_collapse,
Namhyung Kim202e7a62014-03-04 11:01:41 +0900118 .se_sort = sort__comm_sort,
Namhyung Kim14d1ac72012-12-27 18:11:38 +0900119 .se_snprintf = hist_entry__comm_snprintf,
120 .se_width_idx = HISTC_COMM,
121};
122
123/* --sort dso */
124
Roberto Agostino Vitillob5387522012-02-09 23:21:01 +0100125static int64_t _sort__dso_cmp(struct map *map_l, struct map *map_r)
John Kacurdd68ada2009-09-24 18:02:49 +0200126{
Roberto Agostino Vitillob5387522012-02-09 23:21:01 +0100127 struct dso *dso_l = map_l ? map_l->dso : NULL;
128 struct dso *dso_r = map_r ? map_r->dso : NULL;
Arnaldo Carvalho de Melo439d4732009-10-02 03:29:58 -0300129 const char *dso_name_l, *dso_name_r;
John Kacurdd68ada2009-09-24 18:02:49 +0200130
131 if (!dso_l || !dso_r)
Namhyung Kim202e7a62014-03-04 11:01:41 +0900132 return cmp_null(dso_r, dso_l);
John Kacurdd68ada2009-09-24 18:02:49 +0200133
Arnaldo Carvalho de Melo439d4732009-10-02 03:29:58 -0300134 if (verbose) {
135 dso_name_l = dso_l->long_name;
136 dso_name_r = dso_r->long_name;
137 } else {
138 dso_name_l = dso_l->short_name;
139 dso_name_r = dso_r->short_name;
140 }
141
142 return strcmp(dso_name_l, dso_name_r);
John Kacurdd68ada2009-09-24 18:02:49 +0200143}
144
Roberto Agostino Vitillob5387522012-02-09 23:21:01 +0100145static int64_t
146sort__dso_cmp(struct hist_entry *left, struct hist_entry *right)
John Kacurdd68ada2009-09-24 18:02:49 +0200147{
Namhyung Kim202e7a62014-03-04 11:01:41 +0900148 return _sort__dso_cmp(right->ms.map, left->ms.map);
Roberto Agostino Vitillob5387522012-02-09 23:21:01 +0100149}
150
Roberto Agostino Vitillob5387522012-02-09 23:21:01 +0100151static int _hist_entry__dso_snprintf(struct map *map, char *bf,
152 size_t size, unsigned int width)
153{
154 if (map && map->dso) {
155 const char *dso_name = !verbose ? map->dso->short_name :
156 map->dso->long_name;
Namhyung Kim5b591662014-07-31 14:47:38 +0900157 return repsep_snprintf(bf, size, "%-*.*s", width, width, dso_name);
Arnaldo Carvalho de Melo439d4732009-10-02 03:29:58 -0300158 }
John Kacurdd68ada2009-09-24 18:02:49 +0200159
Namhyung Kim5b591662014-07-31 14:47:38 +0900160 return repsep_snprintf(bf, size, "%-*.*s", width, width, "[unknown]");
John Kacurdd68ada2009-09-24 18:02:49 +0200161}
162
Arnaldo Carvalho de Meloc824c432013-10-22 19:01:31 -0300163static int hist_entry__dso_snprintf(struct hist_entry *he, char *bf,
Roberto Agostino Vitillob5387522012-02-09 23:21:01 +0100164 size_t size, unsigned int width)
165{
Arnaldo Carvalho de Meloc824c432013-10-22 19:01:31 -0300166 return _hist_entry__dso_snprintf(he->ms.map, bf, size, width);
Roberto Agostino Vitillob5387522012-02-09 23:21:01 +0100167}
168
Namhyung Kim14d1ac72012-12-27 18:11:38 +0900169struct sort_entry sort_dso = {
170 .se_header = "Shared Object",
171 .se_cmp = sort__dso_cmp,
172 .se_snprintf = hist_entry__dso_snprintf,
173 .se_width_idx = HISTC_DSO,
174};
175
176/* --sort symbol */
177
Namhyung Kim2037be52013-12-18 14:21:09 +0900178static int64_t _sort__addr_cmp(u64 left_ip, u64 right_ip)
179{
180 return (int64_t)(right_ip - left_ip);
181}
182
Namhyung Kim51f27d12013-02-06 14:57:15 +0900183static int64_t _sort__sym_cmp(struct symbol *sym_l, struct symbol *sym_r)
Namhyung Kim14d1ac72012-12-27 18:11:38 +0900184{
185 if (!sym_l || !sym_r)
186 return cmp_null(sym_l, sym_r);
187
188 if (sym_l == sym_r)
189 return 0;
190
Yannick Brosseauc05676c2015-06-17 16:41:10 -0700191 if (sym_l->start != sym_r->start)
192 return (int64_t)(sym_r->start - sym_l->start);
Namhyung Kim14d1ac72012-12-27 18:11:38 +0900193
Yannick Brosseauc05676c2015-06-17 16:41:10 -0700194 return (int64_t)(sym_r->end - sym_l->end);
Namhyung Kim14d1ac72012-12-27 18:11:38 +0900195}
196
197static int64_t
198sort__sym_cmp(struct hist_entry *left, struct hist_entry *right)
199{
Namhyung Kim09600e02013-10-15 11:01:56 +0900200 int64_t ret;
201
Namhyung Kim14d1ac72012-12-27 18:11:38 +0900202 if (!left->ms.sym && !right->ms.sym)
Namhyung Kim2037be52013-12-18 14:21:09 +0900203 return _sort__addr_cmp(left->ip, right->ip);
Namhyung Kim14d1ac72012-12-27 18:11:38 +0900204
Namhyung Kim09600e02013-10-15 11:01:56 +0900205 /*
206 * comparing symbol address alone is not enough since it's a
207 * relative address within a dso.
208 */
Namhyung Kim68f6d022013-12-18 14:21:10 +0900209 if (!sort__has_dso) {
210 ret = sort__dso_cmp(left, right);
211 if (ret != 0)
212 return ret;
213 }
Namhyung Kim09600e02013-10-15 11:01:56 +0900214
Namhyung Kim51f27d12013-02-06 14:57:15 +0900215 return _sort__sym_cmp(left->ms.sym, right->ms.sym);
Namhyung Kim14d1ac72012-12-27 18:11:38 +0900216}
217
Namhyung Kim202e7a62014-03-04 11:01:41 +0900218static int64_t
219sort__sym_sort(struct hist_entry *left, struct hist_entry *right)
220{
221 if (!left->ms.sym || !right->ms.sym)
222 return cmp_null(left->ms.sym, right->ms.sym);
223
224 return strcmp(right->ms.sym->name, left->ms.sym->name);
225}
226
Roberto Agostino Vitillob5387522012-02-09 23:21:01 +0100227static int _hist_entry__sym_snprintf(struct map *map, struct symbol *sym,
228 u64 ip, char level, char *bf, size_t size,
Namhyung Kim43355522012-12-27 18:11:39 +0900229 unsigned int width)
Roberto Agostino Vitillob5387522012-02-09 23:21:01 +0100230{
231 size_t ret = 0;
232
233 if (verbose) {
234 char o = map ? dso__symtab_origin(map->dso) : '!';
235 ret += repsep_snprintf(bf, size, "%-#*llx %c ",
Namhyung Kimded19d52013-04-01 20:35:19 +0900236 BITS_PER_LONG / 4 + 2, ip, o);
Roberto Agostino Vitillob5387522012-02-09 23:21:01 +0100237 }
238
239 ret += repsep_snprintf(bf + ret, size - ret, "[%c] ", level);
Stephane Eranian98a3b322013-01-24 16:10:35 +0100240 if (sym && map) {
241 if (map->type == MAP__VARIABLE) {
242 ret += repsep_snprintf(bf + ret, size - ret, "%s", sym->name);
243 ret += repsep_snprintf(bf + ret, size - ret, "+0x%llx",
Namhyung Kim62667742013-01-24 16:10:42 +0100244 ip - map->unmap_ip(map, sym->start));
Stephane Eranian98a3b322013-01-24 16:10:35 +0100245 ret += repsep_snprintf(bf + ret, size - ret, "%-*s",
246 width - ret, "");
247 } else {
248 ret += repsep_snprintf(bf + ret, size - ret, "%-*s",
249 width - ret,
250 sym->name);
251 }
252 } else {
Roberto Agostino Vitillob5387522012-02-09 23:21:01 +0100253 size_t len = BITS_PER_LONG / 4;
254 ret += repsep_snprintf(bf + ret, size - ret, "%-#.*llx",
255 len, ip);
256 ret += repsep_snprintf(bf + ret, size - ret, "%-*s",
257 width - ret, "");
258 }
259
Namhyung Kim5b591662014-07-31 14:47:38 +0900260 if (ret > width)
261 bf[width] = '\0';
262
263 return width;
Roberto Agostino Vitillob5387522012-02-09 23:21:01 +0100264}
265
Arnaldo Carvalho de Meloc824c432013-10-22 19:01:31 -0300266static int hist_entry__sym_snprintf(struct hist_entry *he, char *bf,
Namhyung Kim43355522012-12-27 18:11:39 +0900267 size_t size, unsigned int width)
Roberto Agostino Vitillob5387522012-02-09 23:21:01 +0100268{
Arnaldo Carvalho de Meloc824c432013-10-22 19:01:31 -0300269 return _hist_entry__sym_snprintf(he->ms.map, he->ms.sym, he->ip,
270 he->level, bf, size, width);
Roberto Agostino Vitillob5387522012-02-09 23:21:01 +0100271}
John Kacurdd68ada2009-09-24 18:02:49 +0200272
Frederic Weisbecker872a8782011-06-29 03:14:52 +0200273struct sort_entry sort_sym = {
274 .se_header = "Symbol",
275 .se_cmp = sort__sym_cmp,
Namhyung Kim202e7a62014-03-04 11:01:41 +0900276 .se_sort = sort__sym_sort,
Frederic Weisbecker872a8782011-06-29 03:14:52 +0200277 .se_snprintf = hist_entry__sym_snprintf,
278 .se_width_idx = HISTC_SYMBOL,
279};
John Kacurdd68ada2009-09-24 18:02:49 +0200280
Arnaldo Carvalho de Melo409a8be2012-05-30 10:33:24 -0300281/* --sort srcline */
282
283static int64_t
284sort__srcline_cmp(struct hist_entry *left, struct hist_entry *right)
285{
Namhyung Kim4adcc432013-09-11 14:09:33 +0900286 if (!left->srcline) {
287 if (!left->ms.map)
288 left->srcline = SRCLINE_UNKNOWN;
289 else {
290 struct map *map = left->ms.map;
291 left->srcline = get_srcline(map->dso,
Andi Kleen85c116a2014-11-12 18:05:27 -0800292 map__rip_2objdump(map, left->ip),
293 left->ms.sym, true);
Namhyung Kim4adcc432013-09-11 14:09:33 +0900294 }
295 }
296 if (!right->srcline) {
297 if (!right->ms.map)
298 right->srcline = SRCLINE_UNKNOWN;
299 else {
300 struct map *map = right->ms.map;
301 right->srcline = get_srcline(map->dso,
Andi Kleen85c116a2014-11-12 18:05:27 -0800302 map__rip_2objdump(map, right->ip),
303 right->ms.sym, true);
Namhyung Kim4adcc432013-09-11 14:09:33 +0900304 }
305 }
Namhyung Kim202e7a62014-03-04 11:01:41 +0900306 return strcmp(right->srcline, left->srcline);
Arnaldo Carvalho de Melo409a8be2012-05-30 10:33:24 -0300307}
308
Arnaldo Carvalho de Meloc824c432013-10-22 19:01:31 -0300309static int hist_entry__srcline_snprintf(struct hist_entry *he, char *bf,
Namhyung Kim5b591662014-07-31 14:47:38 +0900310 size_t size, unsigned int width)
Arnaldo Carvalho de Melo409a8be2012-05-30 10:33:24 -0300311{
Arnaldo Carvalho de Melob2d53672014-11-18 18:02:51 -0300312 return repsep_snprintf(bf, size, "%-*.*s", width, width, he->srcline);
Arnaldo Carvalho de Melo409a8be2012-05-30 10:33:24 -0300313}
314
315struct sort_entry sort_srcline = {
316 .se_header = "Source:Line",
317 .se_cmp = sort__srcline_cmp,
318 .se_snprintf = hist_entry__srcline_snprintf,
319 .se_width_idx = HISTC_SRCLINE,
320};
321
Andi Kleen31191a82015-08-07 15:54:24 -0700322/* --sort srcfile */
323
324static char no_srcfile[1];
325
326static char *get_srcfile(struct hist_entry *e)
327{
328 char *sf, *p;
329 struct map *map = e->ms.map;
330
Andi Kleen2f84b422015-09-01 11:47:19 -0700331 sf = __get_srcline(map->dso, map__rip_2objdump(map, e->ip),
332 e->ms.sym, false, true);
Andi Kleen76b10652015-08-11 06:36:55 -0700333 if (!strcmp(sf, SRCLINE_UNKNOWN))
334 return no_srcfile;
Andi Kleen31191a82015-08-07 15:54:24 -0700335 p = strchr(sf, ':');
336 if (p && *sf) {
337 *p = 0;
338 return sf;
339 }
340 free(sf);
341 return no_srcfile;
342}
343
344static int64_t
345sort__srcfile_cmp(struct hist_entry *left, struct hist_entry *right)
346{
347 if (!left->srcfile) {
348 if (!left->ms.map)
349 left->srcfile = no_srcfile;
350 else
351 left->srcfile = get_srcfile(left);
352 }
353 if (!right->srcfile) {
354 if (!right->ms.map)
355 right->srcfile = no_srcfile;
356 else
357 right->srcfile = get_srcfile(right);
358 }
359 return strcmp(right->srcfile, left->srcfile);
360}
361
362static int hist_entry__srcfile_snprintf(struct hist_entry *he, char *bf,
363 size_t size, unsigned int width)
364{
365 return repsep_snprintf(bf, size, "%-*.*s", width, width, he->srcfile);
366}
367
368struct sort_entry sort_srcfile = {
369 .se_header = "Source File",
370 .se_cmp = sort__srcfile_cmp,
371 .se_snprintf = hist_entry__srcfile_snprintf,
372 .se_width_idx = HISTC_SRCFILE,
373};
374
John Kacurdd68ada2009-09-24 18:02:49 +0200375/* --sort parent */
376
Frederic Weisbecker872a8782011-06-29 03:14:52 +0200377static int64_t
John Kacurdd68ada2009-09-24 18:02:49 +0200378sort__parent_cmp(struct hist_entry *left, struct hist_entry *right)
379{
380 struct symbol *sym_l = left->parent;
381 struct symbol *sym_r = right->parent;
382
383 if (!sym_l || !sym_r)
384 return cmp_null(sym_l, sym_r);
385
Namhyung Kim202e7a62014-03-04 11:01:41 +0900386 return strcmp(sym_r->name, sym_l->name);
John Kacurdd68ada2009-09-24 18:02:49 +0200387}
388
Arnaldo Carvalho de Meloc824c432013-10-22 19:01:31 -0300389static int hist_entry__parent_snprintf(struct hist_entry *he, char *bf,
Arnaldo Carvalho de Meloa4e3b952010-03-31 11:33:40 -0300390 size_t size, unsigned int width)
John Kacurdd68ada2009-09-24 18:02:49 +0200391{
Namhyung Kim5b591662014-07-31 14:47:38 +0900392 return repsep_snprintf(bf, size, "%-*.*s", width, width,
Arnaldo Carvalho de Meloc824c432013-10-22 19:01:31 -0300393 he->parent ? he->parent->name : "[other]");
John Kacurdd68ada2009-09-24 18:02:49 +0200394}
395
Frederic Weisbecker872a8782011-06-29 03:14:52 +0200396struct sort_entry sort_parent = {
397 .se_header = "Parent symbol",
398 .se_cmp = sort__parent_cmp,
399 .se_snprintf = hist_entry__parent_snprintf,
400 .se_width_idx = HISTC_PARENT,
401};
402
Arun Sharmaf60f3592010-06-04 11:27:10 -0300403/* --sort cpu */
404
Frederic Weisbecker872a8782011-06-29 03:14:52 +0200405static int64_t
Arun Sharmaf60f3592010-06-04 11:27:10 -0300406sort__cpu_cmp(struct hist_entry *left, struct hist_entry *right)
407{
408 return right->cpu - left->cpu;
409}
410
Arnaldo Carvalho de Meloc824c432013-10-22 19:01:31 -0300411static int hist_entry__cpu_snprintf(struct hist_entry *he, char *bf,
412 size_t size, unsigned int width)
Arun Sharmaf60f3592010-06-04 11:27:10 -0300413{
Namhyung Kim5b591662014-07-31 14:47:38 +0900414 return repsep_snprintf(bf, size, "%*.*d", width, width, he->cpu);
Arun Sharmaf60f3592010-06-04 11:27:10 -0300415}
416
Frederic Weisbecker872a8782011-06-29 03:14:52 +0200417struct sort_entry sort_cpu = {
418 .se_header = "CPU",
419 .se_cmp = sort__cpu_cmp,
420 .se_snprintf = hist_entry__cpu_snprintf,
421 .se_width_idx = HISTC_CPU,
422};
423
Namhyung Kim14d1ac72012-12-27 18:11:38 +0900424/* sort keys for branch stacks */
425
Roberto Agostino Vitillob5387522012-02-09 23:21:01 +0100426static int64_t
427sort__dso_from_cmp(struct hist_entry *left, struct hist_entry *right)
428{
Jiri Olsa288a4b92014-10-16 16:07:07 +0200429 if (!left->branch_info || !right->branch_info)
430 return cmp_null(left->branch_info, right->branch_info);
431
Roberto Agostino Vitillob5387522012-02-09 23:21:01 +0100432 return _sort__dso_cmp(left->branch_info->from.map,
433 right->branch_info->from.map);
434}
435
Arnaldo Carvalho de Meloc824c432013-10-22 19:01:31 -0300436static int hist_entry__dso_from_snprintf(struct hist_entry *he, char *bf,
Roberto Agostino Vitillob5387522012-02-09 23:21:01 +0100437 size_t size, unsigned int width)
438{
Jiri Olsa288a4b92014-10-16 16:07:07 +0200439 if (he->branch_info)
440 return _hist_entry__dso_snprintf(he->branch_info->from.map,
441 bf, size, width);
442 else
443 return repsep_snprintf(bf, size, "%-*.*s", width, width, "N/A");
Roberto Agostino Vitillob5387522012-02-09 23:21:01 +0100444}
445
Roberto Agostino Vitillob5387522012-02-09 23:21:01 +0100446static int64_t
447sort__dso_to_cmp(struct hist_entry *left, struct hist_entry *right)
448{
Jiri Olsa8b62fa52014-10-16 16:07:06 +0200449 if (!left->branch_info || !right->branch_info)
450 return cmp_null(left->branch_info, right->branch_info);
451
Roberto Agostino Vitillob5387522012-02-09 23:21:01 +0100452 return _sort__dso_cmp(left->branch_info->to.map,
453 right->branch_info->to.map);
454}
455
Arnaldo Carvalho de Meloc824c432013-10-22 19:01:31 -0300456static int hist_entry__dso_to_snprintf(struct hist_entry *he, char *bf,
Roberto Agostino Vitillob5387522012-02-09 23:21:01 +0100457 size_t size, unsigned int width)
458{
Jiri Olsa8b62fa52014-10-16 16:07:06 +0200459 if (he->branch_info)
460 return _hist_entry__dso_snprintf(he->branch_info->to.map,
461 bf, size, width);
462 else
463 return repsep_snprintf(bf, size, "%-*.*s", width, width, "N/A");
Roberto Agostino Vitillob5387522012-02-09 23:21:01 +0100464}
465
466static int64_t
467sort__sym_from_cmp(struct hist_entry *left, struct hist_entry *right)
468{
469 struct addr_map_symbol *from_l = &left->branch_info->from;
470 struct addr_map_symbol *from_r = &right->branch_info->from;
471
Jiri Olsa1b9e97a2014-10-16 16:07:05 +0200472 if (!left->branch_info || !right->branch_info)
473 return cmp_null(left->branch_info, right->branch_info);
474
475 from_l = &left->branch_info->from;
476 from_r = &right->branch_info->from;
477
Roberto Agostino Vitillob5387522012-02-09 23:21:01 +0100478 if (!from_l->sym && !from_r->sym)
Namhyung Kim2037be52013-12-18 14:21:09 +0900479 return _sort__addr_cmp(from_l->addr, from_r->addr);
Roberto Agostino Vitillob5387522012-02-09 23:21:01 +0100480
Namhyung Kim51f27d12013-02-06 14:57:15 +0900481 return _sort__sym_cmp(from_l->sym, from_r->sym);
Roberto Agostino Vitillob5387522012-02-09 23:21:01 +0100482}
483
484static int64_t
485sort__sym_to_cmp(struct hist_entry *left, struct hist_entry *right)
486{
Jiri Olsa38cdbd32014-10-16 16:07:04 +0200487 struct addr_map_symbol *to_l, *to_r;
488
489 if (!left->branch_info || !right->branch_info)
490 return cmp_null(left->branch_info, right->branch_info);
491
492 to_l = &left->branch_info->to;
493 to_r = &right->branch_info->to;
Roberto Agostino Vitillob5387522012-02-09 23:21:01 +0100494
495 if (!to_l->sym && !to_r->sym)
Namhyung Kim2037be52013-12-18 14:21:09 +0900496 return _sort__addr_cmp(to_l->addr, to_r->addr);
Roberto Agostino Vitillob5387522012-02-09 23:21:01 +0100497
Namhyung Kim51f27d12013-02-06 14:57:15 +0900498 return _sort__sym_cmp(to_l->sym, to_r->sym);
Roberto Agostino Vitillob5387522012-02-09 23:21:01 +0100499}
500
Arnaldo Carvalho de Meloc824c432013-10-22 19:01:31 -0300501static int hist_entry__sym_from_snprintf(struct hist_entry *he, char *bf,
Namhyung Kim43355522012-12-27 18:11:39 +0900502 size_t size, unsigned int width)
Roberto Agostino Vitillob5387522012-02-09 23:21:01 +0100503{
Jiri Olsa1b9e97a2014-10-16 16:07:05 +0200504 if (he->branch_info) {
505 struct addr_map_symbol *from = &he->branch_info->from;
Roberto Agostino Vitillob5387522012-02-09 23:21:01 +0100506
Jiri Olsa1b9e97a2014-10-16 16:07:05 +0200507 return _hist_entry__sym_snprintf(from->map, from->sym, from->addr,
508 he->level, bf, size, width);
509 }
510
511 return repsep_snprintf(bf, size, "%-*.*s", width, width, "N/A");
Roberto Agostino Vitillob5387522012-02-09 23:21:01 +0100512}
513
Arnaldo Carvalho de Meloc824c432013-10-22 19:01:31 -0300514static int hist_entry__sym_to_snprintf(struct hist_entry *he, char *bf,
Namhyung Kim43355522012-12-27 18:11:39 +0900515 size_t size, unsigned int width)
Roberto Agostino Vitillob5387522012-02-09 23:21:01 +0100516{
Jiri Olsa38cdbd32014-10-16 16:07:04 +0200517 if (he->branch_info) {
518 struct addr_map_symbol *to = &he->branch_info->to;
Roberto Agostino Vitillob5387522012-02-09 23:21:01 +0100519
Jiri Olsa38cdbd32014-10-16 16:07:04 +0200520 return _hist_entry__sym_snprintf(to->map, to->sym, to->addr,
521 he->level, bf, size, width);
522 }
523
524 return repsep_snprintf(bf, size, "%-*.*s", width, width, "N/A");
Roberto Agostino Vitillob5387522012-02-09 23:21:01 +0100525}
526
Namhyung Kim14d1ac72012-12-27 18:11:38 +0900527struct sort_entry sort_dso_from = {
528 .se_header = "Source Shared Object",
529 .se_cmp = sort__dso_from_cmp,
530 .se_snprintf = hist_entry__dso_from_snprintf,
531 .se_width_idx = HISTC_DSO_FROM,
532};
533
Roberto Agostino Vitillob5387522012-02-09 23:21:01 +0100534struct sort_entry sort_dso_to = {
535 .se_header = "Target Shared Object",
536 .se_cmp = sort__dso_to_cmp,
537 .se_snprintf = hist_entry__dso_to_snprintf,
538 .se_width_idx = HISTC_DSO_TO,
539};
540
541struct sort_entry sort_sym_from = {
542 .se_header = "Source Symbol",
543 .se_cmp = sort__sym_from_cmp,
544 .se_snprintf = hist_entry__sym_from_snprintf,
545 .se_width_idx = HISTC_SYMBOL_FROM,
546};
547
548struct sort_entry sort_sym_to = {
549 .se_header = "Target Symbol",
550 .se_cmp = sort__sym_to_cmp,
551 .se_snprintf = hist_entry__sym_to_snprintf,
552 .se_width_idx = HISTC_SYMBOL_TO,
553};
554
555static int64_t
556sort__mispredict_cmp(struct hist_entry *left, struct hist_entry *right)
557{
Jiri Olsa428560e2014-10-16 16:07:03 +0200558 unsigned char mp, p;
Roberto Agostino Vitillob5387522012-02-09 23:21:01 +0100559
Jiri Olsa428560e2014-10-16 16:07:03 +0200560 if (!left->branch_info || !right->branch_info)
561 return cmp_null(left->branch_info, right->branch_info);
562
563 mp = left->branch_info->flags.mispred != right->branch_info->flags.mispred;
564 p = left->branch_info->flags.predicted != right->branch_info->flags.predicted;
Roberto Agostino Vitillob5387522012-02-09 23:21:01 +0100565 return mp || p;
566}
567
Arnaldo Carvalho de Meloc824c432013-10-22 19:01:31 -0300568static int hist_entry__mispredict_snprintf(struct hist_entry *he, char *bf,
Roberto Agostino Vitillob5387522012-02-09 23:21:01 +0100569 size_t size, unsigned int width){
570 static const char *out = "N/A";
571
Jiri Olsa428560e2014-10-16 16:07:03 +0200572 if (he->branch_info) {
573 if (he->branch_info->flags.predicted)
574 out = "N";
575 else if (he->branch_info->flags.mispred)
576 out = "Y";
577 }
Roberto Agostino Vitillob5387522012-02-09 23:21:01 +0100578
Namhyung Kim5b591662014-07-31 14:47:38 +0900579 return repsep_snprintf(bf, size, "%-*.*s", width, width, out);
Roberto Agostino Vitillob5387522012-02-09 23:21:01 +0100580}
581
Andi Kleen0e332f02015-07-18 08:24:46 -0700582static int64_t
583sort__cycles_cmp(struct hist_entry *left, struct hist_entry *right)
584{
585 return left->branch_info->flags.cycles -
586 right->branch_info->flags.cycles;
587}
588
589static int hist_entry__cycles_snprintf(struct hist_entry *he, char *bf,
590 size_t size, unsigned int width)
591{
592 if (he->branch_info->flags.cycles == 0)
593 return repsep_snprintf(bf, size, "%-*s", width, "-");
594 return repsep_snprintf(bf, size, "%-*hd", width,
595 he->branch_info->flags.cycles);
596}
597
598struct sort_entry sort_cycles = {
599 .se_header = "Basic Block Cycles",
600 .se_cmp = sort__cycles_cmp,
601 .se_snprintf = hist_entry__cycles_snprintf,
602 .se_width_idx = HISTC_CYCLES,
603};
604
Stephane Eranian98a3b322013-01-24 16:10:35 +0100605/* --sort daddr_sym */
606static int64_t
607sort__daddr_cmp(struct hist_entry *left, struct hist_entry *right)
608{
609 uint64_t l = 0, r = 0;
610
611 if (left->mem_info)
612 l = left->mem_info->daddr.addr;
613 if (right->mem_info)
614 r = right->mem_info->daddr.addr;
615
616 return (int64_t)(r - l);
617}
618
Arnaldo Carvalho de Meloc824c432013-10-22 19:01:31 -0300619static int hist_entry__daddr_snprintf(struct hist_entry *he, char *bf,
Stephane Eranian98a3b322013-01-24 16:10:35 +0100620 size_t size, unsigned int width)
621{
622 uint64_t addr = 0;
623 struct map *map = NULL;
624 struct symbol *sym = NULL;
625
Arnaldo Carvalho de Meloc824c432013-10-22 19:01:31 -0300626 if (he->mem_info) {
627 addr = he->mem_info->daddr.addr;
628 map = he->mem_info->daddr.map;
629 sym = he->mem_info->daddr.sym;
Stephane Eranian98a3b322013-01-24 16:10:35 +0100630 }
Arnaldo Carvalho de Meloc824c432013-10-22 19:01:31 -0300631 return _hist_entry__sym_snprintf(map, sym, addr, he->level, bf, size,
Stephane Eranian98a3b322013-01-24 16:10:35 +0100632 width);
633}
634
635static int64_t
636sort__dso_daddr_cmp(struct hist_entry *left, struct hist_entry *right)
637{
638 struct map *map_l = NULL;
639 struct map *map_r = NULL;
640
641 if (left->mem_info)
642 map_l = left->mem_info->daddr.map;
643 if (right->mem_info)
644 map_r = right->mem_info->daddr.map;
645
646 return _sort__dso_cmp(map_l, map_r);
647}
648
Arnaldo Carvalho de Meloc824c432013-10-22 19:01:31 -0300649static int hist_entry__dso_daddr_snprintf(struct hist_entry *he, char *bf,
Stephane Eranian98a3b322013-01-24 16:10:35 +0100650 size_t size, unsigned int width)
651{
652 struct map *map = NULL;
653
Arnaldo Carvalho de Meloc824c432013-10-22 19:01:31 -0300654 if (he->mem_info)
655 map = he->mem_info->daddr.map;
Stephane Eranian98a3b322013-01-24 16:10:35 +0100656
657 return _hist_entry__dso_snprintf(map, bf, size, width);
658}
659
660static int64_t
661sort__locked_cmp(struct hist_entry *left, struct hist_entry *right)
662{
663 union perf_mem_data_src data_src_l;
664 union perf_mem_data_src data_src_r;
665
666 if (left->mem_info)
667 data_src_l = left->mem_info->data_src;
668 else
669 data_src_l.mem_lock = PERF_MEM_LOCK_NA;
670
671 if (right->mem_info)
672 data_src_r = right->mem_info->data_src;
673 else
674 data_src_r.mem_lock = PERF_MEM_LOCK_NA;
675
676 return (int64_t)(data_src_r.mem_lock - data_src_l.mem_lock);
677}
678
Arnaldo Carvalho de Meloc824c432013-10-22 19:01:31 -0300679static int hist_entry__locked_snprintf(struct hist_entry *he, char *bf,
Stephane Eranian98a3b322013-01-24 16:10:35 +0100680 size_t size, unsigned int width)
681{
682 const char *out;
683 u64 mask = PERF_MEM_LOCK_NA;
684
Arnaldo Carvalho de Meloc824c432013-10-22 19:01:31 -0300685 if (he->mem_info)
686 mask = he->mem_info->data_src.mem_lock;
Stephane Eranian98a3b322013-01-24 16:10:35 +0100687
688 if (mask & PERF_MEM_LOCK_NA)
689 out = "N/A";
690 else if (mask & PERF_MEM_LOCK_LOCKED)
691 out = "Yes";
692 else
693 out = "No";
694
695 return repsep_snprintf(bf, size, "%-*s", width, out);
696}
697
698static int64_t
699sort__tlb_cmp(struct hist_entry *left, struct hist_entry *right)
700{
701 union perf_mem_data_src data_src_l;
702 union perf_mem_data_src data_src_r;
703
704 if (left->mem_info)
705 data_src_l = left->mem_info->data_src;
706 else
707 data_src_l.mem_dtlb = PERF_MEM_TLB_NA;
708
709 if (right->mem_info)
710 data_src_r = right->mem_info->data_src;
711 else
712 data_src_r.mem_dtlb = PERF_MEM_TLB_NA;
713
714 return (int64_t)(data_src_r.mem_dtlb - data_src_l.mem_dtlb);
715}
716
717static const char * const tlb_access[] = {
718 "N/A",
719 "HIT",
720 "MISS",
721 "L1",
722 "L2",
723 "Walker",
724 "Fault",
725};
726#define NUM_TLB_ACCESS (sizeof(tlb_access)/sizeof(const char *))
727
Arnaldo Carvalho de Meloc824c432013-10-22 19:01:31 -0300728static int hist_entry__tlb_snprintf(struct hist_entry *he, char *bf,
Stephane Eranian98a3b322013-01-24 16:10:35 +0100729 size_t size, unsigned int width)
730{
731 char out[64];
732 size_t sz = sizeof(out) - 1; /* -1 for null termination */
733 size_t l = 0, i;
734 u64 m = PERF_MEM_TLB_NA;
735 u64 hit, miss;
736
737 out[0] = '\0';
738
Arnaldo Carvalho de Meloc824c432013-10-22 19:01:31 -0300739 if (he->mem_info)
740 m = he->mem_info->data_src.mem_dtlb;
Stephane Eranian98a3b322013-01-24 16:10:35 +0100741
742 hit = m & PERF_MEM_TLB_HIT;
743 miss = m & PERF_MEM_TLB_MISS;
744
745 /* already taken care of */
746 m &= ~(PERF_MEM_TLB_HIT|PERF_MEM_TLB_MISS);
747
748 for (i = 0; m && i < NUM_TLB_ACCESS; i++, m >>= 1) {
749 if (!(m & 0x1))
750 continue;
751 if (l) {
752 strcat(out, " or ");
753 l += 4;
754 }
755 strncat(out, tlb_access[i], sz - l);
756 l += strlen(tlb_access[i]);
757 }
758 if (*out == '\0')
759 strcpy(out, "N/A");
760 if (hit)
761 strncat(out, " hit", sz - l);
762 if (miss)
763 strncat(out, " miss", sz - l);
764
765 return repsep_snprintf(bf, size, "%-*s", width, out);
766}
767
768static int64_t
769sort__lvl_cmp(struct hist_entry *left, struct hist_entry *right)
770{
771 union perf_mem_data_src data_src_l;
772 union perf_mem_data_src data_src_r;
773
774 if (left->mem_info)
775 data_src_l = left->mem_info->data_src;
776 else
777 data_src_l.mem_lvl = PERF_MEM_LVL_NA;
778
779 if (right->mem_info)
780 data_src_r = right->mem_info->data_src;
781 else
782 data_src_r.mem_lvl = PERF_MEM_LVL_NA;
783
784 return (int64_t)(data_src_r.mem_lvl - data_src_l.mem_lvl);
785}
786
787static const char * const mem_lvl[] = {
788 "N/A",
789 "HIT",
790 "MISS",
791 "L1",
792 "LFB",
793 "L2",
794 "L3",
795 "Local RAM",
796 "Remote RAM (1 hop)",
797 "Remote RAM (2 hops)",
798 "Remote Cache (1 hop)",
799 "Remote Cache (2 hops)",
800 "I/O",
801 "Uncached",
802};
803#define NUM_MEM_LVL (sizeof(mem_lvl)/sizeof(const char *))
804
Arnaldo Carvalho de Meloc824c432013-10-22 19:01:31 -0300805static int hist_entry__lvl_snprintf(struct hist_entry *he, char *bf,
Stephane Eranian98a3b322013-01-24 16:10:35 +0100806 size_t size, unsigned int width)
807{
808 char out[64];
809 size_t sz = sizeof(out) - 1; /* -1 for null termination */
810 size_t i, l = 0;
811 u64 m = PERF_MEM_LVL_NA;
812 u64 hit, miss;
813
Arnaldo Carvalho de Meloc824c432013-10-22 19:01:31 -0300814 if (he->mem_info)
815 m = he->mem_info->data_src.mem_lvl;
Stephane Eranian98a3b322013-01-24 16:10:35 +0100816
817 out[0] = '\0';
818
819 hit = m & PERF_MEM_LVL_HIT;
820 miss = m & PERF_MEM_LVL_MISS;
821
822 /* already taken care of */
823 m &= ~(PERF_MEM_LVL_HIT|PERF_MEM_LVL_MISS);
824
825 for (i = 0; m && i < NUM_MEM_LVL; i++, m >>= 1) {
826 if (!(m & 0x1))
827 continue;
828 if (l) {
829 strcat(out, " or ");
830 l += 4;
831 }
832 strncat(out, mem_lvl[i], sz - l);
833 l += strlen(mem_lvl[i]);
834 }
835 if (*out == '\0')
836 strcpy(out, "N/A");
837 if (hit)
838 strncat(out, " hit", sz - l);
839 if (miss)
840 strncat(out, " miss", sz - l);
841
842 return repsep_snprintf(bf, size, "%-*s", width, out);
843}
844
845static int64_t
846sort__snoop_cmp(struct hist_entry *left, struct hist_entry *right)
847{
848 union perf_mem_data_src data_src_l;
849 union perf_mem_data_src data_src_r;
850
851 if (left->mem_info)
852 data_src_l = left->mem_info->data_src;
853 else
854 data_src_l.mem_snoop = PERF_MEM_SNOOP_NA;
855
856 if (right->mem_info)
857 data_src_r = right->mem_info->data_src;
858 else
859 data_src_r.mem_snoop = PERF_MEM_SNOOP_NA;
860
861 return (int64_t)(data_src_r.mem_snoop - data_src_l.mem_snoop);
862}
863
864static const char * const snoop_access[] = {
865 "N/A",
866 "None",
867 "Miss",
868 "Hit",
869 "HitM",
870};
871#define NUM_SNOOP_ACCESS (sizeof(snoop_access)/sizeof(const char *))
872
Arnaldo Carvalho de Meloc824c432013-10-22 19:01:31 -0300873static int hist_entry__snoop_snprintf(struct hist_entry *he, char *bf,
Stephane Eranian98a3b322013-01-24 16:10:35 +0100874 size_t size, unsigned int width)
875{
876 char out[64];
877 size_t sz = sizeof(out) - 1; /* -1 for null termination */
878 size_t i, l = 0;
879 u64 m = PERF_MEM_SNOOP_NA;
880
881 out[0] = '\0';
882
Arnaldo Carvalho de Meloc824c432013-10-22 19:01:31 -0300883 if (he->mem_info)
884 m = he->mem_info->data_src.mem_snoop;
Stephane Eranian98a3b322013-01-24 16:10:35 +0100885
886 for (i = 0; m && i < NUM_SNOOP_ACCESS; i++, m >>= 1) {
887 if (!(m & 0x1))
888 continue;
889 if (l) {
890 strcat(out, " or ");
891 l += 4;
892 }
893 strncat(out, snoop_access[i], sz - l);
894 l += strlen(snoop_access[i]);
895 }
896
897 if (*out == '\0')
898 strcpy(out, "N/A");
899
900 return repsep_snprintf(bf, size, "%-*s", width, out);
901}
902
Don Zickus9b32ba72014-06-01 15:38:29 +0200903static inline u64 cl_address(u64 address)
904{
905 /* return the cacheline of the address */
906 return (address & ~(cacheline_size - 1));
907}
908
909static int64_t
910sort__dcacheline_cmp(struct hist_entry *left, struct hist_entry *right)
911{
912 u64 l, r;
913 struct map *l_map, *r_map;
914
915 if (!left->mem_info) return -1;
916 if (!right->mem_info) return 1;
917
918 /* group event types together */
919 if (left->cpumode > right->cpumode) return -1;
920 if (left->cpumode < right->cpumode) return 1;
921
922 l_map = left->mem_info->daddr.map;
923 r_map = right->mem_info->daddr.map;
924
925 /* if both are NULL, jump to sort on al_addr instead */
926 if (!l_map && !r_map)
927 goto addr;
928
929 if (!l_map) return -1;
930 if (!r_map) return 1;
931
932 if (l_map->maj > r_map->maj) return -1;
933 if (l_map->maj < r_map->maj) return 1;
934
935 if (l_map->min > r_map->min) return -1;
936 if (l_map->min < r_map->min) return 1;
937
938 if (l_map->ino > r_map->ino) return -1;
939 if (l_map->ino < r_map->ino) return 1;
940
941 if (l_map->ino_generation > r_map->ino_generation) return -1;
942 if (l_map->ino_generation < r_map->ino_generation) return 1;
943
944 /*
945 * Addresses with no major/minor numbers are assumed to be
946 * anonymous in userspace. Sort those on pid then address.
947 *
948 * The kernel and non-zero major/minor mapped areas are
949 * assumed to be unity mapped. Sort those on address.
950 */
951
952 if ((left->cpumode != PERF_RECORD_MISC_KERNEL) &&
953 (!(l_map->flags & MAP_SHARED)) &&
954 !l_map->maj && !l_map->min && !l_map->ino &&
955 !l_map->ino_generation) {
956 /* userspace anonymous */
957
958 if (left->thread->pid_ > right->thread->pid_) return -1;
959 if (left->thread->pid_ < right->thread->pid_) return 1;
960 }
961
962addr:
963 /* al_addr does all the right addr - start + offset calculations */
964 l = cl_address(left->mem_info->daddr.al_addr);
965 r = cl_address(right->mem_info->daddr.al_addr);
966
967 if (l > r) return -1;
968 if (l < r) return 1;
969
970 return 0;
971}
972
973static int hist_entry__dcacheline_snprintf(struct hist_entry *he, char *bf,
974 size_t size, unsigned int width)
975{
976
977 uint64_t addr = 0;
978 struct map *map = NULL;
979 struct symbol *sym = NULL;
980 char level = he->level;
981
982 if (he->mem_info) {
983 addr = cl_address(he->mem_info->daddr.al_addr);
984 map = he->mem_info->daddr.map;
985 sym = he->mem_info->daddr.sym;
986
987 /* print [s] for shared data mmaps */
988 if ((he->cpumode != PERF_RECORD_MISC_KERNEL) &&
989 map && (map->type == MAP__VARIABLE) &&
990 (map->flags & MAP_SHARED) &&
991 (map->maj || map->min || map->ino ||
992 map->ino_generation))
993 level = 's';
994 else if (!map)
995 level = 'X';
996 }
997 return _hist_entry__sym_snprintf(map, sym, addr, level, bf, size,
998 width);
999}
1000
Roberto Agostino Vitillob5387522012-02-09 23:21:01 +01001001struct sort_entry sort_mispredict = {
1002 .se_header = "Branch Mispredicted",
1003 .se_cmp = sort__mispredict_cmp,
1004 .se_snprintf = hist_entry__mispredict_snprintf,
1005 .se_width_idx = HISTC_MISPREDICT,
1006};
1007
Andi Kleen05484292013-01-24 16:10:29 +01001008static u64 he_weight(struct hist_entry *he)
1009{
1010 return he->stat.nr_events ? he->stat.weight / he->stat.nr_events : 0;
1011}
1012
1013static int64_t
1014sort__local_weight_cmp(struct hist_entry *left, struct hist_entry *right)
1015{
1016 return he_weight(left) - he_weight(right);
1017}
1018
Arnaldo Carvalho de Meloc824c432013-10-22 19:01:31 -03001019static int hist_entry__local_weight_snprintf(struct hist_entry *he, char *bf,
Andi Kleen05484292013-01-24 16:10:29 +01001020 size_t size, unsigned int width)
1021{
Arnaldo Carvalho de Meloc824c432013-10-22 19:01:31 -03001022 return repsep_snprintf(bf, size, "%-*llu", width, he_weight(he));
Andi Kleen05484292013-01-24 16:10:29 +01001023}
1024
1025struct sort_entry sort_local_weight = {
1026 .se_header = "Local Weight",
1027 .se_cmp = sort__local_weight_cmp,
1028 .se_snprintf = hist_entry__local_weight_snprintf,
1029 .se_width_idx = HISTC_LOCAL_WEIGHT,
1030};
1031
1032static int64_t
1033sort__global_weight_cmp(struct hist_entry *left, struct hist_entry *right)
1034{
1035 return left->stat.weight - right->stat.weight;
1036}
1037
Arnaldo Carvalho de Meloc824c432013-10-22 19:01:31 -03001038static int hist_entry__global_weight_snprintf(struct hist_entry *he, char *bf,
Andi Kleen05484292013-01-24 16:10:29 +01001039 size_t size, unsigned int width)
1040{
Arnaldo Carvalho de Meloc824c432013-10-22 19:01:31 -03001041 return repsep_snprintf(bf, size, "%-*llu", width, he->stat.weight);
Andi Kleen05484292013-01-24 16:10:29 +01001042}
1043
1044struct sort_entry sort_global_weight = {
1045 .se_header = "Weight",
1046 .se_cmp = sort__global_weight_cmp,
1047 .se_snprintf = hist_entry__global_weight_snprintf,
1048 .se_width_idx = HISTC_GLOBAL_WEIGHT,
1049};
1050
Stephane Eranian98a3b322013-01-24 16:10:35 +01001051struct sort_entry sort_mem_daddr_sym = {
1052 .se_header = "Data Symbol",
1053 .se_cmp = sort__daddr_cmp,
1054 .se_snprintf = hist_entry__daddr_snprintf,
1055 .se_width_idx = HISTC_MEM_DADDR_SYMBOL,
1056};
1057
1058struct sort_entry sort_mem_daddr_dso = {
1059 .se_header = "Data Object",
1060 .se_cmp = sort__dso_daddr_cmp,
1061 .se_snprintf = hist_entry__dso_daddr_snprintf,
1062 .se_width_idx = HISTC_MEM_DADDR_SYMBOL,
1063};
1064
1065struct sort_entry sort_mem_locked = {
1066 .se_header = "Locked",
1067 .se_cmp = sort__locked_cmp,
1068 .se_snprintf = hist_entry__locked_snprintf,
1069 .se_width_idx = HISTC_MEM_LOCKED,
1070};
1071
1072struct sort_entry sort_mem_tlb = {
1073 .se_header = "TLB access",
1074 .se_cmp = sort__tlb_cmp,
1075 .se_snprintf = hist_entry__tlb_snprintf,
1076 .se_width_idx = HISTC_MEM_TLB,
1077};
1078
1079struct sort_entry sort_mem_lvl = {
1080 .se_header = "Memory access",
1081 .se_cmp = sort__lvl_cmp,
1082 .se_snprintf = hist_entry__lvl_snprintf,
1083 .se_width_idx = HISTC_MEM_LVL,
1084};
1085
1086struct sort_entry sort_mem_snoop = {
1087 .se_header = "Snoop",
1088 .se_cmp = sort__snoop_cmp,
1089 .se_snprintf = hist_entry__snoop_snprintf,
1090 .se_width_idx = HISTC_MEM_SNOOP,
1091};
1092
Don Zickus9b32ba72014-06-01 15:38:29 +02001093struct sort_entry sort_mem_dcacheline = {
1094 .se_header = "Data Cacheline",
1095 .se_cmp = sort__dcacheline_cmp,
1096 .se_snprintf = hist_entry__dcacheline_snprintf,
1097 .se_width_idx = HISTC_MEM_DCACHELINE,
1098};
1099
Andi Kleenf5d05bc2013-09-20 07:40:41 -07001100static int64_t
1101sort__abort_cmp(struct hist_entry *left, struct hist_entry *right)
1102{
Jiri Olsa49f47442014-10-16 16:07:01 +02001103 if (!left->branch_info || !right->branch_info)
1104 return cmp_null(left->branch_info, right->branch_info);
1105
Andi Kleenf5d05bc2013-09-20 07:40:41 -07001106 return left->branch_info->flags.abort !=
1107 right->branch_info->flags.abort;
1108}
1109
Arnaldo Carvalho de Meloc824c432013-10-22 19:01:31 -03001110static int hist_entry__abort_snprintf(struct hist_entry *he, char *bf,
Andi Kleenf5d05bc2013-09-20 07:40:41 -07001111 size_t size, unsigned int width)
1112{
Jiri Olsa49f47442014-10-16 16:07:01 +02001113 static const char *out = "N/A";
Andi Kleenf5d05bc2013-09-20 07:40:41 -07001114
Jiri Olsa49f47442014-10-16 16:07:01 +02001115 if (he->branch_info) {
1116 if (he->branch_info->flags.abort)
1117 out = "A";
1118 else
1119 out = ".";
1120 }
1121
Andi Kleenf5d05bc2013-09-20 07:40:41 -07001122 return repsep_snprintf(bf, size, "%-*s", width, out);
1123}
1124
1125struct sort_entry sort_abort = {
1126 .se_header = "Transaction abort",
1127 .se_cmp = sort__abort_cmp,
1128 .se_snprintf = hist_entry__abort_snprintf,
1129 .se_width_idx = HISTC_ABORT,
1130};
1131
1132static int64_t
1133sort__in_tx_cmp(struct hist_entry *left, struct hist_entry *right)
1134{
Jiri Olsa0199d242014-10-16 16:07:02 +02001135 if (!left->branch_info || !right->branch_info)
1136 return cmp_null(left->branch_info, right->branch_info);
1137
Andi Kleenf5d05bc2013-09-20 07:40:41 -07001138 return left->branch_info->flags.in_tx !=
1139 right->branch_info->flags.in_tx;
1140}
1141
Arnaldo Carvalho de Meloc824c432013-10-22 19:01:31 -03001142static int hist_entry__in_tx_snprintf(struct hist_entry *he, char *bf,
Andi Kleenf5d05bc2013-09-20 07:40:41 -07001143 size_t size, unsigned int width)
1144{
Jiri Olsa0199d242014-10-16 16:07:02 +02001145 static const char *out = "N/A";
Andi Kleenf5d05bc2013-09-20 07:40:41 -07001146
Jiri Olsa0199d242014-10-16 16:07:02 +02001147 if (he->branch_info) {
1148 if (he->branch_info->flags.in_tx)
1149 out = "T";
1150 else
1151 out = ".";
1152 }
Andi Kleenf5d05bc2013-09-20 07:40:41 -07001153
1154 return repsep_snprintf(bf, size, "%-*s", width, out);
1155}
1156
1157struct sort_entry sort_in_tx = {
1158 .se_header = "Branch in transaction",
1159 .se_cmp = sort__in_tx_cmp,
1160 .se_snprintf = hist_entry__in_tx_snprintf,
1161 .se_width_idx = HISTC_IN_TX,
1162};
1163
Andi Kleen475eeab2013-09-20 07:40:43 -07001164static int64_t
1165sort__transaction_cmp(struct hist_entry *left, struct hist_entry *right)
1166{
1167 return left->transaction - right->transaction;
1168}
1169
1170static inline char *add_str(char *p, const char *str)
1171{
1172 strcpy(p, str);
1173 return p + strlen(str);
1174}
1175
1176static struct txbit {
1177 unsigned flag;
1178 const char *name;
1179 int skip_for_len;
1180} txbits[] = {
1181 { PERF_TXN_ELISION, "EL ", 0 },
1182 { PERF_TXN_TRANSACTION, "TX ", 1 },
1183 { PERF_TXN_SYNC, "SYNC ", 1 },
1184 { PERF_TXN_ASYNC, "ASYNC ", 0 },
1185 { PERF_TXN_RETRY, "RETRY ", 0 },
1186 { PERF_TXN_CONFLICT, "CON ", 0 },
1187 { PERF_TXN_CAPACITY_WRITE, "CAP-WRITE ", 1 },
1188 { PERF_TXN_CAPACITY_READ, "CAP-READ ", 0 },
1189 { 0, NULL, 0 }
1190};
1191
1192int hist_entry__transaction_len(void)
1193{
1194 int i;
1195 int len = 0;
1196
1197 for (i = 0; txbits[i].name; i++) {
1198 if (!txbits[i].skip_for_len)
1199 len += strlen(txbits[i].name);
1200 }
1201 len += 4; /* :XX<space> */
1202 return len;
1203}
1204
Arnaldo Carvalho de Meloc824c432013-10-22 19:01:31 -03001205static int hist_entry__transaction_snprintf(struct hist_entry *he, char *bf,
Andi Kleen475eeab2013-09-20 07:40:43 -07001206 size_t size, unsigned int width)
1207{
Arnaldo Carvalho de Meloc824c432013-10-22 19:01:31 -03001208 u64 t = he->transaction;
Andi Kleen475eeab2013-09-20 07:40:43 -07001209 char buf[128];
1210 char *p = buf;
1211 int i;
1212
1213 buf[0] = 0;
1214 for (i = 0; txbits[i].name; i++)
1215 if (txbits[i].flag & t)
1216 p = add_str(p, txbits[i].name);
1217 if (t && !(t & (PERF_TXN_SYNC|PERF_TXN_ASYNC)))
1218 p = add_str(p, "NEITHER ");
1219 if (t & PERF_TXN_ABORT_MASK) {
1220 sprintf(p, ":%" PRIx64,
1221 (t & PERF_TXN_ABORT_MASK) >>
1222 PERF_TXN_ABORT_SHIFT);
1223 p += strlen(p);
1224 }
1225
1226 return repsep_snprintf(bf, size, "%-*s", width, buf);
1227}
1228
1229struct sort_entry sort_transaction = {
1230 .se_header = "Transaction ",
1231 .se_cmp = sort__transaction_cmp,
1232 .se_snprintf = hist_entry__transaction_snprintf,
1233 .se_width_idx = HISTC_TRANSACTION,
1234};
1235
Frederic Weisbecker872a8782011-06-29 03:14:52 +02001236struct sort_dimension {
1237 const char *name;
1238 struct sort_entry *entry;
1239 int taken;
1240};
1241
Roberto Agostino Vitillob5387522012-02-09 23:21:01 +01001242#define DIM(d, n, func) [d] = { .name = n, .entry = &(func) }
1243
Namhyung Kimfc5871e2012-12-27 18:11:46 +09001244static struct sort_dimension common_sort_dimensions[] = {
Roberto Agostino Vitillob5387522012-02-09 23:21:01 +01001245 DIM(SORT_PID, "pid", sort_thread),
1246 DIM(SORT_COMM, "comm", sort_comm),
1247 DIM(SORT_DSO, "dso", sort_dso),
Roberto Agostino Vitillob5387522012-02-09 23:21:01 +01001248 DIM(SORT_SYM, "symbol", sort_sym),
Roberto Agostino Vitillob5387522012-02-09 23:21:01 +01001249 DIM(SORT_PARENT, "parent", sort_parent),
1250 DIM(SORT_CPU, "cpu", sort_cpu),
Arnaldo Carvalho de Melo409a8be2012-05-30 10:33:24 -03001251 DIM(SORT_SRCLINE, "srcline", sort_srcline),
Andi Kleen31191a82015-08-07 15:54:24 -07001252 DIM(SORT_SRCFILE, "srcfile", sort_srcfile),
Andi Kleenf9ea55d2013-07-18 15:58:53 -07001253 DIM(SORT_LOCAL_WEIGHT, "local_weight", sort_local_weight),
1254 DIM(SORT_GLOBAL_WEIGHT, "weight", sort_global_weight),
Andi Kleen475eeab2013-09-20 07:40:43 -07001255 DIM(SORT_TRANSACTION, "transaction", sort_transaction),
Frederic Weisbecker872a8782011-06-29 03:14:52 +02001256};
1257
Namhyung Kimfc5871e2012-12-27 18:11:46 +09001258#undef DIM
1259
1260#define DIM(d, n, func) [d - __SORT_BRANCH_STACK] = { .name = n, .entry = &(func) }
1261
1262static struct sort_dimension bstack_sort_dimensions[] = {
1263 DIM(SORT_DSO_FROM, "dso_from", sort_dso_from),
1264 DIM(SORT_DSO_TO, "dso_to", sort_dso_to),
1265 DIM(SORT_SYM_FROM, "symbol_from", sort_sym_from),
1266 DIM(SORT_SYM_TO, "symbol_to", sort_sym_to),
1267 DIM(SORT_MISPREDICT, "mispredict", sort_mispredict),
Andi Kleenf5d05bc2013-09-20 07:40:41 -07001268 DIM(SORT_IN_TX, "in_tx", sort_in_tx),
1269 DIM(SORT_ABORT, "abort", sort_abort),
Andi Kleen0e332f02015-07-18 08:24:46 -07001270 DIM(SORT_CYCLES, "cycles", sort_cycles),
Namhyung Kimfc5871e2012-12-27 18:11:46 +09001271};
1272
1273#undef DIM
1274
Namhyung Kimafab87b2013-04-03 21:26:11 +09001275#define DIM(d, n, func) [d - __SORT_MEMORY_MODE] = { .name = n, .entry = &(func) }
1276
1277static struct sort_dimension memory_sort_dimensions[] = {
Namhyung Kimafab87b2013-04-03 21:26:11 +09001278 DIM(SORT_MEM_DADDR_SYMBOL, "symbol_daddr", sort_mem_daddr_sym),
1279 DIM(SORT_MEM_DADDR_DSO, "dso_daddr", sort_mem_daddr_dso),
1280 DIM(SORT_MEM_LOCKED, "locked", sort_mem_locked),
1281 DIM(SORT_MEM_TLB, "tlb", sort_mem_tlb),
1282 DIM(SORT_MEM_LVL, "mem", sort_mem_lvl),
1283 DIM(SORT_MEM_SNOOP, "snoop", sort_mem_snoop),
Don Zickus9b32ba72014-06-01 15:38:29 +02001284 DIM(SORT_MEM_DCACHELINE, "dcacheline", sort_mem_dcacheline),
Namhyung Kimafab87b2013-04-03 21:26:11 +09001285};
1286
1287#undef DIM
1288
Namhyung Kima2ce0672014-03-04 09:06:42 +09001289struct hpp_dimension {
1290 const char *name;
1291 struct perf_hpp_fmt *fmt;
1292 int taken;
1293};
1294
1295#define DIM(d, n) { .name = n, .fmt = &perf_hpp__format[d], }
1296
1297static struct hpp_dimension hpp_sort_dimensions[] = {
1298 DIM(PERF_HPP__OVERHEAD, "overhead"),
1299 DIM(PERF_HPP__OVERHEAD_SYS, "overhead_sys"),
1300 DIM(PERF_HPP__OVERHEAD_US, "overhead_us"),
1301 DIM(PERF_HPP__OVERHEAD_GUEST_SYS, "overhead_guest_sys"),
1302 DIM(PERF_HPP__OVERHEAD_GUEST_US, "overhead_guest_us"),
Namhyung Kim594dcbf2013-10-30 16:06:59 +09001303 DIM(PERF_HPP__OVERHEAD_ACC, "overhead_children"),
Namhyung Kima2ce0672014-03-04 09:06:42 +09001304 DIM(PERF_HPP__SAMPLES, "sample"),
1305 DIM(PERF_HPP__PERIOD, "period"),
1306};
1307
1308#undef DIM
1309
Namhyung Kim8b536992014-03-03 11:46:55 +09001310struct hpp_sort_entry {
1311 struct perf_hpp_fmt hpp;
1312 struct sort_entry *se;
1313};
1314
Namhyung Kima7d945b2014-03-04 10:46:34 +09001315bool perf_hpp__same_sort_entry(struct perf_hpp_fmt *a, struct perf_hpp_fmt *b)
1316{
1317 struct hpp_sort_entry *hse_a;
1318 struct hpp_sort_entry *hse_b;
1319
1320 if (!perf_hpp__is_sort_entry(a) || !perf_hpp__is_sort_entry(b))
1321 return false;
1322
1323 hse_a = container_of(a, struct hpp_sort_entry, hpp);
1324 hse_b = container_of(b, struct hpp_sort_entry, hpp);
1325
1326 return hse_a->se == hse_b->se;
1327}
1328
Namhyung Kime0d66c72014-07-31 14:47:37 +09001329void perf_hpp__reset_sort_width(struct perf_hpp_fmt *fmt, struct hists *hists)
Namhyung Kim678a5002014-03-20 11:18:54 +09001330{
1331 struct hpp_sort_entry *hse;
1332
1333 if (!perf_hpp__is_sort_entry(fmt))
1334 return;
1335
1336 hse = container_of(fmt, struct hpp_sort_entry, hpp);
Namhyung Kim1ecd4452014-07-31 14:47:40 +09001337 hists__new_col_len(hists, hse->se->se_width_idx, strlen(fmt->name));
Namhyung Kim678a5002014-03-20 11:18:54 +09001338}
1339
Namhyung Kim8b536992014-03-03 11:46:55 +09001340static int __sort__hpp_header(struct perf_hpp_fmt *fmt, struct perf_hpp *hpp,
1341 struct perf_evsel *evsel)
1342{
1343 struct hpp_sort_entry *hse;
Namhyung Kim5b591662014-07-31 14:47:38 +09001344 size_t len = fmt->user_len;
Namhyung Kim8b536992014-03-03 11:46:55 +09001345
1346 hse = container_of(fmt, struct hpp_sort_entry, hpp);
Namhyung Kim8b536992014-03-03 11:46:55 +09001347
Namhyung Kim5b591662014-07-31 14:47:38 +09001348 if (!len)
Arnaldo Carvalho de Melo4ea062ed2014-10-09 13:13:41 -03001349 len = hists__col_len(evsel__hists(evsel), hse->se->se_width_idx);
Namhyung Kim5b591662014-07-31 14:47:38 +09001350
Namhyung Kim1ecd4452014-07-31 14:47:40 +09001351 return scnprintf(hpp->buf, hpp->size, "%-*.*s", len, len, fmt->name);
Namhyung Kim8b536992014-03-03 11:46:55 +09001352}
1353
1354static int __sort__hpp_width(struct perf_hpp_fmt *fmt,
1355 struct perf_hpp *hpp __maybe_unused,
1356 struct perf_evsel *evsel)
1357{
1358 struct hpp_sort_entry *hse;
Namhyung Kim5b591662014-07-31 14:47:38 +09001359 size_t len = fmt->user_len;
Namhyung Kim8b536992014-03-03 11:46:55 +09001360
1361 hse = container_of(fmt, struct hpp_sort_entry, hpp);
1362
Namhyung Kim5b591662014-07-31 14:47:38 +09001363 if (!len)
Arnaldo Carvalho de Melo4ea062ed2014-10-09 13:13:41 -03001364 len = hists__col_len(evsel__hists(evsel), hse->se->se_width_idx);
Namhyung Kim5b591662014-07-31 14:47:38 +09001365
1366 return len;
Namhyung Kim8b536992014-03-03 11:46:55 +09001367}
1368
1369static int __sort__hpp_entry(struct perf_hpp_fmt *fmt, struct perf_hpp *hpp,
1370 struct hist_entry *he)
1371{
1372 struct hpp_sort_entry *hse;
Namhyung Kim5b591662014-07-31 14:47:38 +09001373 size_t len = fmt->user_len;
Namhyung Kim8b536992014-03-03 11:46:55 +09001374
1375 hse = container_of(fmt, struct hpp_sort_entry, hpp);
Namhyung Kim5b591662014-07-31 14:47:38 +09001376
1377 if (!len)
1378 len = hists__col_len(he->hists, hse->se->se_width_idx);
Namhyung Kim8b536992014-03-03 11:46:55 +09001379
1380 return hse->se->se_snprintf(he, hpp->buf, hpp->size, len);
1381}
1382
Namhyung Kim87bbdf72015-01-08 09:45:46 +09001383static int64_t __sort__hpp_cmp(struct perf_hpp_fmt *fmt,
1384 struct hist_entry *a, struct hist_entry *b)
1385{
1386 struct hpp_sort_entry *hse;
1387
1388 hse = container_of(fmt, struct hpp_sort_entry, hpp);
1389 return hse->se->se_cmp(a, b);
1390}
1391
1392static int64_t __sort__hpp_collapse(struct perf_hpp_fmt *fmt,
1393 struct hist_entry *a, struct hist_entry *b)
1394{
1395 struct hpp_sort_entry *hse;
1396 int64_t (*collapse_fn)(struct hist_entry *, struct hist_entry *);
1397
1398 hse = container_of(fmt, struct hpp_sort_entry, hpp);
1399 collapse_fn = hse->se->se_collapse ?: hse->se->se_cmp;
1400 return collapse_fn(a, b);
1401}
1402
1403static int64_t __sort__hpp_sort(struct perf_hpp_fmt *fmt,
1404 struct hist_entry *a, struct hist_entry *b)
1405{
1406 struct hpp_sort_entry *hse;
1407 int64_t (*sort_fn)(struct hist_entry *, struct hist_entry *);
1408
1409 hse = container_of(fmt, struct hpp_sort_entry, hpp);
1410 sort_fn = hse->se->se_sort ?: hse->se->se_cmp;
1411 return sort_fn(a, b);
1412}
1413
Namhyung Kima7d945b2014-03-04 10:46:34 +09001414static struct hpp_sort_entry *
1415__sort_dimension__alloc_hpp(struct sort_dimension *sd)
Namhyung Kim8b536992014-03-03 11:46:55 +09001416{
1417 struct hpp_sort_entry *hse;
1418
1419 hse = malloc(sizeof(*hse));
1420 if (hse == NULL) {
1421 pr_err("Memory allocation failed\n");
Namhyung Kima7d945b2014-03-04 10:46:34 +09001422 return NULL;
Namhyung Kim8b536992014-03-03 11:46:55 +09001423 }
1424
1425 hse->se = sd->entry;
Namhyung Kim1ecd4452014-07-31 14:47:40 +09001426 hse->hpp.name = sd->entry->se_header;
Namhyung Kim8b536992014-03-03 11:46:55 +09001427 hse->hpp.header = __sort__hpp_header;
1428 hse->hpp.width = __sort__hpp_width;
1429 hse->hpp.entry = __sort__hpp_entry;
1430 hse->hpp.color = NULL;
1431
Namhyung Kim87bbdf72015-01-08 09:45:46 +09001432 hse->hpp.cmp = __sort__hpp_cmp;
1433 hse->hpp.collapse = __sort__hpp_collapse;
1434 hse->hpp.sort = __sort__hpp_sort;
Namhyung Kim8b536992014-03-03 11:46:55 +09001435
1436 INIT_LIST_HEAD(&hse->hpp.list);
1437 INIT_LIST_HEAD(&hse->hpp.sort_list);
Jiri Olsaf2998422014-05-23 17:15:47 +02001438 hse->hpp.elide = false;
Namhyung Kime0d66c72014-07-31 14:47:37 +09001439 hse->hpp.len = 0;
Namhyung Kim5b591662014-07-31 14:47:38 +09001440 hse->hpp.user_len = 0;
Namhyung Kim8b536992014-03-03 11:46:55 +09001441
Namhyung Kima7d945b2014-03-04 10:46:34 +09001442 return hse;
1443}
1444
1445bool perf_hpp__is_sort_entry(struct perf_hpp_fmt *format)
1446{
1447 return format->header == __sort__hpp_header;
1448}
1449
1450static int __sort_dimension__add_hpp_sort(struct sort_dimension *sd)
1451{
1452 struct hpp_sort_entry *hse = __sort_dimension__alloc_hpp(sd);
1453
1454 if (hse == NULL)
1455 return -1;
1456
Namhyung Kim8b536992014-03-03 11:46:55 +09001457 perf_hpp__register_sort_field(&hse->hpp);
1458 return 0;
1459}
1460
Namhyung Kima7d945b2014-03-04 10:46:34 +09001461static int __sort_dimension__add_hpp_output(struct sort_dimension *sd)
1462{
1463 struct hpp_sort_entry *hse = __sort_dimension__alloc_hpp(sd);
1464
1465 if (hse == NULL)
1466 return -1;
1467
1468 perf_hpp__column_register(&hse->hpp);
1469 return 0;
1470}
1471
Namhyung Kimcfaa1542014-05-19 14:19:30 +09001472static int __sort_dimension__add(struct sort_dimension *sd)
Namhyung Kim2f532d092013-04-03 21:26:10 +09001473{
1474 if (sd->taken)
Namhyung Kim8b536992014-03-03 11:46:55 +09001475 return 0;
1476
Namhyung Kima7d945b2014-03-04 10:46:34 +09001477 if (__sort_dimension__add_hpp_sort(sd) < 0)
Namhyung Kim8b536992014-03-03 11:46:55 +09001478 return -1;
Namhyung Kim2f532d092013-04-03 21:26:10 +09001479
1480 if (sd->entry->se_collapse)
1481 sort__need_collapse = 1;
1482
Namhyung Kim2f532d092013-04-03 21:26:10 +09001483 sd->taken = 1;
Namhyung Kim8b536992014-03-03 11:46:55 +09001484
1485 return 0;
Namhyung Kim2f532d092013-04-03 21:26:10 +09001486}
1487
Namhyung Kima2ce0672014-03-04 09:06:42 +09001488static int __hpp_dimension__add(struct hpp_dimension *hd)
1489{
1490 if (!hd->taken) {
1491 hd->taken = 1;
1492
1493 perf_hpp__register_sort_field(hd->fmt);
1494 }
1495 return 0;
1496}
1497
Namhyung Kima7d945b2014-03-04 10:46:34 +09001498static int __sort_dimension__add_output(struct sort_dimension *sd)
1499{
1500 if (sd->taken)
1501 return 0;
1502
1503 if (__sort_dimension__add_hpp_output(sd) < 0)
1504 return -1;
1505
1506 sd->taken = 1;
1507 return 0;
1508}
1509
1510static int __hpp_dimension__add_output(struct hpp_dimension *hd)
1511{
1512 if (!hd->taken) {
1513 hd->taken = 1;
1514
1515 perf_hpp__column_register(hd->fmt);
1516 }
1517 return 0;
1518}
1519
John Kacurdd68ada2009-09-24 18:02:49 +02001520int sort_dimension__add(const char *tok)
1521{
1522 unsigned int i;
1523
Namhyung Kimfc5871e2012-12-27 18:11:46 +09001524 for (i = 0; i < ARRAY_SIZE(common_sort_dimensions); i++) {
1525 struct sort_dimension *sd = &common_sort_dimensions[i];
John Kacurdd68ada2009-09-24 18:02:49 +02001526
John Kacurdd68ada2009-09-24 18:02:49 +02001527 if (strncasecmp(tok, sd->name, strlen(tok)))
1528 continue;
Namhyung Kimfc5871e2012-12-27 18:11:46 +09001529
John Kacurdd68ada2009-09-24 18:02:49 +02001530 if (sd->entry == &sort_parent) {
1531 int ret = regcomp(&parent_regex, parent_pattern, REG_EXTENDED);
1532 if (ret) {
1533 char err[BUFSIZ];
1534
1535 regerror(ret, &parent_regex, err, sizeof(err));
Arnaldo Carvalho de Melo2aefa4f2010-04-02 12:30:57 -03001536 pr_err("Invalid regex: %s\n%s", parent_pattern, err);
1537 return -EINVAL;
John Kacurdd68ada2009-09-24 18:02:49 +02001538 }
1539 sort__has_parent = 1;
Namhyung Kim930477b2013-04-05 10:26:36 +09001540 } else if (sd->entry == &sort_sym) {
Namhyung Kim1af556402012-09-14 17:35:27 +09001541 sort__has_sym = 1;
Kan Liang94ba4622015-02-09 05:39:44 +00001542 /*
1543 * perf diff displays the performance difference amongst
1544 * two or more perf.data files. Those files could come
1545 * from different binaries. So we should not compare
1546 * their ips, but the name of symbol.
1547 */
1548 if (sort__mode == SORT_MODE__DIFF)
1549 sd->entry->se_collapse = sort__sym_sort;
1550
Namhyung Kim68f6d022013-12-18 14:21:10 +09001551 } else if (sd->entry == &sort_dso) {
1552 sort__has_dso = 1;
John Kacurdd68ada2009-09-24 18:02:49 +02001553 }
1554
Namhyung Kimcfaa1542014-05-19 14:19:30 +09001555 return __sort_dimension__add(sd);
John Kacurdd68ada2009-09-24 18:02:49 +02001556 }
Namhyung Kimfc5871e2012-12-27 18:11:46 +09001557
Namhyung Kima2ce0672014-03-04 09:06:42 +09001558 for (i = 0; i < ARRAY_SIZE(hpp_sort_dimensions); i++) {
1559 struct hpp_dimension *hd = &hpp_sort_dimensions[i];
1560
1561 if (strncasecmp(tok, hd->name, strlen(tok)))
1562 continue;
1563
1564 return __hpp_dimension__add(hd);
1565 }
1566
Namhyung Kimfc5871e2012-12-27 18:11:46 +09001567 for (i = 0; i < ARRAY_SIZE(bstack_sort_dimensions); i++) {
1568 struct sort_dimension *sd = &bstack_sort_dimensions[i];
1569
1570 if (strncasecmp(tok, sd->name, strlen(tok)))
1571 continue;
1572
Namhyung Kim55369fc2013-04-01 20:35:20 +09001573 if (sort__mode != SORT_MODE__BRANCH)
Namhyung Kimfc5871e2012-12-27 18:11:46 +09001574 return -EINVAL;
1575
1576 if (sd->entry == &sort_sym_from || sd->entry == &sort_sym_to)
1577 sort__has_sym = 1;
1578
Namhyung Kimcfaa1542014-05-19 14:19:30 +09001579 __sort_dimension__add(sd);
Namhyung Kimfc5871e2012-12-27 18:11:46 +09001580 return 0;
1581 }
1582
Namhyung Kimafab87b2013-04-03 21:26:11 +09001583 for (i = 0; i < ARRAY_SIZE(memory_sort_dimensions); i++) {
1584 struct sort_dimension *sd = &memory_sort_dimensions[i];
1585
1586 if (strncasecmp(tok, sd->name, strlen(tok)))
1587 continue;
1588
1589 if (sort__mode != SORT_MODE__MEMORY)
1590 return -EINVAL;
1591
1592 if (sd->entry == &sort_mem_daddr_sym)
1593 sort__has_sym = 1;
1594
Namhyung Kimcfaa1542014-05-19 14:19:30 +09001595 __sort_dimension__add(sd);
Namhyung Kimafab87b2013-04-03 21:26:11 +09001596 return 0;
1597 }
1598
John Kacurdd68ada2009-09-24 18:02:49 +02001599 return -ESRCH;
1600}
Arnaldo Carvalho de Meloc8829c72009-12-14 20:09:29 -02001601
Namhyung Kim512ae1b2014-03-18 11:31:39 +09001602static const char *get_default_sort_order(void)
1603{
1604 const char *default_sort_orders[] = {
1605 default_sort_order,
1606 default_branch_sort_order,
1607 default_mem_sort_order,
1608 default_top_sort_order,
1609 default_diff_sort_order,
1610 };
1611
1612 BUG_ON(sort__mode >= ARRAY_SIZE(default_sort_orders));
1613
1614 return default_sort_orders[sort__mode];
1615}
1616
Jiri Olsa1a1c0ff2014-08-23 14:59:48 +02001617static int setup_sort_order(void)
1618{
1619 char *new_sort_order;
1620
1621 /*
1622 * Append '+'-prefixed sort order to the default sort
1623 * order string.
1624 */
1625 if (!sort_order || is_strict_order(sort_order))
1626 return 0;
1627
1628 if (sort_order[1] == '\0') {
1629 error("Invalid --sort key: `+'");
1630 return -EINVAL;
1631 }
1632
1633 /*
1634 * We allocate new sort_order string, but we never free it,
1635 * because it's checked over the rest of the code.
1636 */
1637 if (asprintf(&new_sort_order, "%s,%s",
1638 get_default_sort_order(), sort_order + 1) < 0) {
1639 error("Not enough memory to set up --sort");
1640 return -ENOMEM;
1641 }
1642
1643 sort_order = new_sort_order;
1644 return 0;
1645}
1646
Namhyung Kima7d945b2014-03-04 10:46:34 +09001647static int __setup_sorting(void)
Arnaldo Carvalho de Meloc8829c72009-12-14 20:09:29 -02001648{
Namhyung Kim512ae1b2014-03-18 11:31:39 +09001649 char *tmp, *tok, *str;
Jiri Olsa1a1c0ff2014-08-23 14:59:48 +02001650 const char *sort_keys;
Namhyung Kim55309982013-02-06 14:57:16 +09001651 int ret = 0;
Arnaldo Carvalho de Meloc8829c72009-12-14 20:09:29 -02001652
Jiri Olsa1a1c0ff2014-08-23 14:59:48 +02001653 ret = setup_sort_order();
1654 if (ret)
1655 return ret;
1656
1657 sort_keys = sort_order;
Namhyung Kima7d945b2014-03-04 10:46:34 +09001658 if (sort_keys == NULL) {
Jiri Olsa2f3f9bc2014-08-22 15:58:38 +02001659 if (is_strict_order(field_order)) {
Namhyung Kima7d945b2014-03-04 10:46:34 +09001660 /*
1661 * If user specified field order but no sort order,
1662 * we'll honor it and not add default sort orders.
1663 */
1664 return 0;
1665 }
1666
Namhyung Kim512ae1b2014-03-18 11:31:39 +09001667 sort_keys = get_default_sort_order();
Namhyung Kima7d945b2014-03-04 10:46:34 +09001668 }
Namhyung Kim512ae1b2014-03-18 11:31:39 +09001669
1670 str = strdup(sort_keys);
Namhyung Kim5936f542013-02-06 14:57:17 +09001671 if (str == NULL) {
1672 error("Not enough memory to setup sort keys");
1673 return -ENOMEM;
1674 }
1675
Arnaldo Carvalho de Meloc8829c72009-12-14 20:09:29 -02001676 for (tok = strtok_r(str, ", ", &tmp);
1677 tok; tok = strtok_r(NULL, ", ", &tmp)) {
Namhyung Kim55309982013-02-06 14:57:16 +09001678 ret = sort_dimension__add(tok);
Namhyung Kimfc5871e2012-12-27 18:11:46 +09001679 if (ret == -EINVAL) {
1680 error("Invalid --sort key: `%s'", tok);
Namhyung Kim55309982013-02-06 14:57:16 +09001681 break;
Namhyung Kimfc5871e2012-12-27 18:11:46 +09001682 } else if (ret == -ESRCH) {
Arnaldo Carvalho de Meloc8829c72009-12-14 20:09:29 -02001683 error("Unknown --sort key: `%s'", tok);
Namhyung Kim55309982013-02-06 14:57:16 +09001684 break;
Arnaldo Carvalho de Meloc8829c72009-12-14 20:09:29 -02001685 }
1686 }
1687
1688 free(str);
Namhyung Kim55309982013-02-06 14:57:16 +09001689 return ret;
Arnaldo Carvalho de Meloc8829c72009-12-14 20:09:29 -02001690}
Arnaldo Carvalho de Meloc351c282009-12-16 13:49:27 -02001691
Jiri Olsaf2998422014-05-23 17:15:47 +02001692void perf_hpp__set_elide(int idx, bool elide)
Namhyung Kime67d49a2014-03-18 13:00:59 +09001693{
Jiri Olsaf2998422014-05-23 17:15:47 +02001694 struct perf_hpp_fmt *fmt;
1695 struct hpp_sort_entry *hse;
Namhyung Kime67d49a2014-03-18 13:00:59 +09001696
Jiri Olsaf2998422014-05-23 17:15:47 +02001697 perf_hpp__for_each_format(fmt) {
1698 if (!perf_hpp__is_sort_entry(fmt))
1699 continue;
1700
1701 hse = container_of(fmt, struct hpp_sort_entry, hpp);
1702 if (hse->se->se_width_idx == idx) {
1703 fmt->elide = elide;
1704 break;
1705 }
Namhyung Kime67d49a2014-03-18 13:00:59 +09001706 }
Namhyung Kime67d49a2014-03-18 13:00:59 +09001707}
1708
Jiri Olsaf2998422014-05-23 17:15:47 +02001709static bool __get_elide(struct strlist *list, const char *list_name, FILE *fp)
Arnaldo Carvalho de Meloc351c282009-12-16 13:49:27 -02001710{
1711 if (list && strlist__nr_entries(list) == 1) {
1712 if (fp != NULL)
1713 fprintf(fp, "# %s: %s\n", list_name,
1714 strlist__entry(list, 0)->s);
Jiri Olsaf2998422014-05-23 17:15:47 +02001715 return true;
Arnaldo Carvalho de Meloc351c282009-12-16 13:49:27 -02001716 }
Jiri Olsaf2998422014-05-23 17:15:47 +02001717 return false;
1718}
1719
1720static bool get_elide(int idx, FILE *output)
1721{
1722 switch (idx) {
1723 case HISTC_SYMBOL:
1724 return __get_elide(symbol_conf.sym_list, "symbol", output);
1725 case HISTC_DSO:
1726 return __get_elide(symbol_conf.dso_list, "dso", output);
1727 case HISTC_COMM:
1728 return __get_elide(symbol_conf.comm_list, "comm", output);
1729 default:
1730 break;
1731 }
1732
1733 if (sort__mode != SORT_MODE__BRANCH)
1734 return false;
1735
1736 switch (idx) {
1737 case HISTC_SYMBOL_FROM:
1738 return __get_elide(symbol_conf.sym_from_list, "sym_from", output);
1739 case HISTC_SYMBOL_TO:
1740 return __get_elide(symbol_conf.sym_to_list, "sym_to", output);
1741 case HISTC_DSO_FROM:
1742 return __get_elide(symbol_conf.dso_from_list, "dso_from", output);
1743 case HISTC_DSO_TO:
1744 return __get_elide(symbol_conf.dso_to_list, "dso_to", output);
1745 default:
1746 break;
1747 }
1748
1749 return false;
Arnaldo Carvalho de Meloc351c282009-12-16 13:49:27 -02001750}
Namhyung Kim08e71542013-04-03 21:26:19 +09001751
1752void sort__setup_elide(FILE *output)
1753{
Namhyung Kimcfaa1542014-05-19 14:19:30 +09001754 struct perf_hpp_fmt *fmt;
1755 struct hpp_sort_entry *hse;
Namhyung Kim7524f632013-11-08 17:53:42 +09001756
Jiri Olsaf2998422014-05-23 17:15:47 +02001757 perf_hpp__for_each_format(fmt) {
1758 if (!perf_hpp__is_sort_entry(fmt))
1759 continue;
Namhyung Kim08e71542013-04-03 21:26:19 +09001760
Jiri Olsaf2998422014-05-23 17:15:47 +02001761 hse = container_of(fmt, struct hpp_sort_entry, hpp);
1762 fmt->elide = get_elide(hse->se->se_width_idx, output);
Namhyung Kim08e71542013-04-03 21:26:19 +09001763 }
1764
Namhyung Kim7524f632013-11-08 17:53:42 +09001765 /*
1766 * It makes no sense to elide all of sort entries.
1767 * Just revert them to show up again.
1768 */
Namhyung Kimcfaa1542014-05-19 14:19:30 +09001769 perf_hpp__for_each_format(fmt) {
1770 if (!perf_hpp__is_sort_entry(fmt))
1771 continue;
1772
Jiri Olsaf2998422014-05-23 17:15:47 +02001773 if (!fmt->elide)
Namhyung Kim7524f632013-11-08 17:53:42 +09001774 return;
1775 }
1776
Namhyung Kimcfaa1542014-05-19 14:19:30 +09001777 perf_hpp__for_each_format(fmt) {
1778 if (!perf_hpp__is_sort_entry(fmt))
1779 continue;
1780
Jiri Olsaf2998422014-05-23 17:15:47 +02001781 fmt->elide = false;
Namhyung Kimcfaa1542014-05-19 14:19:30 +09001782 }
Namhyung Kim08e71542013-04-03 21:26:19 +09001783}
Namhyung Kima7d945b2014-03-04 10:46:34 +09001784
1785static int output_field_add(char *tok)
1786{
1787 unsigned int i;
1788
1789 for (i = 0; i < ARRAY_SIZE(common_sort_dimensions); i++) {
1790 struct sort_dimension *sd = &common_sort_dimensions[i];
1791
1792 if (strncasecmp(tok, sd->name, strlen(tok)))
1793 continue;
1794
1795 return __sort_dimension__add_output(sd);
1796 }
1797
1798 for (i = 0; i < ARRAY_SIZE(hpp_sort_dimensions); i++) {
1799 struct hpp_dimension *hd = &hpp_sort_dimensions[i];
1800
1801 if (strncasecmp(tok, hd->name, strlen(tok)))
1802 continue;
1803
1804 return __hpp_dimension__add_output(hd);
1805 }
1806
1807 for (i = 0; i < ARRAY_SIZE(bstack_sort_dimensions); i++) {
1808 struct sort_dimension *sd = &bstack_sort_dimensions[i];
1809
1810 if (strncasecmp(tok, sd->name, strlen(tok)))
1811 continue;
1812
1813 return __sort_dimension__add_output(sd);
1814 }
1815
1816 for (i = 0; i < ARRAY_SIZE(memory_sort_dimensions); i++) {
1817 struct sort_dimension *sd = &memory_sort_dimensions[i];
1818
1819 if (strncasecmp(tok, sd->name, strlen(tok)))
1820 continue;
1821
1822 return __sort_dimension__add_output(sd);
1823 }
1824
1825 return -ESRCH;
1826}
1827
1828static void reset_dimensions(void)
1829{
1830 unsigned int i;
1831
1832 for (i = 0; i < ARRAY_SIZE(common_sort_dimensions); i++)
1833 common_sort_dimensions[i].taken = 0;
1834
1835 for (i = 0; i < ARRAY_SIZE(hpp_sort_dimensions); i++)
1836 hpp_sort_dimensions[i].taken = 0;
1837
1838 for (i = 0; i < ARRAY_SIZE(bstack_sort_dimensions); i++)
1839 bstack_sort_dimensions[i].taken = 0;
1840
1841 for (i = 0; i < ARRAY_SIZE(memory_sort_dimensions); i++)
1842 memory_sort_dimensions[i].taken = 0;
1843}
1844
Jiri Olsa2f3f9bc2014-08-22 15:58:38 +02001845bool is_strict_order(const char *order)
1846{
1847 return order && (*order != '+');
1848}
1849
Namhyung Kima7d945b2014-03-04 10:46:34 +09001850static int __setup_output_field(void)
1851{
Jiri Olsa2f3f9bc2014-08-22 15:58:38 +02001852 char *tmp, *tok, *str, *strp;
1853 int ret = -EINVAL;
Namhyung Kima7d945b2014-03-04 10:46:34 +09001854
1855 if (field_order == NULL)
1856 return 0;
1857
1858 reset_dimensions();
1859
Jiri Olsa2f3f9bc2014-08-22 15:58:38 +02001860 strp = str = strdup(field_order);
Namhyung Kima7d945b2014-03-04 10:46:34 +09001861 if (str == NULL) {
1862 error("Not enough memory to setup output fields");
1863 return -ENOMEM;
1864 }
1865
Jiri Olsa2f3f9bc2014-08-22 15:58:38 +02001866 if (!is_strict_order(field_order))
1867 strp++;
1868
1869 if (!strlen(strp)) {
1870 error("Invalid --fields key: `+'");
1871 goto out;
1872 }
1873
1874 for (tok = strtok_r(strp, ", ", &tmp);
Namhyung Kima7d945b2014-03-04 10:46:34 +09001875 tok; tok = strtok_r(NULL, ", ", &tmp)) {
1876 ret = output_field_add(tok);
1877 if (ret == -EINVAL) {
1878 error("Invalid --fields key: `%s'", tok);
1879 break;
1880 } else if (ret == -ESRCH) {
1881 error("Unknown --fields key: `%s'", tok);
1882 break;
1883 }
1884 }
1885
Jiri Olsa2f3f9bc2014-08-22 15:58:38 +02001886out:
Namhyung Kima7d945b2014-03-04 10:46:34 +09001887 free(str);
1888 return ret;
1889}
1890
1891int setup_sorting(void)
1892{
1893 int err;
1894
1895 err = __setup_sorting();
1896 if (err < 0)
1897 return err;
1898
1899 if (parent_pattern != default_parent_pattern) {
1900 err = sort_dimension__add("parent");
1901 if (err < 0)
1902 return err;
1903 }
1904
1905 reset_dimensions();
1906
1907 /*
1908 * perf diff doesn't use default hpp output fields.
1909 */
1910 if (sort__mode != SORT_MODE__DIFF)
1911 perf_hpp__init();
1912
1913 err = __setup_output_field();
1914 if (err < 0)
1915 return err;
1916
1917 /* copy sort keys to output fields */
1918 perf_hpp__setup_output_field();
1919 /* and then copy output fields to sort keys */
1920 perf_hpp__append_sort_keys();
1921
1922 return 0;
1923}
Namhyung Kim1c89fe92014-05-07 18:42:24 +09001924
1925void reset_output_field(void)
1926{
1927 sort__need_collapse = 0;
1928 sort__has_parent = 0;
1929 sort__has_sym = 0;
1930 sort__has_dso = 0;
1931
Namhyung Kimd69b2962014-05-23 10:59:01 +09001932 field_order = NULL;
1933 sort_order = NULL;
1934
Namhyung Kim1c89fe92014-05-07 18:42:24 +09001935 reset_dimensions();
1936 perf_hpp__reset_output_field();
1937}