annotate.h 4.8 KB
Newer Older
1 2 3 4
#ifndef __PERF_ANNOTATE_H
#define __PERF_ANNOTATE_H

#include <stdbool.h>
5
#include <stdint.h>
B
Borislav Petkov 已提交
6
#include <linux/types.h>
7
#include "symbol.h"
8
#include "hist.h"
9
#include "sort.h"
10 11
#include <linux/list.h>
#include <linux/rbtree.h>
12
#include <pthread.h>
13

14 15
struct ins;

16 17
struct ins_operands {
	char	*raw;
18
	struct {
19
		char	*raw;
20 21
		char	*name;
		u64	addr;
22
		u64	offset;
23
	} target;
24 25 26 27 28 29 30 31 32 33 34
	union {
		struct {
			char	*raw;
			char	*name;
			u64	addr;
		} source;
		struct {
			struct ins *ins;
			struct ins_operands *ops;
		} locked;
	};
35 36
};

37
struct ins_ops {
38
	void (*free)(struct ins_operands *ops);
39
	int (*parse)(struct ins_operands *ops);
40
	int (*scnprintf)(struct ins *ins, char *bf, size_t size,
41
			 struct ins_operands *ops);
42 43 44 45 46 47 48 49
};

struct ins {
	const char     *name;
	struct ins_ops *ops;
};

bool ins__is_jump(const struct ins *ins);
50
bool ins__is_call(const struct ins *ins);
51
int ins__scnprintf(struct ins *ins, char *bf, size_t size, struct ins_operands *ops);
52

53 54
struct annotation;

55
struct disasm_line {
56 57 58 59 60
	struct list_head    node;
	s64		    offset;
	char		    *line;
	char		    *name;
	struct ins	    *ins;
61
	int		    line_nr;
62 63
	float		    ipc;
	u64		    cycles;
64
	struct ins_operands ops;
65 66
};

67 68 69 70 71
static inline bool disasm_line__has_offset(const struct disasm_line *dl)
{
	return dl->ops.target.offset != UINT64_MAX;
}

72 73
void disasm_line__free(struct disasm_line *dl);
struct disasm_line *disasm__get_next_ip_line(struct list_head *head, struct disasm_line *pos);
74
int disasm_line__scnprintf(struct disasm_line *dl, char *bf, size_t size, bool raw);
75
size_t disasm__fprintf(struct list_head *head, FILE *fp);
76
double disasm__calc_percent(struct annotation *notes, int evidx, s64 offset,
77
			    s64 end, const char **path, u64 *nr_samples);
78 79 80 81 82 83

struct sym_hist {
	u64		sum;
	u64		addr[0];
};

84 85 86 87 88 89 90 91 92 93 94
struct cyc_hist {
	u64	start;
	u64	cycles;
	u64	cycles_aggr;
	u32	num;
	u32	num_aggr;
	u8	have_start;
	/* 1 byte padding */
	u16	reset;
};

95
struct source_line_samples {
96
	double		percent;
97
	double		percent_sum;
98
	double          nr;
99 100 101 102
};

struct source_line {
	struct rb_node	node;
103
	char		*path;
104
	int		nr_pcnt;
105
	struct source_line_samples samples[1];
106 107
};

108
/** struct annotated_source - symbols with hits have this attached as in sannotation
109 110
 *
 * @histogram: Array of addr hit histograms per event being monitored
111
 * @lines: If 'print_lines' is specified, per source code line percentages
112
 * @source: source parsed from a disassembler like objdump -dS
113
 * @cyc_hist: Average cycles per basic block
114
 *
115
 * lines is allocated, percentages calculated and all sorted by percentage
116 117 118 119 120
 * when the annotation is about to be presented, so the percentages are for
 * one of the entries in the histogram array, i.e. for the event/counter being
 * presented. It is deallocated right after symbol__{tui,tty,etc}_annotate
 * returns.
 */
121 122 123
struct annotated_source {
	struct list_head   source;
	struct source_line *lines;
124
	int    		   nr_histograms;
125
	size_t		   sizeof_sym_hist;
126
	struct cyc_hist	   *cycles_hist;
127 128 129 130 131 132
	struct sym_hist	   histograms[0];
};

struct annotation {
	pthread_mutex_t		lock;
	struct annotated_source *src;
133 134
};

135 136
static inline struct sym_hist *annotation__histogram(struct annotation *notes, int idx)
{
137 138
	return (((void *)&notes->src->histograms) +
	 	(notes->src->sizeof_sym_hist * idx));
139 140
}

141 142
static inline struct annotation *symbol__annotation(struct symbol *sym)
{
143
	return (void *)sym - symbol_conf.priv_size;
144 145
}

146 147
int addr_map_symbol__inc_samples(struct addr_map_symbol *ams, int evidx);

148 149 150 151
int addr_map_symbol__account_cycles(struct addr_map_symbol *ams,
				    struct addr_map_symbol *start,
				    unsigned cycles);

152 153
int hist_entry__inc_addr_samples(struct hist_entry *he, int evidx, u64 addr);

154
int symbol__alloc_hist(struct symbol *sym);
155
void symbol__annotate_zero_histograms(struct symbol *sym);
156

157
int symbol__annotate(struct symbol *sym, struct map *map, size_t privsize);
158

159
int symbol__annotate_init(struct map *map, struct symbol *sym);
160 161 162
int symbol__annotate_printf(struct symbol *sym, struct map *map,
			    struct perf_evsel *evsel, bool full_paths,
			    int min_pcnt, int max_lines, int context);
163
void symbol__annotate_zero_histogram(struct symbol *sym, int evidx);
164
void symbol__annotate_decay_histogram(struct symbol *sym, int evidx);
165
void disasm__purge(struct list_head *head);
166

167 168
bool ui__has_annotation(void);

169 170 171
int symbol__tty_annotate(struct symbol *sym, struct map *map,
			 struct perf_evsel *evsel, bool print_lines,
			 bool full_paths, int min_pcnt, int max_lines);
172

173
#ifdef HAVE_SLANG_SUPPORT
174 175
int symbol__tui_annotate(struct symbol *sym, struct map *map,
			 struct perf_evsel *evsel,
176
			 struct hist_browser_timer *hbt);
N
Namhyung Kim 已提交
177
#else
178
static inline int symbol__tui_annotate(struct symbol *sym __maybe_unused,
179 180 181 182
				struct map *map __maybe_unused,
				struct perf_evsel *evsel  __maybe_unused,
				struct hist_browser_timer *hbt
				__maybe_unused)
183 184 185 186 187
{
	return 0;
}
#endif

188 189
extern const char	*disassembler_style;

190
#endif	/* __PERF_ANNOTATE_H */