mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/next/linux-next.git
synced 2025-01-17 05:45:20 +00:00
8838abf626
In Makefile.config for unwinding the name dwarf implies either libunwind or libdw. Make it clearer that HAVE_DWARF_SUPPORT is really just defined when libdw is present by renaming to HAVE_LIBDW_SUPPORT. Signed-off-by: Ian Rogers <irogers@google.com> Acked-by: Namhyung Kim <namhyung@kernel.org> Tested-by: Leo Yan <leo.yan@arm.com> Cc: Anup Patel <anup@brainfault.org> Cc: Yang Jihong <yangjihong@bytedance.com> Cc: Palmer Dabbelt <palmer@dabbelt.com> Cc: David S. Miller <davem@davemloft.net> Cc: Albert Ou <aou@eecs.berkeley.edu> Cc: Shenlin Liang <liangshenlin@eswincomputing.com> Cc: Nick Terrell <terrelln@fb.com> Cc: Guilherme Amadio <amadio@gentoo.org> Cc: Steinar H. Gunderson <sesse@google.com> Cc: Changbin Du <changbin.du@huawei.com> Cc: Alexander Lobakin <aleksander.lobakin@intel.com> Cc: Przemek Kitszel <przemyslaw.kitszel@intel.com> Cc: Huacai Chen <chenhuacai@kernel.org> Cc: Guo Ren <guoren@kernel.org> Cc: Masahiro Yamada <masahiroy@kernel.org> Cc: Masami Hiramatsu (Google) <mhiramat@kernel.org> Cc: Will Deacon <will@kernel.org> Cc: James Clark <james.clark@linaro.org> Cc: Mike Leach <mike.leach@linaro.org> Cc: Chen Pei <cp0613@linux.alibaba.com> Cc: Leo Yan <leo.yan@linux.dev> Cc: Oliver Upton <oliver.upton@linux.dev> Cc: Aditya Gupta <adityag@linux.ibm.com> Cc: Kajol Jain <kjain@linux.ibm.com> Cc: Athira Rajeev <atrajeev@linux.vnet.ibm.com> Cc: linux-arm-kernel@lists.infradead.org Cc: linux-riscv@lists.infradead.org Cc: Bibo Mao <maobibo@loongson.cn> Cc: John Garry <john.g.garry@oracle.com> Cc: Atish Patra <atishp@rivosinc.com> Cc: Dima Kogan <dima@secretsauce.net> Cc: Paul Walmsley <paul.walmsley@sifive.com> Cc: Dr. David Alan Gilbert <linux@treblig.org> Cc: linux-csky@vger.kernel.org Link: https://lore.kernel.org/r/20241017001354.56973-11-irogers@google.com Signed-off-by: Namhyung Kim <namhyung@kernel.org>
294 lines
7.6 KiB
C
294 lines
7.6 KiB
C
/* 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_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 */
|