Commit a5a00497 authored by Namhyung Kim's avatar Namhyung Kim Committed by Arnaldo Carvalho de Melo

perf annotate-data: Handle RSP if it's not the FB register

In some cases, the stack pointer on x86 (rsp = reg7) is used to point
variables on stack but it's not the frame base register.  Then it
should handle the register like normal registers (IOW not to access
the other stack variables using offset calculation) but it should not
assume it would have a pointer.

Before:
  -----------------------------------------------------------
  find data type for 0x7c(reg7) at tcp_getsockopt+0xb62
  CU for net/ipv4/tcp.c (die:0x7b5f516)
  frame base: cfa=0 fbreg=6
  no pointer or no type
  check variable "zc" failed (die: 0x7b9580a)
   variable location: base=reg7, offset=0x40
   type='struct tcp_zerocopy_receive' size=0x40 (die:0x7b947f4)

After:
  -----------------------------------------------------------
  find data type for 0x7c(reg7) at tcp_getsockopt+0xb62
  CU for net/ipv4/tcp.c (die:0x7b5f516)
  frame base: cfa=0 fbreg=6
  found "zc" in scope=3/3 (die: 0x7b957fc) type_offset=0x3c
   variable location: base=reg7, offset=0x40
   type='struct tcp_zerocopy_receive' size=0x40 (die:0x7b947f4)

Note that the type-offset was properly calculated to 0x3c as the
variable starts at 0x40.
Signed-off-by: default avatarNamhyung Kim <namhyung@kernel.org>
Cc: Adrian Hunter <adrian.hunter@intel.com>
Cc: Ian Rogers <irogers@google.com>
Cc: Ingo Molnar <mingo@kernel.org>
Cc: Jiri Olsa <jolsa@kernel.org>
Cc: Kan Liang <kan.liang@linux.intel.com>
Cc: Peter Zijlstra <peterz@infradead.org>
Link: https://lore.kernel.org/r/20240412183310.2518474-5-namhyung@kernel.orgSigned-off-by: default avatarArnaldo Carvalho de Melo <acme@redhat.com>
parent 0519fadb
...@@ -25,6 +25,9 @@ ...@@ -25,6 +25,9 @@
#include "symbol_conf.h" #include "symbol_conf.h"
#include "thread.h" #include "thread.h"
/* register number of the stack pointer */
#define X86_REG_SP 7
enum type_state_kind { enum type_state_kind {
TSR_KIND_INVALID = 0, TSR_KIND_INVALID = 0,
TSR_KIND_TYPE, TSR_KIND_TYPE,
...@@ -197,7 +200,7 @@ static void init_type_state(struct type_state *state, struct arch *arch) ...@@ -197,7 +200,7 @@ static void init_type_state(struct type_state *state, struct arch *arch)
state->regs[10].caller_saved = true; state->regs[10].caller_saved = true;
state->regs[11].caller_saved = true; state->regs[11].caller_saved = true;
state->ret_reg = 0; state->ret_reg = 0;
state->stack_reg = 7; state->stack_reg = X86_REG_SP;
} }
} }
...@@ -382,10 +385,18 @@ static bool find_cu_die(struct debuginfo *di, u64 pc, Dwarf_Die *cu_die) ...@@ -382,10 +385,18 @@ static bool find_cu_die(struct debuginfo *di, u64 pc, Dwarf_Die *cu_die)
} }
/* The type info will be saved in @type_die */ /* The type info will be saved in @type_die */
static int check_variable(Dwarf_Die *var_die, Dwarf_Die *type_die, int offset, static int check_variable(struct data_loc_info *dloc, Dwarf_Die *var_die,
bool is_pointer) Dwarf_Die *type_die, int reg, int offset, bool is_fbreg)
{ {
Dwarf_Word size; Dwarf_Word size;
bool is_pointer = true;
if (reg == DWARF_REG_PC)
is_pointer = false;
else if (reg == dloc->fbreg || is_fbreg)
is_pointer = false;
else if (arch__is(dloc->arch, "x86") && reg == X86_REG_SP)
is_pointer = false;
/* Get the type of the variable */ /* Get the type of the variable */
if (die_get_real_type(var_die, type_die) == NULL) { if (die_get_real_type(var_die, type_die) == NULL) {
...@@ -607,7 +618,6 @@ static bool get_global_var_type(Dwarf_Die *cu_die, struct data_loc_info *dloc, ...@@ -607,7 +618,6 @@ static bool get_global_var_type(Dwarf_Die *cu_die, struct data_loc_info *dloc,
{ {
u64 pc; u64 pc;
int offset; int offset;
bool is_pointer = false;
const char *var_name = NULL; const char *var_name = NULL;
struct global_var_entry *gvar; struct global_var_entry *gvar;
Dwarf_Die var_die; Dwarf_Die var_die;
...@@ -623,7 +633,8 @@ static bool get_global_var_type(Dwarf_Die *cu_die, struct data_loc_info *dloc, ...@@ -623,7 +633,8 @@ static bool get_global_var_type(Dwarf_Die *cu_die, struct data_loc_info *dloc,
/* Try to get the variable by address first */ /* Try to get the variable by address first */
if (die_find_variable_by_addr(cu_die, var_addr, &var_die, &offset) && if (die_find_variable_by_addr(cu_die, var_addr, &var_die, &offset) &&
check_variable(&var_die, type_die, offset, is_pointer) == 0) { check_variable(dloc, &var_die, type_die, DWARF_REG_PC, offset,
/*is_fbreg=*/false) == 0) {
var_name = dwarf_diename(&var_die); var_name = dwarf_diename(&var_die);
*var_offset = offset; *var_offset = offset;
goto ok; goto ok;
...@@ -636,7 +647,8 @@ static bool get_global_var_type(Dwarf_Die *cu_die, struct data_loc_info *dloc, ...@@ -636,7 +647,8 @@ static bool get_global_var_type(Dwarf_Die *cu_die, struct data_loc_info *dloc,
/* Try to get the name of global variable */ /* Try to get the name of global variable */
if (die_find_variable_at(cu_die, var_name, pc, &var_die) && if (die_find_variable_at(cu_die, var_name, pc, &var_die) &&
check_variable(&var_die, type_die, *var_offset, is_pointer) == 0) check_variable(dloc, &var_die, type_die, DWARF_REG_PC, *var_offset,
/*is_fbreg=*/false) == 0)
goto ok; goto ok;
return false; return false;
...@@ -1587,8 +1599,7 @@ static int find_data_type_die(struct data_loc_info *dloc, Dwarf_Die *type_die) ...@@ -1587,8 +1599,7 @@ static int find_data_type_die(struct data_loc_info *dloc, Dwarf_Die *type_die)
} }
/* Found a variable, see if it's correct */ /* Found a variable, see if it's correct */
ret = check_variable(&var_die, type_die, offset, ret = check_variable(dloc, &var_die, type_die, reg, offset, is_fbreg);
reg != DWARF_REG_PC && !is_fbreg);
if (ret == 0) { if (ret == 0) {
pr_debug_dtp("found \"%s\" in scope=%d/%d (die: %#lx) ", pr_debug_dtp("found \"%s\" in scope=%d/%d (die: %#lx) ",
dwarf_diename(&var_die), i+1, nr_scopes, dwarf_diename(&var_die), i+1, nr_scopes,
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment