elf_sec__is_a and elf_sec__name maps a symbol back to its section name. It does this by getting the Elf_Scn *, which contains an offset into the strings section.
At the moment we use the Elf_Scn * from the runtime object and the section strings from the debuginfo object, which is wrong. They are not required to match, for example: # readelf --string-dump=.shstrtab /lib/x86_64-linux-gnu/libc-2.19.so | md5sum 55ec050a37f64db113b42979a18a40ef - # readelf --string-dump=.shstrtab /usr/lib/debug/lib/x86_64-linux-gnu/libc-2.19.so | md5sum 5ba4e37aae29b48418297f0c5b76b797 - This issue was first found on ppc64le, where PLT entries (which are labels with no type) were not getting resolved to symbols. This looks to have been introduced in 261360b6e90a (perf symbols: Convert dso__load_syms to take 2 symsrc's) Cc: sta...@vger.kernel.org # 3.8+ Signed-off-by: Anton Blanchard <an...@samba.org> --- Index: b/tools/perf/util/symbol-elf.c =================================================================== --- a/tools/perf/util/symbol-elf.c +++ b/tools/perf/util/symbol-elf.c @@ -569,6 +569,7 @@ int symsrc__init(struct symsrc *ss, stru GElf_Ehdr ehdr; Elf *elf; int fd; + Elf_Scn *sec_strndx; fd = open(name, O_RDONLY); if (fd < 0) @@ -618,6 +619,14 @@ int symsrc__init(struct symsrc *ss, stru if (ss->opdshdr.sh_type != SHT_PROGBITS) ss->opdsec = NULL; + sec_strndx = elf_getscn(elf, ehdr.e_shstrndx); + if (sec_strndx == NULL) + goto out_elf_end; + + ss->secstrs = elf_getdata(sec_strndx, NULL); + if (ss->secstrs == NULL) + goto out_elf_end; + if (dso->kernel == DSO_TYPE_USER) { GElf_Shdr shdr; ss->adjust_symbols = (ehdr.e_type == ET_EXEC || @@ -687,7 +696,7 @@ int dso__load_sym(struct dso *dso, struc struct kmap *kmap = dso->kernel ? map__kmap(map) : NULL; struct map *curr_map = map; struct dso *curr_dso = dso; - Elf_Data *symstrs, *secstrs; + Elf_Data *symstrs; uint32_t nr_syms; int err = -1; uint32_t idx; @@ -695,7 +704,7 @@ int dso__load_sym(struct dso *dso, struc GElf_Shdr shdr; Elf_Data *syms, *opddata = NULL; GElf_Sym sym; - Elf_Scn *sec, *sec_strndx; + Elf_Scn *sec; Elf *elf; int nr = 0; bool remap_kernel = false, adjust_kernel_syms = false; @@ -736,13 +745,6 @@ int dso__load_sym(struct dso *dso, struc if (symstrs == NULL) goto out_elf_end; - sec_strndx = elf_getscn(elf, ehdr.e_shstrndx); - if (sec_strndx == NULL) - goto out_elf_end; - - secstrs = elf_getdata(sec_strndx, NULL); - if (secstrs == NULL) - goto out_elf_end; nr_syms = shdr.sh_size / shdr.sh_entsize; @@ -821,10 +823,10 @@ int dso__load_sym(struct dso *dso, struc gelf_getshdr(sec, &shdr); - if (is_label && !elf_sec__is_a(&shdr, secstrs, map->type)) + if (is_label && !elf_sec__is_a(&shdr, runtime_ss->secstrs, map->type)) continue; - section_name = elf_sec__name(&shdr, secstrs); + section_name = elf_sec__name(&shdr, runtime_ss->secstrs); /* On ARM, symbols for thumb functions have 1 added to * the symbol address as a flag - remove it */ Index: b/tools/perf/util/symbol.h =================================================================== --- a/tools/perf/util/symbol.h +++ b/tools/perf/util/symbol.h @@ -215,6 +215,8 @@ struct symsrc { size_t dynsym_idx; GElf_Shdr dynshdr; + Elf_Data *secstrs; + bool adjust_symbols; bool is_64_bit; #endif -- To unsubscribe from this list: send the line "unsubscribe linux-kernel" in the body of a message to majord...@vger.kernel.org More majordomo info at http://vger.kernel.org/majordomo-info.html Please read the FAQ at http://www.tux.org/lkml/