Commit 3aafe5ae authored by Cody P Schafer's avatar Cody P Schafer Committed by Arnaldo Carvalho de Melo

perf symbols: Use both runtime and debug images

We keep both a 'runtime' elf image as well as a 'debug' elf image around
and generate symbols by looking at both of these.

This eliminates the need for the want_symtab/goto restart mechanism
combined with iterating over and reopening the elf images a second time.

Also give dso__synthsize_plt_symbols() the runtime image (which has
dynsyms) instead of the symbol image (which may only have a symtab and
no dynsyms).

Previously if a debug image was found all runtime images were ignored.

This fixes 2 issues:

 - Symbol resolution to failure on PowerPC systems with debug symbols
   installed, as the debug images lack a '.opd' section which contains
   function descriptors.

 - On all archs, plt synthesis failed when a debug image was loaded and
   that debug image lacks a dynsym section while a runtime image has a
   dynsym section.

Assumptions:

 - If a .opd section exists, it is contained in the highest priority
   image with a dynsym section.

 - This generally implies that the debug image lacks a dynsym section
   (ie: it is marked as NO_BITS).
Signed-off-by: default avatarCody P Schafer <cody@linux.vnet.ibm.com>
Cc: David Hansen <dave@linux.vnet.ibm.com>
Cc: Ingo Molnar <mingo@redhat.com>
Cc: Matt Hellsley <matthltc@us.ibm.com>
Cc: Namhyung Kim <namhyung@kernel.org>
Cc: Paul Mackerras <paulus@samba.org>
Cc: Peter Zijlstra <a.p.zijlstra@chello.nl>
Cc: Sukadev Bhattiprolu <sukadev@linux.vnet.ibm.com>
Link: http://lkml.kernel.org/r/1344637382-22789-17-git-send-email-cody@linux.vnet.ibm.comSigned-off-by: default avatarArnaldo Carvalho de Melo <acme@redhat.com>
parent 261360b6
...@@ -525,6 +525,11 @@ static int dso__swap_init(struct dso *dso, unsigned char eidata) ...@@ -525,6 +525,11 @@ static int dso__swap_init(struct dso *dso, unsigned char eidata)
return 0; return 0;
} }
bool symsrc__possibly_runtime(struct symsrc *ss)
{
return ss->dynsym || ss->opdsec;
}
bool symsrc__has_symtab(struct symsrc *ss) bool symsrc__has_symtab(struct symsrc *ss)
{ {
return ss->symtab != NULL; return ss->symtab != NULL;
......
...@@ -260,6 +260,12 @@ int symsrc__init(struct symsrc *ss, struct dso *dso __used, const char *name, ...@@ -260,6 +260,12 @@ int symsrc__init(struct symsrc *ss, struct dso *dso __used, const char *name,
return -1; return -1;
} }
bool symsrc__possibly_runtime(struct symsrc *ss __used)
{
/* Assume all sym sources could be a runtime image. */
return true;
}
bool symsrc__has_symtab(struct symsrc *ss __used) bool symsrc__has_symtab(struct symsrc *ss __used)
{ {
return false; return false;
......
...@@ -1026,11 +1026,12 @@ int dso__load(struct dso *dso, struct map *map, symbol_filter_t filter) ...@@ -1026,11 +1026,12 @@ int dso__load(struct dso *dso, struct map *map, symbol_filter_t filter)
{ {
char *name; char *name;
int ret = -1; int ret = -1;
struct symsrc ss;
u_int i; u_int i;
struct machine *machine; struct machine *machine;
char *root_dir = (char *) ""; char *root_dir = (char *) "";
int want_symtab; int ss_pos = 0;
struct symsrc ss_[2];
struct symsrc *syms_ss = NULL, *runtime_ss = NULL;
dso__set_loaded(dso, map->type); dso__set_loaded(dso, map->type);
...@@ -1072,12 +1073,12 @@ int dso__load(struct dso *dso, struct map *map, symbol_filter_t filter) ...@@ -1072,12 +1073,12 @@ int dso__load(struct dso *dso, struct map *map, symbol_filter_t filter)
root_dir = machine->root_dir; root_dir = machine->root_dir;
/* Iterate over candidate debug images. /* Iterate over candidate debug images.
* On the first pass, only load images if they have a full symtab. * Keep track of "interesting" ones (those which have a symtab, dynsym,
* Failing that, do a second pass where we accept .dynsym also * and/or opd section) for processing.
*/ */
want_symtab = 1;
restart:
for (i = 0; i < DSO_BINARY_TYPE__SYMTAB_CNT; i++) { for (i = 0; i < DSO_BINARY_TYPE__SYMTAB_CNT; i++) {
struct symsrc *ss = &ss_[ss_pos];
bool next_slot = false;
enum dso_binary_type symtab_type = binary_type_symtab[i]; enum dso_binary_type symtab_type = binary_type_symtab[i];
...@@ -1086,45 +1087,55 @@ int dso__load(struct dso *dso, struct map *map, symbol_filter_t filter) ...@@ -1086,45 +1087,55 @@ int dso__load(struct dso *dso, struct map *map, symbol_filter_t filter)
continue; continue;
/* Name is now the name of the next image to try */ /* Name is now the name of the next image to try */
if (symsrc__init(&ss, dso, name, symtab_type) < 0) if (symsrc__init(ss, dso, name, symtab_type) < 0)
continue; continue;
if (want_symtab && !symsrc__has_symtab(&ss)) { if (!syms_ss && symsrc__has_symtab(ss)) {
symsrc__destroy(&ss); syms_ss = ss;
continue; next_slot = true;
} }
ret = dso__load_sym(dso, map, &ss, &ss, filter, 0); if (!runtime_ss && symsrc__possibly_runtime(ss)) {
runtime_ss = ss;
/* next_slot = true;
* Some people seem to have debuginfo files _WITHOUT_ debug
* info!?!?
*/
if (!ret) {
symsrc__destroy(&ss);
continue;
} }
if (ret > 0) { if (next_slot) {
int nr_plt; ss_pos++;
nr_plt = dso__synthesize_plt_symbols(dso, &ss, map, filter); if (syms_ss && runtime_ss)
if (nr_plt > 0)
ret += nr_plt;
symsrc__destroy(&ss);
break; break;
} }
} }
/* if (!runtime_ss && !syms_ss)
* If we wanted a full symtab but no image had one, goto out_free;
* relax our requirements and repeat the search.
*/ if (runtime_ss && !syms_ss) {
if (ret <= 0 && want_symtab) { syms_ss = runtime_ss;
want_symtab = 0; }
goto restart;
/* We'll have to hope for the best */
if (!runtime_ss && syms_ss)
runtime_ss = syms_ss;
if (syms_ss)
ret = dso__load_sym(dso, map, syms_ss, runtime_ss, filter, 0);
else
ret = -1;
if (ret > 0 && runtime_ss->dynsym) {
int nr_plt;
nr_plt = dso__synthesize_plt_symbols(dso, runtime_ss, map, filter);
if (nr_plt > 0)
ret += nr_plt;
} }
for (; ss_pos > 0; ss_pos--)
symsrc__destroy(&ss_[ss_pos - 1]);
out_free:
free(name); free(name);
if (ret < 0 && strstr(dso->name, " (deleted)") != NULL) if (ret < 0 && strstr(dso->name, " (deleted)") != NULL)
return 0; return 0;
......
...@@ -253,6 +253,7 @@ void symsrc__destroy(struct symsrc *ss); ...@@ -253,6 +253,7 @@ void symsrc__destroy(struct symsrc *ss);
int symsrc__init(struct symsrc *ss, struct dso *dso, const char *name, int symsrc__init(struct symsrc *ss, struct dso *dso, const char *name,
enum dso_binary_type type); enum dso_binary_type type);
bool symsrc__has_symtab(struct symsrc *ss); bool symsrc__has_symtab(struct symsrc *ss);
bool symsrc__possibly_runtime(struct symsrc *ss);
#define DSO__SWAP(dso, type, val) \ #define DSO__SWAP(dso, type, val) \
({ \ ({ \
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment