| /* SPDX-License-Identifier: GPL-2.0 */ |
| #ifndef _PERF_ANNOTATE_DATA_H |
| #define _PERF_ANNOTATE_DATA_H |
| |
| #include <errno.h> |
| #include <linux/compiler.h> |
| #include <linux/rbtree.h> |
| #include <linux/types.h> |
| #include "dwarf-regs.h" |
| #include "annotate.h" |
| |
| #ifdef HAVE_DWARF_SUPPORT |
| #include "debuginfo.h" |
| #endif |
| |
| struct annotated_op_loc; |
| struct debuginfo; |
| struct evsel; |
| struct hist_browser_timer; |
| struct hist_entry; |
| struct map_symbol; |
| struct thread; |
| |
| #define pr_debug_dtp(fmt, ...) \ |
| do { \ |
| if (debug_type_profile) \ |
| pr_info(fmt, ##__VA_ARGS__); \ |
| else \ |
| pr_debug3(fmt, ##__VA_ARGS__); \ |
| } while (0) |
| |
| enum type_state_kind { |
| TSR_KIND_INVALID = 0, |
| TSR_KIND_TYPE, |
| TSR_KIND_PERCPU_BASE, |
| TSR_KIND_CONST, |
| TSR_KIND_POINTER, |
| TSR_KIND_CANARY, |
| }; |
| |
| /** |
| * struct annotated_member - Type of member field |
| * @node: List entry in the parent list |
| * @children: List head for child nodes |
| * @type_name: Name of the member type |
| * @var_name: Name of the member variable |
| * @offset: Offset from the outer data type |
| * @size: Size of the member field |
| * |
| * This represents a member type in a data type. |
| */ |
| struct annotated_member { |
| struct list_head node; |
| struct list_head children; |
| char *type_name; |
| char *var_name; |
| int offset; |
| int size; |
| }; |
| |
| /** |
| * struct type_hist_entry - Histogram entry per offset |
| * @nr_samples: Number of samples |
| * @period: Count of event |
| */ |
| struct type_hist_entry { |
| int nr_samples; |
| u64 period; |
| }; |
| |
| /** |
| * struct type_hist - Type histogram for each event |
| * @nr_samples: Total number of samples in this data type |
| * @period: Total count of the event in this data type |
| * @offset: Array of histogram entry |
| */ |
| struct type_hist { |
| u64 nr_samples; |
| u64 period; |
| struct type_hist_entry addr[]; |
| }; |
| |
| /** |
| * struct annotated_data_type - Data type to profile |
| * @node: RB-tree node for dso->type_tree |
| * @self: Actual type information |
| * @nr_histogram: Number of histogram entries |
| * @histograms: An array of pointers to histograms |
| * |
| * This represents a data type accessed by samples in the profile data. |
| */ |
| struct annotated_data_type { |
| struct rb_node node; |
| struct annotated_member self; |
| int nr_histograms; |
| struct type_hist **histograms; |
| }; |
| |
| extern struct annotated_data_type unknown_type; |
| extern struct annotated_data_type stackop_type; |
| extern struct annotated_data_type canary_type; |
| |
| /** |
| * struct data_loc_info - Data location information |
| * @arch: CPU architecture info |
| * @thread: Thread info |
| * @ms: Map and Symbol info |
| * @ip: Instruction address |
| * @var_addr: Data address (for global variables) |
| * @cpumode: CPU execution mode |
| * @op: Instruction operand location (regs and offset) |
| * @di: Debug info |
| * @fbreg: Frame base register |
| * @fb_cfa: Whether the frame needs to check CFA |
| * @type_offset: Final offset in the type |
| */ |
| struct data_loc_info { |
| /* These are input field, should be filled by caller */ |
| struct arch *arch; |
| struct thread *thread; |
| struct map_symbol *ms; |
| u64 ip; |
| u64 var_addr; |
| u8 cpumode; |
| struct annotated_op_loc *op; |
| struct debuginfo *di; |
| |
| /* These are used internally */ |
| int fbreg; |
| bool fb_cfa; |
| |
| /* This is for the result */ |
| int type_offset; |
| }; |
| |
| /** |
| * struct annotated_data_stat - Debug statistics |
| * @total: Total number of entry |
| * @no_sym: No symbol or map found |
| * @no_insn: Failed to get disasm line |
| * @no_insn_ops: The instruction has no operands |
| * @no_mem_ops: The instruction has no memory operands |
| * @no_reg: Failed to extract a register from the operand |
| * @no_dbginfo: The binary has no debug information |
| * @no_cuinfo: Failed to find a compile_unit |
| * @no_var: Failed to find a matching variable |
| * @no_typeinfo: Failed to get a type info for the variable |
| * @invalid_size: Failed to get a size info of the type |
| * @bad_offset: The access offset is out of the type |
| */ |
| struct annotated_data_stat { |
| int total; |
| int no_sym; |
| int no_insn; |
| int no_insn_ops; |
| int no_mem_ops; |
| int no_reg; |
| int no_dbginfo; |
| int no_cuinfo; |
| int no_var; |
| int no_typeinfo; |
| int invalid_size; |
| int bad_offset; |
| int insn_track; |
| }; |
| extern struct annotated_data_stat ann_data_stat; |
| |
| #ifdef HAVE_DWARF_SUPPORT |
| /* |
| * Type information in a register, valid when @ok is true. |
| * The @caller_saved registers are invalidated after a function call. |
| */ |
| struct type_state_reg { |
| Dwarf_Die type; |
| u32 imm_value; |
| bool ok; |
| bool caller_saved; |
| u8 kind; |
| u8 copied_from; |
| }; |
| |
| /* Type information in a stack location, dynamically allocated */ |
| struct type_state_stack { |
| struct list_head list; |
| Dwarf_Die type; |
| int offset; |
| int size; |
| bool compound; |
| u8 kind; |
| }; |
| |
| /* FIXME: This should be arch-dependent */ |
| #ifdef __powerpc__ |
| #define TYPE_STATE_MAX_REGS 32 |
| #else |
| #define TYPE_STATE_MAX_REGS 16 |
| #endif |
| |
| /* |
| * State table to maintain type info in each register and stack location. |
| * It'll be updated when new variable is allocated or type info is moved |
| * to a new location (register or stack). As it'd be used with the |
| * shortest path of basic blocks, it only maintains a single table. |
| */ |
| struct type_state { |
| /* state of general purpose registers */ |
| struct type_state_reg regs[TYPE_STATE_MAX_REGS]; |
| /* state of stack location */ |
| struct list_head stack_vars; |
| /* return value register */ |
| int ret_reg; |
| /* stack pointer register */ |
| int stack_reg; |
| }; |
| |
| /* Returns data type at the location (ip, reg, offset) */ |
| struct annotated_data_type *find_data_type(struct data_loc_info *dloc); |
| |
| /* Update type access histogram at the given offset */ |
| int annotated_data_type__update_samples(struct annotated_data_type *adt, |
| struct evsel *evsel, int offset, |
| int nr_samples, u64 period); |
| |
| /* Release all data type information in the tree */ |
| void annotated_data_type__tree_delete(struct rb_root *root); |
| |
| /* Release all global variable information in the tree */ |
| void global_var_type__tree_delete(struct rb_root *root); |
| |
| int hist_entry__annotate_data_tty(struct hist_entry *he, struct evsel *evsel); |
| |
| bool has_reg_type(struct type_state *state, int reg); |
| struct type_state_stack *findnew_stack_state(struct type_state *state, |
| int offset, u8 kind, |
| Dwarf_Die *type_die); |
| void set_stack_state(struct type_state_stack *stack, int offset, u8 kind, |
| Dwarf_Die *type_die); |
| struct type_state_stack *find_stack_state(struct type_state *state, |
| int offset); |
| bool get_global_var_type(Dwarf_Die *cu_die, struct data_loc_info *dloc, |
| u64 ip, u64 var_addr, int *var_offset, |
| Dwarf_Die *type_die); |
| bool get_global_var_info(struct data_loc_info *dloc, u64 addr, |
| const char **var_name, int *var_offset); |
| void pr_debug_type_name(Dwarf_Die *die, enum type_state_kind kind); |
| |
| #else /* HAVE_DWARF_SUPPORT */ |
| |
| static inline struct annotated_data_type * |
| find_data_type(struct data_loc_info *dloc __maybe_unused) |
| { |
| return NULL; |
| } |
| |
| static inline int |
| annotated_data_type__update_samples(struct annotated_data_type *adt __maybe_unused, |
| struct evsel *evsel __maybe_unused, |
| int offset __maybe_unused, |
| int nr_samples __maybe_unused, |
| u64 period __maybe_unused) |
| { |
| return -1; |
| } |
| |
| static inline void annotated_data_type__tree_delete(struct rb_root *root __maybe_unused) |
| { |
| } |
| |
| static inline void global_var_type__tree_delete(struct rb_root *root __maybe_unused) |
| { |
| } |
| |
| static inline int hist_entry__annotate_data_tty(struct hist_entry *he __maybe_unused, |
| struct evsel *evsel __maybe_unused) |
| { |
| return -1; |
| } |
| |
| #endif /* HAVE_DWARF_SUPPORT */ |
| |
| #ifdef HAVE_SLANG_SUPPORT |
| int hist_entry__annotate_data_tui(struct hist_entry *he, struct evsel *evsel, |
| struct hist_browser_timer *hbt); |
| #else |
| static inline int hist_entry__annotate_data_tui(struct hist_entry *he __maybe_unused, |
| struct evsel *evsel __maybe_unused, |
| struct hist_browser_timer *hbt __maybe_unused) |
| { |
| return -1; |
| } |
| #endif /* HAVE_SLANG_SUPPORT */ |
| |
| #endif /* _PERF_ANNOTATE_DATA_H */ |