perf buildid-cache: Cache debuginfo
If a stripped binary is placed in the cache, the user is in a situation where there's a cached elf file present, but it doesn't have any symtab to use for name resolution. Grab the debuginfo for binaries that don't end in .ko. This yields a better chance of resolving symbols from older traces. Signed-off-by: Krister Johansen <kjlx@templeofstupid.com> Cc: Alexander Shishkin <alexander.shishkin@linux.intel.com> Cc: Brendan Gregg <brendan.d.gregg@gmail.com> Cc: Peter Zijlstra <peterz@infradead.org> Cc: Thomas-Mich Richter <tmricht@linux.vnet.ibm.com> Link: http://lkml.kernel.org/r/1499305693-1599-7-git-send-email-kjlx@templeofstupid.com Signed-off-by: Arnaldo Carvalho de Melo <acme@redhat.com>
This commit is contained in:
committed by
Arnaldo Carvalho de Melo
parent
f045b8c4b3
commit
d2396999c9
@@ -243,7 +243,7 @@ static bool dso__missing_buildid_cache(struct dso *dso, int parm __maybe_unused)
|
|||||||
char filename[PATH_MAX];
|
char filename[PATH_MAX];
|
||||||
u8 build_id[BUILD_ID_SIZE];
|
u8 build_id[BUILD_ID_SIZE];
|
||||||
|
|
||||||
if (dso__build_id_filename(dso, filename, sizeof(filename)) &&
|
if (dso__build_id_filename(dso, filename, sizeof(filename), false) &&
|
||||||
filename__read_build_id(filename, build_id,
|
filename__read_build_id(filename, build_id,
|
||||||
sizeof(build_id)) != sizeof(build_id)) {
|
sizeof(build_id)) != sizeof(build_id)) {
|
||||||
if (errno == ENOENT)
|
if (errno == ENOENT)
|
||||||
|
|||||||
@@ -1347,7 +1347,7 @@ static int dso__disassemble_filename(struct dso *dso, char *filename, size_t fil
|
|||||||
!dso__is_kcore(dso))
|
!dso__is_kcore(dso))
|
||||||
return SYMBOL_ANNOTATE_ERRNO__NO_VMLINUX;
|
return SYMBOL_ANNOTATE_ERRNO__NO_VMLINUX;
|
||||||
|
|
||||||
build_id_filename = dso__build_id_filename(dso, NULL, 0);
|
build_id_filename = dso__build_id_filename(dso, NULL, 0, false);
|
||||||
if (build_id_filename) {
|
if (build_id_filename) {
|
||||||
__symbol__join_symfs(filename, filename_size, build_id_filename);
|
__symbol__join_symfs(filename, filename_size, build_id_filename);
|
||||||
free(build_id_filename);
|
free(build_id_filename);
|
||||||
|
|||||||
@@ -243,12 +243,15 @@ static bool build_id_cache__valid_id(char *sbuild_id)
|
|||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
static const char *build_id_cache__basename(bool is_kallsyms, bool is_vdso)
|
static const char *build_id_cache__basename(bool is_kallsyms, bool is_vdso,
|
||||||
|
bool is_debug)
|
||||||
{
|
{
|
||||||
return is_kallsyms ? "kallsyms" : (is_vdso ? "vdso" : "elf");
|
return is_kallsyms ? "kallsyms" : (is_vdso ? "vdso" : (is_debug ?
|
||||||
|
"debug" : "elf"));
|
||||||
}
|
}
|
||||||
|
|
||||||
char *dso__build_id_filename(const struct dso *dso, char *bf, size_t size)
|
char *dso__build_id_filename(const struct dso *dso, char *bf, size_t size,
|
||||||
|
bool is_debug)
|
||||||
{
|
{
|
||||||
bool is_kallsyms = dso__is_kallsyms((struct dso *)dso);
|
bool is_kallsyms = dso__is_kallsyms((struct dso *)dso);
|
||||||
bool is_vdso = dso__is_vdso((struct dso *)dso);
|
bool is_vdso = dso__is_vdso((struct dso *)dso);
|
||||||
@@ -270,7 +273,8 @@ char *dso__build_id_filename(const struct dso *dso, char *bf, size_t size)
|
|||||||
ret = asnprintf(&bf, size, "%s", linkname);
|
ret = asnprintf(&bf, size, "%s", linkname);
|
||||||
else
|
else
|
||||||
ret = asnprintf(&bf, size, "%s/%s", linkname,
|
ret = asnprintf(&bf, size, "%s/%s", linkname,
|
||||||
build_id_cache__basename(is_kallsyms, is_vdso));
|
build_id_cache__basename(is_kallsyms, is_vdso,
|
||||||
|
is_debug));
|
||||||
if (ret < 0 || (!alloc && size < (unsigned int)ret))
|
if (ret < 0 || (!alloc && size < (unsigned int)ret))
|
||||||
bf = NULL;
|
bf = NULL;
|
||||||
free(linkname);
|
free(linkname);
|
||||||
@@ -603,12 +607,40 @@ static int build_id_cache__add_sdt_cache(const char *sbuild_id,
|
|||||||
#define build_id_cache__add_sdt_cache(sbuild_id, realname, nsi) (0)
|
#define build_id_cache__add_sdt_cache(sbuild_id, realname, nsi) (0)
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
static char *build_id_cache__find_debug(const char *sbuild_id,
|
||||||
|
struct nsinfo *nsi)
|
||||||
|
{
|
||||||
|
char *realname = NULL;
|
||||||
|
char *debugfile;
|
||||||
|
struct nscookie nsc;
|
||||||
|
size_t len = 0;
|
||||||
|
|
||||||
|
debugfile = calloc(1, PATH_MAX);
|
||||||
|
if (!debugfile)
|
||||||
|
goto out;
|
||||||
|
|
||||||
|
len = __symbol__join_symfs(debugfile, PATH_MAX,
|
||||||
|
"/usr/lib/debug/.build-id/");
|
||||||
|
snprintf(debugfile + len, PATH_MAX - len, "%.2s/%s.debug", sbuild_id,
|
||||||
|
sbuild_id + 2);
|
||||||
|
|
||||||
|
nsinfo__mountns_enter(nsi, &nsc);
|
||||||
|
realname = realpath(debugfile, NULL);
|
||||||
|
if (realname && access(realname, R_OK))
|
||||||
|
zfree(&realname);
|
||||||
|
nsinfo__mountns_exit(&nsc);
|
||||||
|
out:
|
||||||
|
free(debugfile);
|
||||||
|
return realname;
|
||||||
|
}
|
||||||
|
|
||||||
int build_id_cache__add_s(const char *sbuild_id, const char *name,
|
int build_id_cache__add_s(const char *sbuild_id, const char *name,
|
||||||
struct nsinfo *nsi, bool is_kallsyms, bool is_vdso)
|
struct nsinfo *nsi, bool is_kallsyms, bool is_vdso)
|
||||||
{
|
{
|
||||||
const size_t size = PATH_MAX;
|
const size_t size = PATH_MAX;
|
||||||
char *realname = NULL, *filename = NULL, *dir_name = NULL,
|
char *realname = NULL, *filename = NULL, *dir_name = NULL,
|
||||||
*linkname = zalloc(size), *tmp;
|
*linkname = zalloc(size), *tmp;
|
||||||
|
char *debugfile = NULL;
|
||||||
int err = -1;
|
int err = -1;
|
||||||
|
|
||||||
if (!is_kallsyms) {
|
if (!is_kallsyms) {
|
||||||
@@ -635,7 +667,8 @@ int build_id_cache__add_s(const char *sbuild_id, const char *name,
|
|||||||
|
|
||||||
/* Save the allocated buildid dirname */
|
/* Save the allocated buildid dirname */
|
||||||
if (asprintf(&filename, "%s/%s", dir_name,
|
if (asprintf(&filename, "%s/%s", dir_name,
|
||||||
build_id_cache__basename(is_kallsyms, is_vdso)) < 0) {
|
build_id_cache__basename(is_kallsyms, is_vdso,
|
||||||
|
false)) < 0) {
|
||||||
filename = NULL;
|
filename = NULL;
|
||||||
goto out_free;
|
goto out_free;
|
||||||
}
|
}
|
||||||
@@ -652,6 +685,34 @@ int build_id_cache__add_s(const char *sbuild_id, const char *name,
|
|||||||
goto out_free;
|
goto out_free;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Some binaries are stripped, but have .debug files with their symbol
|
||||||
|
* table. Check to see if we can locate one of those, since the elf
|
||||||
|
* file itself may not be very useful to users of our tools without a
|
||||||
|
* symtab.
|
||||||
|
*/
|
||||||
|
if (!is_kallsyms && !is_vdso &&
|
||||||
|
strncmp(".ko", name + strlen(name) - 3, 3)) {
|
||||||
|
debugfile = build_id_cache__find_debug(sbuild_id, nsi);
|
||||||
|
if (debugfile) {
|
||||||
|
zfree(&filename);
|
||||||
|
if (asprintf(&filename, "%s/%s", dir_name,
|
||||||
|
build_id_cache__basename(false, false, true)) < 0) {
|
||||||
|
filename = NULL;
|
||||||
|
goto out_free;
|
||||||
|
}
|
||||||
|
if (access(filename, F_OK)) {
|
||||||
|
if (nsi && nsi->need_setns) {
|
||||||
|
if (copyfile_ns(debugfile, filename,
|
||||||
|
nsi))
|
||||||
|
goto out_free;
|
||||||
|
} else if (link(debugfile, filename) &&
|
||||||
|
errno != EEXIST &&
|
||||||
|
copyfile(debugfile, filename))
|
||||||
|
goto out_free;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (!build_id_cache__linkname(sbuild_id, linkname, size))
|
if (!build_id_cache__linkname(sbuild_id, linkname, size))
|
||||||
goto out_free;
|
goto out_free;
|
||||||
tmp = strrchr(linkname, '/');
|
tmp = strrchr(linkname, '/');
|
||||||
@@ -676,6 +737,7 @@ out_free:
|
|||||||
if (!is_kallsyms)
|
if (!is_kallsyms)
|
||||||
free(realname);
|
free(realname);
|
||||||
free(filename);
|
free(filename);
|
||||||
|
free(debugfile);
|
||||||
free(dir_name);
|
free(dir_name);
|
||||||
free(linkname);
|
free(linkname);
|
||||||
return err;
|
return err;
|
||||||
|
|||||||
@@ -17,7 +17,8 @@ int filename__sprintf_build_id(const char *pathname, char *sbuild_id);
|
|||||||
char *build_id_cache__kallsyms_path(const char *sbuild_id, char *bf,
|
char *build_id_cache__kallsyms_path(const char *sbuild_id, char *bf,
|
||||||
size_t size);
|
size_t size);
|
||||||
|
|
||||||
char *dso__build_id_filename(const struct dso *dso, char *bf, size_t size);
|
char *dso__build_id_filename(const struct dso *dso, char *bf, size_t size,
|
||||||
|
bool is_debug);
|
||||||
|
|
||||||
int build_id__mark_dso_hit(struct perf_tool *tool, union perf_event *event,
|
int build_id__mark_dso_hit(struct perf_tool *tool, union perf_event *event,
|
||||||
struct perf_sample *sample, struct perf_evsel *evsel,
|
struct perf_sample *sample, struct perf_evsel *evsel,
|
||||||
|
|||||||
@@ -32,6 +32,7 @@ char dso__symtab_origin(const struct dso *dso)
|
|||||||
[DSO_BINARY_TYPE__JAVA_JIT] = 'j',
|
[DSO_BINARY_TYPE__JAVA_JIT] = 'j',
|
||||||
[DSO_BINARY_TYPE__DEBUGLINK] = 'l',
|
[DSO_BINARY_TYPE__DEBUGLINK] = 'l',
|
||||||
[DSO_BINARY_TYPE__BUILD_ID_CACHE] = 'B',
|
[DSO_BINARY_TYPE__BUILD_ID_CACHE] = 'B',
|
||||||
|
[DSO_BINARY_TYPE__BUILD_ID_CACHE_DEBUGINFO] = 'D',
|
||||||
[DSO_BINARY_TYPE__FEDORA_DEBUGINFO] = 'f',
|
[DSO_BINARY_TYPE__FEDORA_DEBUGINFO] = 'f',
|
||||||
[DSO_BINARY_TYPE__UBUNTU_DEBUGINFO] = 'u',
|
[DSO_BINARY_TYPE__UBUNTU_DEBUGINFO] = 'u',
|
||||||
[DSO_BINARY_TYPE__OPENEMBEDDED_DEBUGINFO] = 'o',
|
[DSO_BINARY_TYPE__OPENEMBEDDED_DEBUGINFO] = 'o',
|
||||||
@@ -97,7 +98,12 @@ int dso__read_binary_type_filename(const struct dso *dso,
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case DSO_BINARY_TYPE__BUILD_ID_CACHE:
|
case DSO_BINARY_TYPE__BUILD_ID_CACHE:
|
||||||
if (dso__build_id_filename(dso, filename, size) == NULL)
|
if (dso__build_id_filename(dso, filename, size, false) == NULL)
|
||||||
|
ret = -1;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case DSO_BINARY_TYPE__BUILD_ID_CACHE_DEBUGINFO:
|
||||||
|
if (dso__build_id_filename(dso, filename, size, true) == NULL)
|
||||||
ret = -1;
|
ret = -1;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
|||||||
@@ -21,6 +21,7 @@ enum dso_binary_type {
|
|||||||
DSO_BINARY_TYPE__JAVA_JIT,
|
DSO_BINARY_TYPE__JAVA_JIT,
|
||||||
DSO_BINARY_TYPE__DEBUGLINK,
|
DSO_BINARY_TYPE__DEBUGLINK,
|
||||||
DSO_BINARY_TYPE__BUILD_ID_CACHE,
|
DSO_BINARY_TYPE__BUILD_ID_CACHE,
|
||||||
|
DSO_BINARY_TYPE__BUILD_ID_CACHE_DEBUGINFO,
|
||||||
DSO_BINARY_TYPE__FEDORA_DEBUGINFO,
|
DSO_BINARY_TYPE__FEDORA_DEBUGINFO,
|
||||||
DSO_BINARY_TYPE__UBUNTU_DEBUGINFO,
|
DSO_BINARY_TYPE__UBUNTU_DEBUGINFO,
|
||||||
DSO_BINARY_TYPE__BUILDID_DEBUGINFO,
|
DSO_BINARY_TYPE__BUILDID_DEBUGINFO,
|
||||||
|
|||||||
@@ -705,7 +705,8 @@ size_t machine__fprintf_vmlinux_path(struct machine *machine, FILE *fp)
|
|||||||
|
|
||||||
if (kdso->has_build_id) {
|
if (kdso->has_build_id) {
|
||||||
char filename[PATH_MAX];
|
char filename[PATH_MAX];
|
||||||
if (dso__build_id_filename(kdso, filename, sizeof(filename)))
|
if (dso__build_id_filename(kdso, filename, sizeof(filename),
|
||||||
|
false))
|
||||||
printed += fprintf(fp, "[0] %s\n", filename);
|
printed += fprintf(fp, "[0] %s\n", filename);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -53,6 +53,7 @@ static enum dso_binary_type binary_type_symtab[] = {
|
|||||||
DSO_BINARY_TYPE__JAVA_JIT,
|
DSO_BINARY_TYPE__JAVA_JIT,
|
||||||
DSO_BINARY_TYPE__DEBUGLINK,
|
DSO_BINARY_TYPE__DEBUGLINK,
|
||||||
DSO_BINARY_TYPE__BUILD_ID_CACHE,
|
DSO_BINARY_TYPE__BUILD_ID_CACHE,
|
||||||
|
DSO_BINARY_TYPE__BUILD_ID_CACHE_DEBUGINFO,
|
||||||
DSO_BINARY_TYPE__FEDORA_DEBUGINFO,
|
DSO_BINARY_TYPE__FEDORA_DEBUGINFO,
|
||||||
DSO_BINARY_TYPE__UBUNTU_DEBUGINFO,
|
DSO_BINARY_TYPE__UBUNTU_DEBUGINFO,
|
||||||
DSO_BINARY_TYPE__BUILDID_DEBUGINFO,
|
DSO_BINARY_TYPE__BUILDID_DEBUGINFO,
|
||||||
@@ -1418,6 +1419,7 @@ static bool dso__is_compatible_symtab_type(struct dso *dso, bool kmod,
|
|||||||
return kmod && dso->symtab_type == type;
|
return kmod && dso->symtab_type == type;
|
||||||
|
|
||||||
case DSO_BINARY_TYPE__BUILD_ID_CACHE:
|
case DSO_BINARY_TYPE__BUILD_ID_CACHE:
|
||||||
|
case DSO_BINARY_TYPE__BUILD_ID_CACHE_DEBUGINFO:
|
||||||
return true;
|
return true;
|
||||||
|
|
||||||
case DSO_BINARY_TYPE__NOT_FOUND:
|
case DSO_BINARY_TYPE__NOT_FOUND:
|
||||||
@@ -1565,10 +1567,14 @@ int dso__load(struct dso *dso, struct map *map)
|
|||||||
struct symsrc *ss = &ss_[ss_pos];
|
struct symsrc *ss = &ss_[ss_pos];
|
||||||
bool next_slot = false;
|
bool next_slot = false;
|
||||||
bool is_reg;
|
bool is_reg;
|
||||||
|
bool nsexit;
|
||||||
int sirc;
|
int sirc;
|
||||||
|
|
||||||
enum dso_binary_type symtab_type = binary_type_symtab[i];
|
enum dso_binary_type symtab_type = binary_type_symtab[i];
|
||||||
|
|
||||||
|
nsexit = (symtab_type == DSO_BINARY_TYPE__BUILD_ID_CACHE ||
|
||||||
|
symtab_type == DSO_BINARY_TYPE__BUILD_ID_CACHE_DEBUGINFO);
|
||||||
|
|
||||||
if (!dso__is_compatible_symtab_type(dso, kmod, symtab_type))
|
if (!dso__is_compatible_symtab_type(dso, kmod, symtab_type))
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
@@ -1576,13 +1582,13 @@ int dso__load(struct dso *dso, struct map *map)
|
|||||||
root_dir, name, PATH_MAX))
|
root_dir, name, PATH_MAX))
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
if (symtab_type == DSO_BINARY_TYPE__BUILD_ID_CACHE)
|
if (nsexit)
|
||||||
nsinfo__mountns_exit(&nsc);
|
nsinfo__mountns_exit(&nsc);
|
||||||
|
|
||||||
is_reg = is_regular_file(name);
|
is_reg = is_regular_file(name);
|
||||||
sirc = symsrc__init(ss, dso, name, symtab_type);
|
sirc = symsrc__init(ss, dso, name, symtab_type);
|
||||||
|
|
||||||
if (symtab_type == DSO_BINARY_TYPE__BUILD_ID_CACHE)
|
if (nsexit)
|
||||||
nsinfo__mountns_enter(dso->nsinfo, &nsc);
|
nsinfo__mountns_enter(dso->nsinfo, &nsc);
|
||||||
|
|
||||||
if (!is_reg || sirc < 0) {
|
if (!is_reg || sirc < 0) {
|
||||||
@@ -1724,7 +1730,7 @@ int dso__load_vmlinux_path(struct dso *dso, struct map *map)
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (!symbol_conf.ignore_vmlinux_buildid)
|
if (!symbol_conf.ignore_vmlinux_buildid)
|
||||||
filename = dso__build_id_filename(dso, NULL, 0);
|
filename = dso__build_id_filename(dso, NULL, 0, false);
|
||||||
if (filename != NULL) {
|
if (filename != NULL) {
|
||||||
err = dso__load_vmlinux(dso, map, filename, true);
|
err = dso__load_vmlinux(dso, map, filename, true);
|
||||||
if (err > 0)
|
if (err > 0)
|
||||||
|
|||||||
Reference in New Issue
Block a user