Author | Tokens | Token Proportion | Commits | Commit Proportion |
---|---|---|---|---|
Namhyung Kim | 684 | 92.68% | 28 | 75.68% |
Athira Rajeev | 35 | 4.74% | 3 | 8.11% |
Arnaldo Carvalho de Melo | 16 | 2.17% | 5 | 13.51% |
Jiri Olsa | 3 | 0.41% | 1 | 2.70% |
Total | 738 | 37 |
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293
/* 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 */
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.
Created with Cregit http://github.com/cregit/cregit
Version 2.0-RC1