2019-05-27 14:55:05 +08:00
|
|
|
/* SPDX-License-Identifier: GPL-2.0-or-later */
|
2011-06-27 15:27:27 +08:00
|
|
|
#ifndef _DWARF_AUX_H
|
|
|
|
#define _DWARF_AUX_H
|
|
|
|
/*
|
|
|
|
* dwarf-aux.h : libdw auxiliary interfaces
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <dwarf.h>
|
|
|
|
#include <elfutils/libdw.h>
|
|
|
|
#include <elfutils/libdwfl.h>
|
|
|
|
#include <elfutils/version.h>
|
|
|
|
|
2019-08-30 03:18:59 +08:00
|
|
|
struct strbuf;
|
|
|
|
|
2011-06-27 15:27:27 +08:00
|
|
|
/* Find the realpath of the target file */
|
2016-03-24 02:06:35 +08:00
|
|
|
const char *cu_find_realpath(Dwarf_Die *cu_die, const char *fname);
|
2011-06-27 15:27:27 +08:00
|
|
|
|
|
|
|
/* Get DW_AT_comp_dir (should be NULL with older gcc) */
|
2016-03-24 02:06:35 +08:00
|
|
|
const char *cu_get_comp_dir(Dwarf_Die *cu_die);
|
2011-06-27 15:27:27 +08:00
|
|
|
|
|
|
|
/* Get a line number and file name for given address */
|
2021-07-15 14:37:23 +08:00
|
|
|
int cu_find_lineinfo(Dwarf_Die *cudie, Dwarf_Addr addr,
|
2016-03-24 02:06:35 +08:00
|
|
|
const char **fname, int *lineno);
|
2011-06-27 15:27:27 +08:00
|
|
|
|
2021-03-24 00:09:15 +08:00
|
|
|
/* Walk on functions at given address */
|
2016-03-24 02:06:35 +08:00
|
|
|
int cu_walk_functions_at(Dwarf_Die *cu_die, Dwarf_Addr addr,
|
|
|
|
int (*callback)(Dwarf_Die *, void *), void *data);
|
2011-08-11 19:02:59 +08:00
|
|
|
|
2016-09-23 23:35:31 +08:00
|
|
|
/* Get DW_AT_linkage_name (should be NULL for C binary) */
|
|
|
|
const char *die_get_linkage_name(Dwarf_Die *dw_die);
|
|
|
|
|
2019-10-24 17:12:54 +08:00
|
|
|
/* Get the lowest PC in DIE (including range list) */
|
|
|
|
int die_entrypc(Dwarf_Die *dw_die, Dwarf_Addr *addr);
|
|
|
|
|
2012-04-23 11:24:36 +08:00
|
|
|
/* Ensure that this DIE is a subprogram and definition (not declaration) */
|
2016-03-24 02:06:35 +08:00
|
|
|
bool die_is_func_def(Dwarf_Die *dw_die);
|
2012-04-23 11:24:36 +08:00
|
|
|
|
2015-01-30 17:37:44 +08:00
|
|
|
/* Ensure that this DIE is an instance of a subprogram */
|
2016-03-24 02:06:35 +08:00
|
|
|
bool die_is_func_instance(Dwarf_Die *dw_die);
|
2015-01-30 17:37:44 +08:00
|
|
|
|
2011-06-27 15:27:27 +08:00
|
|
|
/* Compare diename and tname */
|
2016-03-24 02:06:35 +08:00
|
|
|
bool die_compare_name(Dwarf_Die *dw_die, const char *tname);
|
2011-06-27 15:27:27 +08:00
|
|
|
|
2015-05-08 09:03:35 +08:00
|
|
|
/* Matching diename with glob pattern */
|
2016-03-24 02:06:35 +08:00
|
|
|
bool die_match_name(Dwarf_Die *dw_die, const char *glob);
|
2015-05-08 09:03:35 +08:00
|
|
|
|
2011-06-27 15:27:27 +08:00
|
|
|
/* Get callsite line number of inline-function instance */
|
2016-03-24 02:06:35 +08:00
|
|
|
int die_get_call_lineno(Dwarf_Die *in_die);
|
2011-06-27 15:27:27 +08:00
|
|
|
|
perf probe: Fix to search nested inlined functions in CU
Fix perf probe to walk through the lines of all nested inlined function
call sites and declared lines when a whole CU is passed to the line
walker.
The die_walk_lines() can have two different type of DIEs, subprogram (or
inlined-subroutine) DIE and CU DIE.
If a caller passes a subprogram DIE, this means that the walker walk on
lines of given subprogram. In this case, it just needs to search on
direct children of DIE tree for finding call-site information of inlined
function which directly called from given subprogram.
On the other hand, if a caller passes a CU DIE to the walker, this means
that the walker have to walk on all lines in the source files included
in given CU DIE. In this case, it has to search whole DIE trees of all
subprograms to find the call-site information of all nested inlined
functions.
Without this patch:
$ perf probe --line kernel/cpu.c:151-157
</home/mhiramat/ksrc/linux-2.6/kernel/cpu.c:151>
static int cpu_notify(unsigned long val, void *v)
{
154 return __cpu_notify(val, v, -1, NULL);
}
With this:
$ perf probe --line kernel/cpu.c:151-157
</home/mhiramat/ksrc/linux-2.6/kernel/cpu.c:151>
152 static int cpu_notify(unsigned long val, void *v)
{
154 return __cpu_notify(val, v, -1, NULL);
}
As you can see, --line option with source line range shows the declared
lines as probe-able.
Cc: Frederic Weisbecker <fweisbec@gmail.com>
Cc: Ingo Molnar <mingo@elte.hu>
Cc: Paul Mackerras <paulus@samba.org>
Cc: Pekka Enberg <penberg@kernel.org>
Cc: Peter Zijlstra <peterz@infradead.org>
Cc: yrl.pp-manager.tt@hitachi.com
Link: http://lkml.kernel.org/r/20110811110241.19900.34994.stgit@fedora15
Signed-off-by: Masami Hiramatsu <masami.hiramatsu.pt@hitachi.com>
Signed-off-by: Arnaldo Carvalho de Melo <acme@redhat.com>
2011-08-11 19:02:41 +08:00
|
|
|
/* Get callsite file name of inlined function instance */
|
2016-03-24 02:06:35 +08:00
|
|
|
const char *die_get_call_file(Dwarf_Die *in_die);
|
perf probe: Fix to search nested inlined functions in CU
Fix perf probe to walk through the lines of all nested inlined function
call sites and declared lines when a whole CU is passed to the line
walker.
The die_walk_lines() can have two different type of DIEs, subprogram (or
inlined-subroutine) DIE and CU DIE.
If a caller passes a subprogram DIE, this means that the walker walk on
lines of given subprogram. In this case, it just needs to search on
direct children of DIE tree for finding call-site information of inlined
function which directly called from given subprogram.
On the other hand, if a caller passes a CU DIE to the walker, this means
that the walker have to walk on all lines in the source files included
in given CU DIE. In this case, it has to search whole DIE trees of all
subprograms to find the call-site information of all nested inlined
functions.
Without this patch:
$ perf probe --line kernel/cpu.c:151-157
</home/mhiramat/ksrc/linux-2.6/kernel/cpu.c:151>
static int cpu_notify(unsigned long val, void *v)
{
154 return __cpu_notify(val, v, -1, NULL);
}
With this:
$ perf probe --line kernel/cpu.c:151-157
</home/mhiramat/ksrc/linux-2.6/kernel/cpu.c:151>
152 static int cpu_notify(unsigned long val, void *v)
{
154 return __cpu_notify(val, v, -1, NULL);
}
As you can see, --line option with source line range shows the declared
lines as probe-able.
Cc: Frederic Weisbecker <fweisbec@gmail.com>
Cc: Ingo Molnar <mingo@elte.hu>
Cc: Paul Mackerras <paulus@samba.org>
Cc: Pekka Enberg <penberg@kernel.org>
Cc: Peter Zijlstra <peterz@infradead.org>
Cc: yrl.pp-manager.tt@hitachi.com
Link: http://lkml.kernel.org/r/20110811110241.19900.34994.stgit@fedora15
Signed-off-by: Masami Hiramatsu <masami.hiramatsu.pt@hitachi.com>
Signed-off-by: Arnaldo Carvalho de Melo <acme@redhat.com>
2011-08-11 19:02:41 +08:00
|
|
|
|
2022-11-01 21:48:49 +08:00
|
|
|
/* Get declared file name of a DIE */
|
|
|
|
const char *die_get_decl_file(Dwarf_Die *dw_die);
|
|
|
|
|
2011-06-27 15:27:27 +08:00
|
|
|
/* Get type die */
|
2016-03-24 02:06:35 +08:00
|
|
|
Dwarf_Die *die_get_type(Dwarf_Die *vr_die, Dwarf_Die *die_mem);
|
2011-06-27 15:27:27 +08:00
|
|
|
|
|
|
|
/* Get a type die, but skip qualifiers and typedef */
|
2016-03-24 02:06:35 +08:00
|
|
|
Dwarf_Die *die_get_real_type(Dwarf_Die *vr_die, Dwarf_Die *die_mem);
|
2011-06-27 15:27:27 +08:00
|
|
|
|
|
|
|
/* Check whether the DIE is signed or not */
|
2016-03-24 02:06:35 +08:00
|
|
|
bool die_is_signed_type(Dwarf_Die *tp_die);
|
2011-06-27 15:27:27 +08:00
|
|
|
|
|
|
|
/* Get data_member_location offset */
|
2016-03-24 02:06:35 +08:00
|
|
|
int die_get_data_member_location(Dwarf_Die *mb_die, Dwarf_Word *offs);
|
2011-06-27 15:27:27 +08:00
|
|
|
|
|
|
|
/* Return values for die_find_child() callbacks */
|
|
|
|
enum {
|
|
|
|
DIE_FIND_CB_END = 0, /* End of Search */
|
|
|
|
DIE_FIND_CB_CHILD = 1, /* Search only children */
|
|
|
|
DIE_FIND_CB_SIBLING = 2, /* Search only siblings */
|
|
|
|
DIE_FIND_CB_CONTINUE = 3, /* Search children and siblings */
|
|
|
|
};
|
|
|
|
|
|
|
|
/* Search child DIEs */
|
2016-03-24 02:06:35 +08:00
|
|
|
Dwarf_Die *die_find_child(Dwarf_Die *rt_die,
|
|
|
|
int (*callback)(Dwarf_Die *, void *),
|
|
|
|
void *data, Dwarf_Die *die_mem);
|
2011-06-27 15:27:27 +08:00
|
|
|
|
|
|
|
/* Search a non-inlined function including given address */
|
2016-03-24 02:06:35 +08:00
|
|
|
Dwarf_Die *die_find_realfunc(Dwarf_Die *cu_die, Dwarf_Addr addr,
|
|
|
|
Dwarf_Die *die_mem);
|
2011-06-27 15:27:27 +08:00
|
|
|
|
2015-04-30 19:42:31 +08:00
|
|
|
/* Search a non-inlined function with tail call at given address */
|
|
|
|
Dwarf_Die *die_find_tailfunc(Dwarf_Die *cu_die, Dwarf_Addr addr,
|
|
|
|
Dwarf_Die *die_mem);
|
|
|
|
|
perf probe: Fix to find line information for probe list
Fix to find the correct (as much as possible) line information for
listing probes. Without this fix, perf probe --list action will show
incorrect line information as below;
probe:getname_flags (on getname_flags@ksrc/linux-3/fs/namei.c)
probe:getname_flags_1 (on getname:-89@x86/include/asm/current.h)
probe:getname_flags_2 (on user_path_at_empty:-2054@x86/include/asm/current.h)
The minus line number is obviously wrong, and current.h is not related
to the probe point. Deeper investigation discovered that there were 2
issues related to this bug, and minor typos too.
The 1st issue is the rack of considering about nested inlined functions,
which causes the wrong (relative) line number.
The 2nd issue is that the dwarf line info is not correct at those
points. It points 14th line of current.h.
Since it seems that the line info includes somewhat unreliable
information, this fixes perf to try to find correct line information
from both of debuginfo and line info as below.
1) Probe address is the entry of a function instance
In this case, the line is set as the function declared line.
2) Probe address is the entry of an expanded inline function block
In this case, the line is set as the function call-site line.
This means that the line number is relative from the entry line
of caller function (which can be an inlined function if nested)
3) Probe address is inside a function instance or an expanded
inline function block
In this case, perf probe queries the line number from lineinfo
and verify the function declared file is same as the file name
queried from lineinfo.
If the file name is different, it is a failure case. The probe
address is shown as symbol+offset.
4) Probe address is not in the any function instance
This is a failure case, the probe address is shown as
symbol+offset.
With this fix, perf probe -l shows correct probe lines as below;
probe:getname_flags (on getname_flags@ksrc/linux-3/fs/namei.c)
probe:getname_flags_1 (on getname:2@ksrc/linux-3/fs/namei.c)
probe:getname_flags_2 (on user_path_at_empty:4@ksrc/linux-3/fs/namei.c)
Changes at v2:
- Fix typos in the function comments. (Thanks to Namhyung Kim)
- Use die_find_top_inlinefunc instead of die_find_inlinefunc_next.
Signed-off-by: Masami Hiramatsu <masami.hiramatsu.pt@hitachi.com>
Cc: Ingo Molnar <mingo@redhat.com>
Cc: Namhyung Kim <namhyung@kernel.org>
Cc: Paul Mackerras <paulus@samba.org>
Cc: Peter Zijlstra <a.p.zijlstra@chello.nl>
Link: http://lkml.kernel.org/r/20130930092144.1693.11058.stgit@udc4-manage.rcp.hitachi.co.jp
Signed-off-by: Arnaldo Carvalho de Melo <acme@redhat.com>
2013-09-30 17:21:44 +08:00
|
|
|
/* Search the top inlined function including given address */
|
2016-03-24 02:06:35 +08:00
|
|
|
Dwarf_Die *die_find_top_inlinefunc(Dwarf_Die *sp_die, Dwarf_Addr addr,
|
|
|
|
Dwarf_Die *die_mem);
|
perf probe: Fix to find line information for probe list
Fix to find the correct (as much as possible) line information for
listing probes. Without this fix, perf probe --list action will show
incorrect line information as below;
probe:getname_flags (on getname_flags@ksrc/linux-3/fs/namei.c)
probe:getname_flags_1 (on getname:-89@x86/include/asm/current.h)
probe:getname_flags_2 (on user_path_at_empty:-2054@x86/include/asm/current.h)
The minus line number is obviously wrong, and current.h is not related
to the probe point. Deeper investigation discovered that there were 2
issues related to this bug, and minor typos too.
The 1st issue is the rack of considering about nested inlined functions,
which causes the wrong (relative) line number.
The 2nd issue is that the dwarf line info is not correct at those
points. It points 14th line of current.h.
Since it seems that the line info includes somewhat unreliable
information, this fixes perf to try to find correct line information
from both of debuginfo and line info as below.
1) Probe address is the entry of a function instance
In this case, the line is set as the function declared line.
2) Probe address is the entry of an expanded inline function block
In this case, the line is set as the function call-site line.
This means that the line number is relative from the entry line
of caller function (which can be an inlined function if nested)
3) Probe address is inside a function instance or an expanded
inline function block
In this case, perf probe queries the line number from lineinfo
and verify the function declared file is same as the file name
queried from lineinfo.
If the file name is different, it is a failure case. The probe
address is shown as symbol+offset.
4) Probe address is not in the any function instance
This is a failure case, the probe address is shown as
symbol+offset.
With this fix, perf probe -l shows correct probe lines as below;
probe:getname_flags (on getname_flags@ksrc/linux-3/fs/namei.c)
probe:getname_flags_1 (on getname:2@ksrc/linux-3/fs/namei.c)
probe:getname_flags_2 (on user_path_at_empty:4@ksrc/linux-3/fs/namei.c)
Changes at v2:
- Fix typos in the function comments. (Thanks to Namhyung Kim)
- Use die_find_top_inlinefunc instead of die_find_inlinefunc_next.
Signed-off-by: Masami Hiramatsu <masami.hiramatsu.pt@hitachi.com>
Cc: Ingo Molnar <mingo@redhat.com>
Cc: Namhyung Kim <namhyung@kernel.org>
Cc: Paul Mackerras <paulus@samba.org>
Cc: Peter Zijlstra <a.p.zijlstra@chello.nl>
Link: http://lkml.kernel.org/r/20130930092144.1693.11058.stgit@udc4-manage.rcp.hitachi.co.jp
Signed-off-by: Arnaldo Carvalho de Melo <acme@redhat.com>
2013-09-30 17:21:44 +08:00
|
|
|
|
|
|
|
/* Search the deepest inlined function including given address */
|
2016-03-24 02:06:35 +08:00
|
|
|
Dwarf_Die *die_find_inlinefunc(Dwarf_Die *sp_die, Dwarf_Addr addr,
|
|
|
|
Dwarf_Die *die_mem);
|
2011-06-27 15:27:27 +08:00
|
|
|
|
2011-08-11 19:03:11 +08:00
|
|
|
/* Walk on the instances of given DIE */
|
2016-03-24 02:06:35 +08:00
|
|
|
int die_walk_instances(Dwarf_Die *in_die,
|
|
|
|
int (*callback)(Dwarf_Die *, void *), void *data);
|
2011-08-11 19:03:11 +08:00
|
|
|
|
2011-06-27 15:27:27 +08:00
|
|
|
/* Walker on lines (Note: line number will not be sorted) */
|
|
|
|
typedef int (* line_walk_callback_t) (const char *fname, int lineno,
|
|
|
|
Dwarf_Addr addr, void *data);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Walk on lines inside given DIE. If the DIE is a subprogram, walk only on
|
|
|
|
* the lines inside the subprogram, otherwise the DIE must be a CU DIE.
|
|
|
|
*/
|
2016-03-24 02:06:35 +08:00
|
|
|
int die_walk_lines(Dwarf_Die *rt_die, line_walk_callback_t callback, void *data);
|
2011-06-27 15:27:27 +08:00
|
|
|
|
|
|
|
/* Find a variable called 'name' at given address */
|
2016-03-24 02:06:35 +08:00
|
|
|
Dwarf_Die *die_find_variable_at(Dwarf_Die *sp_die, const char *name,
|
|
|
|
Dwarf_Addr addr, Dwarf_Die *die_mem);
|
2011-06-27 15:27:27 +08:00
|
|
|
|
|
|
|
/* Find a member called 'name' */
|
2016-03-24 02:06:35 +08:00
|
|
|
Dwarf_Die *die_find_member(Dwarf_Die *st_die, const char *name,
|
|
|
|
Dwarf_Die *die_mem);
|
2011-06-27 15:27:27 +08:00
|
|
|
|
|
|
|
/* Get the name of given variable DIE */
|
2016-03-24 02:06:35 +08:00
|
|
|
int die_get_typename(Dwarf_Die *vr_die, struct strbuf *buf);
|
2011-06-27 15:27:27 +08:00
|
|
|
|
|
|
|
/* Get the name and type of given variable DIE, stored as "type\tname" */
|
2016-03-24 02:06:35 +08:00
|
|
|
int die_get_varname(Dwarf_Die *vr_die, struct strbuf *buf);
|
|
|
|
int die_get_var_range(Dwarf_Die *sp_die, Dwarf_Die *vr_die, struct strbuf *buf);
|
2016-08-30 16:39:37 +08:00
|
|
|
|
|
|
|
/* Check if target program is compiled with optimization */
|
|
|
|
bool die_is_optimized_target(Dwarf_Die *cu_die);
|
|
|
|
|
|
|
|
/* Use next address after prologue as probe location */
|
|
|
|
void die_skip_prologue(Dwarf_Die *sp_die, Dwarf_Die *cu_die,
|
|
|
|
Dwarf_Addr *entrypc);
|
|
|
|
|
2011-06-27 15:27:27 +08:00
|
|
|
#endif
|