perf callchain: Make get_srcline fall back to sym+offset
authorAndi Kleen <ak@linux.intel.com>
Thu, 13 Nov 2014 02:05:27 +0000 (18:05 -0800)
committerArnaldo Carvalho de Melo <acme@redhat.com>
Mon, 24 Nov 2014 21:03:47 +0000 (18:03 -0300)
When the source line is not found fall back to sym + offset.  This is
generally much more useful than a raw address.

For this we need to pass in the symbol from the caller.

For some callers it's awkward to compute, so we stay at the old
behaviour.

Signed-off-by: Andi Kleen <ak@linux.intel.com>
Cc: Jiri Olsa <jolsa@redhat.com>
Cc: Namhyung Kim <namhyung@kernel.org>
Link: http://lkml.kernel.org/r/1415844328-4884-10-git-send-email-andi@firstfloor.org
Signed-off-by: Arnaldo Carvalho de Melo <acme@redhat.com>
tools/perf/util/annotate.c
tools/perf/util/callchain.c
tools/perf/util/map.c
tools/perf/util/sort.c
tools/perf/util/srcline.c
tools/perf/util/util.h

index e5670f1af737d9ec40916349066cf12c66c2ed70..79999ceaf2be08e5f4880e853d467bf59874340c 100644 (file)
@@ -1192,7 +1192,7 @@ static int symbol__get_source_line(struct symbol *sym, struct map *map,
                        goto next;
 
                offset = start + i;
-               src_line->path = get_srcline(map->dso, offset);
+               src_line->path = get_srcline(map->dso, offset, NULL, false);
                insert_source_line(&tmp_root, src_line);
 
        next:
index b6624aeaaca9b5b865606e231620a1e4c6ba62fc..517ed84db97a9b895771b69437168d9c6a4edcca 100644 (file)
@@ -819,7 +819,8 @@ char *callchain_list__sym_name(struct callchain_list *cl,
                    cl->ms.map && !cl->srcline)
                        cl->srcline = get_srcline(cl->ms.map->dso,
                                                  map__rip_2objdump(cl->ms.map,
-                                                                   cl->ip));
+                                                                   cl->ip),
+                                                 cl->ms.sym, false);
                if (cl->srcline)
                        printed = scnprintf(bf, bfsize, "%s %s",
                                        cl->ms.sym->name, cl->srcline);
index 040a785c857b6a7f6fd0eba42f741864dba5955d..62ca9f2607d557804a0bc260e74a5e5c5dbc32e1 100644 (file)
@@ -360,7 +360,7 @@ int map__fprintf_srcline(struct map *map, u64 addr, const char *prefix,
 
        if (map && map->dso) {
                srcline = get_srcline(map->dso,
-                                     map__rip_2objdump(map, addr));
+                                     map__rip_2objdump(map, addr), NULL, true);
                if (srcline != SRCLINE_UNKNOWN)
                        ret = fprintf(fp, "%s%s", prefix, srcline);
                free_srcline(srcline);
index 82a5596241a75211fd035238bb8d41059f0c8c1e..9139dda9f9a37afd7ae16921928e224604950cf5 100644 (file)
@@ -291,7 +291,8 @@ sort__srcline_cmp(struct hist_entry *left, struct hist_entry *right)
                else {
                        struct map *map = left->ms.map;
                        left->srcline = get_srcline(map->dso,
-                                           map__rip_2objdump(map, left->ip));
+                                          map__rip_2objdump(map, left->ip),
+                                                   left->ms.sym, true);
                }
        }
        if (!right->srcline) {
@@ -300,7 +301,8 @@ sort__srcline_cmp(struct hist_entry *left, struct hist_entry *right)
                else {
                        struct map *map = right->ms.map;
                        right->srcline = get_srcline(map->dso,
-                                           map__rip_2objdump(map, right->ip));
+                                            map__rip_2objdump(map, right->ip),
+                                                    right->ms.sym, true);
                }
        }
        return strcmp(right->srcline, left->srcline);
index ac877f96fed7d23a5c074615391ed186a1f3c909..e73b6a5c9e0fa8772862fdd3535701b67325c5dc 100644 (file)
@@ -8,6 +8,8 @@
 #include "util/util.h"
 #include "util/debug.h"
 
+#include "symbol.h"
+
 #ifdef HAVE_LIBBFD_SUPPORT
 
 /*
@@ -250,7 +252,8 @@ void dso__free_a2l(struct dso *dso __maybe_unused)
  */
 #define A2L_FAIL_LIMIT 123
 
-char *get_srcline(struct dso *dso, unsigned long addr)
+char *get_srcline(struct dso *dso, unsigned long addr, struct symbol *sym,
+                 bool show_sym)
 {
        char *file = NULL;
        unsigned line = 0;
@@ -289,7 +292,11 @@ out:
                dso->has_srcline = 0;
                dso__free_a2l(dso);
        }
-       if (asprintf(&srcline, "%s[%lx]", dso->short_name, addr) < 0)
+       if (sym) {
+               if (asprintf(&srcline, "%s+%ld", show_sym ? sym->name : "",
+                                       addr - sym->start) < 0)
+                       return SRCLINE_UNKNOWN;
+       } else if (asprintf(&srcline, "%s[%lx]", dso->short_name, addr) < 0)
                return SRCLINE_UNKNOWN;
        return srcline;
 }
index 76d23d83eae59eaf63e39e969d480febd2a390a5..419bee030f835c295fee546a4211afaa19f84ace 100644 (file)
@@ -337,8 +337,10 @@ static inline int path__join3(char *bf, size_t size,
 }
 
 struct dso;
+struct symbol;
 
-char *get_srcline(struct dso *dso, unsigned long addr);
+char *get_srcline(struct dso *dso, unsigned long addr, struct symbol *sym,
+                 bool show_sym);
 void free_srcline(char *srcline);
 
 int filename__read_int(const char *filename, int *value);