blob: 11ad7321153846ad7b658fba24e9da0a21f273ff [file] [log] [blame]
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001/*
2 * Copyright (C) 2011, Red Hat Inc, Arnaldo Carvalho de Melo <acme@redhat.com>
3 *
4 * Parts came from builtin-annotate.c, see those files for further
5 * copyright notes.
6 *
7 * Released under the GPL v2. (and only v2, not any later version)
8 */
9
Arnaldo Carvalho de Meloa43783a2017-04-18 10:46:11 -030010#include <errno.h>
Arnaldo Carvalho de Melofd20e812017-04-17 15:23:08 -030011#include <inttypes.h>
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -020012#include "util.h"
Namhyung Kim48c65bd2014-02-20 10:32:53 +090013#include "ui/ui.h"
14#include "sort.h"
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -020015#include "build-id.h"
16#include "color.h"
17#include "cache.h"
18#include "symbol.h"
19#include "debug.h"
20#include "annotate.h"
Namhyung Kimdb8fd072013-03-05 14:53:21 +090021#include "evsel.h"
Peter Zijlstra70fbe052016-09-05 16:08:12 -030022#include "block-range.h"
Arnaldo Carvalho de Meloa0675582017-04-17 16:51:59 -030023#include "string2.h"
Arnaldo Carvalho de Melo786c1b52016-11-16 15:39:50 -030024#include "arch/common.h"
Andi Kleene5924882014-11-12 18:05:26 -080025#include <regex.h>
Arnaldo Carvalho de Meloce6f4fa2011-02-08 13:27:39 -020026#include <pthread.h>
Jiri Olsa4383db82012-10-27 23:18:29 +020027#include <linux/bitops.h>
Arnaldo Carvalho de Melo877a7a12017-04-17 11:39:06 -030028#include <linux/kernel.h>
Arnaldo Carvalho de Meloce6f4fa2011-02-08 13:27:39 -020029
Arnaldo Carvalho de Meloa1e9b742018-03-15 19:12:39 -030030/* FIXME: For the HE_COLORSET */
31#include "ui/browser.h"
32
33/*
34 * FIXME: Using the same values as slang.h,
35 * but that header may not be available everywhere
36 */
Arnaldo Carvalho de Meloc2983042018-03-15 23:14:51 -030037#define LARROW_CHAR ((unsigned char)',')
38#define RARROW_CHAR ((unsigned char)'+')
39#define DARROW_CHAR ((unsigned char)'.')
40#define UARROW_CHAR ((unsigned char)'-')
Arnaldo Carvalho de Meloa1e9b742018-03-15 19:12:39 -030041
Arnaldo Carvalho de Melo3d689ed2017-04-17 16:10:49 -030042#include "sane_ctype.h"
43
Andi Kleenf69b64f2011-09-15 14:31:41 -070044const char *disassembler_style;
Maciek Borzecki7a4ec932012-09-04 12:32:30 +020045const char *objdump_path;
Andi Kleene5924882014-11-12 18:05:26 -080046static regex_t file_lineno;
Andi Kleenf69b64f2011-09-15 14:31:41 -070047
Arnaldo Carvalho de Melo75b49202016-11-24 11:16:06 -030048static struct ins_ops *ins__find(struct arch *arch, const char *name);
Arnaldo Carvalho de Melo2a1ff812016-11-24 11:37:08 -030049static void ins__sort(struct arch *arch);
Arnaldo Carvalho de Melo75b49202016-11-24 11:16:06 -030050static int disasm_line__parse(char *line, const char **namep, char **rawp);
Arnaldo Carvalho de Melo7a997fe2012-05-12 13:15:34 -030051
Arnaldo Carvalho de Melo786c1b52016-11-16 15:39:50 -030052struct arch {
53 const char *name;
Arnaldo Carvalho de Melo763d8962016-11-17 12:31:51 -030054 struct ins *instructions;
55 size_t nr_instructions;
Arnaldo Carvalho de Melo2a1ff812016-11-24 11:37:08 -030056 size_t nr_instructions_allocated;
57 struct ins_ops *(*associate_instruction_ops)(struct arch *arch, const char *name);
Arnaldo Carvalho de Melo763d8962016-11-17 12:31:51 -030058 bool sorted_instructions;
Arnaldo Carvalho de Melo0781ea92016-11-18 12:34:26 -030059 bool initialized;
60 void *priv;
Jin Yao69fb09f2017-07-07 13:06:34 +080061 unsigned int model;
62 unsigned int family;
Jiri Olsa696e2452017-10-11 17:01:24 +020063 int (*init)(struct arch *arch, char *cpuid);
Jin Yao69fb09f2017-07-07 13:06:34 +080064 bool (*ins_is_fused)(struct arch *arch, const char *ins1,
65 const char *ins2);
Arnaldo Carvalho de Melo786c1b52016-11-16 15:39:50 -030066 struct {
67 char comment_char;
Arnaldo Carvalho de Melo9c2fb452016-11-16 15:50:38 -030068 char skip_functions_char;
Arnaldo Carvalho de Melo786c1b52016-11-16 15:39:50 -030069 } objdump;
70};
71
Arnaldo Carvalho de Melo763d8962016-11-17 12:31:51 -030072static struct ins_ops call_ops;
73static struct ins_ops dec_ops;
74static struct ins_ops jump_ops;
75static struct ins_ops mov_ops;
76static struct ins_ops nop_ops;
77static struct ins_ops lock_ops;
78static struct ins_ops ret_ops;
79
Arnaldo Carvalho de Melo2a1ff812016-11-24 11:37:08 -030080static int arch__grow_instructions(struct arch *arch)
81{
82 struct ins *new_instructions;
83 size_t new_nr_allocated;
84
85 if (arch->nr_instructions_allocated == 0 && arch->instructions)
86 goto grow_from_non_allocated_table;
87
88 new_nr_allocated = arch->nr_instructions_allocated + 128;
89 new_instructions = realloc(arch->instructions, new_nr_allocated * sizeof(struct ins));
90 if (new_instructions == NULL)
91 return -1;
92
93out_update_instructions:
94 arch->instructions = new_instructions;
95 arch->nr_instructions_allocated = new_nr_allocated;
96 return 0;
97
98grow_from_non_allocated_table:
99 new_nr_allocated = arch->nr_instructions + 128;
100 new_instructions = calloc(new_nr_allocated, sizeof(struct ins));
101 if (new_instructions == NULL)
102 return -1;
103
104 memcpy(new_instructions, arch->instructions, arch->nr_instructions);
105 goto out_update_instructions;
106}
107
Arnaldo Carvalho de Meloacc9bfb2016-11-18 16:54:10 -0300108static int arch__associate_ins_ops(struct arch* arch, const char *name, struct ins_ops *ops)
Arnaldo Carvalho de Melo2a1ff812016-11-24 11:37:08 -0300109{
110 struct ins *ins;
111
112 if (arch->nr_instructions == arch->nr_instructions_allocated &&
113 arch__grow_instructions(arch))
114 return -1;
115
116 ins = &arch->instructions[arch->nr_instructions];
117 ins->name = strdup(name);
118 if (!ins->name)
119 return -1;
120
121 ins->ops = ops;
122 arch->nr_instructions++;
123
124 ins__sort(arch);
125 return 0;
126}
127
Arnaldo Carvalho de Melo763d8962016-11-17 12:31:51 -0300128#include "arch/arm/annotate/instructions.c"
Kim Phillips0fcb1da2016-11-30 09:23:44 -0600129#include "arch/arm64/annotate/instructions.c"
Arnaldo Carvalho de Melo763d8962016-11-17 12:31:51 -0300130#include "arch/x86/annotate/instructions.c"
Ravi Bangoriadbdebdc2016-11-23 21:33:46 +0530131#include "arch/powerpc/annotate/instructions.c"
Christian Borntraegerd9f8dfa2017-04-06 09:51:52 +0200132#include "arch/s390/annotate/instructions.c"
Arnaldo Carvalho de Melo763d8962016-11-17 12:31:51 -0300133
Arnaldo Carvalho de Melo786c1b52016-11-16 15:39:50 -0300134static struct arch architectures[] = {
135 {
136 .name = "arm",
Arnaldo Carvalho de Meloacc9bfb2016-11-18 16:54:10 -0300137 .init = arm__annotate_init,
Arnaldo Carvalho de Melo786c1b52016-11-16 15:39:50 -0300138 },
139 {
Kim Phillips0fcb1da2016-11-30 09:23:44 -0600140 .name = "arm64",
141 .init = arm64__annotate_init,
142 },
143 {
Arnaldo Carvalho de Melo786c1b52016-11-16 15:39:50 -0300144 .name = "x86",
Jiri Olsa696e2452017-10-11 17:01:24 +0200145 .init = x86__annotate_init,
Arnaldo Carvalho de Melo763d8962016-11-17 12:31:51 -0300146 .instructions = x86__instructions,
147 .nr_instructions = ARRAY_SIZE(x86__instructions),
Jin Yao69fb09f2017-07-07 13:06:34 +0800148 .ins_is_fused = x86__ins_is_fused,
Arnaldo Carvalho de Melo786c1b52016-11-16 15:39:50 -0300149 .objdump = {
150 .comment_char = '#',
151 },
152 },
Ravi Bangoriadbdebdc2016-11-23 21:33:46 +0530153 {
154 .name = "powerpc",
155 .init = powerpc__annotate_init,
156 },
Christian Borntraegere77852b2017-04-06 09:51:51 +0200157 {
158 .name = "s390",
Christian Borntraegerd9f8dfa2017-04-06 09:51:52 +0200159 .init = s390__annotate_init,
Christian Borntraegere77852b2017-04-06 09:51:51 +0200160 .objdump = {
161 .comment_char = '#',
162 },
163 },
Arnaldo Carvalho de Melo786c1b52016-11-16 15:39:50 -0300164};
165
Arnaldo Carvalho de Meloc46219ac2012-05-12 13:26:20 -0300166static void ins__delete(struct ins_operands *ops)
167{
Arnaldo Carvalho de Melo39956142015-03-05 15:27:28 -0300168 if (ops == NULL)
169 return;
Arnaldo Carvalho de Melo74cf2492013-12-27 16:55:14 -0300170 zfree(&ops->source.raw);
171 zfree(&ops->source.name);
172 zfree(&ops->target.raw);
173 zfree(&ops->target.name);
Arnaldo Carvalho de Meloc46219ac2012-05-12 13:26:20 -0300174}
175
Arnaldo Carvalho de Melo54170722012-05-07 18:54:16 -0300176static int ins__raw_scnprintf(struct ins *ins, char *bf, size_t size,
177 struct ins_operands *ops)
178{
Ravi Bangoria648388a2017-11-14 08:55:40 +0530179 return scnprintf(bf, size, "%-6s %s", ins->name, ops->raw);
Arnaldo Carvalho de Melo54170722012-05-07 18:54:16 -0300180}
181
182int ins__scnprintf(struct ins *ins, char *bf, size_t size,
183 struct ins_operands *ops)
184{
185 if (ins->ops->scnprintf)
186 return ins->ops->scnprintf(ins, bf, size, ops);
187
188 return ins__raw_scnprintf(ins, bf, size, ops);
189}
190
Jin Yao69fb09f2017-07-07 13:06:34 +0800191bool ins__is_fused(struct arch *arch, const char *ins1, const char *ins2)
192{
193 if (!arch || !arch->ins_is_fused)
194 return false;
195
196 return arch->ins_is_fused(arch, ins1, ins2);
197}
198
Arnaldo Carvalho de Melo9c2fb452016-11-16 15:50:38 -0300199static int call__parse(struct arch *arch, struct ins_operands *ops, struct map *map)
Arnaldo Carvalho de Melod86b0592012-04-18 16:07:38 -0300200{
Arnaldo Carvalho de Melod2232882012-04-20 15:26:47 -0300201 char *endptr, *tok, *name;
Arnaldo Carvalho de Melo696703a2018-03-02 11:59:36 -0300202 struct addr_map_symbol target = {
203 .map = map,
204 };
Arnaldo Carvalho de Melod2232882012-04-20 15:26:47 -0300205
Arnaldo Carvalho de Melo44d1a3e2012-04-25 08:00:23 -0300206 ops->target.addr = strtoull(ops->raw, &endptr, 16);
Arnaldo Carvalho de Melod2232882012-04-20 15:26:47 -0300207
208 name = strchr(endptr, '<');
209 if (name == NULL)
210 goto indirect_call;
211
212 name++;
213
Arnaldo Carvalho de Melo9c2fb452016-11-16 15:50:38 -0300214 if (arch->objdump.skip_functions_char &&
215 strchr(name, arch->objdump.skip_functions_char))
Russell Kingcfef25b2015-12-06 23:07:13 +0000216 return -1;
Russell Kingcfef25b2015-12-06 23:07:13 +0000217
Arnaldo Carvalho de Melod2232882012-04-20 15:26:47 -0300218 tok = strchr(name, '>');
219 if (tok == NULL)
220 return -1;
221
222 *tok = '\0';
Arnaldo Carvalho de Melo44d1a3e2012-04-25 08:00:23 -0300223 ops->target.name = strdup(name);
Arnaldo Carvalho de Melod2232882012-04-20 15:26:47 -0300224 *tok = '>';
225
Arnaldo Carvalho de Melo696703a2018-03-02 11:59:36 -0300226 if (ops->target.name == NULL)
227 return -1;
228find_target:
229 target.addr = map__objdump_2mem(map, ops->target.addr);
230
231 if (map_groups__find_ams(&target) == 0 &&
232 map__rip_2objdump(target.map, map->map_ip(target.map, target.addr)) == ops->target.addr)
233 ops->target.sym = target.sym;
234
235 return 0;
Arnaldo Carvalho de Melod2232882012-04-20 15:26:47 -0300236
237indirect_call:
Ravi Bangoria88a7fcf2016-08-19 18:29:35 +0530238 tok = strchr(endptr, '*');
Arnaldo Carvalho de Melo696703a2018-03-02 11:59:36 -0300239 if (tok != NULL)
240 ops->target.addr = strtoull(tok + 1, NULL, 16);
241 goto find_target;
Arnaldo Carvalho de Melod86b0592012-04-18 16:07:38 -0300242}
243
Arnaldo Carvalho de Melod2232882012-04-20 15:26:47 -0300244static int call__scnprintf(struct ins *ins, char *bf, size_t size,
Arnaldo Carvalho de Melo54170722012-05-07 18:54:16 -0300245 struct ins_operands *ops)
Arnaldo Carvalho de Melod2232882012-04-20 15:26:47 -0300246{
Arnaldo Carvalho de Melo696703a2018-03-02 11:59:36 -0300247 if (ops->target.sym)
248 return scnprintf(bf, size, "%-6s %s", ins->name, ops->target.sym->name);
Arnaldo Carvalho de Melod2232882012-04-20 15:26:47 -0300249
Arnaldo Carvalho de Meloe8ea1562012-05-11 12:28:55 -0300250 if (ops->target.addr == 0)
251 return ins__raw_scnprintf(ins, bf, size, ops);
252
Arnaldo Carvalho de Melo4c9cb2c2018-03-16 13:28:09 -0300253 if (ops->target.name)
254 return scnprintf(bf, size, "%-6s %s", ins->name, ops->target.name);
255
Ravi Bangoria648388a2017-11-14 08:55:40 +0530256 return scnprintf(bf, size, "%-6s *%" PRIx64, ins->name, ops->target.addr);
Arnaldo Carvalho de Melod2232882012-04-20 15:26:47 -0300257}
258
Arnaldo Carvalho de Melod86b0592012-04-18 16:07:38 -0300259static struct ins_ops call_ops = {
Arnaldo Carvalho de Melod2232882012-04-20 15:26:47 -0300260 .parse = call__parse,
261 .scnprintf = call__scnprintf,
Arnaldo Carvalho de Melod86b0592012-04-18 16:07:38 -0300262};
263
264bool ins__is_call(const struct ins *ins)
265{
Thomas Richter0b58a772018-03-07 14:43:25 +0100266 return ins->ops == &call_ops || ins->ops == &s390_call_ops;
Arnaldo Carvalho de Melod86b0592012-04-18 16:07:38 -0300267}
268
Arnaldo Carvalho de Melo786c1b52016-11-16 15:39:50 -0300269static int jump__parse(struct arch *arch __maybe_unused, struct ins_operands *ops, struct map *map __maybe_unused)
Arnaldo Carvalho de Melo4f9d0322012-04-18 13:58:34 -0300270{
Arnaldo Carvalho de Meloc7e6ead2012-04-20 14:38:46 -0300271 const char *s = strchr(ops->raw, '+');
Ravi Bangoria3ee2eb62016-12-05 21:26:46 +0530272 const char *c = strchr(ops->raw, ',');
Arnaldo Carvalho de Melo4f9d0322012-04-18 13:58:34 -0300273
Kim Phillipsb13bbee2017-06-01 09:29:59 -0500274 /*
275 * skip over possible up to 2 operands to get to address, e.g.:
276 * tbnz w0, #26, ffff0000083cd190 <security_file_permission+0xd0>
277 */
278 if (c++ != NULL) {
Ravi Bangoria3ee2eb62016-12-05 21:26:46 +0530279 ops->target.addr = strtoull(c, NULL, 16);
Kim Phillipsb13bbee2017-06-01 09:29:59 -0500280 if (!ops->target.addr) {
281 c = strchr(c, ',');
282 if (c++ != NULL)
283 ops->target.addr = strtoull(c, NULL, 16);
284 }
285 } else {
Ravi Bangoria3ee2eb62016-12-05 21:26:46 +0530286 ops->target.addr = strtoull(ops->raw, NULL, 16);
Kim Phillipsb13bbee2017-06-01 09:29:59 -0500287 }
Arnaldo Carvalho de Melo4f9d0322012-04-18 13:58:34 -0300288
Ravi Bangoriae2168742016-12-05 21:26:47 +0530289 if (s++ != NULL) {
Adrian Hunterbbb7f842013-08-07 14:38:54 +0300290 ops->target.offset = strtoull(s, NULL, 16);
Ravi Bangoriae2168742016-12-05 21:26:47 +0530291 ops->target.offset_avail = true;
292 } else {
293 ops->target.offset_avail = false;
294 }
Arnaldo Carvalho de Melofb29fa52012-04-25 14:16:03 -0300295
Arnaldo Carvalho de Melo4f9d0322012-04-18 13:58:34 -0300296 return 0;
297}
298
Arnaldo Carvalho de Meloc7e6ead2012-04-20 14:38:46 -0300299static int jump__scnprintf(struct ins *ins, char *bf, size_t size,
Arnaldo Carvalho de Melo54170722012-05-07 18:54:16 -0300300 struct ins_operands *ops)
Arnaldo Carvalho de Melo28548d72012-04-19 10:16:27 -0300301{
Kim Phillipsb13bbee2017-06-01 09:29:59 -0500302 const char *c = strchr(ops->raw, ',');
303
Ravi Bangoriae2168742016-12-05 21:26:47 +0530304 if (!ops->target.addr || ops->target.offset < 0)
Ravi Bangoriabec60e52016-12-05 21:26:45 +0530305 return ins__raw_scnprintf(ins, bf, size, ops);
306
Kim Phillipsb13bbee2017-06-01 09:29:59 -0500307 if (c != NULL) {
308 const char *c2 = strchr(c + 1, ',');
309
310 /* check for 3-op insn */
311 if (c2 != NULL)
312 c = c2;
313 c++;
314
315 /* mirror arch objdump's space-after-comma style */
316 if (*c == ' ')
317 c++;
318 }
319
Ravi Bangoria648388a2017-11-14 08:55:40 +0530320 return scnprintf(bf, size, "%-6s %.*s%" PRIx64,
Kim Phillipsb13bbee2017-06-01 09:29:59 -0500321 ins->name, c ? c - ops->raw : 0, ops->raw,
322 ops->target.offset);
Arnaldo Carvalho de Melo28548d72012-04-19 10:16:27 -0300323}
324
Arnaldo Carvalho de Melo4f9d0322012-04-18 13:58:34 -0300325static struct ins_ops jump_ops = {
Arnaldo Carvalho de Meloc7e6ead2012-04-20 14:38:46 -0300326 .parse = jump__parse,
327 .scnprintf = jump__scnprintf,
Arnaldo Carvalho de Melo4f9d0322012-04-18 13:58:34 -0300328};
329
330bool ins__is_jump(const struct ins *ins)
331{
332 return ins->ops == &jump_ops;
333}
334
Arnaldo Carvalho de Melo6de783b2012-05-11 16:48:49 -0300335static int comment__symbol(char *raw, char *comment, u64 *addrp, char **namep)
336{
337 char *endptr, *name, *t;
338
339 if (strstr(raw, "(%rip)") == NULL)
340 return 0;
341
342 *addrp = strtoull(comment, &endptr, 16);
Thomas Richter35a8a142017-11-28 08:56:32 +0100343 if (endptr == comment)
344 return 0;
Arnaldo Carvalho de Melo6de783b2012-05-11 16:48:49 -0300345 name = strchr(endptr, '<');
346 if (name == NULL)
347 return -1;
348
349 name++;
350
351 t = strchr(name, '>');
352 if (t == NULL)
353 return 0;
354
355 *t = '\0';
356 *namep = strdup(name);
357 *t = '>';
358
359 return 0;
360}
361
Arnaldo Carvalho de Melo786c1b52016-11-16 15:39:50 -0300362static int lock__parse(struct arch *arch, struct ins_operands *ops, struct map *map)
Arnaldo Carvalho de Melo7a997fe2012-05-12 13:15:34 -0300363{
Arnaldo Carvalho de Melo7a997fe2012-05-12 13:15:34 -0300364 ops->locked.ops = zalloc(sizeof(*ops->locked.ops));
365 if (ops->locked.ops == NULL)
366 return 0;
367
Arnaldo Carvalho de Melo75b49202016-11-24 11:16:06 -0300368 if (disasm_line__parse(ops->raw, &ops->locked.ins.name, &ops->locked.ops->raw) < 0)
Arnaldo Carvalho de Melo7a997fe2012-05-12 13:15:34 -0300369 goto out_free_ops;
370
Arnaldo Carvalho de Melo75b49202016-11-24 11:16:06 -0300371 ops->locked.ins.ops = ins__find(arch, ops->locked.ins.name);
Rabin Vincent0fb9f2a2015-01-18 20:00:21 +0100372
Arnaldo Carvalho de Melo75b49202016-11-24 11:16:06 -0300373 if (ops->locked.ins.ops == NULL)
Namhyung Kim2ba34aa2012-11-10 02:27:13 +0900374 goto out_free_ops;
Arnaldo Carvalho de Melo7a997fe2012-05-12 13:15:34 -0300375
Arnaldo Carvalho de Melo75b49202016-11-24 11:16:06 -0300376 if (ops->locked.ins.ops->parse &&
377 ops->locked.ins.ops->parse(arch, ops->locked.ops, map) < 0)
Rabin Vincentbe819082015-01-18 20:00:20 +0100378 goto out_free_ops;
Arnaldo Carvalho de Melo7a997fe2012-05-12 13:15:34 -0300379
380 return 0;
381
382out_free_ops:
Arnaldo Carvalho de Melo04662522013-12-26 17:41:15 -0300383 zfree(&ops->locked.ops);
Arnaldo Carvalho de Melo7a997fe2012-05-12 13:15:34 -0300384 return 0;
385}
386
387static int lock__scnprintf(struct ins *ins, char *bf, size_t size,
388 struct ins_operands *ops)
389{
390 int printed;
391
Arnaldo Carvalho de Melo75b49202016-11-24 11:16:06 -0300392 if (ops->locked.ins.ops == NULL)
Arnaldo Carvalho de Melo7a997fe2012-05-12 13:15:34 -0300393 return ins__raw_scnprintf(ins, bf, size, ops);
394
Ravi Bangoria648388a2017-11-14 08:55:40 +0530395 printed = scnprintf(bf, size, "%-6s ", ins->name);
Arnaldo Carvalho de Melo75b49202016-11-24 11:16:06 -0300396 return printed + ins__scnprintf(&ops->locked.ins, bf + printed,
Arnaldo Carvalho de Melo7a997fe2012-05-12 13:15:34 -0300397 size - printed, ops->locked.ops);
398}
399
Arnaldo Carvalho de Meloc46219ac2012-05-12 13:26:20 -0300400static void lock__delete(struct ins_operands *ops)
401{
Arnaldo Carvalho de Melo75b49202016-11-24 11:16:06 -0300402 struct ins *ins = &ops->locked.ins;
Rabin Vincent0fb9f2a2015-01-18 20:00:21 +0100403
Arnaldo Carvalho de Melo75b49202016-11-24 11:16:06 -0300404 if (ins->ops && ins->ops->free)
Rabin Vincent0fb9f2a2015-01-18 20:00:21 +0100405 ins->ops->free(ops->locked.ops);
406 else
407 ins__delete(ops->locked.ops);
408
Arnaldo Carvalho de Melo74cf2492013-12-27 16:55:14 -0300409 zfree(&ops->locked.ops);
410 zfree(&ops->target.raw);
411 zfree(&ops->target.name);
Arnaldo Carvalho de Meloc46219ac2012-05-12 13:26:20 -0300412}
413
Arnaldo Carvalho de Melo7a997fe2012-05-12 13:15:34 -0300414static struct ins_ops lock_ops = {
Arnaldo Carvalho de Meloc46219ac2012-05-12 13:26:20 -0300415 .free = lock__delete,
Arnaldo Carvalho de Melo7a997fe2012-05-12 13:15:34 -0300416 .parse = lock__parse,
417 .scnprintf = lock__scnprintf,
418};
419
Arnaldo Carvalho de Melo786c1b52016-11-16 15:39:50 -0300420static int mov__parse(struct arch *arch, struct ins_operands *ops, struct map *map __maybe_unused)
Arnaldo Carvalho de Melo6de783b2012-05-11 16:48:49 -0300421{
422 char *s = strchr(ops->raw, ','), *target, *comment, prev;
423
424 if (s == NULL)
425 return -1;
426
427 *s = '\0';
428 ops->source.raw = strdup(ops->raw);
429 *s = ',';
Arnaldo Carvalho de Melo48000a12014-12-17 17:24:45 -0300430
Arnaldo Carvalho de Melo6de783b2012-05-11 16:48:49 -0300431 if (ops->source.raw == NULL)
432 return -1;
433
434 target = ++s;
Arnaldo Carvalho de Melo786c1b52016-11-16 15:39:50 -0300435 comment = strchr(s, arch->objdump.comment_char);
Arnaldo Carvalho de Melo6de783b2012-05-11 16:48:49 -0300436
Alex Converse1e2bb042014-08-14 14:03:00 -0700437 if (comment != NULL)
438 s = comment - 1;
439 else
440 s = strchr(s, '\0') - 1;
441
442 while (s > target && isspace(s[0]))
443 --s;
444 s++;
Arnaldo Carvalho de Melo6de783b2012-05-11 16:48:49 -0300445 prev = *s;
446 *s = '\0';
447
448 ops->target.raw = strdup(target);
449 *s = prev;
450
451 if (ops->target.raw == NULL)
452 goto out_free_source;
453
Arnaldo Carvalho de Melo6de783b2012-05-11 16:48:49 -0300454 if (comment == NULL)
455 return 0;
456
Taeung Song4597cf02017-04-08 09:52:24 +0900457 comment = ltrim(comment);
Thomas Richter35a8a142017-11-28 08:56:32 +0100458 comment__symbol(ops->source.raw, comment + 1, &ops->source.addr, &ops->source.name);
459 comment__symbol(ops->target.raw, comment + 1, &ops->target.addr, &ops->target.name);
Arnaldo Carvalho de Melo6de783b2012-05-11 16:48:49 -0300460
461 return 0;
462
463out_free_source:
Arnaldo Carvalho de Melo04662522013-12-26 17:41:15 -0300464 zfree(&ops->source.raw);
Arnaldo Carvalho de Melo6de783b2012-05-11 16:48:49 -0300465 return -1;
466}
467
468static int mov__scnprintf(struct ins *ins, char *bf, size_t size,
469 struct ins_operands *ops)
470{
Ravi Bangoria648388a2017-11-14 08:55:40 +0530471 return scnprintf(bf, size, "%-6s %s,%s", ins->name,
Arnaldo Carvalho de Melo6de783b2012-05-11 16:48:49 -0300472 ops->source.name ?: ops->source.raw,
473 ops->target.name ?: ops->target.raw);
474}
475
476static struct ins_ops mov_ops = {
477 .parse = mov__parse,
478 .scnprintf = mov__scnprintf,
479};
480
Arnaldo Carvalho de Melo786c1b52016-11-16 15:39:50 -0300481static int dec__parse(struct arch *arch __maybe_unused, struct ins_operands *ops, struct map *map __maybe_unused)
Arnaldo Carvalho de Meloa43712c2012-05-11 17:21:09 -0300482{
483 char *target, *comment, *s, prev;
484
485 target = s = ops->raw;
486
487 while (s[0] != '\0' && !isspace(s[0]))
488 ++s;
489 prev = *s;
490 *s = '\0';
491
492 ops->target.raw = strdup(target);
493 *s = prev;
494
495 if (ops->target.raw == NULL)
496 return -1;
497
Kim Phillips859afa62016-11-30 09:23:33 -0600498 comment = strchr(s, arch->objdump.comment_char);
Arnaldo Carvalho de Meloa43712c2012-05-11 17:21:09 -0300499 if (comment == NULL)
500 return 0;
501
Taeung Song4597cf02017-04-08 09:52:24 +0900502 comment = ltrim(comment);
Thomas Richter35a8a142017-11-28 08:56:32 +0100503 comment__symbol(ops->target.raw, comment + 1, &ops->target.addr, &ops->target.name);
Arnaldo Carvalho de Meloa43712c2012-05-11 17:21:09 -0300504
505 return 0;
506}
507
508static int dec__scnprintf(struct ins *ins, char *bf, size_t size,
509 struct ins_operands *ops)
510{
Ravi Bangoria648388a2017-11-14 08:55:40 +0530511 return scnprintf(bf, size, "%-6s %s", ins->name,
Arnaldo Carvalho de Meloa43712c2012-05-11 17:21:09 -0300512 ops->target.name ?: ops->target.raw);
513}
514
515static struct ins_ops dec_ops = {
516 .parse = dec__parse,
517 .scnprintf = dec__scnprintf,
518};
519
Irina Tirdea1d037ca2012-09-11 01:15:03 +0300520static int nop__scnprintf(struct ins *ins __maybe_unused, char *bf, size_t size,
521 struct ins_operands *ops __maybe_unused)
Arnaldo Carvalho de Melob9818e92012-05-07 18:57:02 -0300522{
Ravi Bangoria648388a2017-11-14 08:55:40 +0530523 return scnprintf(bf, size, "%-6s", "nop");
Arnaldo Carvalho de Melob9818e92012-05-07 18:57:02 -0300524}
525
526static struct ins_ops nop_ops = {
527 .scnprintf = nop__scnprintf,
528};
529
Naveen N. Rao6ef94922016-06-24 17:23:58 +0530530static struct ins_ops ret_ops = {
531 .scnprintf = ins__raw_scnprintf,
532};
533
534bool ins__is_ret(const struct ins *ins)
535{
536 return ins->ops == &ret_ops;
537}
538
Jin Yao7e63a132017-07-07 13:06:35 +0800539bool ins__is_lock(const struct ins *ins)
540{
541 return ins->ops == &lock_ops;
542}
543
Chris Ryder7e4c1492016-05-19 17:59:46 +0100544static int ins__key_cmp(const void *name, const void *insp)
Arnaldo Carvalho de Melo4f9d0322012-04-18 13:58:34 -0300545{
546 const struct ins *ins = insp;
547
548 return strcmp(name, ins->name);
549}
550
Chris Ryder7e4c1492016-05-19 17:59:46 +0100551static int ins__cmp(const void *a, const void *b)
552{
553 const struct ins *ia = a;
554 const struct ins *ib = b;
555
556 return strcmp(ia->name, ib->name);
557}
558
Arnaldo Carvalho de Melo763d8962016-11-17 12:31:51 -0300559static void ins__sort(struct arch *arch)
Arnaldo Carvalho de Melo4f9d0322012-04-18 13:58:34 -0300560{
Arnaldo Carvalho de Melo763d8962016-11-17 12:31:51 -0300561 const int nmemb = arch->nr_instructions;
Arnaldo Carvalho de Melo4f9d0322012-04-18 13:58:34 -0300562
Arnaldo Carvalho de Melo763d8962016-11-17 12:31:51 -0300563 qsort(arch->instructions, nmemb, sizeof(struct ins), ins__cmp);
Chris Ryder7e4c1492016-05-19 17:59:46 +0100564}
565
Arnaldo Carvalho de Melo2a1ff812016-11-24 11:37:08 -0300566static struct ins_ops *__ins__find(struct arch *arch, const char *name)
Chris Ryder7e4c1492016-05-19 17:59:46 +0100567{
Arnaldo Carvalho de Melo75b49202016-11-24 11:16:06 -0300568 struct ins *ins;
Arnaldo Carvalho de Melo763d8962016-11-17 12:31:51 -0300569 const int nmemb = arch->nr_instructions;
Chris Ryder7e4c1492016-05-19 17:59:46 +0100570
Arnaldo Carvalho de Melo763d8962016-11-17 12:31:51 -0300571 if (!arch->sorted_instructions) {
572 ins__sort(arch);
573 arch->sorted_instructions = true;
Chris Ryder7e4c1492016-05-19 17:59:46 +0100574 }
575
Arnaldo Carvalho de Melo75b49202016-11-24 11:16:06 -0300576 ins = bsearch(name, arch->instructions, nmemb, sizeof(struct ins), ins__key_cmp);
577 return ins ? ins->ops : NULL;
Arnaldo Carvalho de Melo4f9d0322012-04-18 13:58:34 -0300578}
579
Arnaldo Carvalho de Melo2a1ff812016-11-24 11:37:08 -0300580static struct ins_ops *ins__find(struct arch *arch, const char *name)
581{
582 struct ins_ops *ops = __ins__find(arch, name);
583
584 if (!ops && arch->associate_instruction_ops)
585 ops = arch->associate_instruction_ops(arch, name);
586
587 return ops;
588}
589
Arnaldo Carvalho de Melo786c1b52016-11-16 15:39:50 -0300590static int arch__key_cmp(const void *name, const void *archp)
591{
592 const struct arch *arch = archp;
593
594 return strcmp(name, arch->name);
595}
596
597static int arch__cmp(const void *a, const void *b)
598{
599 const struct arch *aa = a;
600 const struct arch *ab = b;
601
602 return strcmp(aa->name, ab->name);
603}
604
605static void arch__sort(void)
606{
607 const int nmemb = ARRAY_SIZE(architectures);
608
609 qsort(architectures, nmemb, sizeof(struct arch), arch__cmp);
610}
611
612static struct arch *arch__find(const char *name)
613{
614 const int nmemb = ARRAY_SIZE(architectures);
615 static bool sorted;
616
617 if (!sorted) {
618 arch__sort();
619 sorted = true;
620 }
621
622 return bsearch(name, architectures, nmemb, sizeof(struct arch), arch__key_cmp);
623}
624
Arnaldo Carvalho de Melod04b35f2011-11-11 22:17:32 -0200625int symbol__alloc_hist(struct symbol *sym)
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -0200626{
627 struct annotation *notes = symbol__annotation(sym);
Ravi Bangoria331c7cb2017-10-24 19:50:06 +0530628 size_t size = symbol__size(sym);
Cody Schafer86963292012-07-19 20:05:25 -0700629 size_t sizeof_sym_hist;
630
Ravi Bangoria331c7cb2017-10-24 19:50:06 +0530631 /*
632 * Add buffer of one element for zero length symbol.
633 * When sample is taken from first instruction of
634 * zero length symbol, perf still resolves it and
635 * shows symbol name in perf report and allows to
636 * annotate it.
637 */
638 if (size == 0)
639 size = 1;
640
Cody Schafer86963292012-07-19 20:05:25 -0700641 /* Check for overflow when calculating sizeof_sym_hist */
Taeung Song896bccd2017-07-20 06:36:45 +0900642 if (size > (SIZE_MAX - sizeof(struct sym_hist)) / sizeof(struct sym_hist_entry))
Cody Schafer86963292012-07-19 20:05:25 -0700643 return -1;
644
Taeung Song896bccd2017-07-20 06:36:45 +0900645 sizeof_sym_hist = (sizeof(struct sym_hist) + size * sizeof(struct sym_hist_entry));
Cody Schafer86963292012-07-19 20:05:25 -0700646
647 /* Check for overflow in zalloc argument */
648 if (sizeof_sym_hist > (SIZE_MAX - sizeof(*notes->src))
649 / symbol_conf.nr_events)
650 return -1;
Arnaldo Carvalho de Meloce6f4fa2011-02-08 13:27:39 -0200651
Arnaldo Carvalho de Melod04b35f2011-11-11 22:17:32 -0200652 notes->src = zalloc(sizeof(*notes->src) + symbol_conf.nr_events * sizeof_sym_hist);
Arnaldo Carvalho de Meloce6f4fa2011-02-08 13:27:39 -0200653 if (notes->src == NULL)
654 return -1;
655 notes->src->sizeof_sym_hist = sizeof_sym_hist;
Arnaldo Carvalho de Melod04b35f2011-11-11 22:17:32 -0200656 notes->src->nr_histograms = symbol_conf.nr_events;
Arnaldo Carvalho de Meloce6f4fa2011-02-08 13:27:39 -0200657 INIT_LIST_HEAD(&notes->src->source);
658 return 0;
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -0200659}
660
Andi Kleend4957632015-07-18 08:24:48 -0700661/* The cycles histogram is lazily allocated. */
662static int symbol__alloc_hist_cycles(struct symbol *sym)
663{
664 struct annotation *notes = symbol__annotation(sym);
665 const size_t size = symbol__size(sym);
666
667 notes->src->cycles_hist = calloc(size, sizeof(struct cyc_hist));
668 if (notes->src->cycles_hist == NULL)
669 return -1;
670 return 0;
671}
672
Arnaldo Carvalho de Melo36532462011-02-06 14:54:44 -0200673void symbol__annotate_zero_histograms(struct symbol *sym)
674{
675 struct annotation *notes = symbol__annotation(sym);
676
Arnaldo Carvalho de Meloce6f4fa2011-02-08 13:27:39 -0200677 pthread_mutex_lock(&notes->lock);
Andi Kleend4957632015-07-18 08:24:48 -0700678 if (notes->src != NULL) {
Arnaldo Carvalho de Meloce6f4fa2011-02-08 13:27:39 -0200679 memset(notes->src->histograms, 0,
680 notes->src->nr_histograms * notes->src->sizeof_sym_hist);
Andi Kleend4957632015-07-18 08:24:48 -0700681 if (notes->src->cycles_hist)
682 memset(notes->src->cycles_hist, 0,
683 symbol__size(sym) * sizeof(struct cyc_hist));
684 }
Arnaldo Carvalho de Meloce6f4fa2011-02-08 13:27:39 -0200685 pthread_mutex_unlock(&notes->lock);
Arnaldo Carvalho de Melo36532462011-02-06 14:54:44 -0200686}
687
Andi Kleend4957632015-07-18 08:24:48 -0700688static int __symbol__account_cycles(struct annotation *notes,
689 u64 start,
690 unsigned offset, unsigned cycles,
691 unsigned have_start)
692{
693 struct cyc_hist *ch;
694
695 ch = notes->src->cycles_hist;
696 /*
697 * For now we can only account one basic block per
698 * final jump. But multiple could be overlapping.
699 * Always account the longest one. So when
700 * a shorter one has been already seen throw it away.
701 *
702 * We separately always account the full cycles.
703 */
704 ch[offset].num_aggr++;
705 ch[offset].cycles_aggr += cycles;
706
707 if (!have_start && ch[offset].have_start)
708 return 0;
709 if (ch[offset].num) {
710 if (have_start && (!ch[offset].have_start ||
711 ch[offset].start > start)) {
712 ch[offset].have_start = 0;
713 ch[offset].cycles = 0;
714 ch[offset].num = 0;
715 if (ch[offset].reset < 0xffff)
716 ch[offset].reset++;
717 } else if (have_start &&
718 ch[offset].start < start)
719 return 0;
720 }
721 ch[offset].have_start = have_start;
722 ch[offset].start = start;
723 ch[offset].cycles += cycles;
724 ch[offset].num++;
725 return 0;
726}
727
Arnaldo Carvalho de Melob66d8c02013-12-18 15:37:41 -0300728static int __symbol__inc_addr_samples(struct symbol *sym, struct map *map,
Taeung Songbab89f62017-07-20 16:28:53 -0300729 struct annotation *notes, int evidx, u64 addr,
Taeung Song461c17f2017-07-20 17:18:05 -0300730 struct perf_sample *sample)
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -0200731{
Arnaldo Carvalho de Melo2f525d02011-02-04 13:43:24 -0200732 unsigned offset;
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -0200733 struct sym_hist *h;
734
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -0200735 pr_debug3("%s: addr=%#" PRIx64 "\n", __func__, map->unmap_ip(map, addr));
736
Ravi Bangoriaedee44b2016-11-22 14:10:50 +0530737 if ((addr < sym->start || addr >= sym->end) &&
738 (addr != sym->end || sym->start != sym->end)) {
Arnaldo Carvalho de Meloe3d006c2015-10-21 15:45:13 -0300739 pr_debug("%s(%d): ERANGE! sym->name=%s, start=%#" PRIx64 ", addr=%#" PRIx64 ", end=%#" PRIx64 "\n",
740 __func__, __LINE__, sym->name, sym->start, addr, sym->end);
Arnaldo Carvalho de Melo31d68e72012-03-27 12:55:57 -0300741 return -ERANGE;
Arnaldo Carvalho de Meloe3d006c2015-10-21 15:45:13 -0300742 }
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -0200743
Arnaldo Carvalho de Melo2f525d02011-02-04 13:43:24 -0200744 offset = addr - sym->start;
745 h = annotation__histogram(notes, evidx);
Taeung Song8158683d2017-07-20 06:36:51 +0900746 h->nr_samples++;
Taeung Song896bccd2017-07-20 06:36:45 +0900747 h->addr[offset].nr_samples++;
Taeung Song461c17f2017-07-20 17:18:05 -0300748 h->period += sample->period;
749 h->addr[offset].period += sample->period;
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -0200750
751 pr_debug3("%#" PRIx64 " %s: period++ [addr: %#" PRIx64 ", %#" PRIx64
Taeung Song461c17f2017-07-20 17:18:05 -0300752 ", evidx=%d] => nr_samples: %" PRIu64 ", period: %" PRIu64 "\n",
753 sym->start, sym->name, addr, addr - sym->start, evidx,
754 h->addr[offset].nr_samples, h->addr[offset].period);
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -0200755 return 0;
756}
757
Andi Kleend4957632015-07-18 08:24:48 -0700758static struct annotation *symbol__get_annotation(struct symbol *sym, bool cycles)
Andi Kleen83be34a2015-05-27 10:51:46 -0700759{
760 struct annotation *notes = symbol__annotation(sym);
761
762 if (notes->src == NULL) {
763 if (symbol__alloc_hist(sym) < 0)
764 return NULL;
765 }
Andi Kleend4957632015-07-18 08:24:48 -0700766 if (!notes->src->cycles_hist && cycles) {
767 if (symbol__alloc_hist_cycles(sym) < 0)
768 return NULL;
769 }
Andi Kleen83be34a2015-05-27 10:51:46 -0700770 return notes;
771}
772
Arnaldo Carvalho de Melo44e83032013-12-18 17:12:24 -0300773static int symbol__inc_addr_samples(struct symbol *sym, struct map *map,
Taeung Songbab89f62017-07-20 16:28:53 -0300774 int evidx, u64 addr,
775 struct perf_sample *sample)
Arnaldo Carvalho de Melob66d8c02013-12-18 15:37:41 -0300776{
777 struct annotation *notes;
778
Namhyung Kim48c65bd2014-02-20 10:32:53 +0900779 if (sym == NULL)
Arnaldo Carvalho de Melob66d8c02013-12-18 15:37:41 -0300780 return 0;
Andi Kleend4957632015-07-18 08:24:48 -0700781 notes = symbol__get_annotation(sym, false);
Andi Kleen83be34a2015-05-27 10:51:46 -0700782 if (notes == NULL)
783 return -ENOMEM;
Taeung Songbab89f62017-07-20 16:28:53 -0300784 return __symbol__inc_addr_samples(sym, map, notes, evidx, addr, sample);
Arnaldo Carvalho de Melob66d8c02013-12-18 15:37:41 -0300785}
786
Andi Kleend4957632015-07-18 08:24:48 -0700787static int symbol__account_cycles(u64 addr, u64 start,
788 struct symbol *sym, unsigned cycles)
789{
790 struct annotation *notes;
791 unsigned offset;
792
793 if (sym == NULL)
794 return 0;
795 notes = symbol__get_annotation(sym, true);
796 if (notes == NULL)
797 return -ENOMEM;
798 if (addr < sym->start || addr >= sym->end)
799 return -ERANGE;
800
801 if (start) {
802 if (start < sym->start || start >= sym->end)
803 return -ERANGE;
804 if (start >= addr)
805 start = 0;
806 }
807 offset = addr - sym->start;
808 return __symbol__account_cycles(notes,
809 start ? start - sym->start : 0,
810 offset, cycles,
811 !!start);
812}
813
814int addr_map_symbol__account_cycles(struct addr_map_symbol *ams,
815 struct addr_map_symbol *start,
816 unsigned cycles)
817{
Adrian Hunter3d7245b2015-08-14 10:11:34 +0300818 u64 saddr = 0;
Andi Kleend4957632015-07-18 08:24:48 -0700819 int err;
820
821 if (!cycles)
822 return 0;
823
824 /*
825 * Only set start when IPC can be computed. We can only
826 * compute it when the basic block is completely in a single
827 * function.
828 * Special case the case when the jump is elsewhere, but
829 * it starts on the function start.
830 */
831 if (start &&
832 (start->sym == ams->sym ||
833 (ams->sym &&
834 start->addr == ams->sym->start + ams->map->start)))
835 saddr = start->al_addr;
836 if (saddr == 0)
Adrian Hunter3d7245b2015-08-14 10:11:34 +0300837 pr_debug2("BB with bad start: addr %"PRIx64" start %"PRIx64" sym %"PRIx64" saddr %"PRIx64"\n",
Andi Kleend4957632015-07-18 08:24:48 -0700838 ams->addr,
839 start ? start->addr : 0,
840 ams->sym ? ams->sym->start + ams->map->start : 0,
841 saddr);
842 err = symbol__account_cycles(ams->al_addr, saddr, ams->sym, cycles);
843 if (err)
844 pr_debug2("account_cycles failed %d\n", err);
845 return err;
846}
847
Arnaldo Carvalho de Melof56c0832018-03-15 11:46:23 -0300848static unsigned annotation__count_insn(struct annotation *notes, u64 start, u64 end)
849{
850 unsigned n_insn = 0;
851 u64 offset;
852
853 for (offset = start; offset <= end; offset++) {
854 if (notes->offsets[offset])
855 n_insn++;
856 }
857 return n_insn;
858}
859
860static void annotation__count_and_fill(struct annotation *notes, u64 start, u64 end, struct cyc_hist *ch)
861{
862 unsigned n_insn;
863 u64 offset;
864
865 n_insn = annotation__count_insn(notes, start, end);
866 if (n_insn && ch->num && ch->cycles) {
867 float ipc = n_insn / ((double)ch->cycles / (double)ch->num);
868
869 /* Hide data when there are too many overlaps. */
870 if (ch->reset >= 0x7fff || ch->reset >= ch->num / 2)
871 return;
872
873 for (offset = start; offset <= end; offset++) {
874 struct annotation_line *al = notes->offsets[offset];
875
876 if (al)
877 al->ipc = ipc;
878 }
879 }
880}
881
882void annotation__compute_ipc(struct annotation *notes, size_t size)
883{
884 u64 offset;
885
886 if (!notes->src || !notes->src->cycles_hist)
887 return;
888
889 pthread_mutex_lock(&notes->lock);
890 for (offset = 0; offset < size; ++offset) {
891 struct cyc_hist *ch;
892
893 ch = &notes->src->cycles_hist[offset];
894 if (ch && ch->cycles) {
895 struct annotation_line *al;
896
897 if (ch->have_start)
898 annotation__count_and_fill(notes, ch->start, offset, ch);
899 al = notes->offsets[offset];
900 if (al && ch->num_aggr)
901 al->cycles = ch->cycles_aggr / ch->num_aggr;
902 notes->have_cycles = true;
903 }
904 }
905 pthread_mutex_unlock(&notes->lock);
906}
907
Taeung Songbab89f62017-07-20 16:28:53 -0300908int addr_map_symbol__inc_samples(struct addr_map_symbol *ams, struct perf_sample *sample,
909 int evidx)
Arnaldo Carvalho de Melo0f4e7a22013-12-18 16:48:29 -0300910{
Taeung Songbab89f62017-07-20 16:28:53 -0300911 return symbol__inc_addr_samples(ams->sym, ams->map, evidx, ams->al_addr, sample);
Arnaldo Carvalho de Melo0f4e7a22013-12-18 16:48:29 -0300912}
913
Taeung Songbab89f62017-07-20 16:28:53 -0300914int hist_entry__inc_addr_samples(struct hist_entry *he, struct perf_sample *sample,
915 int evidx, u64 ip)
Arnaldo Carvalho de Melof626adf2013-12-18 17:10:15 -0300916{
Taeung Songbab89f62017-07-20 16:28:53 -0300917 return symbol__inc_addr_samples(he->ms.sym, he->ms.map, evidx, ip, sample);
Arnaldo Carvalho de Melof626adf2013-12-18 17:10:15 -0300918}
919
Arnaldo Carvalho de Melo786c1b52016-11-16 15:39:50 -0300920static void disasm_line__init_ins(struct disasm_line *dl, struct arch *arch, struct map *map)
Arnaldo Carvalho de Melo4f9d0322012-04-18 13:58:34 -0300921{
Arnaldo Carvalho de Melo75b49202016-11-24 11:16:06 -0300922 dl->ins.ops = ins__find(arch, dl->ins.name);
Arnaldo Carvalho de Melo4f9d0322012-04-18 13:58:34 -0300923
Arnaldo Carvalho de Melo75b49202016-11-24 11:16:06 -0300924 if (!dl->ins.ops)
Arnaldo Carvalho de Melo4f9d0322012-04-18 13:58:34 -0300925 return;
926
Arnaldo Carvalho de Melo75b49202016-11-24 11:16:06 -0300927 if (dl->ins.ops->parse && dl->ins.ops->parse(arch, &dl->ops, map) < 0)
928 dl->ins.ops = NULL;
Arnaldo Carvalho de Melo4f9d0322012-04-18 13:58:34 -0300929}
930
Arnaldo Carvalho de Melo75b49202016-11-24 11:16:06 -0300931static int disasm_line__parse(char *line, const char **namep, char **rawp)
Arnaldo Carvalho de Melo7a997fe2012-05-12 13:15:34 -0300932{
Taeung Song4597cf02017-04-08 09:52:24 +0900933 char tmp, *name = ltrim(line);
Arnaldo Carvalho de Melo7a997fe2012-05-12 13:15:34 -0300934
935 if (name[0] == '\0')
936 return -1;
937
938 *rawp = name + 1;
939
940 while ((*rawp)[0] != '\0' && !isspace((*rawp)[0]))
941 ++*rawp;
942
943 tmp = (*rawp)[0];
944 (*rawp)[0] = '\0';
945 *namep = strdup(name);
946
947 if (*namep == NULL)
948 goto out_free_name;
949
950 (*rawp)[0] = tmp;
Taeung Song4597cf02017-04-08 09:52:24 +0900951 *rawp = ltrim(*rawp);
Arnaldo Carvalho de Melo7a997fe2012-05-12 13:15:34 -0300952
953 return 0;
954
955out_free_name:
Arnaldo Carvalho de Melo75b49202016-11-24 11:16:06 -0300956 free((void *)namep);
957 *namep = NULL;
Arnaldo Carvalho de Melo7a997fe2012-05-12 13:15:34 -0300958 return -1;
959}
960
Jiri Olsaea07c5a2017-10-11 17:01:29 +0200961struct annotate_args {
962 size_t privsize;
Jiri Olsa24fe7b82017-10-11 17:01:30 +0200963 struct arch *arch;
Jiri Olsa1a04db72017-10-11 17:01:31 +0200964 struct map *map;
Jiri Olsad03a6862017-10-11 17:01:33 +0200965 struct perf_evsel *evsel;
Jiri Olsa47488342017-10-11 17:01:32 +0200966 s64 offset;
967 char *line;
968 int line_nr;
Jiri Olsaea07c5a2017-10-11 17:01:29 +0200969};
970
Jiri Olsac835e192017-10-11 17:01:37 +0200971static void annotation_line__delete(struct annotation_line *al)
972{
973 void *ptr = (void *) al - al->privsize;
974
Jiri Olsa8b4c74d2017-10-11 17:01:41 +0200975 free_srcline(al->path);
Jiri Olsac835e192017-10-11 17:01:37 +0200976 zfree(&al->line);
977 free(ptr);
978}
979
980/*
981 * Allocating the annotation line data with following
982 * structure:
983 *
984 * --------------------------------------
985 * private space | struct annotation_line
986 * --------------------------------------
987 *
988 * Size of the private space is stored in 'struct annotation_line'.
989 *
990 */
991static struct annotation_line *
992annotation_line__new(struct annotate_args *args, size_t privsize)
993{
994 struct annotation_line *al;
Jiri Olsa7e304552017-10-11 17:01:39 +0200995 struct perf_evsel *evsel = args->evsel;
Jiri Olsac835e192017-10-11 17:01:37 +0200996 size_t size = privsize + sizeof(*al);
Jiri Olsa7e304552017-10-11 17:01:39 +0200997 int nr = 1;
998
999 if (perf_evsel__is_group_event(evsel))
1000 nr = evsel->nr_members;
1001
1002 size += sizeof(al->samples[0]) * nr;
Jiri Olsac835e192017-10-11 17:01:37 +02001003
1004 al = zalloc(size);
1005 if (al) {
1006 al = (void *) al + privsize;
1007 al->privsize = privsize;
1008 al->offset = args->offset;
1009 al->line = strdup(args->line);
1010 al->line_nr = args->line_nr;
Jiri Olsa7e304552017-10-11 17:01:39 +02001011 al->samples_nr = nr;
Jiri Olsac835e192017-10-11 17:01:37 +02001012 }
1013
1014 return al;
1015}
1016
1017/*
1018 * Allocating the disasm annotation line data with
1019 * following structure:
1020 *
1021 * ------------------------------------------------------------
1022 * privsize space | struct disasm_line | struct annotation_line
1023 * ------------------------------------------------------------
1024 *
1025 * We have 'struct annotation_line' member as last member
1026 * of 'struct disasm_line' to have an easy access.
1027 *
1028 */
Jiri Olsa47488342017-10-11 17:01:32 +02001029static struct disasm_line *disasm_line__new(struct annotate_args *args)
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001030{
Jiri Olsac835e192017-10-11 17:01:37 +02001031 struct disasm_line *dl = NULL;
1032 struct annotation_line *al;
1033 size_t privsize = args->privsize + offsetof(struct disasm_line, al);
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001034
Jiri Olsac835e192017-10-11 17:01:37 +02001035 al = annotation_line__new(args, privsize);
1036 if (al != NULL) {
1037 dl = disasm_line(al);
Jiri Olsad5490b92017-10-11 17:01:26 +02001038
1039 if (dl->al.line == NULL)
Arnaldo Carvalho de Melo058b4cc2012-04-02 12:59:01 -03001040 goto out_delete;
Arnaldo Carvalho de Melo51454182012-04-15 15:52:18 -03001041
Jiri Olsa47488342017-10-11 17:01:32 +02001042 if (args->offset != -1) {
Jiri Olsad5490b92017-10-11 17:01:26 +02001043 if (disasm_line__parse(dl->al.line, &dl->ins.name, &dl->ops.raw) < 0)
Arnaldo Carvalho de Melo51454182012-04-15 15:52:18 -03001044 goto out_free_line;
1045
Jiri Olsa1a04db72017-10-11 17:01:31 +02001046 disasm_line__init_ins(dl, args->arch, args->map);
Arnaldo Carvalho de Melo51454182012-04-15 15:52:18 -03001047 }
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001048 }
1049
Arnaldo Carvalho de Melo29ed6e72012-04-15 15:24:39 -03001050 return dl;
Arnaldo Carvalho de Melo51454182012-04-15 15:52:18 -03001051
1052out_free_line:
Jiri Olsad5490b92017-10-11 17:01:26 +02001053 zfree(&dl->al.line);
Arnaldo Carvalho de Melo058b4cc2012-04-02 12:59:01 -03001054out_delete:
Arnaldo Carvalho de Melo29ed6e72012-04-15 15:24:39 -03001055 free(dl);
Arnaldo Carvalho de Melo058b4cc2012-04-02 12:59:01 -03001056 return NULL;
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001057}
1058
Arnaldo Carvalho de Melo29ed6e72012-04-15 15:24:39 -03001059void disasm_line__free(struct disasm_line *dl)
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001060{
Arnaldo Carvalho de Melo75b49202016-11-24 11:16:06 -03001061 if (dl->ins.ops && dl->ins.ops->free)
1062 dl->ins.ops->free(&dl->ops);
Arnaldo Carvalho de Meloc46219ac2012-05-12 13:26:20 -03001063 else
1064 ins__delete(&dl->ops);
Arnaldo Carvalho de Melo75b49202016-11-24 11:16:06 -03001065 free((void *)dl->ins.name);
1066 dl->ins.name = NULL;
Jiri Olsac835e192017-10-11 17:01:37 +02001067 annotation_line__delete(&dl->al);
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001068}
1069
Arnaldo Carvalho de Melo54170722012-05-07 18:54:16 -03001070int disasm_line__scnprintf(struct disasm_line *dl, char *bf, size_t size, bool raw)
1071{
Arnaldo Carvalho de Melo75b49202016-11-24 11:16:06 -03001072 if (raw || !dl->ins.ops)
Ravi Bangoria648388a2017-11-14 08:55:40 +05301073 return scnprintf(bf, size, "%-6s %s", dl->ins.name, dl->ops.raw);
Arnaldo Carvalho de Melo54170722012-05-07 18:54:16 -03001074
Arnaldo Carvalho de Melo75b49202016-11-24 11:16:06 -03001075 return ins__scnprintf(&dl->ins, bf, size, &dl->ops);
Arnaldo Carvalho de Melo54170722012-05-07 18:54:16 -03001076}
1077
Jiri Olsa82b9d7f2017-10-11 17:01:35 +02001078static void annotation_line__add(struct annotation_line *al, struct list_head *head)
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001079{
Jiri Olsa82b9d7f2017-10-11 17:01:35 +02001080 list_add_tail(&al->node, head);
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001081}
1082
Jiri Olsac4c72432017-10-11 17:01:34 +02001083struct annotation_line *
1084annotation_line__next(struct annotation_line *pos, struct list_head *head)
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001085{
Jiri Olsac4c72432017-10-11 17:01:34 +02001086 list_for_each_entry_continue(pos, head, node)
1087 if (pos->offset >= 0)
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001088 return pos;
1089
1090 return NULL;
1091}
1092
Peter Zijlstra70fbe052016-09-05 16:08:12 -03001093static const char *annotate__address_color(struct block_range *br)
1094{
1095 double cov = block_range__coverage(br);
1096
1097 if (cov >= 0) {
1098 /* mark red for >75% coverage */
1099 if (cov > 0.75)
1100 return PERF_COLOR_RED;
1101
1102 /* mark dull for <1% coverage */
1103 if (cov < 0.01)
1104 return PERF_COLOR_NORMAL;
1105 }
1106
1107 return PERF_COLOR_MAGENTA;
1108}
1109
1110static const char *annotate__asm_color(struct block_range *br)
1111{
1112 double cov = block_range__coverage(br);
1113
1114 if (cov >= 0) {
1115 /* mark dull for <1% coverage */
1116 if (cov < 0.01)
1117 return PERF_COLOR_NORMAL;
1118 }
1119
1120 return PERF_COLOR_BLUE;
1121}
1122
1123static void annotate__branch_printf(struct block_range *br, u64 addr)
1124{
1125 bool emit_comment = true;
1126
1127 if (!br)
1128 return;
1129
1130#if 1
1131 if (br->is_target && br->start == addr) {
1132 struct block_range *branch = br;
1133 double p;
1134
1135 /*
1136 * Find matching branch to our target.
1137 */
1138 while (!branch->is_branch)
1139 branch = block_range__next(branch);
1140
1141 p = 100 *(double)br->entry / branch->coverage;
1142
1143 if (p > 0.1) {
1144 if (emit_comment) {
1145 emit_comment = false;
1146 printf("\t#");
1147 }
1148
1149 /*
1150 * The percentage of coverage joined at this target in relation
1151 * to the next branch.
1152 */
1153 printf(" +%.2f%%", p);
1154 }
1155 }
1156#endif
1157 if (br->is_branch && br->end == addr) {
1158 double p = 100*(double)br->taken / br->coverage;
1159
1160 if (p > 0.1) {
1161 if (emit_comment) {
1162 emit_comment = false;
1163 printf("\t#");
1164 }
1165
1166 /*
1167 * The percentage of coverage leaving at this branch, and
1168 * its prediction ratio.
1169 */
1170 printf(" -%.2f%% (p:%.2f%%)", p, 100*(double)br->pred / br->taken);
1171 }
1172 }
1173}
1174
Jiri Olsaf48e7c42017-10-11 17:01:58 +02001175static int disasm_line__print(struct disasm_line *dl, u64 start, int addr_fmt_width)
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001176{
Jiri Olsa29971f92017-10-11 17:01:47 +02001177 s64 offset = dl->al.offset;
1178 const u64 addr = start + offset;
1179 struct block_range *br;
1180
1181 br = block_range__find(addr);
Jiri Olsaf48e7c42017-10-11 17:01:58 +02001182 color_fprintf(stdout, annotate__address_color(br), " %*" PRIx64 ":", addr_fmt_width, addr);
Jiri Olsa29971f92017-10-11 17:01:47 +02001183 color_fprintf(stdout, annotate__asm_color(br), "%s", dl->al.line);
1184 annotate__branch_printf(br, addr);
1185 return 0;
1186}
1187
1188static int
1189annotation_line__print(struct annotation_line *al, struct symbol *sym, u64 start,
1190 struct perf_evsel *evsel, u64 len, int min_pcnt, int printed,
Jiri Olsaf48e7c42017-10-11 17:01:58 +02001191 int max_lines, struct annotation_line *queue, int addr_fmt_width)
Jiri Olsa29971f92017-10-11 17:01:47 +02001192{
1193 struct disasm_line *dl = container_of(al, struct disasm_line, al);
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001194 static const char *prev_line;
1195 static const char *prev_color;
1196
Jiri Olsa29971f92017-10-11 17:01:47 +02001197 if (al->offset != -1) {
Jiri Olsaf681d592017-10-11 17:01:42 +02001198 double max_percent = 0.0;
Namhyung Kimb1dd4432013-03-05 14:53:25 +09001199 int i, nr_percent = 1;
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001200 const char *color;
1201 struct annotation *notes = symbol__annotation(sym);
Arnaldo Carvalho de Meloce6f4fa2011-02-08 13:27:39 -02001202
Jiri Olsa29971f92017-10-11 17:01:47 +02001203 for (i = 0; i < al->samples_nr; i++) {
1204 struct annotation_data *sample = &al->samples[i];
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001205
Jiri Olsaf681d592017-10-11 17:01:42 +02001206 if (sample->percent > max_percent)
1207 max_percent = sample->percent;
Namhyung Kimb1dd4432013-03-05 14:53:25 +09001208 }
1209
1210 if (max_percent < min_pcnt)
Arnaldo Carvalho de Melo36532462011-02-06 14:54:44 -02001211 return -1;
1212
Arnaldo Carvalho de Meloe3087b82011-02-08 15:01:39 -02001213 if (max_lines && printed >= max_lines)
Arnaldo Carvalho de Melo36532462011-02-06 14:54:44 -02001214 return 1;
Arnaldo Carvalho de Melod040bd32011-02-05 15:37:31 -02001215
Arnaldo Carvalho de Melod5e3d742011-02-08 15:29:25 -02001216 if (queue != NULL) {
Jiri Olsa29971f92017-10-11 17:01:47 +02001217 list_for_each_entry_from(queue, &notes->src->source, node) {
1218 if (queue == al)
Arnaldo Carvalho de Melod5e3d742011-02-08 15:29:25 -02001219 break;
Jiri Olsa29971f92017-10-11 17:01:47 +02001220 annotation_line__print(queue, sym, start, evsel, len,
Jiri Olsaf48e7c42017-10-11 17:01:58 +02001221 0, 0, 1, NULL, addr_fmt_width);
Arnaldo Carvalho de Melod5e3d742011-02-08 15:29:25 -02001222 }
1223 }
1224
Namhyung Kimb1dd4432013-03-05 14:53:25 +09001225 color = get_percent_color(max_percent);
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001226
1227 /*
1228 * Also color the filename and line if needed, with
1229 * the same color than the percentage. Don't print it
1230 * twice for close colored addr with the same filename:line
1231 */
Jiri Olsa29971f92017-10-11 17:01:47 +02001232 if (al->path) {
1233 if (!prev_line || strcmp(prev_line, al->path)
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001234 || color != prev_color) {
Jiri Olsa29971f92017-10-11 17:01:47 +02001235 color_fprintf(stdout, color, " %s", al->path);
1236 prev_line = al->path;
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001237 prev_color = color;
1238 }
1239 }
1240
Namhyung Kimb1dd4432013-03-05 14:53:25 +09001241 for (i = 0; i < nr_percent; i++) {
Jiri Olsa29971f92017-10-11 17:01:47 +02001242 struct annotation_data *sample = &al->samples[i];
Jiri Olsaf681d592017-10-11 17:01:42 +02001243
1244 color = get_percent_color(sample->percent);
Martin Liška0c4a5bc2015-06-19 16:10:43 -03001245
1246 if (symbol_conf.show_total_period)
Arnaldo Carvalho de Meloce9ee4a2017-07-26 17:16:46 -03001247 color_fprintf(stdout, color, " %11" PRIu64,
Jiri Olsaf681d592017-10-11 17:01:42 +02001248 sample->he.period);
Taeung Song1ac39372017-08-18 17:46:48 +09001249 else if (symbol_conf.show_nr_samples)
1250 color_fprintf(stdout, color, " %7" PRIu64,
Jiri Olsaf681d592017-10-11 17:01:42 +02001251 sample->he.nr_samples);
Martin Liška0c4a5bc2015-06-19 16:10:43 -03001252 else
Jiri Olsaf681d592017-10-11 17:01:42 +02001253 color_fprintf(stdout, color, " %7.2f", sample->percent);
Namhyung Kimb1dd4432013-03-05 14:53:25 +09001254 }
1255
Jiri Olsaf48e7c42017-10-11 17:01:58 +02001256 printf(" : ");
Peter Zijlstra70fbe052016-09-05 16:08:12 -03001257
Jiri Olsaf48e7c42017-10-11 17:01:58 +02001258 disasm_line__print(dl, start, addr_fmt_width);
Peter Zijlstra70fbe052016-09-05 16:08:12 -03001259 printf("\n");
Arnaldo Carvalho de Meloe3087b82011-02-08 15:01:39 -02001260 } else if (max_lines && printed >= max_lines)
Arnaldo Carvalho de Melo36532462011-02-06 14:54:44 -02001261 return 1;
1262 else {
Arnaldo Carvalho de Meloce9ee4a2017-07-26 17:16:46 -03001263 int width = symbol_conf.show_total_period ? 12 : 8;
Namhyung Kimb1dd4432013-03-05 14:53:25 +09001264
Arnaldo Carvalho de Melod5e3d742011-02-08 15:29:25 -02001265 if (queue)
1266 return -1;
1267
Namhyung Kim759ff492013-03-05 14:53:26 +09001268 if (perf_evsel__is_group_event(evsel))
Namhyung Kimb1dd4432013-03-05 14:53:25 +09001269 width *= evsel->nr_members;
1270
Jiri Olsa29971f92017-10-11 17:01:47 +02001271 if (!*al->line)
Namhyung Kimb1dd4432013-03-05 14:53:25 +09001272 printf(" %*s:\n", width, " ");
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001273 else
Jiri Olsaf48e7c42017-10-11 17:01:58 +02001274 printf(" %*s: %*s %s\n", width, " ", addr_fmt_width, " ", al->line);
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001275 }
Arnaldo Carvalho de Melo36532462011-02-06 14:54:44 -02001276
1277 return 0;
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001278}
1279
Namhyung Kim3aec1502013-03-05 14:53:22 +09001280/*
1281 * symbol__parse_objdump_line() parses objdump output (with -d --no-show-raw)
1282 * which looks like following
1283 *
1284 * 0000000000415500 <_init>:
1285 * 415500: sub $0x8,%rsp
1286 * 415504: mov 0x2f5ad5(%rip),%rax # 70afe0 <_DYNAMIC+0x2f8>
1287 * 41550b: test %rax,%rax
1288 * 41550e: je 415515 <_init+0x15>
1289 * 415510: callq 416e70 <__gmon_start__@plt>
1290 * 415515: add $0x8,%rsp
1291 * 415519: retq
1292 *
1293 * it will be parsed and saved into struct disasm_line as
1294 * <offset> <name> <ops.raw>
1295 *
1296 * The offset will be a relative offset from the start of the symbol and -1
1297 * means that it's not a disassembly line so should be treated differently.
1298 * The ops.raw part will be parsed further according to type of the instruction.
1299 */
Jiri Olsa1a04db72017-10-11 17:01:31 +02001300static int symbol__parse_objdump_line(struct symbol *sym, FILE *file,
Jiri Olsaea07c5a2017-10-11 17:01:29 +02001301 struct annotate_args *args,
Andi Kleene5924882014-11-12 18:05:26 -08001302 int *line_nr)
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001303{
Jiri Olsa1a04db72017-10-11 17:01:31 +02001304 struct map *map = args->map;
Arnaldo Carvalho de Meloce6f4fa2011-02-08 13:27:39 -02001305 struct annotation *notes = symbol__annotation(sym);
Arnaldo Carvalho de Melo29ed6e72012-04-15 15:24:39 -03001306 struct disasm_line *dl;
Taeung Song4597cf02017-04-08 09:52:24 +09001307 char *line = NULL, *parsed_line, *tmp, *tmp2;
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001308 size_t line_len;
1309 s64 line_ip, offset = -1;
Andi Kleene5924882014-11-12 18:05:26 -08001310 regmatch_t match[2];
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001311
1312 if (getline(&line, &line_len, file) < 0)
1313 return -1;
1314
1315 if (!line)
1316 return -1;
1317
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001318 line_ip = -1;
Taeung Song4597cf02017-04-08 09:52:24 +09001319 parsed_line = rtrim(line);
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001320
Andi Kleene5924882014-11-12 18:05:26 -08001321 /* /filename:linenr ? Save line number and ignore. */
Taeung Song986a5bc2017-04-08 09:52:25 +09001322 if (regexec(&file_lineno, parsed_line, 2, match, 0) == 0) {
1323 *line_nr = atoi(parsed_line + match[1].rm_so);
Andi Kleene5924882014-11-12 18:05:26 -08001324 return 0;
1325 }
1326
Taeung Song4597cf02017-04-08 09:52:24 +09001327 tmp = ltrim(parsed_line);
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001328 if (*tmp) {
1329 /*
1330 * Parse hexa addresses followed by ':'
1331 */
1332 line_ip = strtoull(tmp, &tmp2, 16);
1333 if (*tmp2 != ':' || tmp == tmp2 || tmp2[1] == '\0')
1334 line_ip = -1;
1335 }
1336
1337 if (line_ip != -1) {
1338 u64 start = map__rip_2objdump(map, sym->start),
1339 end = map__rip_2objdump(map, sym->end);
1340
1341 offset = line_ip - start;
Arnaldo Carvalho de Melo2c241bd2014-10-14 17:19:44 -03001342 if ((u64)line_ip < start || (u64)line_ip >= end)
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001343 offset = -1;
Arnaldo Carvalho de Melo058b4cc2012-04-02 12:59:01 -03001344 else
1345 parsed_line = tmp2 + 1;
Namhyung Kima31b7cc2012-04-11 17:04:59 -03001346 }
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001347
Jiri Olsa47488342017-10-11 17:01:32 +02001348 args->offset = offset;
1349 args->line = parsed_line;
1350 args->line_nr = *line_nr;
1351
1352 dl = disasm_line__new(args);
Arnaldo Carvalho de Melo058b4cc2012-04-02 12:59:01 -03001353 free(line);
Andi Kleene5924882014-11-12 18:05:26 -08001354 (*line_nr)++;
Arnaldo Carvalho de Melo058b4cc2012-04-02 12:59:01 -03001355
Arnaldo Carvalho de Melo29ed6e72012-04-15 15:24:39 -03001356 if (dl == NULL)
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001357 return -1;
Arnaldo Carvalho de Melo058b4cc2012-04-02 12:59:01 -03001358
Ravi Bangoriae2168742016-12-05 21:26:47 +05301359 if (!disasm_line__has_offset(dl)) {
Adrian Hunterbbb7f842013-08-07 14:38:54 +03001360 dl->ops.target.offset = dl->ops.target.addr -
1361 map__rip_2objdump(map, sym->start);
Ravi Bangoriae2168742016-12-05 21:26:47 +05301362 dl->ops.target.offset_avail = true;
1363 }
Adrian Hunterbbb7f842013-08-07 14:38:54 +03001364
Arnaldo Carvalho de Melo696703a2018-03-02 11:59:36 -03001365 /* kcore has no symbols, so add the call target symbol */
1366 if (dl->ins.ops && ins__is_call(&dl->ins) && !dl->ops.target.sym) {
Adrian Hunter6e427ab2013-10-14 13:43:40 +03001367 struct addr_map_symbol target = {
1368 .map = map,
1369 .addr = dl->ops.target.addr,
1370 };
Adrian Hunterb1781702013-08-07 14:38:57 +03001371
Arnaldo Carvalho de Melobe39db92016-09-01 19:25:52 -03001372 if (!map_groups__find_ams(&target) &&
Adrian Hunter6e427ab2013-10-14 13:43:40 +03001373 target.sym->start == target.al_addr)
Arnaldo Carvalho de Melo696703a2018-03-02 11:59:36 -03001374 dl->ops.target.sym = target.sym;
Adrian Hunterb1781702013-08-07 14:38:57 +03001375 }
1376
Jiri Olsa82b9d7f2017-10-11 17:01:35 +02001377 annotation_line__add(&dl->al, &notes->src->source);
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001378
1379 return 0;
1380}
1381
Andi Kleene5924882014-11-12 18:05:26 -08001382static __attribute__((constructor)) void symbol__init_regexpr(void)
1383{
1384 regcomp(&file_lineno, "^/[^:]+:([0-9]+)", REG_EXTENDED);
1385}
1386
Adrian Hunter484a5e72013-08-07 14:38:56 +03001387static void delete_last_nop(struct symbol *sym)
1388{
1389 struct annotation *notes = symbol__annotation(sym);
1390 struct list_head *list = &notes->src->source;
1391 struct disasm_line *dl;
1392
1393 while (!list_empty(list)) {
Jiri Olsaa17c4ca2017-10-11 17:01:25 +02001394 dl = list_entry(list->prev, struct disasm_line, al.node);
Adrian Hunter484a5e72013-08-07 14:38:56 +03001395
Arnaldo Carvalho de Melo75b49202016-11-24 11:16:06 -03001396 if (dl->ins.ops) {
1397 if (dl->ins.ops != &nop_ops)
Adrian Hunter484a5e72013-08-07 14:38:56 +03001398 return;
1399 } else {
Jiri Olsad5490b92017-10-11 17:01:26 +02001400 if (!strstr(dl->al.line, " nop ") &&
1401 !strstr(dl->al.line, " nopl ") &&
1402 !strstr(dl->al.line, " nopw "))
Adrian Hunter484a5e72013-08-07 14:38:56 +03001403 return;
1404 }
1405
Jiri Olsaa17c4ca2017-10-11 17:01:25 +02001406 list_del(&dl->al.node);
Adrian Hunter484a5e72013-08-07 14:38:56 +03001407 disasm_line__free(dl);
1408 }
1409}
1410
Arnaldo Carvalho de Meloee51d852016-07-29 16:27:18 -03001411int symbol__strerror_disassemble(struct symbol *sym __maybe_unused, struct map *map,
1412 int errnum, char *buf, size_t buflen)
1413{
1414 struct dso *dso = map->dso;
1415
1416 BUG_ON(buflen == 0);
1417
1418 if (errnum >= 0) {
1419 str_error_r(errnum, buf, buflen);
1420 return 0;
1421 }
1422
1423 switch (errnum) {
1424 case SYMBOL_ANNOTATE_ERRNO__NO_VMLINUX: {
1425 char bf[SBUILD_ID_SIZE + 15] = " with build id ";
1426 char *build_id_msg = NULL;
1427
1428 if (dso->has_build_id) {
1429 build_id__sprintf(dso->build_id,
1430 sizeof(dso->build_id), bf + 15);
1431 build_id_msg = bf;
1432 }
1433 scnprintf(buf, buflen,
1434 "No vmlinux file%s\nwas found in the path.\n\n"
1435 "Note that annotation using /proc/kcore requires CAP_SYS_RAWIO capability.\n\n"
1436 "Please use:\n\n"
1437 " perf buildid-cache -vu vmlinux\n\n"
1438 "or:\n\n"
1439 " --vmlinux vmlinux\n", build_id_msg ?: "");
1440 }
1441 break;
1442 default:
1443 scnprintf(buf, buflen, "Internal error: Invalid %d error code\n", errnum);
1444 break;
1445 }
1446
1447 return 0;
1448}
1449
Arnaldo Carvalho de Melo05ed3ac2016-08-09 15:32:53 -03001450static int dso__disassemble_filename(struct dso *dso, char *filename, size_t filename_size)
1451{
1452 char linkname[PATH_MAX];
1453 char *build_id_filename;
Taeung Song6ebd2542017-03-27 16:10:36 +09001454 char *build_id_path = NULL;
Namhyung Kim3619ef72017-06-08 16:31:01 +09001455 char *pos;
Arnaldo Carvalho de Melo05ed3ac2016-08-09 15:32:53 -03001456
1457 if (dso->symtab_type == DSO_BINARY_TYPE__KALLSYMS &&
1458 !dso__is_kcore(dso))
1459 return SYMBOL_ANNOTATE_ERRNO__NO_VMLINUX;
1460
Krister Johansend2396992017-07-05 18:48:13 -07001461 build_id_filename = dso__build_id_filename(dso, NULL, 0, false);
Arnaldo Carvalho de Melo05ed3ac2016-08-09 15:32:53 -03001462 if (build_id_filename) {
1463 __symbol__join_symfs(filename, filename_size, build_id_filename);
1464 free(build_id_filename);
1465 } else {
1466 if (dso->has_build_id)
1467 return ENOMEM;
1468 goto fallback;
1469 }
1470
Taeung Song6ebd2542017-03-27 16:10:36 +09001471 build_id_path = strdup(filename);
1472 if (!build_id_path)
1473 return -1;
1474
Namhyung Kim3619ef72017-06-08 16:31:01 +09001475 /*
1476 * old style build-id cache has name of XX/XXXXXXX.. while
1477 * new style has XX/XXXXXXX../{elf,kallsyms,vdso}.
1478 * extract the build-id part of dirname in the new style only.
1479 */
1480 pos = strrchr(build_id_path, '/');
1481 if (pos && strlen(pos) < SBUILD_ID_SIZE - 2)
1482 dirname(build_id_path);
Taeung Song6ebd2542017-03-27 16:10:36 +09001483
Arnaldo Carvalho de Melo05ed3ac2016-08-09 15:32:53 -03001484 if (dso__is_kcore(dso) ||
Taeung Song6ebd2542017-03-27 16:10:36 +09001485 readlink(build_id_path, linkname, sizeof(linkname)) < 0 ||
Arnaldo Carvalho de Melo05ed3ac2016-08-09 15:32:53 -03001486 strstr(linkname, DSO__NAME_KALLSYMS) ||
1487 access(filename, R_OK)) {
1488fallback:
1489 /*
1490 * If we don't have build-ids or the build-id file isn't in the
1491 * cache, or is just a kallsyms file, well, lets hope that this
1492 * DSO is the same as when 'perf record' ran.
1493 */
1494 __symbol__join_symfs(filename, filename_size, dso->long_name);
1495 }
1496
Taeung Song6ebd2542017-03-27 16:10:36 +09001497 free(build_id_path);
Arnaldo Carvalho de Melo05ed3ac2016-08-09 15:32:53 -03001498 return 0;
1499}
1500
Jiri Olsa1a04db72017-10-11 17:01:31 +02001501static int symbol__disassemble(struct symbol *sym, struct annotate_args *args)
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001502{
Jiri Olsa1a04db72017-10-11 17:01:31 +02001503 struct map *map = args->map;
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001504 struct dso *dso = map->dso;
Arnaldo Carvalho de Melo68101582018-03-14 10:34:11 -03001505 char *command;
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001506 FILE *file;
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001507 char symfs_filename[PATH_MAX];
Adrian Hunterafba19d2013-10-09 15:01:12 +03001508 struct kcore_extract kce;
1509 bool delete_extract = false;
Arnaldo Carvalho de Melo9955d0b2016-06-15 15:48:08 -03001510 int stdout_fd[2];
Andi Kleene5924882014-11-12 18:05:26 -08001511 int lineno = 0;
Andi Kleen62ec9b32015-11-05 19:06:07 -08001512 int nline;
Arnaldo Carvalho de Melo9955d0b2016-06-15 15:48:08 -03001513 pid_t pid;
Arnaldo Carvalho de Melo05ed3ac2016-08-09 15:32:53 -03001514 int err = dso__disassemble_filename(dso, symfs_filename, sizeof(symfs_filename));
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001515
Arnaldo Carvalho de Melo05ed3ac2016-08-09 15:32:53 -03001516 if (err)
1517 return err;
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001518
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001519 pr_debug("%s: filename=%s, sym=%s, start=%#" PRIx64 ", end=%#" PRIx64 "\n", __func__,
Arnaldo Carvalho de Melo3caee092016-08-09 15:16:37 -03001520 symfs_filename, sym->name, map->unmap_ip(map, sym->start),
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001521 map->unmap_ip(map, sym->end));
1522
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001523 pr_debug("annotating [%p] %30s : [%p] %30s\n",
1524 dso, dso->long_name, sym, sym->name);
1525
Adrian Hunterafba19d2013-10-09 15:01:12 +03001526 if (dso__is_kcore(dso)) {
1527 kce.kcore_filename = symfs_filename;
1528 kce.addr = map__rip_2objdump(map, sym->start);
1529 kce.offs = sym->start;
Arnaldo Carvalho de Melo2c241bd2014-10-14 17:19:44 -03001530 kce.len = sym->end - sym->start;
Adrian Hunterafba19d2013-10-09 15:01:12 +03001531 if (!kcore_extract__create(&kce)) {
1532 delete_extract = true;
1533 strlcpy(symfs_filename, kce.extract_filename,
1534 sizeof(symfs_filename));
Adrian Hunterafba19d2013-10-09 15:01:12 +03001535 }
Jiri Olsa2c7da8c2015-03-02 12:56:12 -05001536 } else if (dso__needs_decompress(dso)) {
Namhyung Kim3c84fd52017-06-08 16:31:04 +09001537 char tmp[KMOD_DECOMP_LEN];
Jiri Olsa2c7da8c2015-03-02 12:56:12 -05001538
Namhyung Kim3c84fd52017-06-08 16:31:04 +09001539 if (dso__decompress_kmodule_path(dso, symfs_filename,
1540 tmp, sizeof(tmp)) < 0)
Arnaldo Carvalho de Melo3caee092016-08-09 15:16:37 -03001541 goto out;
Jiri Olsa2c7da8c2015-03-02 12:56:12 -05001542
1543 strcpy(symfs_filename, tmp);
Adrian Hunterafba19d2013-10-09 15:01:12 +03001544 }
1545
Arnaldo Carvalho de Melo68101582018-03-14 10:34:11 -03001546 err = asprintf(&command,
Maciek Borzecki7a4ec932012-09-04 12:32:30 +02001547 "%s %s%s --start-address=0x%016" PRIx64
Stephane Eranian3e6a2a72011-05-17 17:32:07 +02001548 " --stop-address=0x%016" PRIx64
Ravi Bangoria7b4500b2017-05-05 15:44:17 +05301549 " -l -d %s %s -C \"%s\" 2>/dev/null|grep -v \"%s:\"|expand",
Maciek Borzecki7a4ec932012-09-04 12:32:30 +02001550 objdump_path ? objdump_path : "objdump",
Andi Kleenf69b64f2011-09-15 14:31:41 -07001551 disassembler_style ? "-M " : "",
1552 disassembler_style ? disassembler_style : "",
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001553 map__rip_2objdump(map, sym->start),
Arnaldo Carvalho de Melo2c241bd2014-10-14 17:19:44 -03001554 map__rip_2objdump(map, sym->end),
Stephane Eranian3e6a2a72011-05-17 17:32:07 +02001555 symbol_conf.annotate_asm_raw ? "" : "--no-show-raw",
1556 symbol_conf.annotate_src ? "-S" : "",
Arnaldo Carvalho de Melo3caee092016-08-09 15:16:37 -03001557 symfs_filename, symfs_filename);
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001558
Arnaldo Carvalho de Melo68101582018-03-14 10:34:11 -03001559 if (err < 0) {
1560 pr_err("Failure allocating memory for the command to run\n");
1561 goto out_remove_tmp;
1562 }
1563
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001564 pr_debug("Executing: %s\n", command);
1565
Arnaldo Carvalho de Melo9955d0b2016-06-15 15:48:08 -03001566 err = -1;
1567 if (pipe(stdout_fd) < 0) {
1568 pr_err("Failure creating the pipe to run %s\n", command);
Arnaldo Carvalho de Melo68101582018-03-14 10:34:11 -03001569 goto out_free_command;
Arnaldo Carvalho de Melo9955d0b2016-06-15 15:48:08 -03001570 }
1571
1572 pid = fork();
1573 if (pid < 0) {
1574 pr_err("Failure forking to run %s\n", command);
1575 goto out_close_stdout;
1576 }
1577
1578 if (pid == 0) {
1579 close(stdout_fd[0]);
1580 dup2(stdout_fd[1], 1);
1581 close(stdout_fd[1]);
1582 execl("/bin/sh", "sh", "-c", command, NULL);
1583 perror(command);
1584 exit(-1);
1585 }
1586
1587 close(stdout_fd[1]);
1588
1589 file = fdopen(stdout_fd[0], "r");
Andi Kleen62ec9b32015-11-05 19:06:07 -08001590 if (!file) {
Arnaldo Carvalho de Melo9955d0b2016-06-15 15:48:08 -03001591 pr_err("Failure creating FILE stream for %s\n", command);
Andi Kleen62ec9b32015-11-05 19:06:07 -08001592 /*
1593 * If we were using debug info should retry with
1594 * original binary.
1595 */
Arnaldo Carvalho de Melo68101582018-03-14 10:34:11 -03001596 goto out_free_command;
Andi Kleen62ec9b32015-11-05 19:06:07 -08001597 }
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001598
Andi Kleen62ec9b32015-11-05 19:06:07 -08001599 nline = 0;
1600 while (!feof(file)) {
Arnaldo Carvalho de Meloed7b3392017-03-21 16:00:50 -03001601 /*
1602 * The source code line number (lineno) needs to be kept in
1603 * accross calls to symbol__parse_objdump_line(), so that it
1604 * can associate it with the instructions till the next one.
1605 * See disasm_line__new() and struct disasm_line::line_nr.
1606 */
Jiri Olsa1a04db72017-10-11 17:01:31 +02001607 if (symbol__parse_objdump_line(sym, file, args, &lineno) < 0)
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001608 break;
Andi Kleen62ec9b32015-11-05 19:06:07 -08001609 nline++;
1610 }
1611
1612 if (nline == 0)
1613 pr_err("No output from %s\n", command);
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001614
Adrian Hunter484a5e72013-08-07 14:38:56 +03001615 /*
1616 * kallsyms does not have symbol sizes so there may a nop at the end.
1617 * Remove it.
1618 */
1619 if (dso__is_kcore(dso))
1620 delete_last_nop(sym);
1621
Arnaldo Carvalho de Melo9955d0b2016-06-15 15:48:08 -03001622 fclose(file);
1623 err = 0;
Arnaldo Carvalho de Melo68101582018-03-14 10:34:11 -03001624out_free_command:
1625 free(command);
Jiri Olsa2c7da8c2015-03-02 12:56:12 -05001626out_remove_tmp:
Arnaldo Carvalho de Melo9955d0b2016-06-15 15:48:08 -03001627 close(stdout_fd[0]);
1628
Jiri Olsa2c7da8c2015-03-02 12:56:12 -05001629 if (dso__needs_decompress(dso))
1630 unlink(symfs_filename);
Arnaldo Carvalho de Melo3caee092016-08-09 15:16:37 -03001631
Adrian Hunterafba19d2013-10-09 15:01:12 +03001632 if (delete_extract)
1633 kcore_extract__delete(&kce);
Arnaldo Carvalho de Meloc12944f2016-08-09 14:56:13 -03001634out:
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001635 return err;
Arnaldo Carvalho de Melo9955d0b2016-06-15 15:48:08 -03001636
1637out_close_stdout:
1638 close(stdout_fd[1]);
Arnaldo Carvalho de Melo68101582018-03-14 10:34:11 -03001639 goto out_free_command;
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001640}
1641
Jiri Olsa073ae602017-10-11 17:01:40 +02001642static void calc_percent(struct sym_hist *hist,
1643 struct annotation_data *sample,
1644 s64 offset, s64 end)
1645{
1646 unsigned int hits = 0;
1647 u64 period = 0;
1648
1649 while (offset < end) {
1650 hits += hist->addr[offset].nr_samples;
1651 period += hist->addr[offset].period;
1652 ++offset;
1653 }
1654
1655 if (hist->nr_samples) {
1656 sample->he.period = period;
1657 sample->he.nr_samples = hits;
1658 sample->percent = 100.0 * hits / hist->nr_samples;
1659 }
1660}
1661
Jiri Olsa9e4e0a92017-11-15 12:05:59 +01001662static void annotation__calc_percent(struct annotation *notes,
1663 struct perf_evsel *evsel, s64 len)
Jiri Olsa073ae602017-10-11 17:01:40 +02001664{
1665 struct annotation_line *al, *next;
1666
1667 list_for_each_entry(al, &notes->src->source, node) {
1668 s64 end;
1669 int i;
1670
1671 if (al->offset == -1)
1672 continue;
1673
1674 next = annotation_line__next(al, &notes->src->source);
1675 end = next ? next->offset : len;
1676
1677 for (i = 0; i < al->samples_nr; i++) {
1678 struct annotation_data *sample;
1679 struct sym_hist *hist;
1680
1681 hist = annotation__histogram(notes, evsel->idx + i);
1682 sample = &al->samples[i];
1683
1684 calc_percent(hist, sample, al->offset, end);
1685 }
1686 }
Jiri Olsa073ae602017-10-11 17:01:40 +02001687}
1688
Jiri Olsa9e4e0a92017-11-15 12:05:59 +01001689void symbol__calc_percent(struct symbol *sym, struct perf_evsel *evsel)
Jiri Olsa073ae602017-10-11 17:01:40 +02001690{
1691 struct annotation *notes = symbol__annotation(sym);
1692
Jiri Olsa9e4e0a92017-11-15 12:05:59 +01001693 annotation__calc_percent(notes, evsel, symbol__size(sym));
Jiri Olsa073ae602017-10-11 17:01:40 +02001694}
1695
Jiri Olsac34df252017-10-11 17:01:28 +02001696int symbol__annotate(struct symbol *sym, struct map *map,
Jiri Olsad03a6862017-10-11 17:01:33 +02001697 struct perf_evsel *evsel, size_t privsize,
Arnaldo Carvalho de Melo5449f132017-12-11 12:46:11 -03001698 struct arch **parch)
Jiri Olsac34df252017-10-11 17:01:28 +02001699{
Jiri Olsaea07c5a2017-10-11 17:01:29 +02001700 struct annotate_args args = {
1701 .privsize = privsize,
Jiri Olsa1a04db72017-10-11 17:01:31 +02001702 .map = map,
Jiri Olsad03a6862017-10-11 17:01:33 +02001703 .evsel = evsel,
Jiri Olsaea07c5a2017-10-11 17:01:29 +02001704 };
Arnaldo Carvalho de Melo5449f132017-12-11 12:46:11 -03001705 struct perf_env *env = perf_evsel__env(evsel);
Arnaldo Carvalho de Melo3285deb2017-12-11 12:52:17 -03001706 const char *arch_name = perf_env__arch(env);
Jiri Olsac34df252017-10-11 17:01:28 +02001707 struct arch *arch;
1708 int err;
1709
Jiri Olsac34df252017-10-11 17:01:28 +02001710 if (!arch_name)
1711 return -1;
1712
Jiri Olsa24fe7b82017-10-11 17:01:30 +02001713 args.arch = arch = arch__find(arch_name);
Jiri Olsac34df252017-10-11 17:01:28 +02001714 if (arch == NULL)
1715 return -ENOTSUP;
1716
1717 if (parch)
1718 *parch = arch;
1719
1720 if (arch->init) {
Arnaldo Carvalho de Melo5449f132017-12-11 12:46:11 -03001721 err = arch->init(arch, env ? env->cpuid : NULL);
Jiri Olsac34df252017-10-11 17:01:28 +02001722 if (err) {
1723 pr_err("%s: failed to initialize %s arch priv area\n", __func__, arch->name);
1724 return err;
1725 }
1726 }
1727
Jiri Olsa05d3f1a2017-11-15 12:20:08 +01001728 return symbol__disassemble(sym, &args);
Jiri Olsac34df252017-10-11 17:01:28 +02001729}
1730
Jiri Olsa8b4c74d2017-10-11 17:01:41 +02001731static void insert_source_line(struct rb_root *root, struct annotation_line *al)
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001732{
Jiri Olsa8b4c74d2017-10-11 17:01:41 +02001733 struct annotation_line *iter;
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001734 struct rb_node **p = &root->rb_node;
1735 struct rb_node *parent = NULL;
Namhyung Kim1491c222013-03-05 14:53:28 +09001736 int i, ret;
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001737
1738 while (*p != NULL) {
1739 parent = *p;
Jiri Olsa8b4c74d2017-10-11 17:01:41 +02001740 iter = rb_entry(parent, struct annotation_line, rb_node);
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001741
Jiri Olsa8b4c74d2017-10-11 17:01:41 +02001742 ret = strcmp(iter->path, al->path);
Namhyung Kim41127962012-11-09 14:58:49 +09001743 if (ret == 0) {
Jiri Olsa8b4c74d2017-10-11 17:01:41 +02001744 for (i = 0; i < al->samples_nr; i++)
1745 iter->samples[i].percent_sum += al->samples[i].percent;
Namhyung Kim41127962012-11-09 14:58:49 +09001746 return;
1747 }
1748
1749 if (ret < 0)
1750 p = &(*p)->rb_left;
1751 else
1752 p = &(*p)->rb_right;
1753 }
1754
Jiri Olsa8b4c74d2017-10-11 17:01:41 +02001755 for (i = 0; i < al->samples_nr; i++)
1756 al->samples[i].percent_sum = al->samples[i].percent;
Namhyung Kim41127962012-11-09 14:58:49 +09001757
Jiri Olsa8b4c74d2017-10-11 17:01:41 +02001758 rb_link_node(&al->rb_node, parent, p);
1759 rb_insert_color(&al->rb_node, root);
Namhyung Kim41127962012-11-09 14:58:49 +09001760}
1761
Jiri Olsa8b4c74d2017-10-11 17:01:41 +02001762static int cmp_source_line(struct annotation_line *a, struct annotation_line *b)
Namhyung Kim1491c222013-03-05 14:53:28 +09001763{
1764 int i;
1765
Jiri Olsa8b4c74d2017-10-11 17:01:41 +02001766 for (i = 0; i < a->samples_nr; i++) {
Arnaldo Carvalho de Melo276af92f2015-06-19 16:36:12 -03001767 if (a->samples[i].percent_sum == b->samples[i].percent_sum)
Namhyung Kim1491c222013-03-05 14:53:28 +09001768 continue;
Arnaldo Carvalho de Melo276af92f2015-06-19 16:36:12 -03001769 return a->samples[i].percent_sum > b->samples[i].percent_sum;
Namhyung Kim1491c222013-03-05 14:53:28 +09001770 }
1771
1772 return 0;
1773}
1774
Jiri Olsa8b4c74d2017-10-11 17:01:41 +02001775static void __resort_source_line(struct rb_root *root, struct annotation_line *al)
Namhyung Kim41127962012-11-09 14:58:49 +09001776{
Jiri Olsa8b4c74d2017-10-11 17:01:41 +02001777 struct annotation_line *iter;
Namhyung Kim41127962012-11-09 14:58:49 +09001778 struct rb_node **p = &root->rb_node;
1779 struct rb_node *parent = NULL;
1780
1781 while (*p != NULL) {
1782 parent = *p;
Jiri Olsa8b4c74d2017-10-11 17:01:41 +02001783 iter = rb_entry(parent, struct annotation_line, rb_node);
Namhyung Kim41127962012-11-09 14:58:49 +09001784
Jiri Olsa8b4c74d2017-10-11 17:01:41 +02001785 if (cmp_source_line(al, iter))
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001786 p = &(*p)->rb_left;
1787 else
1788 p = &(*p)->rb_right;
1789 }
1790
Jiri Olsa8b4c74d2017-10-11 17:01:41 +02001791 rb_link_node(&al->rb_node, parent, p);
1792 rb_insert_color(&al->rb_node, root);
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001793}
1794
Namhyung Kim41127962012-11-09 14:58:49 +09001795static void resort_source_line(struct rb_root *dest_root, struct rb_root *src_root)
1796{
Jiri Olsa8b4c74d2017-10-11 17:01:41 +02001797 struct annotation_line *al;
Namhyung Kim41127962012-11-09 14:58:49 +09001798 struct rb_node *node;
1799
1800 node = rb_first(src_root);
1801 while (node) {
1802 struct rb_node *next;
1803
Jiri Olsa8b4c74d2017-10-11 17:01:41 +02001804 al = rb_entry(node, struct annotation_line, rb_node);
Namhyung Kim41127962012-11-09 14:58:49 +09001805 next = rb_next(node);
1806 rb_erase(node, src_root);
1807
Jiri Olsa8b4c74d2017-10-11 17:01:41 +02001808 __resort_source_line(dest_root, al);
Namhyung Kim41127962012-11-09 14:58:49 +09001809 node = next;
1810 }
1811}
1812
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001813static void print_summary(struct rb_root *root, const char *filename)
1814{
Jiri Olsa8b4c74d2017-10-11 17:01:41 +02001815 struct annotation_line *al;
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001816 struct rb_node *node;
1817
1818 printf("\nSorted summary for file %s\n", filename);
1819 printf("----------------------------------------------\n\n");
1820
1821 if (RB_EMPTY_ROOT(root)) {
1822 printf(" Nothing higher than %1.1f%%\n", MIN_GREEN);
1823 return;
1824 }
1825
1826 node = rb_first(root);
1827 while (node) {
Namhyung Kim1491c222013-03-05 14:53:28 +09001828 double percent, percent_max = 0.0;
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001829 const char *color;
1830 char *path;
Namhyung Kim1491c222013-03-05 14:53:28 +09001831 int i;
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001832
Jiri Olsa8b4c74d2017-10-11 17:01:41 +02001833 al = rb_entry(node, struct annotation_line, rb_node);
1834 for (i = 0; i < al->samples_nr; i++) {
1835 percent = al->samples[i].percent_sum;
Namhyung Kim1491c222013-03-05 14:53:28 +09001836 color = get_percent_color(percent);
1837 color_fprintf(stdout, color, " %7.2f", percent);
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001838
Namhyung Kim1491c222013-03-05 14:53:28 +09001839 if (percent > percent_max)
1840 percent_max = percent;
1841 }
1842
Jiri Olsa8b4c74d2017-10-11 17:01:41 +02001843 path = al->path;
Namhyung Kim1491c222013-03-05 14:53:28 +09001844 color = get_percent_color(percent_max);
Namhyung Kimf048d542013-09-11 14:09:28 +09001845 color_fprintf(stdout, color, " %s\n", path);
Namhyung Kim1491c222013-03-05 14:53:28 +09001846
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001847 node = rb_next(node);
1848 }
1849}
1850
Namhyung Kimdb8fd072013-03-05 14:53:21 +09001851static void symbol__annotate_hits(struct symbol *sym, struct perf_evsel *evsel)
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001852{
1853 struct annotation *notes = symbol__annotation(sym);
Namhyung Kimdb8fd072013-03-05 14:53:21 +09001854 struct sym_hist *h = annotation__histogram(notes, evsel->idx);
Arnaldo Carvalho de Melo1b2e2df2012-04-19 10:57:06 -03001855 u64 len = symbol__size(sym), offset;
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001856
1857 for (offset = 0; offset < len; ++offset)
Taeung Song896bccd2017-07-20 06:36:45 +09001858 if (h->addr[offset].nr_samples != 0)
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001859 printf("%*" PRIx64 ": %" PRIu64 "\n", BITS_PER_LONG / 2,
Taeung Song896bccd2017-07-20 06:36:45 +09001860 sym->start + offset, h->addr[offset].nr_samples);
Taeung Song8158683d2017-07-20 06:36:51 +09001861 printf("%*s: %" PRIu64 "\n", BITS_PER_LONG / 2, "h->nr_samples", h->nr_samples);
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001862}
1863
Jiri Olsaf48e7c42017-10-11 17:01:58 +02001864static int annotated_source__addr_fmt_width(struct list_head *lines, u64 start)
1865{
1866 char bf[32];
1867 struct annotation_line *line;
1868
1869 list_for_each_entry_reverse(line, lines, node) {
1870 if (line->offset != -1)
1871 return scnprintf(bf, sizeof(bf), "%" PRIx64, start + line->offset);
1872 }
1873
1874 return 0;
1875}
1876
Namhyung Kimdb8fd072013-03-05 14:53:21 +09001877int symbol__annotate_printf(struct symbol *sym, struct map *map,
1878 struct perf_evsel *evsel, bool full_paths,
1879 int min_pcnt, int max_lines, int context)
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001880{
1881 struct dso *dso = map->dso;
David Ahernbfd14b92012-09-08 09:06:50 -06001882 char *filename;
1883 const char *d_filename;
Arnaldo Carvalho de Melo9cdbadce2014-03-18 11:50:21 -03001884 const char *evsel_name = perf_evsel__name(evsel);
Arnaldo Carvalho de Meloce6f4fa2011-02-08 13:27:39 -02001885 struct annotation *notes = symbol__annotation(sym);
Peter Zijlstra (Intel)135cce12016-06-30 10:29:55 +02001886 struct sym_hist *h = annotation__histogram(notes, evsel->idx);
Jiri Olsa8f25b812017-10-11 17:01:46 +02001887 struct annotation_line *pos, *queue = NULL;
Arnaldo Carvalho de Melo058b4cc2012-04-02 12:59:01 -03001888 u64 start = map__rip_2objdump(map, sym->start);
Jiri Olsaf48e7c42017-10-11 17:01:58 +02001889 int printed = 2, queue_len = 0, addr_fmt_width;
Arnaldo Carvalho de Melo36532462011-02-06 14:54:44 -02001890 int more = 0;
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001891 u64 len;
Arnaldo Carvalho de Meloce9ee4a2017-07-26 17:16:46 -03001892 int width = symbol_conf.show_total_period ? 12 : 8;
Peter Zijlstra (Intel)53dd9b52016-06-30 09:17:26 -03001893 int graph_dotted_len;
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001894
David Ahernbfd14b92012-09-08 09:06:50 -06001895 filename = strdup(dso->long_name);
1896 if (!filename)
1897 return -ENOMEM;
1898
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001899 if (full_paths)
1900 d_filename = filename;
1901 else
1902 d_filename = basename(filename);
1903
Arnaldo Carvalho de Melo1b2e2df2012-04-19 10:57:06 -03001904 len = symbol__size(sym);
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001905
Namhyung Kim759ff492013-03-05 14:53:26 +09001906 if (perf_evsel__is_group_event(evsel))
Namhyung Kimb1dd4432013-03-05 14:53:25 +09001907 width *= evsel->nr_members;
1908
Peter Zijlstra (Intel)135cce12016-06-30 10:29:55 +02001909 graph_dotted_len = printf(" %-*.*s| Source code & Disassembly of %s for %s (%" PRIu64 " samples)\n",
Taeung Song1ac39372017-08-18 17:46:48 +09001910 width, width, symbol_conf.show_total_period ? "Period" :
1911 symbol_conf.show_nr_samples ? "Samples" : "Percent",
Taeung Song38d2dcd2017-07-25 06:28:42 +09001912 d_filename, evsel_name, h->nr_samples);
Arnaldo Carvalho de Melo9cdbadce2014-03-18 11:50:21 -03001913
Peter Zijlstra (Intel)53dd9b52016-06-30 09:17:26 -03001914 printf("%-*.*s----\n",
Arnaldo Carvalho de Melo9cdbadce2014-03-18 11:50:21 -03001915 graph_dotted_len, graph_dotted_len, graph_dotted_line);
Arnaldo Carvalho de Melof1e27012011-02-05 18:51:38 -02001916
Namhyung Kimbb963e12017-02-17 17:17:38 +09001917 if (verbose > 0)
Namhyung Kimdb8fd072013-03-05 14:53:21 +09001918 symbol__annotate_hits(sym, evsel);
Arnaldo Carvalho de Melof1e27012011-02-05 18:51:38 -02001919
Jiri Olsaf48e7c42017-10-11 17:01:58 +02001920 addr_fmt_width = annotated_source__addr_fmt_width(&notes->src->source, start);
1921
Jiri Olsa8f25b812017-10-11 17:01:46 +02001922 list_for_each_entry(pos, &notes->src->source, node) {
1923 int err;
1924
Arnaldo Carvalho de Melod5e3d742011-02-08 15:29:25 -02001925 if (context && queue == NULL) {
1926 queue = pos;
1927 queue_len = 0;
1928 }
1929
Jiri Olsa8f25b812017-10-11 17:01:46 +02001930 err = annotation_line__print(pos, sym, start, evsel, len,
1931 min_pcnt, printed, max_lines,
Jiri Olsaf48e7c42017-10-11 17:01:58 +02001932 queue, addr_fmt_width);
Jiri Olsa8f25b812017-10-11 17:01:46 +02001933
1934 switch (err) {
Arnaldo Carvalho de Melo36532462011-02-06 14:54:44 -02001935 case 0:
1936 ++printed;
Arnaldo Carvalho de Melod5e3d742011-02-08 15:29:25 -02001937 if (context) {
1938 printed += queue_len;
1939 queue = NULL;
1940 queue_len = 0;
1941 }
Arnaldo Carvalho de Melof1e27012011-02-05 18:51:38 -02001942 break;
Arnaldo Carvalho de Melo36532462011-02-06 14:54:44 -02001943 case 1:
1944 /* filtered by max_lines */
1945 ++more;
1946 break;
1947 case -1:
1948 default:
Arnaldo Carvalho de Melod5e3d742011-02-08 15:29:25 -02001949 /*
1950 * Filtered by min_pcnt or non IP lines when
1951 * context != 0
1952 */
1953 if (!context)
1954 break;
1955 if (queue_len == context)
Jiri Olsa8f25b812017-10-11 17:01:46 +02001956 queue = list_entry(queue->node.next, typeof(*queue), node);
Arnaldo Carvalho de Melod5e3d742011-02-08 15:29:25 -02001957 else
1958 ++queue_len;
Arnaldo Carvalho de Melo36532462011-02-06 14:54:44 -02001959 break;
1960 }
1961 }
Arnaldo Carvalho de Melof1e27012011-02-05 18:51:38 -02001962
David Ahernbfd14b92012-09-08 09:06:50 -06001963 free(filename);
1964
Arnaldo Carvalho de Melo36532462011-02-06 14:54:44 -02001965 return more;
1966}
1967
1968void symbol__annotate_zero_histogram(struct symbol *sym, int evidx)
1969{
1970 struct annotation *notes = symbol__annotation(sym);
1971 struct sym_hist *h = annotation__histogram(notes, evidx);
1972
Arnaldo Carvalho de Meloce6f4fa2011-02-08 13:27:39 -02001973 memset(h, 0, notes->src->sizeof_sym_hist);
Arnaldo Carvalho de Melo36532462011-02-06 14:54:44 -02001974}
1975
Arnaldo Carvalho de Meloce6f4fa2011-02-08 13:27:39 -02001976void symbol__annotate_decay_histogram(struct symbol *sym, int evidx)
Arnaldo Carvalho de Melo36532462011-02-06 14:54:44 -02001977{
1978 struct annotation *notes = symbol__annotation(sym);
1979 struct sym_hist *h = annotation__histogram(notes, evidx);
Arnaldo Carvalho de Melo1b2e2df2012-04-19 10:57:06 -03001980 int len = symbol__size(sym), offset;
Arnaldo Carvalho de Melo36532462011-02-06 14:54:44 -02001981
Taeung Song8158683d2017-07-20 06:36:51 +09001982 h->nr_samples = 0;
Arnaldo Carvalho de Melo8b84a562012-04-05 16:15:59 -03001983 for (offset = 0; offset < len; ++offset) {
Taeung Song896bccd2017-07-20 06:36:45 +09001984 h->addr[offset].nr_samples = h->addr[offset].nr_samples * 7 / 8;
Taeung Song8158683d2017-07-20 06:36:51 +09001985 h->nr_samples += h->addr[offset].nr_samples;
Arnaldo Carvalho de Melof1e27012011-02-05 18:51:38 -02001986 }
1987}
1988
Jiri Olsaf8eb37b2017-10-11 17:01:38 +02001989void annotated_source__purge(struct annotated_source *as)
Arnaldo Carvalho de Melof1e27012011-02-05 18:51:38 -02001990{
Jiri Olsaf8eb37b2017-10-11 17:01:38 +02001991 struct annotation_line *al, *n;
Arnaldo Carvalho de Melof1e27012011-02-05 18:51:38 -02001992
Jiri Olsaf8eb37b2017-10-11 17:01:38 +02001993 list_for_each_entry_safe(al, n, &as->source, node) {
1994 list_del(&al->node);
1995 disasm_line__free(disasm_line(al));
Arnaldo Carvalho de Melof1e27012011-02-05 18:51:38 -02001996 }
1997}
1998
Arnaldo Carvalho de Melo51454182012-04-15 15:52:18 -03001999static size_t disasm_line__fprintf(struct disasm_line *dl, FILE *fp)
2000{
2001 size_t printed;
2002
Jiri Olsad5490b92017-10-11 17:01:26 +02002003 if (dl->al.offset == -1)
2004 return fprintf(fp, "%s\n", dl->al.line);
Arnaldo Carvalho de Melo51454182012-04-15 15:52:18 -03002005
Jiri Olsad5490b92017-10-11 17:01:26 +02002006 printed = fprintf(fp, "%#" PRIx64 " %s", dl->al.offset, dl->ins.name);
Arnaldo Carvalho de Melo51454182012-04-15 15:52:18 -03002007
Arnaldo Carvalho de Meloc7e6ead2012-04-20 14:38:46 -03002008 if (dl->ops.raw[0] != '\0') {
Arnaldo Carvalho de Melo51454182012-04-15 15:52:18 -03002009 printed += fprintf(fp, "%.*s %s\n", 6 - (int)printed, " ",
Arnaldo Carvalho de Meloc7e6ead2012-04-20 14:38:46 -03002010 dl->ops.raw);
Arnaldo Carvalho de Melo51454182012-04-15 15:52:18 -03002011 }
2012
2013 return printed + fprintf(fp, "\n");
2014}
2015
2016size_t disasm__fprintf(struct list_head *head, FILE *fp)
2017{
2018 struct disasm_line *pos;
2019 size_t printed = 0;
2020
Jiri Olsaa17c4ca2017-10-11 17:01:25 +02002021 list_for_each_entry(pos, head, al.node)
Arnaldo Carvalho de Melo51454182012-04-15 15:52:18 -03002022 printed += disasm_line__fprintf(pos, fp);
2023
2024 return printed;
2025}
2026
Arnaldo Carvalho de Melo0db45bc2018-03-15 15:31:56 -03002027
2028bool disasm_line__is_valid_jump(struct disasm_line *dl, struct symbol *sym)
2029{
2030 if (!dl || !dl->ins.ops || !ins__is_jump(&dl->ins) ||
2031 !disasm_line__has_offset(dl) || dl->ops.target.offset < 0 ||
2032 dl->ops.target.offset >= (s64)symbol__size(sym))
2033 return false;
2034
2035 return true;
2036}
2037
2038void annotation__mark_jump_targets(struct annotation *notes, struct symbol *sym)
2039{
2040 u64 offset, size = symbol__size(sym);
2041
2042 /* PLT symbols contain external offsets */
2043 if (strstr(sym->name, "@plt"))
2044 return;
2045
2046 for (offset = 0; offset < size; ++offset) {
2047 struct annotation_line *al = notes->offsets[offset];
2048 struct disasm_line *dl;
2049
2050 dl = disasm_line(al);
2051
2052 if (!disasm_line__is_valid_jump(dl, sym))
2053 continue;
2054
2055 al = notes->offsets[dl->ops.target.offset];
2056
2057 /*
2058 * FIXME: Oops, no jump target? Buggy disassembler? Or do we
2059 * have to adjust to the previous offset?
2060 */
2061 if (al == NULL)
2062 continue;
2063
2064 if (++al->jump_sources > notes->max_jump_sources)
2065 notes->max_jump_sources = al->jump_sources;
2066
2067 ++notes->nr_jumps;
2068 }
2069}
2070
Arnaldo Carvalho de Melo5bc49f62018-03-15 15:59:01 -03002071void annotation__set_offsets(struct annotation *notes, s64 size)
2072{
2073 struct annotation_line *al;
2074
2075 notes->max_line_len = 0;
2076
2077 list_for_each_entry(al, &notes->src->source, node) {
2078 size_t line_len = strlen(al->line);
2079
2080 if (notes->max_line_len < line_len)
2081 notes->max_line_len = line_len;
2082 al->idx = notes->nr_entries++;
2083 if (al->offset != -1) {
2084 al->idx_asm = notes->nr_asm_entries++;
2085 /*
2086 * FIXME: short term bandaid to cope with assembly
2087 * routines that comes with labels in the same column
2088 * as the address in objdump, sigh.
2089 *
2090 * E.g. copy_user_generic_unrolled
2091 */
2092 if (al->offset < size)
2093 notes->offsets[al->offset] = al;
2094 } else
2095 al->idx_asm = -1;
2096 }
2097}
2098
Arnaldo Carvalho de Melob8b0d812018-03-15 16:26:29 -03002099static inline int width_jumps(int n)
2100{
2101 if (n >= 100)
2102 return 5;
2103 if (n / 10)
2104 return 2;
2105 return 1;
2106}
2107
2108void annotation__init_column_widths(struct annotation *notes, struct symbol *sym)
2109{
2110 notes->widths.addr = notes->widths.target =
2111 notes->widths.min_addr = hex_width(symbol__size(sym));
2112 notes->widths.max_addr = hex_width(sym->end);
2113 notes->widths.jumps = width_jumps(notes->max_jump_sources);
2114}
2115
Arnaldo Carvalho de Melo7232bf72018-03-15 16:19:59 -03002116void annotation__update_column_widths(struct annotation *notes)
2117{
2118 if (notes->options->use_offset)
2119 notes->widths.target = notes->widths.min_addr;
2120 else
2121 notes->widths.target = notes->widths.max_addr;
2122
2123 notes->widths.addr = notes->widths.target;
2124
2125 if (notes->options->show_nr_jumps)
2126 notes->widths.addr += notes->widths.jumps + 1;
2127}
2128
Jiri Olsa8b4c74d2017-10-11 17:01:41 +02002129static void annotation__calc_lines(struct annotation *notes, struct map *map,
2130 struct rb_root *root, u64 start)
2131{
2132 struct annotation_line *al;
2133 struct rb_root tmp_root = RB_ROOT;
2134
2135 list_for_each_entry(al, &notes->src->source, node) {
2136 double percent_max = 0.0;
2137 int i;
2138
2139 for (i = 0; i < al->samples_nr; i++) {
2140 struct annotation_data *sample;
2141
2142 sample = &al->samples[i];
2143
2144 if (sample->percent > percent_max)
2145 percent_max = sample->percent;
2146 }
2147
2148 if (percent_max <= 0.5)
2149 continue;
2150
Jin Yao935f5a92017-12-30 00:26:52 +08002151 al->path = get_srcline(map->dso, start + al->offset, NULL,
2152 false, true, start + al->offset);
Jiri Olsa8b4c74d2017-10-11 17:01:41 +02002153 insert_source_line(&tmp_root, al);
2154 }
2155
2156 resort_source_line(root, &tmp_root);
2157}
2158
2159static void symbol__calc_lines(struct symbol *sym, struct map *map,
2160 struct rb_root *root)
2161{
2162 struct annotation *notes = symbol__annotation(sym);
2163 u64 start = map__rip_2objdump(map, sym->start);
2164
2165 annotation__calc_lines(notes, map, root, start);
2166}
2167
Namhyung Kimdb8fd072013-03-05 14:53:21 +09002168int symbol__tty_annotate(struct symbol *sym, struct map *map,
2169 struct perf_evsel *evsel, bool print_lines,
2170 bool full_paths, int min_pcnt, int max_lines)
Arnaldo Carvalho de Melof1e27012011-02-05 18:51:38 -02002171{
2172 struct dso *dso = map->dso;
Arnaldo Carvalho de Melof1e27012011-02-05 18:51:38 -02002173 struct rb_root source_line = RB_ROOT;
Arnaldo Carvalho de Melof1e27012011-02-05 18:51:38 -02002174
Arnaldo Carvalho de Melo5449f132017-12-11 12:46:11 -03002175 if (symbol__annotate(sym, map, evsel, 0, NULL) < 0)
Arnaldo Carvalho de Melof1e27012011-02-05 18:51:38 -02002176 return -1;
2177
Jiri Olsa05d3f1a2017-11-15 12:20:08 +01002178 symbol__calc_percent(sym, evsel);
2179
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02002180 if (print_lines) {
Michael Petlan4a4c03c2015-11-09 16:33:31 +01002181 srcline_full_filename = full_paths;
Jiri Olsa8b4c74d2017-10-11 17:01:41 +02002182 symbol__calc_lines(sym, map, &source_line);
Namhyung Kim86c98ca2013-09-11 14:09:30 +09002183 print_summary(&source_line, dso->long_name);
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02002184 }
2185
Namhyung Kimdb8fd072013-03-05 14:53:21 +09002186 symbol__annotate_printf(sym, map, evsel, full_paths,
Arnaldo Carvalho de Melod5e3d742011-02-08 15:29:25 -02002187 min_pcnt, max_lines, 0);
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02002188
Jiri Olsaf8eb37b2017-10-11 17:01:38 +02002189 annotated_source__purge(symbol__annotation(sym)->src);
Arnaldo Carvalho de Melof1e27012011-02-05 18:51:38 -02002190
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02002191 return 0;
2192}
Arnaldo Carvalho de Melof626adf2013-12-18 17:10:15 -03002193
Namhyung Kim48c65bd2014-02-20 10:32:53 +09002194bool ui__has_annotation(void)
2195{
Jiri Olsa2e0453a2016-05-03 13:54:44 +02002196 return use_browser == 1 && perf_hpp_list.sym;
Namhyung Kim48c65bd2014-02-20 10:32:53 +09002197}
Arnaldo Carvalho de Meloecda45b2018-03-15 16:54:11 -03002198
Arnaldo Carvalho de Melo2f025ea2018-03-15 17:04:53 -03002199
2200double annotation_line__max_percent(struct annotation_line *al, struct annotation *notes)
2201{
2202 double percent_max = 0.0;
2203 int i;
2204
2205 for (i = 0; i < notes->nr_events; i++) {
2206 if (al->samples[i].percent > percent_max)
2207 percent_max = al->samples[i].percent;
2208 }
2209
2210 return percent_max;
2211}
2212
Arnaldo Carvalho de Meloa1e9b742018-03-15 19:12:39 -03002213static void disasm_line__write(struct disasm_line *dl, struct annotation *notes,
2214 void *obj, char *bf, size_t size,
2215 void (*obj__printf)(void *obj, const char *fmt, ...),
2216 void (*obj__write_graph)(void *obj, int graph))
2217{
2218 if (dl->ins.ops && dl->ins.ops->scnprintf) {
2219 if (ins__is_jump(&dl->ins)) {
2220 bool fwd = dl->ops.target.offset > dl->al.offset;
2221
2222 obj__write_graph(obj, fwd ? DARROW_CHAR : UARROW_CHAR);
2223 obj__printf(obj, " ");
2224 } else if (ins__is_call(&dl->ins)) {
2225 obj__write_graph(obj, RARROW_CHAR);
2226 obj__printf(obj, " ");
2227 } else if (ins__is_ret(&dl->ins)) {
2228 obj__write_graph(obj, LARROW_CHAR);
2229 obj__printf(obj, " ");
2230 } else {
2231 obj__printf(obj, " ");
2232 }
2233 } else {
2234 obj__printf(obj, " ");
2235 }
2236
2237 disasm_line__scnprintf(dl, bf, size, !notes->options->use_offset);
2238}
2239
Arnaldo Carvalho de Meloc2983042018-03-15 23:14:51 -03002240static void __annotation_line__write(struct annotation_line *al, struct annotation *notes,
2241 bool first_line, bool current_entry, bool change_color, int width,
2242 void *obj,
2243 int (*obj__set_color)(void *obj, int color),
2244 void (*obj__set_percent_color)(void *obj, double percent, bool current),
2245 int (*obj__set_jumps_percent_color)(void *obj, int nr, bool current),
2246 void (*obj__printf)(void *obj, const char *fmt, ...),
2247 void (*obj__write_graph)(void *obj, int graph))
2248
Arnaldo Carvalho de Melo2ba5eca2018-03-15 17:54:36 -03002249{
2250 double percent_max = annotation_line__max_percent(al, notes);
Arnaldo Carvalho de Meloa1e9b742018-03-15 19:12:39 -03002251 int pcnt_width = annotation__pcnt_width(notes),
2252 cycles_width = annotation__cycles_width(notes);
Arnaldo Carvalho de Melo2ba5eca2018-03-15 17:54:36 -03002253 bool show_title = false;
Arnaldo Carvalho de Meloa1e9b742018-03-15 19:12:39 -03002254 char bf[256];
2255 int printed;
Arnaldo Carvalho de Melo2ba5eca2018-03-15 17:54:36 -03002256
2257 if (first_line && (al->offset == -1 || percent_max == 0.0)) {
2258 if (notes->have_cycles) {
2259 if (al->ipc == 0.0 && al->cycles == 0)
2260 show_title = true;
2261 } else
2262 show_title = true;
2263 }
2264
Arnaldo Carvalho de Melo2ba5eca2018-03-15 17:54:36 -03002265 if (al->offset != -1 && percent_max != 0.0) {
2266 int i;
2267
2268 for (i = 0; i < notes->nr_events; i++) {
2269 obj__set_percent_color(obj, al->samples[i].percent, current_entry);
2270 if (notes->options->show_total_period) {
2271 obj__printf(obj, "%11" PRIu64 " ", al->samples[i].he.period);
2272 } else if (notes->options->show_nr_samples) {
2273 obj__printf(obj, "%6" PRIu64 " ",
2274 al->samples[i].he.nr_samples);
2275 } else {
2276 obj__printf(obj, "%6.2f ",
2277 al->samples[i].percent);
2278 }
2279 }
2280 } else {
Arnaldo Carvalho de Melo2ba5eca2018-03-15 17:54:36 -03002281 obj__set_percent_color(obj, 0, current_entry);
2282
2283 if (!show_title)
Arnaldo Carvalho de Meloa1e9b742018-03-15 19:12:39 -03002284 obj__printf(obj, "%-*s", pcnt_width, " ");
Arnaldo Carvalho de Melo2ba5eca2018-03-15 17:54:36 -03002285 else {
Arnaldo Carvalho de Meloa1e9b742018-03-15 19:12:39 -03002286 obj__printf(obj, "%-*s", pcnt_width,
Arnaldo Carvalho de Melo2ba5eca2018-03-15 17:54:36 -03002287 notes->options->show_total_period ? "Period" :
2288 notes->options->show_nr_samples ? "Samples" : "Percent");
2289 }
2290 }
2291
2292 if (notes->have_cycles) {
2293 if (al->ipc)
2294 obj__printf(obj, "%*.2f ", ANNOTATION__IPC_WIDTH - 1, al->ipc);
2295 else if (!show_title)
2296 obj__printf(obj, "%*s", ANNOTATION__IPC_WIDTH, " ");
2297 else
2298 obj__printf(obj, "%*s ", ANNOTATION__IPC_WIDTH - 1, "IPC");
2299
2300 if (al->cycles)
2301 obj__printf(obj, "%*" PRIu64 " ",
2302 ANNOTATION__CYCLES_WIDTH - 1, al->cycles);
2303 else if (!show_title)
2304 obj__printf(obj, "%*s", ANNOTATION__CYCLES_WIDTH, " ");
2305 else
2306 obj__printf(obj, "%*s ", ANNOTATION__CYCLES_WIDTH - 1, "Cycle");
2307 }
2308
2309 obj__printf(obj, " ");
Arnaldo Carvalho de Meloa1e9b742018-03-15 19:12:39 -03002310
2311 if (!*al->line)
2312 obj__printf(obj, "%-*s", width - pcnt_width - cycles_width, " ");
2313 else if (al->offset == -1) {
2314 if (al->line_nr && notes->options->show_linenr)
2315 printed = scnprintf(bf, sizeof(bf), "%-*d ", notes->widths.addr + 1, al->line_nr);
2316 else
2317 printed = scnprintf(bf, sizeof(bf), "%-*s ", notes->widths.addr, " ");
2318 obj__printf(obj, bf);
2319 obj__printf(obj, "%-*s", width - printed - pcnt_width - cycles_width + 1, al->line);
2320 } else {
2321 u64 addr = al->offset;
2322 int color = -1;
2323
2324 if (!notes->options->use_offset)
2325 addr += notes->start;
2326
2327 if (!notes->options->use_offset) {
2328 printed = scnprintf(bf, sizeof(bf), "%" PRIx64 ": ", addr);
2329 } else {
2330 if (al->jump_sources) {
2331 if (notes->options->show_nr_jumps) {
2332 int prev;
2333 printed = scnprintf(bf, sizeof(bf), "%*d ",
2334 notes->widths.jumps,
2335 al->jump_sources);
2336 prev = obj__set_jumps_percent_color(obj, al->jump_sources,
2337 current_entry);
2338 obj__printf(obj, bf);
2339 obj__set_color(obj, prev);
2340 }
2341
2342 printed = scnprintf(bf, sizeof(bf), "%*" PRIx64 ": ",
2343 notes->widths.target, addr);
2344 } else {
2345 printed = scnprintf(bf, sizeof(bf), "%-*s ",
2346 notes->widths.addr, " ");
2347 }
2348 }
2349
2350 if (change_color)
2351 color = obj__set_color(obj, HE_COLORSET_ADDR);
2352 obj__printf(obj, bf);
2353 if (change_color)
2354 obj__set_color(obj, color);
2355
2356 disasm_line__write(disasm_line(al), notes, obj, bf, sizeof(bf), obj__printf, obj__write_graph);
2357
2358 obj__printf(obj, "%-*s", width - pcnt_width - cycles_width - 3 - printed, bf);
2359 }
2360
Arnaldo Carvalho de Melo2ba5eca2018-03-15 17:54:36 -03002361}
2362
Arnaldo Carvalho de Meloc2983042018-03-15 23:14:51 -03002363void annotation_line__write(struct annotation_line *al, struct annotation *notes,
2364 struct annotation_write_ops *ops)
2365{
2366 __annotation_line__write(al, notes, ops->first_line, ops->current_entry,
2367 ops->change_color, ops->width, ops->obj,
2368 ops->set_color, ops->set_percent_color,
2369 ops->set_jumps_percent_color, ops->printf,
2370 ops->write_graph);
2371}
2372
Arnaldo Carvalho de Meloecda45b2018-03-15 16:54:11 -03002373int symbol__annotate2(struct symbol *sym, struct map *map, struct perf_evsel *evsel,
2374 struct annotation_options *options, struct arch **parch)
2375{
2376 struct annotation *notes = symbol__annotation(sym);
2377 size_t size = symbol__size(sym);
2378 int nr_pcnt = 1, err;
2379
2380 notes->offsets = zalloc(size * sizeof(struct annotation_line *));
2381 if (notes->offsets == NULL)
2382 return -1;
2383
2384 if (perf_evsel__is_group_event(evsel))
2385 nr_pcnt = evsel->nr_members;
2386
2387 err = symbol__annotate(sym, map, evsel, 0, parch);
2388 if (err)
2389 goto out_free_offsets;
2390
2391 notes->options = options;
2392
2393 symbol__calc_percent(sym, evsel);
2394
2395 notes->start = map__rip_2objdump(map, sym->start);
2396
2397 annotation__set_offsets(notes, size);
2398 annotation__mark_jump_targets(notes, sym);
2399 annotation__compute_ipc(notes, size);
2400 annotation__init_column_widths(notes, sym);
2401 notes->nr_events = nr_pcnt;
2402
2403 annotation__update_column_widths(notes);
2404
2405 return 0;
2406
2407out_free_offsets:
2408 zfree(&notes->offsets);
2409 return -1;
2410}