Skip to content

Commit

Permalink
perf tools: Check recorded kernel version when finding vmlinux
Browse files Browse the repository at this point in the history
Currently vmlinux_path__init() only tries to find vmlinux file from
current directory, /boot and some canonical directories with version
number of the running kernel.  This can be a problem when reporting old
data recorded on a kernel version not running currently.

We can use --symfs option for this but it's annoying for user to do it
always.  As we already have the info in the perf.data file, it can be
changed to use it for the search automatically.

Before:

  $ perf report
  ...
  # Samples: 4K of event 'cpu-clock'
  # Event count (approx.): 1067250000
  #
  # Overhead  Command     Shared Object      Symbol
  # ........  ..........  .................  ..............................
      71.87%     swapper  [kernel.kallsyms]  [k] recover_probed_instruction

After:

  # Overhead  Command     Shared Object      Symbol
  # ........  ..........  .................  ....................
      71.87%     swapper  [kernel.kallsyms]  [k] native_safe_halt

This requires to change signature of symbol__init() to receive struct
perf_session_env *.

Reported-by: Minchan Kim <minchan@kernel.org>
Signed-off-by: Namhyung Kim <namhyung@kernel.org>
Cc: Adrian Hunter <adrian.hunter@intel.com>
Cc: David Ahern <dsahern@gmail.com>
Cc: Ingo Molnar <mingo@kernel.org>
Cc: Jiri Olsa <jolsa@redhat.com>
Cc: Masami Hiramatsu <masami.hiramatsu.pt@hitachi.com>
Cc: Minchan Kim <minchan@kernel.org>
Cc: Namhyung Kim <namhyung.kim@lge.com>
Cc: Paul Mackerras <paulus@samba.org>
Cc: Peter Zijlstra <a.p.zijlstra@chello.nl>
Cc: Stephane Eranian <eranian@google.com>
Link: http://lkml.kernel.org/r/1407825645-24586-14-git-send-email-namhyung@kernel.org
Signed-off-by: Arnaldo Carvalho de Melo <acme@redhat.com>
  • Loading branch information
Namhyung Kim authored and Arnaldo Carvalho de Melo committed Aug 13, 2014
1 parent cb2ffae commit 0a7e6d1
Show file tree
Hide file tree
Showing 19 changed files with 39 additions and 30 deletions.
2 changes: 1 addition & 1 deletion tools/perf/builtin-annotate.c
Original file line number Diff line number Diff line change
Expand Up @@ -345,7 +345,7 @@ int cmd_annotate(int argc, const char **argv, const char *prefix __maybe_unused)
symbol_conf.priv_size = sizeof(struct annotation);
symbol_conf.try_vmlinux_path = true;

ret = symbol__init();
ret = symbol__init(&annotate.session->header.env);
if (ret < 0)
goto out_delete;

Expand Down
2 changes: 1 addition & 1 deletion tools/perf/builtin-buildid-cache.c
Original file line number Diff line number Diff line change
Expand Up @@ -329,7 +329,7 @@ int cmd_buildid_cache(int argc, const char **argv,
return -1;
}

if (symbol__init() < 0)
if (symbol__init(session ? &session->header.env : NULL) < 0)
goto out;

setup_pager();
Expand Down
2 changes: 1 addition & 1 deletion tools/perf/builtin-diff.c
Original file line number Diff line number Diff line change
Expand Up @@ -1143,7 +1143,7 @@ int cmd_diff(int argc, const char **argv, const char *prefix __maybe_unused)

argc = parse_options(argc, argv, options, diff_usage, 0);

if (symbol__init() < 0)
if (symbol__init(NULL) < 0)
return -1;

if (data_init(argc, argv) < 0)
Expand Down
2 changes: 1 addition & 1 deletion tools/perf/builtin-inject.c
Original file line number Diff line number Diff line change
Expand Up @@ -462,7 +462,7 @@ int cmd_inject(int argc, const char **argv, const char *prefix __maybe_unused)
if (inject.session == NULL)
return -ENOMEM;

if (symbol__init() < 0)
if (symbol__init(&inject.session->header.env) < 0)
return -1;

ret = __cmd_inject(&inject);
Expand Down
4 changes: 2 additions & 2 deletions tools/perf/builtin-kmem.c
Original file line number Diff line number Diff line change
Expand Up @@ -692,15 +692,15 @@ int cmd_kmem(int argc, const char **argv, const char *prefix __maybe_unused)
usage_with_options(kmem_usage, kmem_options);

if (!strncmp(argv[0], "rec", 3)) {
symbol__init();
symbol__init(NULL);
return __cmd_record(argc, argv);
}

session = perf_session__new(&file, false, &perf_kmem);
if (session == NULL)
return -ENOMEM;

symbol__init();
symbol__init(&session->header.env);

