blob: 6f01e611793692fae233b5f34691e69eb85b7242 [file] [log] [blame]
Greg Kroah-Hartmanb2441312017-11-01 15:07:57 +01001/* SPDX-License-Identifier: GPL-2.0 */
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02002#ifndef __PERF_ANNOTATE_H
3#define __PERF_ANNOTATE_H
4
5#include <stdbool.h>
Arnaldo Carvalho de Melofb29fa52012-04-25 14:16:03 -03006#include <stdint.h>
Borislav Petkovd944c4e2014-04-25 21:31:02 +02007#include <linux/types.h>
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -02008#include "symbol.h"
Namhyung Kim9783adf2012-11-02 14:50:05 +09009#include "hist.h"
Namhyung Kim2b676bf2013-02-07 18:02:08 +090010#include "sort.h"
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -020011#include <linux/list.h>
12#include <linux/rbtree.h>
Irina Tirdea27683dc2012-09-08 03:43:19 +030013#include <pthread.h>
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -020014
Arnaldo Carvalho de Melo75b49202016-11-24 11:16:06 -030015struct ins_ops;
16
17struct ins {
18 const char *name;
19 struct ins_ops *ops;
20};
Arnaldo Carvalho de Melo28548d72012-04-19 10:16:27 -030021
Arnaldo Carvalho de Meloc7e6ead2012-04-20 14:38:46 -030022struct ins_operands {
23 char *raw;
Arnaldo Carvalho de Melo44d1a3e2012-04-25 08:00:23 -030024 struct {
Arnaldo Carvalho de Melo6de783b2012-05-11 16:48:49 -030025 char *raw;
Arnaldo Carvalho de Melo44d1a3e2012-04-25 08:00:23 -030026 char *name;
Arnaldo Carvalho de Melo44d1a3e2012-04-25 08:00:23 -030027 u64 addr;
Ravi Bangoriae2168742016-12-05 21:26:47 +053028 s64 offset;
29 bool offset_avail;
Arnaldo Carvalho de Melo44d1a3e2012-04-25 08:00:23 -030030 } target;
Arnaldo Carvalho de Melo7a997fe2012-05-12 13:15:34 -030031 union {
32 struct {
33 char *raw;
34 char *name;
35 u64 addr;
36 } source;
37 struct {
Arnaldo Carvalho de Melo75b49202016-11-24 11:16:06 -030038 struct ins ins;
Arnaldo Carvalho de Melo7a997fe2012-05-12 13:15:34 -030039 struct ins_operands *ops;
40 } locked;
41 };
Arnaldo Carvalho de Meloc7e6ead2012-04-20 14:38:46 -030042};
43
Arnaldo Carvalho de Melo786c1b52016-11-16 15:39:50 -030044struct arch;
45
Arnaldo Carvalho de Melo4f9d0322012-04-18 13:58:34 -030046struct ins_ops {
Arnaldo Carvalho de Meloc46219ac2012-05-12 13:26:20 -030047 void (*free)(struct ins_operands *ops);
Arnaldo Carvalho de Melo786c1b52016-11-16 15:39:50 -030048 int (*parse)(struct arch *arch, struct ins_operands *ops, struct map *map);
Arnaldo Carvalho de Melo28548d72012-04-19 10:16:27 -030049 int (*scnprintf)(struct ins *ins, char *bf, size_t size,
Arnaldo Carvalho de Melo54170722012-05-07 18:54:16 -030050 struct ins_operands *ops);
Arnaldo Carvalho de Melo4f9d0322012-04-18 13:58:34 -030051};
52
Arnaldo Carvalho de Melo4f9d0322012-04-18 13:58:34 -030053bool ins__is_jump(const struct ins *ins);
Arnaldo Carvalho de Melod86b0592012-04-18 16:07:38 -030054bool ins__is_call(const struct ins *ins);
Naveen N. Rao6ef94922016-06-24 17:23:58 +053055bool ins__is_ret(const struct ins *ins);
Jin Yao7e63a132017-07-07 13:06:35 +080056bool ins__is_lock(const struct ins *ins);
Arnaldo Carvalho de Melo54170722012-05-07 18:54:16 -030057int ins__scnprintf(struct ins *ins, char *bf, size_t size, struct ins_operands *ops);
Jin Yao69fb09f2017-07-07 13:06:34 +080058bool ins__is_fused(struct arch *arch, const char *ins1, const char *ins2);
Arnaldo Carvalho de Melo4f9d0322012-04-18 13:58:34 -030059
Namhyung Kime64aa752013-03-05 14:53:30 +090060struct annotation;
61
Jiri Olsaa17c4ca2017-10-11 17:01:25 +020062struct annotation_line {
63 struct list_head node;
Jiri Olsa5b12adc2017-10-11 17:01:36 +020064 struct rb_node rb_node;
Jiri Olsad5490b92017-10-11 17:01:26 +020065 s64 offset;
66 char *line;
67 int line_nr;
Jiri Olsa37236d52017-10-11 17:01:27 +020068 float ipc;
69 u64 cycles;
Jiri Olsaa17c4ca2017-10-11 17:01:25 +020070};
71
Arnaldo Carvalho de Melo29ed6e72012-04-15 15:24:39 -030072struct disasm_line {
Jiri Olsaa17c4ca2017-10-11 17:01:25 +020073 struct annotation_line al;
Jiri Olsaa17c4ca2017-10-11 17:01:25 +020074 struct ins ins;
Jiri Olsaa17c4ca2017-10-11 17:01:25 +020075 struct ins_operands ops;
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -020076};
77
Arnaldo Carvalho de Melofb29fa52012-04-25 14:16:03 -030078static inline bool disasm_line__has_offset(const struct disasm_line *dl)
79{
Ravi Bangoriae2168742016-12-05 21:26:47 +053080 return dl->ops.target.offset_avail;
Arnaldo Carvalho de Melofb29fa52012-04-25 14:16:03 -030081}
82
Taeung Song896bccd2017-07-20 06:36:45 +090083struct sym_hist_entry {
84 u64 nr_samples;
85 u64 period;
86};
87
Arnaldo Carvalho de Melo29ed6e72012-04-15 15:24:39 -030088void disasm_line__free(struct disasm_line *dl);
Jiri Olsac4c72432017-10-11 17:01:34 +020089struct annotation_line *
90annotation_line__next(struct annotation_line *pos, struct list_head *head);
Arnaldo Carvalho de Melo54170722012-05-07 18:54:16 -030091int disasm_line__scnprintf(struct disasm_line *dl, char *bf, size_t size, bool raw);
Arnaldo Carvalho de Melo51454182012-04-15 15:52:18 -030092size_t disasm__fprintf(struct list_head *head, FILE *fp);
Namhyung Kime64aa752013-03-05 14:53:30 +090093double disasm__calc_percent(struct annotation *notes, int evidx, s64 offset,
Taeung Song896bccd2017-07-20 06:36:45 +090094 s64 end, const char **path, struct sym_hist_entry *sample);
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -020095
96struct sym_hist {
Taeung Song8158683d2017-07-20 06:36:51 +090097 u64 nr_samples;
Taeung Song461c17f2017-07-20 17:18:05 -030098 u64 period;
Taeung Song896bccd2017-07-20 06:36:45 +090099 struct sym_hist_entry addr[0];
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -0200100};
101
Andi Kleend4957632015-07-18 08:24:48 -0700102struct cyc_hist {
103 u64 start;
104 u64 cycles;
105 u64 cycles_aggr;
106 u32 num;
107 u32 num_aggr;
108 u8 have_start;
109 /* 1 byte padding */
110 u16 reset;
111};
112
Arnaldo Carvalho de Melo276af92f2015-06-19 16:36:12 -0300113struct source_line_samples {
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -0200114 double percent;
Namhyung Kim41127962012-11-09 14:58:49 +0900115 double percent_sum;
Taeung Song99094a5e2017-03-28 21:12:05 +0900116 u64 nr;
Namhyung Kimc5a83682013-03-05 14:53:27 +0900117};
118
119struct source_line {
120 struct rb_node node;
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -0200121 char *path;
Namhyung Kim1491c222013-03-05 14:53:28 +0900122 int nr_pcnt;
Arnaldo Carvalho de Melo276af92f2015-06-19 16:36:12 -0300123 struct source_line_samples samples[1];
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -0200124};
125
Arnaldo Carvalho de Meloce6f4fa2011-02-08 13:27:39 -0200126/** struct annotated_source - symbols with hits have this attached as in sannotation
Arnaldo Carvalho de Melo2f525d02011-02-04 13:43:24 -0200127 *
128 * @histogram: Array of addr hit histograms per event being monitored
Arnaldo Carvalho de Meloce6f4fa2011-02-08 13:27:39 -0200129 * @lines: If 'print_lines' is specified, per source code line percentages
Arnaldo Carvalho de Melo29ed6e72012-04-15 15:24:39 -0300130 * @source: source parsed from a disassembler like objdump -dS
Andi Kleend4957632015-07-18 08:24:48 -0700131 * @cyc_hist: Average cycles per basic block
Arnaldo Carvalho de Melo2f525d02011-02-04 13:43:24 -0200132 *
Arnaldo Carvalho de Meloce6f4fa2011-02-08 13:27:39 -0200133 * lines is allocated, percentages calculated and all sorted by percentage
Arnaldo Carvalho de Melo2f525d02011-02-04 13:43:24 -0200134 * when the annotation is about to be presented, so the percentages are for
135 * one of the entries in the histogram array, i.e. for the event/counter being
136 * presented. It is deallocated right after symbol__{tui,tty,etc}_annotate
137 * returns.
138 */
Arnaldo Carvalho de Meloce6f4fa2011-02-08 13:27:39 -0200139struct annotated_source {
140 struct list_head source;
141 struct source_line *lines;
Arnaldo Carvalho de Melo36532462011-02-06 14:54:44 -0200142 int nr_histograms;
Jiri Olsa5ec45022015-10-05 20:06:03 +0200143 size_t sizeof_sym_hist;
Andi Kleend4957632015-07-18 08:24:48 -0700144 struct cyc_hist *cycles_hist;
Arnaldo Carvalho de Meloce6f4fa2011-02-08 13:27:39 -0200145 struct sym_hist histograms[0];
146};
147
148struct annotation {
149 pthread_mutex_t lock;
Peter Zijlstra70fbe052016-09-05 16:08:12 -0300150 u64 max_coverage;
Arnaldo Carvalho de Meloce6f4fa2011-02-08 13:27:39 -0200151 struct annotated_source *src;
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -0200152};
153
Arnaldo Carvalho de Melo2f525d02011-02-04 13:43:24 -0200154static inline struct sym_hist *annotation__histogram(struct annotation *notes, int idx)
155{
Arnaldo Carvalho de Meloce6f4fa2011-02-08 13:27:39 -0200156 return (((void *)&notes->src->histograms) +
157 (notes->src->sizeof_sym_hist * idx));
Arnaldo Carvalho de Melo2f525d02011-02-04 13:43:24 -0200158}
159
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -0200160static inline struct annotation *symbol__annotation(struct symbol *sym)
161{
Namhyung Kim813ccd12015-01-14 20:18:05 +0900162 return (void *)sym - symbol_conf.priv_size;
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -0200163}
164
Taeung Songbab89f62017-07-20 16:28:53 -0300165int addr_map_symbol__inc_samples(struct addr_map_symbol *ams, struct perf_sample *sample,
166 int evidx);
Arnaldo Carvalho de Melo0f4e7a22013-12-18 16:48:29 -0300167
Andi Kleend4957632015-07-18 08:24:48 -0700168int addr_map_symbol__account_cycles(struct addr_map_symbol *ams,
169 struct addr_map_symbol *start,
170 unsigned cycles);
171
Taeung Songbab89f62017-07-20 16:28:53 -0300172int hist_entry__inc_addr_samples(struct hist_entry *he, struct perf_sample *sample,
173 int evidx, u64 addr);
Arnaldo Carvalho de Melof626adf2013-12-18 17:10:15 -0300174
Arnaldo Carvalho de Melod04b35f2011-11-11 22:17:32 -0200175int symbol__alloc_hist(struct symbol *sym);
Arnaldo Carvalho de Melo36532462011-02-06 14:54:44 -0200176void symbol__annotate_zero_histograms(struct symbol *sym);
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -0200177
Jiri Olsac34df252017-10-11 17:01:28 +0200178int symbol__annotate(struct symbol *sym, struct map *map,
Jiri Olsad03a6862017-10-11 17:01:33 +0200179 struct perf_evsel *evsel, size_t privsize,
Jiri Olsac34df252017-10-11 17:01:28 +0200180 struct arch **parch, char *cpuid);
Arnaldo Carvalho de Melof626adf2013-12-18 17:10:15 -0300181
Arnaldo Carvalho de Meloee51d852016-07-29 16:27:18 -0300182enum symbol_disassemble_errno {
183 SYMBOL_ANNOTATE_ERRNO__SUCCESS = 0,
184
185 /*
186 * Choose an arbitrary negative big number not to clash with standard
187 * errno since SUS requires the errno has distinct positive values.
188 * See 'Issue 6' in the link below.
189 *
190 * http://pubs.opengroup.org/onlinepubs/9699919799/basedefs/errno.h.html
191 */
192 __SYMBOL_ANNOTATE_ERRNO__START = -10000,
193
194 SYMBOL_ANNOTATE_ERRNO__NO_VMLINUX = __SYMBOL_ANNOTATE_ERRNO__START,
195
196 __SYMBOL_ANNOTATE_ERRNO__END,
197};
198
199int symbol__strerror_disassemble(struct symbol *sym, struct map *map,
200 int errnum, char *buf, size_t buflen);
201
Namhyung Kimdb8fd072013-03-05 14:53:21 +0900202int symbol__annotate_printf(struct symbol *sym, struct map *map,
203 struct perf_evsel *evsel, bool full_paths,
204 int min_pcnt, int max_lines, int context);
Arnaldo Carvalho de Melo36532462011-02-06 14:54:44 -0200205void symbol__annotate_zero_histogram(struct symbol *sym, int evidx);
Arnaldo Carvalho de Meloce6f4fa2011-02-08 13:27:39 -0200206void symbol__annotate_decay_histogram(struct symbol *sym, int evidx);
Arnaldo Carvalho de Melo29ed6e72012-04-15 15:24:39 -0300207void disasm__purge(struct list_head *head);
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -0200208
Namhyung Kim48c65bd2014-02-20 10:32:53 +0900209bool ui__has_annotation(void);
210
Namhyung Kimdb8fd072013-03-05 14:53:21 +0900211int symbol__tty_annotate(struct symbol *sym, struct map *map,
212 struct perf_evsel *evsel, bool print_lines,
213 bool full_paths, int min_pcnt, int max_lines);
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -0200214
Ingo Molnar89fe8082013-09-30 12:07:11 +0200215#ifdef HAVE_SLANG_SUPPORT
Namhyung Kimdb8fd072013-03-05 14:53:21 +0900216int symbol__tui_annotate(struct symbol *sym, struct map *map,
217 struct perf_evsel *evsel,
Namhyung Kim9783adf2012-11-02 14:50:05 +0900218 struct hist_browser_timer *hbt);
Namhyung Kim1254b51e2012-09-28 18:32:02 +0900219#else
Irina Tirdea1d037ca2012-09-11 01:15:03 +0300220static inline int symbol__tui_annotate(struct symbol *sym __maybe_unused,
Namhyung Kimdb8fd072013-03-05 14:53:21 +0900221 struct map *map __maybe_unused,
222 struct perf_evsel *evsel __maybe_unused,
223 struct hist_browser_timer *hbt
224 __maybe_unused)
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -0200225{
226 return 0;
227}
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -0200228#endif
229
Andi Kleenf69b64f2011-09-15 14:31:41 -0700230extern const char *disassembler_style;
231
Arnaldo Carvalho de Melo78f7def2011-02-04 09:45:46 -0200232#endif /* __PERF_ANNOTATE_H */