blob: 87e4cadc5d2758ce4603428560c9f8548e4fc851 [file] [log] [blame]
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02001#ifndef __PERF_ANNOTATE_H
2#define __PERF_ANNOTATE_H
3
4#include <stdbool.h>
Arnaldo Carvalho de Melofb29fa52012-04-25 14:16:03 -03005#include <stdint.h>
Borislav Petkovd944c4e2014-04-25 21:31:02 +02006#include <linux/types.h>
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02007#include "symbol.h"
Namhyung Kim9783adf2012-11-02 14:50:05 +09008#include "hist.h"
Namhyung Kim2b676bf2013-02-07 18:02:08 +09009#include "sort.h"
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -020010#include <linux/list.h>
11#include <linux/rbtree.h>
Irina Tirdea27683dc2012-09-08 03:43:19 +030012#include <pthread.h>
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -020013
Arnaldo Carvalho de Melo75b49202016-11-24 11:16:06 -030014struct ins_ops;
15
16struct ins {
17 const char *name;
18 struct ins_ops *ops;
19};
Arnaldo Carvalho de Melo28548d72012-04-19 10:16:27 -030020
Arnaldo Carvalho de Meloc7e6ead2012-04-20 14:38:46 -030021struct ins_operands {
22 char *raw;
Arnaldo Carvalho de Melo44d1a3e2012-04-25 08:00:23 -030023 struct {
Arnaldo Carvalho de Melo6de783b2012-05-11 16:48:49 -030024 char *raw;
Arnaldo Carvalho de Melo44d1a3e2012-04-25 08:00:23 -030025 char *name;
Arnaldo Carvalho de Melo44d1a3e2012-04-25 08:00:23 -030026 u64 addr;
Arnaldo Carvalho de Melo6de783b2012-05-11 16:48:49 -030027 u64 offset;
Arnaldo Carvalho de Melo44d1a3e2012-04-25 08:00:23 -030028 } target;
Arnaldo Carvalho de Melo7a997fe2012-05-12 13:15:34 -030029 union {
30 struct {
31 char *raw;
32 char *name;
33 u64 addr;
34 } source;
35 struct {
Arnaldo Carvalho de Melo75b49202016-11-24 11:16:06 -030036 struct ins ins;
Arnaldo Carvalho de Melo7a997fe2012-05-12 13:15:34 -030037 struct ins_operands *ops;
38 } locked;
39 };
Arnaldo Carvalho de Meloc7e6ead2012-04-20 14:38:46 -030040};
41
Arnaldo Carvalho de Melo786c1b52016-11-16 15:39:50 -030042struct arch;
43
Arnaldo Carvalho de Melo4f9d0322012-04-18 13:58:34 -030044struct ins_ops {
Arnaldo Carvalho de Meloc46219ac2012-05-12 13:26:20 -030045 void (*free)(struct ins_operands *ops);
Arnaldo Carvalho de Melo786c1b52016-11-16 15:39:50 -030046 int (*parse)(struct arch *arch, struct ins_operands *ops, struct map *map);
Arnaldo Carvalho de Melo28548d72012-04-19 10:16:27 -030047 int (*scnprintf)(struct ins *ins, char *bf, size_t size,
Arnaldo Carvalho de Melo54170722012-05-07 18:54:16 -030048 struct ins_operands *ops);
Arnaldo Carvalho de Melo4f9d0322012-04-18 13:58:34 -030049};
50
Arnaldo Carvalho de Melo4f9d0322012-04-18 13:58:34 -030051bool ins__is_jump(const struct ins *ins);
Arnaldo Carvalho de Melod86b0592012-04-18 16:07:38 -030052bool ins__is_call(const struct ins *ins);
Naveen N. Rao6ef94922016-06-24 17:23:58 +053053bool ins__is_ret(const struct ins *ins);
Arnaldo Carvalho de Melo54170722012-05-07 18:54:16 -030054int ins__scnprintf(struct ins *ins, char *bf, size_t size, struct ins_operands *ops);
Arnaldo Carvalho de Melo4f9d0322012-04-18 13:58:34 -030055
Namhyung Kime64aa752013-03-05 14:53:30 +090056struct annotation;
57
Arnaldo Carvalho de Melo29ed6e72012-04-15 15:24:39 -030058struct disasm_line {
Arnaldo Carvalho de Meloc7e6ead2012-04-20 14:38:46 -030059 struct list_head node;
60 s64 offset;
61 char *line;
Arnaldo Carvalho de Melo75b49202016-11-24 11:16:06 -030062 struct ins ins;
Andi Kleene5924882014-11-12 18:05:26 -080063 int line_nr;
Andi Kleen30e863b2015-07-18 08:24:50 -070064 float ipc;
65 u64 cycles;
Arnaldo Carvalho de Meloc7e6ead2012-04-20 14:38:46 -030066 struct ins_operands ops;
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -020067};
68
Arnaldo Carvalho de Melofb29fa52012-04-25 14:16:03 -030069static inline bool disasm_line__has_offset(const struct disasm_line *dl)
70{
71 return dl->ops.target.offset != UINT64_MAX;
72}
73
Arnaldo Carvalho de Melo29ed6e72012-04-15 15:24:39 -030074void disasm_line__free(struct disasm_line *dl);
75struct disasm_line *disasm__get_next_ip_line(struct list_head *head, struct disasm_line *pos);
Arnaldo Carvalho de Melo54170722012-05-07 18:54:16 -030076int disasm_line__scnprintf(struct disasm_line *dl, char *bf, size_t size, bool raw);
Arnaldo Carvalho de Melo51454182012-04-15 15:52:18 -030077size_t disasm__fprintf(struct list_head *head, FILE *fp);
Namhyung Kime64aa752013-03-05 14:53:30 +090078double disasm__calc_percent(struct annotation *notes, int evidx, s64 offset,
Martin Liška0c4a5bc2015-06-19 16:10:43 -030079 s64 end, const char **path, u64 *nr_samples);
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -020080
81struct sym_hist {
82 u64 sum;
83 u64 addr[0];
84};
85
Andi Kleend4957632015-07-18 08:24:48 -070086struct cyc_hist {
87 u64 start;
88 u64 cycles;
89 u64 cycles_aggr;
90 u32 num;
91 u32 num_aggr;
92 u8 have_start;
93 /* 1 byte padding */
94 u16 reset;
95};
96
Arnaldo Carvalho de Melo276af922015-06-19 16:36:12 -030097struct source_line_samples {
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -020098 double percent;
Namhyung Kim41127962012-11-09 14:58:49 +090099 double percent_sum;
Arnaldo Carvalho de Melo276af922015-06-19 16:36:12 -0300100 double nr;
Namhyung Kimc5a83682013-03-05 14:53:27 +0900101};
102
103struct source_line {
104 struct rb_node node;
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -0200105 char *path;
Namhyung Kim1491c222013-03-05 14:53:28 +0900106 int nr_pcnt;
Arnaldo Carvalho de Melo276af922015-06-19 16:36:12 -0300107 struct source_line_samples samples[1];
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -0200108};
109
Arnaldo Carvalho de Meloce6f4fa2011-02-08 13:27:39 -0200110/** struct annotated_source - symbols with hits have this attached as in sannotation
Arnaldo Carvalho de Melo2f525d02011-02-04 13:43:24 -0200111 *
112 * @histogram: Array of addr hit histograms per event being monitored
Arnaldo Carvalho de Meloce6f4fa2011-02-08 13:27:39 -0200113 * @lines: If 'print_lines' is specified, per source code line percentages
Arnaldo Carvalho de Melo29ed6e72012-04-15 15:24:39 -0300114 * @source: source parsed from a disassembler like objdump -dS
Andi Kleend4957632015-07-18 08:24:48 -0700115 * @cyc_hist: Average cycles per basic block
Arnaldo Carvalho de Melo2f525d02011-02-04 13:43:24 -0200116 *
Arnaldo Carvalho de Meloce6f4fa2011-02-08 13:27:39 -0200117 * lines is allocated, percentages calculated and all sorted by percentage
Arnaldo Carvalho de Melo2f525d02011-02-04 13:43:24 -0200118 * when the annotation is about to be presented, so the percentages are for
119 * one of the entries in the histogram array, i.e. for the event/counter being
120 * presented. It is deallocated right after symbol__{tui,tty,etc}_annotate
121 * returns.
122 */
Arnaldo Carvalho de Meloce6f4fa2011-02-08 13:27:39 -0200123struct annotated_source {
124 struct list_head source;
125 struct source_line *lines;
Arnaldo Carvalho de Melo36532462011-02-06 14:54:44 -0200126 int nr_histograms;
Jiri Olsa5ec45022015-10-05 20:06:03 +0200127 size_t sizeof_sym_hist;
Andi Kleend4957632015-07-18 08:24:48 -0700128 struct cyc_hist *cycles_hist;
Arnaldo Carvalho de Meloce6f4fa2011-02-08 13:27:39 -0200129 struct sym_hist histograms[0];
130};
131
132struct annotation {
133 pthread_mutex_t lock;
Peter Zijlstra70fbe052016-09-05 16:08:12 -0300134 u64 max_coverage;
Arnaldo Carvalho de Meloce6f4fa2011-02-08 13:27:39 -0200135 struct annotated_source *src;
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -0200136};
137
Arnaldo Carvalho de Melo2f525d02011-02-04 13:43:24 -0200138static inline struct sym_hist *annotation__histogram(struct annotation *notes, int idx)
139{
Arnaldo Carvalho de Meloce6f4fa2011-02-08 13:27:39 -0200140 return (((void *)&notes->src->histograms) +
141 (notes->src->sizeof_sym_hist * idx));
Arnaldo Carvalho de Melo2f525d02011-02-04 13:43:24 -0200142}
143
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -0200144static inline struct annotation *symbol__annotation(struct symbol *sym)
145{
Namhyung Kim813ccd12015-01-14 20:18:05 +0900146 return (void *)sym - symbol_conf.priv_size;
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -0200147}
148
Arnaldo Carvalho de Melo0f4e7a22013-12-18 16:48:29 -0300149int addr_map_symbol__inc_samples(struct addr_map_symbol *ams, int evidx);
150
Andi Kleend4957632015-07-18 08:24:48 -0700151int addr_map_symbol__account_cycles(struct addr_map_symbol *ams,
152 struct addr_map_symbol *start,
153 unsigned cycles);
154
Arnaldo Carvalho de Melof626adf2013-12-18 17:10:15 -0300155int hist_entry__inc_addr_samples(struct hist_entry *he, int evidx, u64 addr);
156
Arnaldo Carvalho de Melod04b35f2011-11-11 22:17:32 -0200157int symbol__alloc_hist(struct symbol *sym);
Arnaldo Carvalho de Melo36532462011-02-06 14:54:44 -0200158void symbol__annotate_zero_histograms(struct symbol *sym);
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -0200159
Arnaldo Carvalho de Melo786c1b52016-11-16 15:39:50 -0300160int symbol__disassemble(struct symbol *sym, struct map *map, const char *arch_name, size_t privsize);
Arnaldo Carvalho de Melof626adf2013-12-18 17:10:15 -0300161
Arnaldo Carvalho de Meloee51d852016-07-29 16:27:18 -0300162enum symbol_disassemble_errno {
163 SYMBOL_ANNOTATE_ERRNO__SUCCESS = 0,
164
165 /*
166 * Choose an arbitrary negative big number not to clash with standard
167 * errno since SUS requires the errno has distinct positive values.
168 * See 'Issue 6' in the link below.
169 *
170 * http://pubs.opengroup.org/onlinepubs/9699919799/basedefs/errno.h.html
171 */
172 __SYMBOL_ANNOTATE_ERRNO__START = -10000,
173
174 SYMBOL_ANNOTATE_ERRNO__NO_VMLINUX = __SYMBOL_ANNOTATE_ERRNO__START,
175
176 __SYMBOL_ANNOTATE_ERRNO__END,
177};
178
179int symbol__strerror_disassemble(struct symbol *sym, struct map *map,
180 int errnum, char *buf, size_t buflen);
181
Namhyung Kimdb8fd072013-03-05 14:53:21 +0900182int symbol__annotate_printf(struct symbol *sym, struct map *map,
183 struct perf_evsel *evsel, bool full_paths,
184 int min_pcnt, int max_lines, int context);
Arnaldo Carvalho de Melo36532462011-02-06 14:54:44 -0200185void symbol__annotate_zero_histogram(struct symbol *sym, int evidx);
Arnaldo Carvalho de Meloce6f4fa2011-02-08 13:27:39 -0200186void symbol__annotate_decay_histogram(struct symbol *sym, int evidx);
Arnaldo Carvalho de Melo29ed6e72012-04-15 15:24:39 -0300187void disasm__purge(struct list_head *head);
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -0200188
Namhyung Kim48c65bd2014-02-20 10:32:53 +0900189bool ui__has_annotation(void);
190
Namhyung Kimdb8fd072013-03-05 14:53:21 +0900191int symbol__tty_annotate(struct symbol *sym, struct map *map,
192 struct perf_evsel *evsel, bool print_lines,
193 bool full_paths, int min_pcnt, int max_lines);
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -0200194
Ingo Molnar89fe8082013-09-30 12:07:11 +0200195#ifdef HAVE_SLANG_SUPPORT
Namhyung Kimdb8fd072013-03-05 14:53:21 +0900196int symbol__tui_annotate(struct symbol *sym, struct map *map,
197 struct perf_evsel *evsel,
Namhyung Kim9783adf2012-11-02 14:50:05 +0900198 struct hist_browser_timer *hbt);
Namhyung Kim1254b51e2012-09-28 18:32:02 +0900199#else
Irina Tirdea1d037ca2012-09-11 01:15:03 +0300200static inline int symbol__tui_annotate(struct symbol *sym __maybe_unused,
Namhyung Kimdb8fd072013-03-05 14:53:21 +0900201 struct map *map __maybe_unused,
202 struct perf_evsel *evsel __maybe_unused,
203 struct hist_browser_timer *hbt
204 __maybe_unused)
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -0200205{
206 return 0;
207}
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -0200208#endif
209
Andi Kleenf69b64f2011-09-15 14:31:41 -0700210extern const char *disassembler_style;
211
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -0200212#endif /* __PERF_ANNOTATE_H */