if (!strcmp(argv[0], "stat")) {
if (cpu__setup_cpunode_map())
Expand Down
4 changes: 2 additions & 2 deletions tools/perf/builtin-kvm.c
Original file line number Diff line number Diff line change
Expand Up @@ -1064,7 +1064,7 @@ static int read_events(struct perf_kvm_stat *kvm)
return -EINVAL;
}

symbol__init();
symbol__init(&kvm->session->header.env);

if (!perf_session__has_traces(kvm->session, "kvm record"))
return -EINVAL;
Expand Down Expand Up @@ -1314,7 +1314,7 @@ static int kvm_events_live(struct perf_kvm_stat *kvm,
kvm->opts.target.uid_str = NULL;
kvm->opts.target.uid = UINT_MAX;

symbol__init();
symbol__init(NULL);
disable_buildid_cache();

use_browser = 0;
Expand Down
2 changes: 1 addition & 1 deletion tools/perf/builtin-lock.c
Original file line number Diff line number Diff line change
Expand Up @@ -865,7 +865,7 @@ static int __cmd_report(bool display_info)
return -ENOMEM;
}

symbol__init();
symbol__init(&session->header.env);

if (!perf_session__has_traces(session, "lock record"))
goto out_delete;
Expand Down
2 changes: 1 addition & 1 deletion tools/perf/builtin-mem.c
Original file line number Diff line number Diff line change
Expand Up @@ -133,7 +133,7 @@ static int report_raw_events(struct perf_mem *mem)
goto out_delete;
}

if (symbol__init() < 0)
if (symbol__init(&session->header.env) < 0)
return -1;

printf("# PID, TID, IP, ADDR, LOCAL WEIGHT, DSRC, SYMBOL\n");
Expand Down
2 changes: 1 addition & 1 deletion tools/perf/builtin-record.c
Original file line number Diff line number Diff line change
Expand Up @@ -908,7 +908,7 @@ int cmd_record(int argc, const char **argv, const char *prefix __maybe_unused)
usage_with_options(record_usage, record_options);
}

symbol__init();
symbol__init(NULL);

if (symbol_conf.kptr_restrict)
pr_warning(
Expand Down
2 changes: 1 addition & 1 deletion tools/perf/builtin-report.c
Original file line number Diff line number Diff line change
Expand Up @@ -798,7 +798,7 @@ int cmd_report(int argc, const char **argv, const char *prefix __maybe_unused)
}
}

if (symbol__init() < 0)
if (symbol__init(&session->header.env) < 0)
goto error;

if (argc) {
Expand Down
2 changes: 1 addition & 1 deletion tools/perf/builtin-sched.c
Original file line number Diff line number Diff line change
Expand Up @@ -1462,7 +1462,7 @@ static int perf_sched__read_events(struct perf_sched *sched,
return -1;
}

symbol__init();
symbol__init(&session->header.env);

if (perf_session__set_tracepoints_handlers(session, handlers))
goto out_delete;
Expand Down
2 changes: 1 addition & 1 deletion tools/perf/builtin-script.c
Original file line number Diff line number Diff line change
Expand Up @@ -1732,7 +1732,7 @@ int cmd_script(int argc, const char **argv, const char *prefix __maybe_unused)
goto out_delete;
}

if (symbol__init() < 0)
if (symbol__init(&session->header.env) < 0)
goto out_delete;

script.session = session;
Expand Down
2 changes: 1 addition & 1 deletion tools/perf/builtin-timechart.c
Original file line number Diff line number Diff line change
Expand Up @@ -1607,7 +1607,7 @@ static int __cmd_timechart(struct timechart *tchart, const char *output_name)
if (session == NULL)
return -ENOMEM;

symbol__init();
symbol__init(&session->header.env);

