/* SPDX-License-Identifier: GPL-2.0 */ #ifndef _PERF_ANNOTATE_DATA_H #define _PERF_ANNOTATE_DATA_H #include #include #include #include #include "dwarf-regs.h" #include "annotate.h" #ifdef HAVE_LIBDW_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_LIBDW_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_LIBDW_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_LIBDW_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 */