perf probe: Search concrete out-of-line instances
gcc 4.6 generates a concrete out-of-line instance when there is a function which is implicitly inlined somewhere but also has its own instance. The concrete out-of-line instance means that it has an abstract origin of the function which is referred by not only inlined-subroutines but also a concrete subprogram. Since current dwarf_func_inline_instances() can find only instances of inlined-subroutines, this introduces new die_walk_instances() to find both of subprogram and inlined-subroutines. e.g. without this, Available variables at sched_group_rt_period @<cpu_rt_period_read_uint+9> struct task_group* tg perf probe failed to find actual subprogram instance of sched_group_rt_period(). With this, Available variables at sched_group_rt_period @<cpu_rt_period_read_uint+9> struct task_group* tg @<sched_group_rt_period+0> struct task_group* tg Now it found the sched_group_rt_period() itself. 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/20110811110311.19900.63997.stgit@fedora15 Signed-off-by: Masami Hiramatsu <masami.hiramatsu.pt@hitachi.com> Signed-off-by: Arnaldo Carvalho de Melo <acme@redhat.com>
This commit is contained in:
committed by
Arnaldo Carvalho de Melo
parent
f182e3e13c
commit
db0d2c6420
@@ -453,6 +453,64 @@ Dwarf_Die *die_find_inlinefunc(Dwarf_Die *sp_die, Dwarf_Addr addr,
|
||||
return die_mem;
|
||||
}
|
||||
|
||||
struct __instance_walk_param {
|
||||
void *addr;
|
||||
int (*callback)(Dwarf_Die *, void *);
|
||||
void *data;
|
||||
int retval;
|
||||
};
|
||||
|
||||
static int __die_walk_instances_cb(Dwarf_Die *inst, void *data)
|
||||
{
|
||||
struct __instance_walk_param *iwp = data;
|
||||
Dwarf_Attribute attr_mem;
|
||||
Dwarf_Die origin_mem;
|
||||
Dwarf_Attribute *attr;
|
||||
Dwarf_Die *origin;
|
||||
|
||||
attr = dwarf_attr(inst, DW_AT_abstract_origin, &attr_mem);
|
||||
if (attr == NULL)
|
||||
return DIE_FIND_CB_CONTINUE;
|
||||
|
||||
origin = dwarf_formref_die(attr, &origin_mem);
|
||||
if (origin == NULL || origin->addr != iwp->addr)
|
||||
return DIE_FIND_CB_CONTINUE;
|
||||
|
||||
iwp->retval = iwp->callback(inst, iwp->data);
|
||||
|
||||
return (iwp->retval) ? DIE_FIND_CB_END : DIE_FIND_CB_CONTINUE;
|
||||
}
|
||||
|
||||
/**
|
||||
* die_walk_instances - Walk on instances of given DIE
|
||||
* @or_die: an abstract original DIE
|
||||
* @callback: a callback function which is called with instance DIE
|
||||
* @data: user data
|
||||
*
|
||||
* Walk on the instances of give @in_die. @in_die must be an inlined function
|
||||
* declartion. This returns the return value of @callback if it returns
|
||||
* non-zero value, or -ENOENT if there is no instance.
|
||||
*/
|
||||
int die_walk_instances(Dwarf_Die *or_die, int (*callback)(Dwarf_Die *, void *),
|
||||
void *data)
|
||||
{
|
||||
Dwarf_Die cu_die;
|
||||
Dwarf_Die die_mem;
|
||||
struct __instance_walk_param iwp = {
|
||||
.addr = or_die->addr,
|
||||
.callback = callback,
|
||||
.data = data,
|
||||
.retval = -ENOENT,
|
||||
};
|
||||
|
||||
if (dwarf_diecu(or_die, &cu_die, NULL, NULL) == NULL)
|
||||
return -ENOENT;
|
||||
|
||||
die_find_child(&cu_die, __die_walk_instances_cb, &iwp, &die_mem);
|
||||
|
||||
return iwp.retval;
|
||||
}
|
||||
|
||||
/* Line walker internal parameters */
|
||||
struct __line_walk_param {
|
||||
bool recursive;
|
||||
|
Reference in New Issue
Block a user