Greg Kroah-Hartman | b244131 | 2017-11-01 15:07:57 +0100 | [diff] [blame] | 1 | /* SPDX-License-Identifier: GPL-2.0 */ |
Arnaldo Carvalho de Melo | 78f7def | 2011-02-04 09:45:46 -0200 | [diff] [blame] | 2 | #ifndef __PERF_ANNOTATE_H |
| 3 | #define __PERF_ANNOTATE_H |
| 4 | |
| 5 | #include <stdbool.h> |
Arnaldo Carvalho de Melo | fb29fa5 | 2012-04-25 14:16:03 -0300 | [diff] [blame] | 6 | #include <stdint.h> |
Borislav Petkov | d944c4e | 2014-04-25 21:31:02 +0200 | [diff] [blame] | 7 | #include <linux/types.h> |
Arnaldo Carvalho de Melo | 78f7def | 2011-02-04 09:45:46 -0200 | [diff] [blame] | 8 | #include "symbol.h" |
Namhyung Kim | 9783adf | 2012-11-02 14:50:05 +0900 | [diff] [blame] | 9 | #include "hist.h" |
Namhyung Kim | 2b676bf | 2013-02-07 18:02:08 +0900 | [diff] [blame] | 10 | #include "sort.h" |
Arnaldo Carvalho de Melo | 78f7def | 2011-02-04 09:45:46 -0200 | [diff] [blame] | 11 | #include <linux/list.h> |
| 12 | #include <linux/rbtree.h> |
Irina Tirdea | 27683dc | 2012-09-08 03:43:19 +0300 | [diff] [blame] | 13 | #include <pthread.h> |
Arnaldo Carvalho de Melo | 78f7def | 2011-02-04 09:45:46 -0200 | [diff] [blame] | 14 | |
Arnaldo Carvalho de Melo | 75b4920 | 2016-11-24 11:16:06 -0300 | [diff] [blame] | 15 | struct ins_ops; |
| 16 | |
| 17 | struct ins { |
| 18 | const char *name; |
| 19 | struct ins_ops *ops; |
| 20 | }; |
Arnaldo Carvalho de Melo | 28548d7 | 2012-04-19 10:16:27 -0300 | [diff] [blame] | 21 | |
Arnaldo Carvalho de Melo | c7e6ead | 2012-04-20 14:38:46 -0300 | [diff] [blame] | 22 | struct ins_operands { |
| 23 | char *raw; |
Arnaldo Carvalho de Melo | 44d1a3e | 2012-04-25 08:00:23 -0300 | [diff] [blame] | 24 | struct { |
Arnaldo Carvalho de Melo | 6de783b | 2012-05-11 16:48:49 -0300 | [diff] [blame] | 25 | char *raw; |
Arnaldo Carvalho de Melo | 44d1a3e | 2012-04-25 08:00:23 -0300 | [diff] [blame] | 26 | char *name; |
Arnaldo Carvalho de Melo | 44d1a3e | 2012-04-25 08:00:23 -0300 | [diff] [blame] | 27 | u64 addr; |
Ravi Bangoria | e216874 | 2016-12-05 21:26:47 +0530 | [diff] [blame] | 28 | s64 offset; |
| 29 | bool offset_avail; |
Arnaldo Carvalho de Melo | 44d1a3e | 2012-04-25 08:00:23 -0300 | [diff] [blame] | 30 | } target; |
Arnaldo Carvalho de Melo | 7a997fe | 2012-05-12 13:15:34 -0300 | [diff] [blame] | 31 | union { |
| 32 | struct { |
| 33 | char *raw; |
| 34 | char *name; |
| 35 | u64 addr; |
| 36 | } source; |
| 37 | struct { |
Arnaldo Carvalho de Melo | 75b4920 | 2016-11-24 11:16:06 -0300 | [diff] [blame] | 38 | struct ins ins; |
Arnaldo Carvalho de Melo | 7a997fe | 2012-05-12 13:15:34 -0300 | [diff] [blame] | 39 | struct ins_operands *ops; |
| 40 | } locked; |
| 41 | }; |
Arnaldo Carvalho de Melo | c7e6ead | 2012-04-20 14:38:46 -0300 | [diff] [blame] | 42 | }; |
| 43 | |
Arnaldo Carvalho de Melo | 786c1b5 | 2016-11-16 15:39:50 -0300 | [diff] [blame] | 44 | struct arch; |
| 45 | |
Arnaldo Carvalho de Melo | 4f9d032 | 2012-04-18 13:58:34 -0300 | [diff] [blame] | 46 | struct ins_ops { |
Arnaldo Carvalho de Melo | c46219ac | 2012-05-12 13:26:20 -0300 | [diff] [blame] | 47 | void (*free)(struct ins_operands *ops); |
Arnaldo Carvalho de Melo | 786c1b5 | 2016-11-16 15:39:50 -0300 | [diff] [blame] | 48 | int (*parse)(struct arch *arch, struct ins_operands *ops, struct map *map); |
Arnaldo Carvalho de Melo | 28548d7 | 2012-04-19 10:16:27 -0300 | [diff] [blame] | 49 | int (*scnprintf)(struct ins *ins, char *bf, size_t size, |
Arnaldo Carvalho de Melo | 5417072 | 2012-05-07 18:54:16 -0300 | [diff] [blame] | 50 | struct ins_operands *ops); |
Arnaldo Carvalho de Melo | 4f9d032 | 2012-04-18 13:58:34 -0300 | [diff] [blame] | 51 | }; |
| 52 | |
Arnaldo Carvalho de Melo | 4f9d032 | 2012-04-18 13:58:34 -0300 | [diff] [blame] | 53 | bool ins__is_jump(const struct ins *ins); |
Arnaldo Carvalho de Melo | d86b059 | 2012-04-18 16:07:38 -0300 | [diff] [blame] | 54 | bool ins__is_call(const struct ins *ins); |
Naveen N. Rao | 6ef9492 | 2016-06-24 17:23:58 +0530 | [diff] [blame] | 55 | bool ins__is_ret(const struct ins *ins); |
Jin Yao | 7e63a13 | 2017-07-07 13:06:35 +0800 | [diff] [blame] | 56 | bool ins__is_lock(const struct ins *ins); |
Arnaldo Carvalho de Melo | 5417072 | 2012-05-07 18:54:16 -0300 | [diff] [blame] | 57 | int ins__scnprintf(struct ins *ins, char *bf, size_t size, struct ins_operands *ops); |
Jin Yao | 69fb09f | 2017-07-07 13:06:34 +0800 | [diff] [blame] | 58 | bool ins__is_fused(struct arch *arch, const char *ins1, const char *ins2); |
Arnaldo Carvalho de Melo | 4f9d032 | 2012-04-18 13:58:34 -0300 | [diff] [blame] | 59 | |
Namhyung Kim | e64aa75 | 2013-03-05 14:53:30 +0900 | [diff] [blame] | 60 | struct annotation; |
| 61 | |
Jiri Olsa | 7e30455 | 2017-10-11 17:01:39 +0200 | [diff] [blame] | 62 | struct sym_hist_entry { |
| 63 | u64 nr_samples; |
| 64 | u64 period; |
| 65 | }; |
| 66 | |
| 67 | struct annotation_data { |
| 68 | double percent; |
Jiri Olsa | 8b4c74d | 2017-10-11 17:01:41 +0200 | [diff] [blame] | 69 | double percent_sum; |
Jiri Olsa | 7e30455 | 2017-10-11 17:01:39 +0200 | [diff] [blame] | 70 | struct sym_hist_entry he; |
| 71 | }; |
| 72 | |
Jiri Olsa | a17c4ca | 2017-10-11 17:01:25 +0200 | [diff] [blame] | 73 | struct annotation_line { |
| 74 | struct list_head node; |
Jiri Olsa | 5b12adc | 2017-10-11 17:01:36 +0200 | [diff] [blame] | 75 | struct rb_node rb_node; |
Jiri Olsa | d5490b9 | 2017-10-11 17:01:26 +0200 | [diff] [blame] | 76 | s64 offset; |
| 77 | char *line; |
| 78 | int line_nr; |
Jiri Olsa | 37236d5 | 2017-10-11 17:01:27 +0200 | [diff] [blame] | 79 | float ipc; |
| 80 | u64 cycles; |
Jiri Olsa | c835e19 | 2017-10-11 17:01:37 +0200 | [diff] [blame] | 81 | size_t privsize; |
Jiri Olsa | 8b4c74d | 2017-10-11 17:01:41 +0200 | [diff] [blame] | 82 | char *path; |
Jiri Olsa | 7e30455 | 2017-10-11 17:01:39 +0200 | [diff] [blame] | 83 | int samples_nr; |
| 84 | struct annotation_data samples[0]; |
Jiri Olsa | a17c4ca | 2017-10-11 17:01:25 +0200 | [diff] [blame] | 85 | }; |
| 86 | |
Arnaldo Carvalho de Melo | 29ed6e7 | 2012-04-15 15:24:39 -0300 | [diff] [blame] | 87 | struct disasm_line { |
Jiri Olsa | a17c4ca | 2017-10-11 17:01:25 +0200 | [diff] [blame] | 88 | struct ins ins; |
Jiri Olsa | a17c4ca | 2017-10-11 17:01:25 +0200 | [diff] [blame] | 89 | struct ins_operands ops; |
Jiri Olsa | c835e19 | 2017-10-11 17:01:37 +0200 | [diff] [blame] | 90 | |
| 91 | /* This needs to be at the end. */ |
| 92 | struct annotation_line al; |
Arnaldo Carvalho de Melo | 78f7def | 2011-02-04 09:45:46 -0200 | [diff] [blame] | 93 | }; |
| 94 | |
Jiri Olsa | c835e19 | 2017-10-11 17:01:37 +0200 | [diff] [blame] | 95 | static inline struct disasm_line *disasm_line(struct annotation_line *al) |
| 96 | { |
| 97 | return al ? container_of(al, struct disasm_line, al) : NULL; |
| 98 | } |
| 99 | |
Arnaldo Carvalho de Melo | fb29fa5 | 2012-04-25 14:16:03 -0300 | [diff] [blame] | 100 | static inline bool disasm_line__has_offset(const struct disasm_line *dl) |
| 101 | { |
Ravi Bangoria | e216874 | 2016-12-05 21:26:47 +0530 | [diff] [blame] | 102 | return dl->ops.target.offset_avail; |
Arnaldo Carvalho de Melo | fb29fa5 | 2012-04-25 14:16:03 -0300 | [diff] [blame] | 103 | } |
| 104 | |
Arnaldo Carvalho de Melo | 29ed6e7 | 2012-04-15 15:24:39 -0300 | [diff] [blame] | 105 | void disasm_line__free(struct disasm_line *dl); |
Jiri Olsa | c4c7243 | 2017-10-11 17:01:34 +0200 | [diff] [blame] | 106 | struct annotation_line * |
| 107 | annotation_line__next(struct annotation_line *pos, struct list_head *head); |
Arnaldo Carvalho de Melo | 5417072 | 2012-05-07 18:54:16 -0300 | [diff] [blame] | 108 | int disasm_line__scnprintf(struct disasm_line *dl, char *bf, size_t size, bool raw); |
Arnaldo Carvalho de Melo | 5145418 | 2012-04-15 15:52:18 -0300 | [diff] [blame] | 109 | size_t disasm__fprintf(struct list_head *head, FILE *fp); |
Jiri Olsa | 9e4e0a9 | 2017-11-15 12:05:59 +0100 | [diff] [blame] | 110 | void symbol__calc_percent(struct symbol *sym, struct perf_evsel *evsel); |
Arnaldo Carvalho de Melo | 78f7def | 2011-02-04 09:45:46 -0200 | [diff] [blame] | 111 | |
| 112 | struct sym_hist { |
Taeung Song | 8158683d | 2017-07-20 06:36:51 +0900 | [diff] [blame] | 113 | u64 nr_samples; |
Taeung Song | 461c17f | 2017-07-20 17:18:05 -0300 | [diff] [blame] | 114 | u64 period; |
Taeung Song | 896bccd | 2017-07-20 06:36:45 +0900 | [diff] [blame] | 115 | struct sym_hist_entry addr[0]; |
Arnaldo Carvalho de Melo | 78f7def | 2011-02-04 09:45:46 -0200 | [diff] [blame] | 116 | }; |
| 117 | |
Andi Kleen | d495763 | 2015-07-18 08:24:48 -0700 | [diff] [blame] | 118 | struct cyc_hist { |
| 119 | u64 start; |
| 120 | u64 cycles; |
| 121 | u64 cycles_aggr; |
| 122 | u32 num; |
| 123 | u32 num_aggr; |
| 124 | u8 have_start; |
| 125 | /* 1 byte padding */ |
| 126 | u16 reset; |
| 127 | }; |
| 128 | |
Arnaldo Carvalho de Melo | ce6f4fa | 2011-02-08 13:27:39 -0200 | [diff] [blame] | 129 | /** struct annotated_source - symbols with hits have this attached as in sannotation |
Arnaldo Carvalho de Melo | 2f525d0 | 2011-02-04 13:43:24 -0200 | [diff] [blame] | 130 | * |
| 131 | * @histogram: Array of addr hit histograms per event being monitored |
Arnaldo Carvalho de Melo | ce6f4fa | 2011-02-08 13:27:39 -0200 | [diff] [blame] | 132 | * @lines: If 'print_lines' is specified, per source code line percentages |
Arnaldo Carvalho de Melo | 29ed6e7 | 2012-04-15 15:24:39 -0300 | [diff] [blame] | 133 | * @source: source parsed from a disassembler like objdump -dS |
Andi Kleen | d495763 | 2015-07-18 08:24:48 -0700 | [diff] [blame] | 134 | * @cyc_hist: Average cycles per basic block |
Arnaldo Carvalho de Melo | 2f525d0 | 2011-02-04 13:43:24 -0200 | [diff] [blame] | 135 | * |
Arnaldo Carvalho de Melo | ce6f4fa | 2011-02-08 13:27:39 -0200 | [diff] [blame] | 136 | * lines is allocated, percentages calculated and all sorted by percentage |
Arnaldo Carvalho de Melo | 2f525d0 | 2011-02-04 13:43:24 -0200 | [diff] [blame] | 137 | * when the annotation is about to be presented, so the percentages are for |
| 138 | * one of the entries in the histogram array, i.e. for the event/counter being |
| 139 | * presented. It is deallocated right after symbol__{tui,tty,etc}_annotate |
| 140 | * returns. |
| 141 | */ |
Arnaldo Carvalho de Melo | ce6f4fa | 2011-02-08 13:27:39 -0200 | [diff] [blame] | 142 | struct annotated_source { |
| 143 | struct list_head source; |
Arnaldo Carvalho de Melo | 3653246 | 2011-02-06 14:54:44 -0200 | [diff] [blame] | 144 | int nr_histograms; |
Jiri Olsa | 5ec4502 | 2015-10-05 20:06:03 +0200 | [diff] [blame] | 145 | size_t sizeof_sym_hist; |
Andi Kleen | d495763 | 2015-07-18 08:24:48 -0700 | [diff] [blame] | 146 | struct cyc_hist *cycles_hist; |
Arnaldo Carvalho de Melo | ce6f4fa | 2011-02-08 13:27:39 -0200 | [diff] [blame] | 147 | struct sym_hist histograms[0]; |
| 148 | }; |
| 149 | |
| 150 | struct annotation { |
| 151 | pthread_mutex_t lock; |
Peter Zijlstra | 70fbe05 | 2016-09-05 16:08:12 -0300 | [diff] [blame] | 152 | u64 max_coverage; |
Arnaldo Carvalho de Melo | ce6f4fa | 2011-02-08 13:27:39 -0200 | [diff] [blame] | 153 | struct annotated_source *src; |
Arnaldo Carvalho de Melo | 78f7def | 2011-02-04 09:45:46 -0200 | [diff] [blame] | 154 | }; |
| 155 | |
Arnaldo Carvalho de Melo | 2f525d0 | 2011-02-04 13:43:24 -0200 | [diff] [blame] | 156 | static inline struct sym_hist *annotation__histogram(struct annotation *notes, int idx) |
| 157 | { |
Arnaldo Carvalho de Melo | ce6f4fa | 2011-02-08 13:27:39 -0200 | [diff] [blame] | 158 | return (((void *)¬es->src->histograms) + |
| 159 | (notes->src->sizeof_sym_hist * idx)); |
Arnaldo Carvalho de Melo | 2f525d0 | 2011-02-04 13:43:24 -0200 | [diff] [blame] | 160 | } |
| 161 | |
Arnaldo Carvalho de Melo | 78f7def | 2011-02-04 09:45:46 -0200 | [diff] [blame] | 162 | static inline struct annotation *symbol__annotation(struct symbol *sym) |
| 163 | { |
Namhyung Kim | 813ccd1 | 2015-01-14 20:18:05 +0900 | [diff] [blame] | 164 | return (void *)sym - symbol_conf.priv_size; |
Arnaldo Carvalho de Melo | 78f7def | 2011-02-04 09:45:46 -0200 | [diff] [blame] | 165 | } |
| 166 | |
Taeung Song | bab89f6 | 2017-07-20 16:28:53 -0300 | [diff] [blame] | 167 | int addr_map_symbol__inc_samples(struct addr_map_symbol *ams, struct perf_sample *sample, |
| 168 | int evidx); |
Arnaldo Carvalho de Melo | 0f4e7a2 | 2013-12-18 16:48:29 -0300 | [diff] [blame] | 169 | |
Andi Kleen | d495763 | 2015-07-18 08:24:48 -0700 | [diff] [blame] | 170 | int addr_map_symbol__account_cycles(struct addr_map_symbol *ams, |
| 171 | struct addr_map_symbol *start, |
| 172 | unsigned cycles); |
| 173 | |
Taeung Song | bab89f6 | 2017-07-20 16:28:53 -0300 | [diff] [blame] | 174 | int hist_entry__inc_addr_samples(struct hist_entry *he, struct perf_sample *sample, |
| 175 | int evidx, u64 addr); |
Arnaldo Carvalho de Melo | f626adf | 2013-12-18 17:10:15 -0300 | [diff] [blame] | 176 | |
Arnaldo Carvalho de Melo | d04b35f | 2011-11-11 22:17:32 -0200 | [diff] [blame] | 177 | int symbol__alloc_hist(struct symbol *sym); |
Arnaldo Carvalho de Melo | 3653246 | 2011-02-06 14:54:44 -0200 | [diff] [blame] | 178 | void symbol__annotate_zero_histograms(struct symbol *sym); |
Arnaldo Carvalho de Melo | 78f7def | 2011-02-04 09:45:46 -0200 | [diff] [blame] | 179 | |
Jiri Olsa | c34df25 | 2017-10-11 17:01:28 +0200 | [diff] [blame] | 180 | int symbol__annotate(struct symbol *sym, struct map *map, |
Jiri Olsa | d03a686 | 2017-10-11 17:01:33 +0200 | [diff] [blame] | 181 | struct perf_evsel *evsel, size_t privsize, |
Arnaldo Carvalho de Melo | 5449f13 | 2017-12-11 12:46:11 -0300 | [diff] [blame] | 182 | struct arch **parch); |
Arnaldo Carvalho de Melo | f626adf | 2013-12-18 17:10:15 -0300 | [diff] [blame] | 183 | |
Arnaldo Carvalho de Melo | ee51d85 | 2016-07-29 16:27:18 -0300 | [diff] [blame] | 184 | enum symbol_disassemble_errno { |
| 185 | SYMBOL_ANNOTATE_ERRNO__SUCCESS = 0, |
| 186 | |
| 187 | /* |
| 188 | * Choose an arbitrary negative big number not to clash with standard |
| 189 | * errno since SUS requires the errno has distinct positive values. |
| 190 | * See 'Issue 6' in the link below. |
| 191 | * |
| 192 | * http://pubs.opengroup.org/onlinepubs/9699919799/basedefs/errno.h.html |
| 193 | */ |
| 194 | __SYMBOL_ANNOTATE_ERRNO__START = -10000, |
| 195 | |
| 196 | SYMBOL_ANNOTATE_ERRNO__NO_VMLINUX = __SYMBOL_ANNOTATE_ERRNO__START, |
| 197 | |
| 198 | __SYMBOL_ANNOTATE_ERRNO__END, |
| 199 | }; |
| 200 | |
| 201 | int symbol__strerror_disassemble(struct symbol *sym, struct map *map, |
| 202 | int errnum, char *buf, size_t buflen); |
| 203 | |
Namhyung Kim | db8fd07 | 2013-03-05 14:53:21 +0900 | [diff] [blame] | 204 | int symbol__annotate_printf(struct symbol *sym, struct map *map, |
| 205 | struct perf_evsel *evsel, bool full_paths, |
| 206 | int min_pcnt, int max_lines, int context); |
Arnaldo Carvalho de Melo | 3653246 | 2011-02-06 14:54:44 -0200 | [diff] [blame] | 207 | void symbol__annotate_zero_histogram(struct symbol *sym, int evidx); |
Arnaldo Carvalho de Melo | ce6f4fa | 2011-02-08 13:27:39 -0200 | [diff] [blame] | 208 | void symbol__annotate_decay_histogram(struct symbol *sym, int evidx); |
Jiri Olsa | f8eb37b | 2017-10-11 17:01:38 +0200 | [diff] [blame] | 209 | void annotated_source__purge(struct annotated_source *as); |
Arnaldo Carvalho de Melo | 78f7def | 2011-02-04 09:45:46 -0200 | [diff] [blame] | 210 | |
Namhyung Kim | 48c65bd | 2014-02-20 10:32:53 +0900 | [diff] [blame] | 211 | bool ui__has_annotation(void); |
| 212 | |
Namhyung Kim | db8fd07 | 2013-03-05 14:53:21 +0900 | [diff] [blame] | 213 | int symbol__tty_annotate(struct symbol *sym, struct map *map, |
| 214 | struct perf_evsel *evsel, bool print_lines, |
| 215 | bool full_paths, int min_pcnt, int max_lines); |
Arnaldo Carvalho de Melo | 78f7def | 2011-02-04 09:45:46 -0200 | [diff] [blame] | 216 | |
Ingo Molnar | 89fe808 | 2013-09-30 12:07:11 +0200 | [diff] [blame] | 217 | #ifdef HAVE_SLANG_SUPPORT |
Namhyung Kim | db8fd07 | 2013-03-05 14:53:21 +0900 | [diff] [blame] | 218 | int symbol__tui_annotate(struct symbol *sym, struct map *map, |
| 219 | struct perf_evsel *evsel, |
Namhyung Kim | 9783adf | 2012-11-02 14:50:05 +0900 | [diff] [blame] | 220 | struct hist_browser_timer *hbt); |
Namhyung Kim | 1254b51e | 2012-09-28 18:32:02 +0900 | [diff] [blame] | 221 | #else |
Irina Tirdea | 1d037ca | 2012-09-11 01:15:03 +0300 | [diff] [blame] | 222 | static inline int symbol__tui_annotate(struct symbol *sym __maybe_unused, |
Namhyung Kim | db8fd07 | 2013-03-05 14:53:21 +0900 | [diff] [blame] | 223 | struct map *map __maybe_unused, |
| 224 | struct perf_evsel *evsel __maybe_unused, |
| 225 | struct hist_browser_timer *hbt |
| 226 | __maybe_unused) |
Arnaldo Carvalho de Melo | 78f7def | 2011-02-04 09:45:46 -0200 | [diff] [blame] | 227 | { |
| 228 | return 0; |
| 229 | } |
Arnaldo Carvalho de Melo | 78f7def | 2011-02-04 09:45:46 -0200 | [diff] [blame] | 230 | #endif |
| 231 | |
Andi Kleen | f69b64f | 2011-09-15 14:31:41 -0700 | [diff] [blame] | 232 | extern const char *disassembler_style; |
| 233 | |
Arnaldo Carvalho de Melo | 78f7def | 2011-02-04 09:45:46 -0200 | [diff] [blame] | 234 | #endif /* __PERF_ANNOTATE_H */ |