diff options
author | Eric B Munson <ebmunson@us.ibm.com> | 2010-06-14 09:56:33 -0400 |
---|---|---|
committer | Arnaldo Carvalho de Melo <acme@redhat.com> | 2010-06-17 09:06:27 -0400 |
commit | 70c3856b2f1304e0abc65f1b96a8c60ddfc0fb9e (patch) | |
tree | 69230e53848da5a55e6148f57657c6bc3b531123 /tools/perf/util/symbol.c | |
parent | cf103a14dd2ab23f847e998c8881ea4a5f8090bf (diff) |
perf symbols: Function descriptor symbol lookup
Currently symbol resolution does not work for 64-bit programs on architectures
that use function descriptors such as ppc64.
The problem is that a symbol doesn't point to a text address, it points to a
data area that contains (amongst other things) a pointer to the text address.
We look for a section called ".opd" which is the function descriptor area. To
create the full symbol table, when we see a symbol in the function descriptor
section we load the first pointer and use that as the text address.
Cc: Ingo Molnar <mingo@elte.hu>
Cc: Paul Mackerras <paulus@samba.org>
Cc: Peter Zijlstra <peterz@infradead.org>
LKML-Reference: <1276523793-15422-1-git-send-email-ebmunson@us.ibm.com>
Signed-off-by: Anton Blanchard <anton@samba.org>
Signed-off-by: Eric B Munson <ebmunson@us.ibm.com>
Signed-off-by: Arnaldo Carvalho de Melo <acme@redhat.com>
Diffstat (limited to 'tools/perf/util/symbol.c')
-rw-r--r-- | tools/perf/util/symbol.c | 37 |
1 files changed, 34 insertions, 3 deletions
diff --git a/tools/perf/util/symbol.c b/tools/perf/util/symbol.c index b63e5713849f..971d0a05d6b4 100644 --- a/tools/perf/util/symbol.c +++ b/tools/perf/util/symbol.c | |||
@@ -933,6 +933,25 @@ static bool elf_sec__is_a(GElf_Shdr *self, Elf_Data *secstrs, enum map_type type | |||
933 | } | 933 | } |
934 | } | 934 | } |
935 | 935 | ||
936 | static size_t elf_addr_to_index(Elf *elf, GElf_Addr addr) | ||
937 | { | ||
938 | Elf_Scn *sec = NULL; | ||
939 | GElf_Shdr shdr; | ||
940 | size_t cnt = 1; | ||
941 | |||
942 | while ((sec = elf_nextscn(elf, sec)) != NULL) { | ||
943 | gelf_getshdr(sec, &shdr); | ||
944 | |||
945 | if ((addr >= shdr.sh_addr) && | ||
946 | (addr < (shdr.sh_addr + shdr.sh_size))) | ||
947 | return cnt; | ||
948 | |||
949 | ++cnt; | ||
950 | } | ||
951 | |||
952 | return -1; | ||
953 | } | ||
954 | |||
936 | static int dso__load_sym(struct dso *self, struct map *map, const char *name, | 955 | static int dso__load_sym(struct dso *self, struct map *map, const char *name, |
937 | int fd, symbol_filter_t filter, int kmodule) | 956 | int fd, symbol_filter_t filter, int kmodule) |
938 | { | 957 | { |
@@ -944,12 +963,13 @@ static int dso__load_sym(struct dso *self, struct map *map, const char *name, | |||
944 | int err = -1; | 963 | int err = -1; |
945 | uint32_t idx; | 964 | uint32_t idx; |
946 | GElf_Ehdr ehdr; | 965 | GElf_Ehdr ehdr; |
947 | GElf_Shdr shdr; | 966 | GElf_Shdr shdr, opdshdr; |
948 | Elf_Data *syms; | 967 | Elf_Data *syms, *opddata = NULL; |
949 | GElf_Sym sym; | 968 | GElf_Sym sym; |
950 | Elf_Scn *sec, *sec_strndx; | 969 | Elf_Scn *sec, *sec_strndx, *opdsec; |
951 | Elf *elf; | 970 | Elf *elf; |
952 | int nr = 0; | 971 | int nr = 0; |
972 | size_t opdidx = 0; | ||
953 | 973 | ||
954 | elf = elf_begin(fd, PERF_ELF_C_READ_MMAP, NULL); | 974 | elf = elf_begin(fd, PERF_ELF_C_READ_MMAP, NULL); |
955 | if (elf == NULL) { | 975 | if (elf == NULL) { |
@@ -969,6 +989,10 @@ static int dso__load_sym(struct dso *self, struct map *map, const char *name, | |||
969 | goto out_elf_end; | 989 | goto out_elf_end; |
970 | } | 990 | } |
971 | 991 | ||
992 | opdsec = elf_section_by_name(elf, &ehdr, &opdshdr, ".opd", &opdidx); | ||
993 | if (opdsec) | ||
994 | opddata = elf_rawdata(opdsec, NULL); | ||
995 | |||
972 | syms = elf_getdata(sec, NULL); | 996 | syms = elf_getdata(sec, NULL); |
973 | if (syms == NULL) | 997 | if (syms == NULL) |
974 | goto out_elf_end; | 998 | goto out_elf_end; |
@@ -1013,6 +1037,13 @@ static int dso__load_sym(struct dso *self, struct map *map, const char *name, | |||
1013 | if (!is_label && !elf_sym__is_a(&sym, map->type)) | 1037 | if (!is_label && !elf_sym__is_a(&sym, map->type)) |
1014 | continue; | 1038 | continue; |
1015 | 1039 | ||
1040 | if (opdsec && sym.st_shndx == opdidx) { | ||
1041 | u32 offset = sym.st_value - opdshdr.sh_addr; | ||
1042 | u64 *opd = opddata->d_buf + offset; | ||
1043 | sym.st_value = *opd; | ||
1044 | sym.st_shndx = elf_addr_to_index(elf, sym.st_value); | ||
1045 | } | ||
1046 | |||
1016 | sec = elf_getscn(elf, sym.st_shndx); | 1047 | sec = elf_getscn(elf, sym.st_shndx); |
1017 | if (!sec) | 1048 | if (!sec) |
1018 | goto out_elf_end; | 1049 | goto out_elf_end; |