Skip to content

Commit

Permalink
perf callchain: Refactor inline_list to store srcline string directly
Browse files Browse the repository at this point in the history
This is a preparation for the creation of real callchain entries for
inlined frames. The rest of the perf code uses the srcline string. As
such, using that also for the srcline API allows us to simplify some of
the upcoming code. Most notably, it will allow us to cache the srcline
for a given inline node and reuse it for different callchain entries.

Signed-off-by: Milian Wolff <milian.wolff@kdab.com>
Reviewed-by: Jiri Olsa <jolsa@redhat.com>
Reviewed-by: Namhyung Kim <namhyung@kernel.org>
Cc: David Ahern <dsahern@gmail.com>
Cc: Peter Zijlstra <peterz@infradead.org>
Cc: Yao Jin <yao.jin@linux.intel.com>
Link: http://lkml.kernel.org/r/20171009203310.17362-5-milian.wolff@kdab.com
Signed-off-by: Arnaldo Carvalho de Melo <acme@redhat.com>
  • Loading branch information
milianw authored and acmel committed Oct 24, 2017
1 parent fea0cf8 commit 2be8832
Show file tree
Hide file tree
Showing 2 changed files with 41 additions and 16 deletions.
54 changes: 40 additions & 14 deletions tools/perf/util/srcline.c
Original file line number Diff line number Diff line change
Expand Up @@ -33,8 +33,8 @@ static const char *dso__name(struct dso *dso)
return dso_name;
}

static int inline_list__append(struct symbol *symbol, char *filename,
int line_nr, struct inline_node *node)
static int inline_list__append(struct symbol *symbol, char *srcline,
struct inline_node *node)
{
struct inline_list *ilist;

Expand All @@ -43,8 +43,7 @@ static int inline_list__append(struct symbol *symbol, char *filename,
return -1;

ilist->symbol = symbol;
ilist->filename = filename;
ilist->line_nr = line_nr;
ilist->srcline = srcline;

if (callchain_param.order == ORDER_CALLEE)
list_add_tail(&ilist->list, &node->val);
Expand All @@ -54,6 +53,30 @@ static int inline_list__append(struct symbol *symbol, char *filename,
return 0;
}

/* basename version that takes a const input string */
static const char *gnu_basename(const char *path)
{
const char *base = strrchr(path, '/');

return base ? base + 1 : path;
}

static char *srcline_from_fileline(const char *file, unsigned int line)
{
char *srcline;

if (!file)
return NULL;

if (!srcline_full_filename)
file = gnu_basename(file);

if (asprintf(&srcline, "%s:%u", file, line) < 0)
return NULL;

return srcline;
}

#ifdef HAVE_LIBBFD_SUPPORT

/*
Expand Down Expand Up @@ -237,9 +260,12 @@ static int inline_list__append_dso_a2l(struct dso *dso,
{
struct a2l_data *a2l = dso->a2l;
struct symbol *inline_sym = new_inline_sym(dso, sym, a2l->funcname);
char *srcline = NULL;

return inline_list__append(inline_sym, strdup(a2l->filename),
a2l->line, node);
if (a2l->filename)
srcline = srcline_from_fileline(a2l->filename, a2l->line);

return inline_list__append(inline_sym, srcline, node);
}

static int addr2line(const char *dso_name, u64 addr,
Expand Down Expand Up @@ -437,13 +463,15 @@ static struct inline_node *addr2inlines(const char *dso_name, u64 addr,
node->addr = addr;

while (getline(&filename, &len, fp) != -1) {
char *srcline;

if (filename_split(filename, &line_nr) != 1) {
free(filename);
goto out;
}

if (inline_list__append(sym, filename, line_nr, node) != 0)
srcline = srcline_from_fileline(filename, line_nr);
if (inline_list__append(sym, srcline, node) != 0)
goto out;

filename = NULL;
Expand Down Expand Up @@ -487,16 +515,14 @@ char *__get_srcline(struct dso *dso, u64 addr, struct symbol *sym,
unwind_inlines, NULL, sym))
goto out;

if (asprintf(&srcline, "%s:%u",
srcline_full_filename ? file : basename(file),
line) < 0) {
free(file);
srcline = srcline_from_fileline(file, line);
free(file);

if (!srcline)
goto out;
}

dso->a2l_fails = 0;

free(file);
return srcline;

out:
Expand Down Expand Up @@ -548,7 +574,7 @@ void inline_node__delete(struct inline_node *node)

list_for_each_entry_safe(ilist, tmp, &node->val, list) {
list_del_init(&ilist->list);
zfree(&ilist->filename);
free_srcline(ilist->srcline);
/* only the inlined symbols are owned by the list */
if (ilist->symbol && ilist->symbol->inlined)
symbol__delete(ilist->symbol);
Expand Down
3 changes: 1 addition & 2 deletions tools/perf/util/srcline.h
Original file line number Diff line number Diff line change
Expand Up @@ -18,8 +18,7 @@ void free_srcline(char *srcline);

struct inline_list {
struct symbol *symbol;
char *filename;
unsigned int line_nr;
char *srcline;
struct list_head list;
};

Expand Down

0 comments on commit 2be8832

Please sign in to comment.