(void)perf_header__process_sections(&session->header,
perf_data_file__fd(session->file),
Expand Down
2 changes: 1 addition & 1 deletion tools/perf/builtin-top.c
Original file line number Diff line number Diff line change
Expand Up @@ -1234,7 +1234,7 @@ int cmd_top(int argc, const char **argv, const char *prefix __maybe_unused)
symbol_conf.priv_size = sizeof(struct annotation);

symbol_conf.try_vmlinux_path = (symbol_conf.vmlinux_name == NULL);
if (symbol__init() < 0)
if (symbol__init(NULL) < 0)
return -1;

sort__setup_elide(stdout);
Expand Down
4 changes: 2 additions & 2 deletions tools/perf/builtin-trace.c
Original file line number Diff line number Diff line change
Expand Up @@ -1411,7 +1411,7 @@ static int trace__tool_process(struct perf_tool *tool,

static int trace__symbols_init(struct trace *trace, struct perf_evlist *evlist)
{
int err = symbol__init();
int err = symbol__init(NULL);

if (err)
return err;
Expand Down Expand Up @@ -2245,7 +2245,7 @@ static int trace__replay(struct trace *trace)
if (session == NULL)
return -ENOMEM;

if (symbol__init() < 0)
if (symbol__init(&session->header.env) < 0)
goto out;

trace->host = &session->machines.host;
Expand Down
2 changes: 1 addition & 1 deletion tools/perf/tests/builtin-test.c
Original file line number Diff line number Diff line change
Expand Up @@ -297,7 +297,7 @@ int cmd_test(int argc, const char **argv, const char *prefix __maybe_unused)
symbol_conf.sort_by_name = true;
symbol_conf.try_vmlinux_path = true;

if (symbol__init() < 0)
if (symbol__init(NULL) < 0)
return -1;

if (skip != NULL)
Expand Down
2 changes: 1 addition & 1 deletion tools/perf/util/probe-event.c
Original file line number Diff line number Diff line change
Expand Up @@ -79,7 +79,7 @@ static int init_symbol_maps(bool user_only)
int ret;

symbol_conf.sort_by_name = true;
ret = symbol__init();
ret = symbol__init(NULL);
if (ret < 0) {
pr_debug("Failed to init symbol map.\n");
goto out;
Expand Down
26 changes: 17 additions & 9 deletions tools/perf/util/symbol.c
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,7 @@
#include "machine.h"
#include "symbol.h"
#include "strlist.h"
#include "header.h"

#include <elf.h>
#include <limits.h>
Expand Down Expand Up @@ -1749,10 +1750,11 @@ static void vmlinux_path__exit(void)
zfree(&vmlinux_path);
}

static int vmlinux_path__init(void)
static int vmlinux_path__init(struct perf_session_env *env)
{
struct utsname uts;
char bf[PATH_MAX];
char *kernel_version;

vmlinux_path = malloc(sizeof(char *) * 5);
if (vmlinux_path == NULL)
Expand All @@ -1767,25 +1769,31 @@ static int vmlinux_path__init(void)
goto out_fail;
++vmlinux_path__nr_entries;

/* only try running kernel version if no symfs was given */
/* only try kernel version if no symfs was given */
if (symbol_conf.symfs[0] != 0)
return 0;

if (uname(&uts) < 0)
goto out_fail;
if (env) {
kernel_version = env->os_release;
} else {
if (uname(&uts) < 0)
goto out_fail;

kernel_version = uts.release;
}

snprintf(bf, sizeof(bf), "/boot/vmlinux-%s", uts.release);
snprintf(bf, sizeof(bf), "/boot/vmlinux-%s", kernel_version);
vmlinux_path[vmlinux_path__nr_entries] = strdup(bf);
if (vmlinux_path[vmlinux_path__nr_entries] == NULL)
goto out_fail;
++vmlinux_path__nr_entries;
snprintf(bf, sizeof(bf), "/lib/modules/%s/build/vmlinux", uts.release);
snprintf(bf, sizeof(bf), "/lib/modules/%s/build/vmlinux", kernel_version);
vmlinux_path[vmlinux_path__nr_entries] = strdup(bf);
if (vmlinux_path[vmlinux_path__nr_entries] == NULL)
goto out_fail;
++vmlinux_path__nr_entries;
snprintf(bf, sizeof(bf), "/usr/lib/debug/lib/modules/%s/vmlinux",
uts.release);
kernel_version);
vmlinux_path[vmlinux_path__nr_entries] = strdup(bf);
if (vmlinux_path[vmlinux_path__nr_entries] == NULL)
goto out_fail;
Expand Down Expand Up @@ -1831,7 +1839,7 @@ static bool symbol__read_kptr_restrict(void)
return value;
}

int symbol__init(void)
int symbol__init(struct perf_session_env *env)
{
const char *symfs;

Expand All @@ -1846,7 +1854,7 @@ int symbol__init(void)
symbol_conf.priv_size += (sizeof(struct symbol_name_rb_node) -
sizeof(struct symbol));

if (symbol_conf.try_vmlinux_path && vmlinux_path__init() < 0)
if (symbol_conf.try_vmlinux_path && vmlinux_path__init(env) < 0)
return -1;

if (symbol_conf.field_sep && *symbol_conf.field_sep == '.') {
Expand Down
3 changes: 2 additions & 1 deletion tools/perf/util/symbol.h
Original file line number Diff line number Diff line change
Expand Up @@ -262,7 +262,8 @@ int modules__parse(const char *filename, void *arg,
int filename__read_debuglink(const char *filename, char *debuglink,
size_t size);

int symbol__init(void);
struct perf_session_env;
int symbol__init(struct perf_session_env *env);
void symbol__exit(void);
void symbol__elf_init(void);
struct symbol *symbol__new(u64 start, u64 len, u8 binding, const char *name);
Expand Down

0 comments on commit 0a7e6d1

Please sign in to comment.