Commit 5ab6d715 authored by Ian Rogers's avatar Ian Rogers Committed by Arnaldo Carvalho de Melo

perf maps: Add functions to access maps

Introduce functions to access struct maps. These functions reduce the
number of places reference counting is necessary. While tidying APIs do
some small const-ification, in particlar to unwind_libunwind_ops.

Committer notes:

Fixed up tools/perf/util/unwind-libunwind.c:

-               return ops->get_entries(cb, arg, thread, data, max_stack);
+               return ops->get_entries(cb, arg, thread, data, max_stack, best_effort);
Signed-off-by: default avatarIan Rogers <irogers@google.com>
Cc: Adrian Hunter <adrian.hunter@intel.com>
Cc: Alexander Shishkin <alexander.shishkin@linux.intel.com>
Cc: Alexey Bayduraev <alexey.v.bayduraev@linux.intel.com>
Cc: Andi Kleen <ak@linux.intel.com>
Cc: Andrew Morton <akpm@linux-foundation.org>
Cc: Andy Shevchenko <andriy.shevchenko@linux.intel.com>
Cc: Darren Hart <dvhart@infradead.org>
Cc: Davidlohr Bueso <dave@stgolabs.net>
Cc: Dmitriy Vyukov <dvyukov@google.com>
Cc: Eric Dumazet <edumazet@google.com>
Cc: German Gomez <german.gomez@arm.com>
Cc: Hao Luo <haoluo@google.com>
Cc: Ingo Molnar <mingo@redhat.com>
Cc: James Clark <james.clark@arm.com>
Cc: Jiri Olsa <jolsa@kernel.org>
Cc: John Garry <john.g.garry@oracle.com>
Cc: Kajol Jain <kjain@linux.ibm.com>
Cc: Kan Liang <kan.liang@linux.intel.com>
Cc: Leo Yan <leo.yan@linaro.org>
Cc: Madhavan Srinivasan <maddy@linux.ibm.com>
Cc: Mark Rutland <mark.rutland@arm.com>
Cc: Masami Hiramatsu <mhiramat@kernel.org>
Cc: Miaoqian Lin <linmq006@gmail.com>
Cc: Namhyung Kim <namhyung@kernel.org>
Cc: Peter Zijlstra <peterz@infradead.org>
Cc: Riccardo Mancini <rickyman7@gmail.com>
Cc: Shunsuke Nakamura <nakamura.shun@fujitsu.com>
Cc: Song Liu <song@kernel.org>
Cc: Stephane Eranian <eranian@google.com>
Cc: Stephen Brennan <stephen.s.brennan@oracle.com>
Cc: Steven Rostedt (VMware) <rostedt@goodmis.org>
Cc: Thomas Gleixner <tglx@linutronix.de>
Cc: Thomas Richter <tmricht@linux.ibm.com>
Cc: Yury Norov <yury.norov@gmail.com>
Link: https://lore.kernel.org/r/20230320212248.1175731-2-irogers@google.comSigned-off-by: default avatarArnaldo Carvalho de Melo <acme@redhat.com>
parent ff583dc4
...@@ -100,10 +100,11 @@ static PyObject *perf_sample_insn(PyObject *obj, PyObject *args) ...@@ -100,10 +100,11 @@ static PyObject *perf_sample_insn(PyObject *obj, PyObject *args)
if (!c) if (!c)
return NULL; return NULL;
if (c->sample->ip && !c->sample->insn_len && if (c->sample->ip && !c->sample->insn_len && c->al->thread->maps) {
c->al->thread->maps && c->al->thread->maps->machine) struct machine *machine = maps__machine(c->al->thread->maps);
script_fetch_insn(c->sample, c->al->thread, c->al->thread->maps->machine);
script_fetch_insn(c->sample, c->al->thread, machine);
}
if (!c->sample->insn_len) if (!c->sample->insn_len)
Py_RETURN_NONE; /* N.B. This is a return statement */ Py_RETURN_NONE; /* N.B. This is a return statement */
......
...@@ -269,7 +269,7 @@ static int read_object_code(u64 addr, size_t len, u8 cpumode, ...@@ -269,7 +269,7 @@ static int read_object_code(u64 addr, size_t len, u8 cpumode,
len = al.map->end - addr; len = al.map->end - addr;
/* Read the object code using perf */ /* Read the object code using perf */
ret_len = dso__data_read_offset(al.map->dso, thread->maps->machine, ret_len = dso__data_read_offset(al.map->dso, maps__machine(thread->maps),
al.addr, buf1, len); al.addr, buf1, len);
if (ret_len != len) { if (ret_len != len) {
pr_debug("dso__data_read_offset failed\n"); pr_debug("dso__data_read_offset failed\n");
......
...@@ -3139,7 +3139,8 @@ static int evsel__hists_browse(struct evsel *evsel, int nr_events, const char *h ...@@ -3139,7 +3139,8 @@ static int evsel__hists_browse(struct evsel *evsel, int nr_events, const char *h
continue; continue;
case 'k': case 'k':
if (browser->selection != NULL) if (browser->selection != NULL)
hists_browser__zoom_map(browser, browser->selection->maps->machine->vmlinux_map); hists_browser__zoom_map(browser,
maps__machine(browser->selection->maps)->vmlinux_map);
continue; continue;
case 'V': case 'V':
verbose = (verbose + 1) % 4; verbose = (verbose + 1) % 4;
......
...@@ -1112,6 +1112,8 @@ int hist_entry__append_callchain(struct hist_entry *he, struct perf_sample *samp ...@@ -1112,6 +1112,8 @@ int hist_entry__append_callchain(struct hist_entry *he, struct perf_sample *samp
int fill_callchain_info(struct addr_location *al, struct callchain_cursor_node *node, int fill_callchain_info(struct addr_location *al, struct callchain_cursor_node *node,
bool hide_unresolved) bool hide_unresolved)
{ {
struct machine *machine = maps__machine(node->ms.maps);
al->maps = node->ms.maps; al->maps = node->ms.maps;
al->map = node->ms.map; al->map = node->ms.map;
al->sym = node->ms.sym; al->sym = node->ms.sym;
...@@ -1124,9 +1126,8 @@ int fill_callchain_info(struct addr_location *al, struct callchain_cursor_node * ...@@ -1124,9 +1126,8 @@ int fill_callchain_info(struct addr_location *al, struct callchain_cursor_node *
if (al->map == NULL) if (al->map == NULL)
goto out; goto out;
} }
if (al->maps == machine__kernel_maps(machine)) {
if (al->maps == machine__kernel_maps(al->maps->machine)) { if (machine__is_host(machine)) {
if (machine__is_host(al->maps->machine)) {
al->cpumode = PERF_RECORD_MISC_KERNEL; al->cpumode = PERF_RECORD_MISC_KERNEL;
al->level = 'k'; al->level = 'k';
} else { } else {
...@@ -1134,7 +1135,7 @@ int fill_callchain_info(struct addr_location *al, struct callchain_cursor_node * ...@@ -1134,7 +1135,7 @@ int fill_callchain_info(struct addr_location *al, struct callchain_cursor_node *
al->level = 'g'; al->level = 'g';
} }
} else { } else {
if (machine__is_host(al->maps->machine)) { if (machine__is_host(machine)) {
al->cpumode = PERF_RECORD_MISC_USER; al->cpumode = PERF_RECORD_MISC_USER;
al->level = '.'; al->level = '.';
} else if (perf_guest) { } else if (perf_guest) {
......
...@@ -181,7 +181,7 @@ static int db_ids_from_al(struct db_export *dbe, struct addr_location *al, ...@@ -181,7 +181,7 @@ static int db_ids_from_al(struct db_export *dbe, struct addr_location *al,
if (al->map) { if (al->map) {
struct dso *dso = al->map->dso; struct dso *dso = al->map->dso;
err = db_export__dso(dbe, dso, al->maps->machine); err = db_export__dso(dbe, dso, maps__machine(al->maps));
if (err) if (err)
return err; return err;
*dso_db_id = dso->db_id; *dso_db_id = dso->db_id;
...@@ -354,19 +354,21 @@ int db_export__sample(struct db_export *dbe, union perf_event *event, ...@@ -354,19 +354,21 @@ int db_export__sample(struct db_export *dbe, union perf_event *event,
}; };
struct thread *main_thread; struct thread *main_thread;
struct comm *comm = NULL; struct comm *comm = NULL;
struct machine *machine;
int err; int err;
err = db_export__evsel(dbe, evsel); err = db_export__evsel(dbe, evsel);
if (err) if (err)
return err; return err;
err = db_export__machine(dbe, al->maps->machine); machine = maps__machine(al->maps);
err = db_export__machine(dbe, machine);
if (err) if (err)
return err; return err;
main_thread = thread__main_thread(al->maps->machine, thread); main_thread = thread__main_thread(machine, thread);
err = db_export__threads(dbe, thread, main_thread, al->maps->machine, &comm); err = db_export__threads(dbe, thread, main_thread, machine, &comm);
if (err) if (err)
goto out_put; goto out_put;
...@@ -380,7 +382,7 @@ int db_export__sample(struct db_export *dbe, union perf_event *event, ...@@ -380,7 +382,7 @@ int db_export__sample(struct db_export *dbe, union perf_event *event,
goto out_put; goto out_put;
if (dbe->cpr) { if (dbe->cpr) {
struct call_path *cp = call_path_from_sample(dbe, al->maps->machine, struct call_path *cp = call_path_from_sample(dbe, machine,
thread, sample, thread, sample,
evsel); evsel);
if (cp) { if (cp) {
......
...@@ -197,8 +197,12 @@ static const __u8 *dlfilter__insn(void *ctx, __u32 *len) ...@@ -197,8 +197,12 @@ static const __u8 *dlfilter__insn(void *ctx, __u32 *len)
if (!al->thread && machine__resolve(d->machine, al, d->sample) < 0) if (!al->thread && machine__resolve(d->machine, al, d->sample) < 0)
return NULL; return NULL;
if (al->thread->maps && al->thread->maps->machine) if (al->thread->maps) {
script_fetch_insn(d->sample, al->thread, al->thread->maps->machine); struct machine *machine = maps__machine(al->thread->maps);
if (machine)
script_fetch_insn(d->sample, al->thread, machine);
}
} }
if (!d->sample->insn_len) if (!d->sample->insn_len)
......
...@@ -572,7 +572,7 @@ struct map *thread__find_map(struct thread *thread, u8 cpumode, u64 addr, ...@@ -572,7 +572,7 @@ struct map *thread__find_map(struct thread *thread, u8 cpumode, u64 addr,
struct addr_location *al) struct addr_location *al)
{ {
struct maps *maps = thread->maps; struct maps *maps = thread->maps;
struct machine *machine = maps->machine; struct machine *machine = maps__machine(maps);
bool load_map = false; bool load_map = false;
al->maps = maps; al->maps = maps;
...@@ -637,7 +637,7 @@ struct map *thread__find_map_fb(struct thread *thread, u8 cpumode, u64 addr, ...@@ -637,7 +637,7 @@ struct map *thread__find_map_fb(struct thread *thread, u8 cpumode, u64 addr,
struct addr_location *al) struct addr_location *al)
{ {
struct map *map = thread__find_map(thread, cpumode, addr, al); struct map *map = thread__find_map(thread, cpumode, addr, al);
struct machine *machine = thread->maps->machine; struct machine *machine = maps__machine(thread->maps);
u8 addr_cpumode = machine__addr_cpumode(machine, cpumode, addr); u8 addr_cpumode = machine__addr_cpumode(machine, cpumode, addr);
if (map || addr_cpumode == cpumode) if (map || addr_cpumode == cpumode)
......
...@@ -241,7 +241,7 @@ void hists__calc_col_len(struct hists *hists, struct hist_entry *h) ...@@ -241,7 +241,7 @@ void hists__calc_col_len(struct hists *hists, struct hist_entry *h)
if (h->cgroup) { if (h->cgroup) {
const char *cgrp_name = "unknown"; const char *cgrp_name = "unknown";
struct cgroup *cgrp = cgroup__find(h->ms.maps->machine->env, struct cgroup *cgrp = cgroup__find(maps__machine(h->ms.maps)->env,
h->cgroup); h->cgroup);
if (cgrp != NULL) if (cgrp != NULL)
cgrp_name = cgrp->name; cgrp_name = cgrp->name;
......
...@@ -2847,7 +2847,7 @@ static int find_prev_cpumode(struct ip_callchain *chain, struct thread *thread, ...@@ -2847,7 +2847,7 @@ static int find_prev_cpumode(struct ip_callchain *chain, struct thread *thread,
static u64 get_leaf_frame_caller(struct perf_sample *sample, static u64 get_leaf_frame_caller(struct perf_sample *sample,
struct thread *thread, int usr_idx) struct thread *thread, int usr_idx)
{ {
if (machine__normalized_is(thread->maps->machine, "arm64")) if (machine__normalized_is(maps__machine(thread->maps), "arm64"))
return get_leaf_frame_caller_aarch64(sample, thread, usr_idx); return get_leaf_frame_caller_aarch64(sample, thread, usr_idx);
else else
return 0; return 0;
......
...@@ -234,7 +234,7 @@ bool __map__is_kernel(const struct map *map) ...@@ -234,7 +234,7 @@ bool __map__is_kernel(const struct map *map)
{ {
if (!map->dso->kernel) if (!map->dso->kernel)
return false; return false;
return machine__kernel_map(map__kmaps((struct map *)map)->machine) == map; return machine__kernel_map(maps__machine(map__kmaps((struct map *)map))) == map;
} }
bool __map__is_extra_kernel_map(const struct map *map) bool __map__is_extra_kernel_map(const struct map *map)
...@@ -475,12 +475,16 @@ u64 map__rip_2objdump(struct map *map, u64 rip) ...@@ -475,12 +475,16 @@ u64 map__rip_2objdump(struct map *map, u64 rip)
* kcore may not either. However the trampoline object code is on the * kcore may not either. However the trampoline object code is on the
* main kernel map, so just use that instead. * main kernel map, so just use that instead.
*/ */
if (kmap && is_entry_trampoline(kmap->name) && kmap->kmaps && kmap->kmaps->machine) { if (kmap && is_entry_trampoline(kmap->name) && kmap->kmaps) {
struct map *kernel_map = machine__kernel_map(kmap->kmaps->machine); struct machine *machine = maps__machine(kmap->kmaps);
if (machine) {
struct map *kernel_map = machine__kernel_map(machine);
if (kernel_map) if (kernel_map)
map = kernel_map; map = kernel_map;
} }
}
if (!map->dso->adjust_symbols) if (!map->dso->adjust_symbols)
return rip; return rip;
......
...@@ -13,7 +13,7 @@ ...@@ -13,7 +13,7 @@
static void maps__init(struct maps *maps, struct machine *machine) static void maps__init(struct maps *maps, struct machine *machine)
{ {
maps->entries = RB_ROOT; maps->entries = RB_ROOT;
init_rwsem(&maps->lock); init_rwsem(maps__lock(maps));
maps->machine = machine; maps->machine = machine;
maps->last_search_by_name = NULL; maps->last_search_by_name = NULL;
maps->nr_maps = 0; maps->nr_maps = 0;
...@@ -32,7 +32,7 @@ static void __maps__free_maps_by_name(struct maps *maps) ...@@ -32,7 +32,7 @@ static void __maps__free_maps_by_name(struct maps *maps)
static int __maps__insert(struct maps *maps, struct map *map) static int __maps__insert(struct maps *maps, struct map *map)
{ {
struct rb_node **p = &maps->entries.rb_node; struct rb_node **p = &maps__entries(maps)->rb_node;
struct rb_node *parent = NULL; struct rb_node *parent = NULL;
const u64 ip = map->start; const u64 ip = map->start;
struct map_rb_node *m, *new_rb_node; struct map_rb_node *m, *new_rb_node;
...@@ -54,7 +54,7 @@ static int __maps__insert(struct maps *maps, struct map *map) ...@@ -54,7 +54,7 @@ static int __maps__insert(struct maps *maps, struct map *map)
} }
rb_link_node(&new_rb_node->rb_node, parent, p); rb_link_node(&new_rb_node->rb_node, parent, p);
rb_insert_color(&new_rb_node->rb_node, &maps->entries); rb_insert_color(&new_rb_node->rb_node, maps__entries(maps));
map__get(map); map__get(map);
return 0; return 0;
} }
...@@ -63,7 +63,7 @@ int maps__insert(struct maps *maps, struct map *map) ...@@ -63,7 +63,7 @@ int maps__insert(struct maps *maps, struct map *map)
{ {
int err; int err;
down_write(&maps->lock); down_write(maps__lock(maps));
err = __maps__insert(maps, map); err = __maps__insert(maps, map);
if (err) if (err)
goto out; goto out;
...@@ -84,10 +84,11 @@ int maps__insert(struct maps *maps, struct map *map) ...@@ -84,10 +84,11 @@ int maps__insert(struct maps *maps, struct map *map)
* If we already performed some search by name, then we need to add the just * If we already performed some search by name, then we need to add the just
* inserted map and resort. * inserted map and resort.
*/ */
if (maps->maps_by_name) { if (maps__maps_by_name(maps)) {
if (maps->nr_maps > maps->nr_maps_allocated) { if (maps__nr_maps(maps) > maps->nr_maps_allocated) {
int nr_allocate = maps->nr_maps * 2; int nr_allocate = maps__nr_maps(maps) * 2;
struct map **maps_by_name = realloc(maps->maps_by_name, nr_allocate * sizeof(map)); struct map **maps_by_name = realloc(maps__maps_by_name(maps),
nr_allocate * sizeof(map));
if (maps_by_name == NULL) { if (maps_by_name == NULL) {
__maps__free_maps_by_name(maps); __maps__free_maps_by_name(maps);
...@@ -97,18 +98,18 @@ int maps__insert(struct maps *maps, struct map *map) ...@@ -97,18 +98,18 @@ int maps__insert(struct maps *maps, struct map *map)
maps->maps_by_name = maps_by_name; maps->maps_by_name = maps_by_name;
maps->nr_maps_allocated = nr_allocate; maps->nr_maps_allocated = nr_allocate;
} }
maps->maps_by_name[maps->nr_maps - 1] = map; maps__maps_by_name(maps)[maps__nr_maps(maps) - 1] = map;
__maps__sort_by_name(maps); __maps__sort_by_name(maps);
} }
out: out:
up_write(&maps->lock); up_write(maps__lock(maps));
return err; return err;
} }
static void __maps__remove(struct maps *maps, struct map_rb_node *rb_node) static void __maps__remove(struct maps *maps, struct map_rb_node *rb_node)
{ {
rb_erase_init(&rb_node->rb_node, &maps->entries); rb_erase_init(&rb_node->rb_node, maps__entries(maps));
map__put(rb_node->map); map__put(rb_node->map);
free(rb_node); free(rb_node);
} }
...@@ -117,7 +118,7 @@ void maps__remove(struct maps *maps, struct map *map) ...@@ -117,7 +118,7 @@ void maps__remove(struct maps *maps, struct map *map)
{ {
struct map_rb_node *rb_node; struct map_rb_node *rb_node;
down_write(&maps->lock); down_write(maps__lock(maps));
if (maps->last_search_by_name == map) if (maps->last_search_by_name == map)
maps->last_search_by_name = NULL; maps->last_search_by_name = NULL;
...@@ -125,9 +126,9 @@ void maps__remove(struct maps *maps, struct map *map) ...@@ -125,9 +126,9 @@ void maps__remove(struct maps *maps, struct map *map)
assert(rb_node->map == map); assert(rb_node->map == map);
__maps__remove(maps, rb_node); __maps__remove(maps, rb_node);
--maps->nr_maps; --maps->nr_maps;
if (maps->maps_by_name) if (maps__maps_by_name(maps))
__maps__free_maps_by_name(maps); __maps__free_maps_by_name(maps);
up_write(&maps->lock); up_write(maps__lock(maps));
} }
static void __maps__purge(struct maps *maps) static void __maps__purge(struct maps *maps)
...@@ -135,7 +136,7 @@ static void __maps__purge(struct maps *maps) ...@@ -135,7 +136,7 @@ static void __maps__purge(struct maps *maps)
struct map_rb_node *pos, *next; struct map_rb_node *pos, *next;
maps__for_each_entry_safe(maps, pos, next) { maps__for_each_entry_safe(maps, pos, next) {
rb_erase_init(&pos->rb_node, &maps->entries); rb_erase_init(&pos->rb_node, maps__entries(maps));
map__put(pos->map); map__put(pos->map);
free(pos); free(pos);
} }
...@@ -143,9 +144,9 @@ static void __maps__purge(struct maps *maps) ...@@ -143,9 +144,9 @@ static void __maps__purge(struct maps *maps)
static void maps__exit(struct maps *maps) static void maps__exit(struct maps *maps)
{ {
down_write(&maps->lock); down_write(maps__lock(maps));
__maps__purge(maps); __maps__purge(maps);
up_write(&maps->lock); up_write(maps__lock(maps));
} }
bool maps__empty(struct maps *maps) bool maps__empty(struct maps *maps)
...@@ -170,6 +171,14 @@ void maps__delete(struct maps *maps) ...@@ -170,6 +171,14 @@ void maps__delete(struct maps *maps)
free(maps); free(maps);
} }
struct maps *maps__get(struct maps *maps)
{
if (maps)
refcount_inc(&maps->refcnt);
return maps;
}
void maps__put(struct maps *maps) void maps__put(struct maps *maps)
{ {
if (maps && refcount_dec_and_test(&maps->refcnt)) if (maps && refcount_dec_and_test(&maps->refcnt))
...@@ -195,7 +204,7 @@ struct symbol *maps__find_symbol_by_name(struct maps *maps, const char *name, st ...@@ -195,7 +204,7 @@ struct symbol *maps__find_symbol_by_name(struct maps *maps, const char *name, st
struct symbol *sym; struct symbol *sym;
struct map_rb_node *pos; struct map_rb_node *pos;
down_read(&maps->lock); down_read(maps__lock(maps));
maps__for_each_entry(maps, pos) { maps__for_each_entry(maps, pos) {
sym = map__find_symbol_by_name(pos->map, name); sym = map__find_symbol_by_name(pos->map, name);
...@@ -213,7 +222,7 @@ struct symbol *maps__find_symbol_by_name(struct maps *maps, const char *name, st ...@@ -213,7 +222,7 @@ struct symbol *maps__find_symbol_by_name(struct maps *maps, const char *name, st
sym = NULL; sym = NULL;
out: out:
up_read(&maps->lock); up_read(maps__lock(maps));
return sym; return sym;
} }
...@@ -238,7 +247,7 @@ size_t maps__fprintf(struct maps *maps, FILE *fp) ...@@ -238,7 +247,7 @@ size_t maps__fprintf(struct maps *maps, FILE *fp)
size_t printed = 0; size_t printed = 0;
struct map_rb_node *pos; struct map_rb_node *pos;
down_read(&maps->lock); down_read(maps__lock(maps));
maps__for_each_entry(maps, pos) { maps__for_each_entry(maps, pos) {
printed += fprintf(fp, "Map:"); printed += fprintf(fp, "Map:");
...@@ -249,7 +258,7 @@ size_t maps__fprintf(struct maps *maps, FILE *fp) ...@@ -249,7 +258,7 @@ size_t maps__fprintf(struct maps *maps, FILE *fp)
} }
} }
up_read(&maps->lock); up_read(maps__lock(maps));
return printed; return printed;
} }
...@@ -260,9 +269,9 @@ int maps__fixup_overlappings(struct maps *maps, struct map *map, FILE *fp) ...@@ -260,9 +269,9 @@ int maps__fixup_overlappings(struct maps *maps, struct map *map, FILE *fp)
struct rb_node *next, *first; struct rb_node *next, *first;
int err = 0; int err = 0;
down_write(&maps->lock); down_write(maps__lock(maps));
root = &maps->entries; root = maps__entries(maps);
/* /*
* Find first map where end > map->start. * Find first map where end > map->start.
...@@ -358,7 +367,7 @@ int maps__fixup_overlappings(struct maps *maps, struct map *map, FILE *fp) ...@@ -358,7 +367,7 @@ int maps__fixup_overlappings(struct maps *maps, struct map *map, FILE *fp)
err = 0; err = 0;
out: out:
up_write(&maps->lock); up_write(maps__lock(maps));
return err; return err;
} }
...@@ -371,7 +380,7 @@ int maps__clone(struct thread *thread, struct maps *parent) ...@@ -371,7 +380,7 @@ int maps__clone(struct thread *thread, struct maps *parent)
int err; int err;
struct map_rb_node *rb_node; struct map_rb_node *rb_node;
down_read(&parent->lock); down_read(maps__lock(parent));
maps__for_each_entry(parent, rb_node) { maps__for_each_entry(parent, rb_node) {
struct map *new = map__clone(rb_node->map); struct map *new = map__clone(rb_node->map);
...@@ -394,7 +403,7 @@ int maps__clone(struct thread *thread, struct maps *parent) ...@@ -394,7 +403,7 @@ int maps__clone(struct thread *thread, struct maps *parent)
err = 0; err = 0;
out_unlock: out_unlock:
up_read(&parent->lock); up_read(maps__lock(parent));
return err; return err;
} }
...@@ -415,9 +424,9 @@ struct map *maps__find(struct maps *maps, u64 ip) ...@@ -415,9 +424,9 @@ struct map *maps__find(struct maps *maps, u64 ip)
struct map_rb_node *m; struct map_rb_node *m;
down_read(&maps->lock); down_read(maps__lock(maps));
p = maps->entries.rb_node; p = maps__entries(maps)->rb_node;
while (p != NULL) { while (p != NULL) {
m = rb_entry(p, struct map_rb_node, rb_node); m = rb_entry(p, struct map_rb_node, rb_node);
if (ip < m->map->start) if (ip < m->map->start)
...@@ -430,13 +439,13 @@ struct map *maps__find(struct maps *maps, u64 ip) ...@@ -430,13 +439,13 @@ struct map *maps__find(struct maps *maps, u64 ip)
m = NULL; m = NULL;
out: out:
up_read(&maps->lock); up_read(maps__lock(maps));
return m ? m->map : NULL; return m ? m->map : NULL;
} }
struct map_rb_node *maps__first(struct maps *maps) struct map_rb_node *maps__first(struct maps *maps)
{ {
struct rb_node *first = rb_first(&maps->entries); struct rb_node *first = rb_first(maps__entries(maps));
if (first) if (first)
return rb_entry(first, struct map_rb_node, rb_node); return rb_entry(first, struct map_rb_node, rb_node);
......
...@@ -43,7 +43,7 @@ struct maps { ...@@ -43,7 +43,7 @@ struct maps {
unsigned int nr_maps_allocated; unsigned int nr_maps_allocated;
#ifdef HAVE_LIBUNWIND_SUPPORT #ifdef HAVE_LIBUNWIND_SUPPORT
void *addr_space; void *addr_space;
struct unwind_libunwind_ops *unwind_libunwind_ops; const struct unwind_libunwind_ops *unwind_libunwind_ops;
#endif #endif
}; };
...@@ -58,20 +58,51 @@ struct kmap { ...@@ -58,20 +58,51 @@ struct kmap {
struct maps *maps__new(struct machine *machine); struct maps *maps__new(struct machine *machine);
void maps__delete(struct maps *maps); void maps__delete(struct maps *maps);
bool maps__empty(struct maps *maps); bool maps__empty(struct maps *maps);
int maps__clone(struct thread *thread, struct maps *parent);
struct maps *maps__get(struct maps *maps);
void maps__put(struct maps *maps);
static inline struct maps *maps__get(struct maps *maps) static inline struct rb_root *maps__entries(struct maps *maps)
{ {
if (maps) return &maps->entries;
refcount_inc(&maps->refcnt);
return maps;
} }
void maps__put(struct maps *maps); static inline struct machine *maps__machine(struct maps *maps)
int maps__clone(struct thread *thread, struct maps *parent); {
return maps->machine;
}
static inline struct rw_semaphore *maps__lock(struct maps *maps)
{
return &maps->lock;
}
static inline struct map **maps__maps_by_name(struct maps *maps)
{
return maps->maps_by_name;
}
static inline unsigned int maps__nr_maps(const struct maps *maps)
{
return maps->nr_maps;
}
#ifdef HAVE_LIBUNWIND_SUPPORT
static inline void *maps__addr_space(struct maps *maps)
{
return maps->addr_space;
}
static inline const struct unwind_libunwind_ops *maps__unwind_libunwind_ops(const struct maps *maps)
{
return maps->unwind_libunwind_ops;
}
#endif
size_t maps__fprintf(struct maps *maps, FILE *fp); size_t maps__fprintf(struct maps *maps, FILE *fp);
int maps__insert(struct maps *maps, struct map *map); int maps__insert(struct maps *maps, struct map *map);
void maps__remove(struct maps *maps, struct map *map); void maps__remove(struct maps *maps, struct map *map);
struct symbol *maps__find_symbol(struct maps *maps, u64 addr, struct map **mapp); struct symbol *maps__find_symbol(struct maps *maps, u64 addr, struct map **mapp);
......
...@@ -1288,7 +1288,7 @@ static void python_export_sample_table(struct db_export *dbe, ...@@ -1288,7 +1288,7 @@ static void python_export_sample_table(struct db_export *dbe,
tuple_set_d64(t, 0, es->db_id); tuple_set_d64(t, 0, es->db_id);
tuple_set_d64(t, 1, es->evsel->db_id); tuple_set_d64(t, 1, es->evsel->db_id);
tuple_set_d64(t, 2, es->al->maps->machine->db_id); tuple_set_d64(t, 2, maps__machine(es->al->maps)->db_id);
tuple_set_d64(t, 3, es->al->thread->db_id); tuple_set_d64(t, 3, es->al->thread->db_id);
tuple_set_d64(t, 4, es->comm_db_id); tuple_set_d64(t, 4, es->comm_db_id);
tuple_set_d64(t, 5, es->dso_db_id); tuple_set_d64(t, 5, es->dso_db_id);
......
...@@ -803,7 +803,7 @@ static int hist_entry__cgroup_snprintf(struct hist_entry *he, ...@@ -803,7 +803,7 @@ static int hist_entry__cgroup_snprintf(struct hist_entry *he,
const char *cgrp_name = "N/A"; const char *cgrp_name = "N/A";
if (he->cgroup) { if (he->cgroup) {
struct cgroup *cgrp = cgroup__find(he->ms.maps->machine->env, struct cgroup *cgrp = cgroup__find(maps__machine(he->ms.maps)->env,
he->cgroup); he->cgroup);
if (cgrp != NULL) if (cgrp != NULL)
cgrp_name = cgrp->name; cgrp_name = cgrp->name;
......
...@@ -1428,7 +1428,7 @@ static int dso__process_kernel_symbol(struct dso *dso, struct map *map, ...@@ -1428,7 +1428,7 @@ static int dso__process_kernel_symbol(struct dso *dso, struct map *map,
* we still are sure to have a reference to this DSO via * we still are sure to have a reference to this DSO via
* *curr_map->dso. * *curr_map->dso.
*/ */
dsos__add(&kmaps->machine->dsos, curr_dso); dsos__add(&maps__machine(kmaps)->dsos, curr_dso);
/* kmaps already got it */ /* kmaps already got it */
map__put(curr_map); map__put(curr_map);
dso__set_loaded(curr_dso); dso__set_loaded(curr_dso);
......
...@@ -275,7 +275,7 @@ void maps__fixup_end(struct maps *maps) ...@@ -275,7 +275,7 @@ void maps__fixup_end(struct maps *maps)
{ {
struct map_rb_node *prev = NULL, *curr; struct map_rb_node *prev = NULL, *curr;
down_write(&maps->lock); down_write(maps__lock(maps));
maps__for_each_entry(maps, curr) { maps__for_each_entry(maps, curr) {
if (prev != NULL && !prev->map->end) if (prev != NULL && !prev->map->end)
...@@ -291,7 +291,7 @@ void maps__fixup_end(struct maps *maps) ...@@ -291,7 +291,7 @@ void maps__fixup_end(struct maps *maps)
if (curr && !curr->map->end) if (curr && !curr->map->end)
curr->map->end = ~0ULL; curr->map->end = ~0ULL;
up_write(&maps->lock); up_write(maps__lock(maps));
} }
struct symbol *symbol__new(u64 start, u64 len, u8 binding, u8 type, const char *name) struct symbol *symbol__new(u64 start, u64 len, u8 binding, u8 type, const char *name)
...@@ -844,7 +844,7 @@ static int maps__split_kallsyms(struct maps *kmaps, struct dso *dso, u64 delta, ...@@ -844,7 +844,7 @@ static int maps__split_kallsyms(struct maps *kmaps, struct dso *dso, u64 delta,
if (!kmaps) if (!kmaps)
return -1; return -1;
machine = kmaps->machine; machine = maps__machine(kmaps);
x86_64 = machine__is(machine, "x86_64"); x86_64 = machine__is(machine, "x86_64");
...@@ -968,7 +968,7 @@ static int maps__split_kallsyms(struct maps *kmaps, struct dso *dso, u64 delta, ...@@ -968,7 +968,7 @@ static int maps__split_kallsyms(struct maps *kmaps, struct dso *dso, u64 delta,
if (curr_map != initial_map && if (curr_map != initial_map &&
dso->kernel == DSO_SPACE__KERNEL_GUEST && dso->kernel == DSO_SPACE__KERNEL_GUEST &&
machine__is_default_guest(kmaps->machine)) { machine__is_default_guest(maps__machine(kmaps))) {
dso__set_loaded(curr_map->dso); dso__set_loaded(curr_map->dso);
} }
...@@ -1365,7 +1365,7 @@ static int dso__load_kcore(struct dso *dso, struct map *map, ...@@ -1365,7 +1365,7 @@ static int dso__load_kcore(struct dso *dso, struct map *map,
if (!kmaps) if (!kmaps)
return -EINVAL; return -EINVAL;
machine = kmaps->machine; machine = maps__machine(kmaps);
/* This function requires that the map is the kernel map */ /* This function requires that the map is the kernel map */
if (!__map__is_kernel(map)) if (!__map__is_kernel(map))
...@@ -1894,7 +1894,7 @@ int dso__load(struct dso *dso, struct map *map) ...@@ -1894,7 +1894,7 @@ int dso__load(struct dso *dso, struct map *map)
else if (dso->kernel == DSO_SPACE__KERNEL_GUEST) else if (dso->kernel == DSO_SPACE__KERNEL_GUEST)
ret = dso__load_guest_kernel_sym(dso, map); ret = dso__load_guest_kernel_sym(dso, map);
machine = map__kmaps(map)->machine; machine = maps__machine(map__kmaps(map));
if (machine__is(machine, "x86_64")) if (machine__is(machine, "x86_64"))
machine__map_x86_64_entry_trampolines(machine, dso); machine__map_x86_64_entry_trampolines(machine, dso);
goto out; goto out;
...@@ -2059,32 +2059,32 @@ static int map__strcmp_name(const void *name, const void *b) ...@@ -2059,32 +2059,32 @@ static int map__strcmp_name(const void *name, const void *b)
void __maps__sort_by_name(struct maps *maps) void __maps__sort_by_name(struct maps *maps)
{ {
qsort(maps->maps_by_name, maps->nr_maps, sizeof(struct map *), map__strcmp); qsort(maps__maps_by_name(maps), maps__nr_maps(maps), sizeof(struct map *), map__strcmp);
} }
static int map__groups__sort_by_name_from_rbtree(struct maps *maps) static int map__groups__sort_by_name_from_rbtree(struct maps *maps)
{ {
struct map_rb_node *rb_node; struct map_rb_node *rb_node;
struct map **maps_by_name = realloc(maps->maps_by_name, struct map **maps_by_name = realloc(maps__maps_by_name(maps),
maps->nr_maps * sizeof(struct map *)); maps__nr_maps(maps) * sizeof(struct map *));
int i = 0; int i = 0;
if (maps_by_name == NULL) if (maps_by_name == NULL)
return -1; return -1;
up_read(&maps->lock); up_read(maps__lock(maps));
down_write(&maps->lock); down_write(maps__lock(maps));
maps->maps_by_name = maps_by_name; maps->maps_by_name = maps_by_name;
maps->nr_maps_allocated = maps->nr_maps; maps->nr_maps_allocated = maps__nr_maps(maps);
maps__for_each_entry(maps, rb_node) maps__for_each_entry(maps, rb_node)
maps_by_name[i++] = rb_node->map; maps_by_name[i++] = rb_node->map;
__maps__sort_by_name(maps); __maps__sort_by_name(maps);
up_write(&maps->lock); up_write(maps__lock(maps));
down_read(&maps->lock); down_read(maps__lock(maps));
return 0; return 0;
} }
...@@ -2093,11 +2093,12 @@ static struct map *__maps__find_by_name(struct maps *maps, const char *name) ...@@ -2093,11 +2093,12 @@ static struct map *__maps__find_by_name(struct maps *maps, const char *name)
{ {
struct map **mapp; struct map **mapp;
if (maps->maps_by_name == NULL && if (maps__maps_by_name(maps) == NULL &&
map__groups__sort_by_name_from_rbtree(maps)) map__groups__sort_by_name_from_rbtree(maps))
return NULL; return NULL;
mapp = bsearch(name, maps->maps_by_name, maps->nr_maps, sizeof(*mapp), map__strcmp_name); mapp = bsearch(name, maps__maps_by_name(maps), maps__nr_maps(maps),
sizeof(*mapp), map__strcmp_name);
if (mapp) if (mapp)
return *mapp; return *mapp;
return NULL; return NULL;
...@@ -2108,9 +2109,10 @@ struct map *maps__find_by_name(struct maps *maps, const char *name) ...@@ -2108,9 +2109,10 @@ struct map *maps__find_by_name(struct maps *maps, const char *name)
struct map_rb_node *rb_node; struct map_rb_node *rb_node;
struct map *map; struct map *map;
down_read(&maps->lock); down_read(maps__lock(maps));
if (maps->last_search_by_name && strcmp(maps->last_search_by_name->dso->short_name, name) == 0) { if (maps->last_search_by_name &&
strcmp(maps->last_search_by_name->dso->short_name, name) == 0) {
map = maps->last_search_by_name; map = maps->last_search_by_name;
goto out_unlock; goto out_unlock;
} }
...@@ -2120,7 +2122,7 @@ struct map *maps__find_by_name(struct maps *maps, const char *name) ...@@ -2120,7 +2122,7 @@ struct map *maps__find_by_name(struct maps *maps, const char *name)
* made. * made.
*/ */
map = __maps__find_by_name(maps, name); map = __maps__find_by_name(maps, name);
if (map || maps->maps_by_name != NULL) if (map || maps__maps_by_name(maps) != NULL)
goto out_unlock; goto out_unlock;
/* Fallback to traversing the rbtree... */ /* Fallback to traversing the rbtree... */
...@@ -2134,7 +2136,7 @@ struct map *maps__find_by_name(struct maps *maps, const char *name) ...@@ -2134,7 +2136,7 @@ struct map *maps__find_by_name(struct maps *maps, const char *name)
map = NULL; map = NULL;
out_unlock: out_unlock:
up_read(&maps->lock); up_read(maps__lock(maps));
return map; return map;
} }
...@@ -2386,7 +2388,7 @@ static int dso__load_guest_kernel_sym(struct dso *dso, struct map *map) ...@@ -2386,7 +2388,7 @@ static int dso__load_guest_kernel_sym(struct dso *dso, struct map *map)
{ {
int err; int err;
const char *kallsyms_filename; const char *kallsyms_filename;
struct machine *machine = map__kmaps(map)->machine; struct machine *machine = maps__machine(map__kmaps(map));
char path[PATH_MAX]; char path[PATH_MAX];
if (machine->kallsyms_filename) { if (machine->kallsyms_filename) {
......
...@@ -155,8 +155,8 @@ static int thread_stack__init(struct thread_stack *ts, struct thread *thread, ...@@ -155,8 +155,8 @@ static int thread_stack__init(struct thread_stack *ts, struct thread *thread,
ts->br_stack_sz = br_stack_sz; ts->br_stack_sz = br_stack_sz;
} }
if (thread->maps && thread->maps->machine) { if (thread->maps && maps__machine(thread->maps)) {
struct machine *machine = thread->maps->machine; struct machine *machine = maps__machine(thread->maps);
const char *arch = perf_env__arch(machine->env); const char *arch = perf_env__arch(machine->env);
ts->kernel_start = machine__kernel_start(machine); ts->kernel_start = machine__kernel_start(machine);
......
...@@ -362,7 +362,7 @@ static int __thread__prepare_access(struct thread *thread) ...@@ -362,7 +362,7 @@ static int __thread__prepare_access(struct thread *thread)
struct maps *maps = thread->maps; struct maps *maps = thread->maps;
struct map_rb_node *rb_node; struct map_rb_node *rb_node;
down_read(&maps->lock); down_read(maps__lock(maps));
maps__for_each_entry(maps, rb_node) { maps__for_each_entry(maps, rb_node) {
err = unwind__prepare_access(thread->maps, rb_node->map, &initialized); err = unwind__prepare_access(thread->maps, rb_node->map, &initialized);
...@@ -370,7 +370,7 @@ static int __thread__prepare_access(struct thread *thread) ...@@ -370,7 +370,7 @@ static int __thread__prepare_access(struct thread *thread)
break; break;
} }
up_read(&maps->lock); up_read(maps__lock(maps));
return err; return err;
} }
......
...@@ -667,24 +667,26 @@ static unw_accessors_t accessors = { ...@@ -667,24 +667,26 @@ static unw_accessors_t accessors = {
static int _unwind__prepare_access(struct maps *maps) static int _unwind__prepare_access(struct maps *maps)
{ {
maps->addr_space = unw_create_addr_space(&accessors, 0); void *addr_space = unw_create_addr_space(&accessors, 0);
if (!maps->addr_space) {
maps->addr_space = addr_space;
if (!addr_space) {
pr_err("unwind: Can't create unwind address space.\n"); pr_err("unwind: Can't create unwind address space.\n");
return -ENOMEM; return -ENOMEM;
} }
unw_set_caching_policy(maps->addr_space, UNW_CACHE_GLOBAL); unw_set_caching_policy(addr_space, UNW_CACHE_GLOBAL);
return 0; return 0;
} }
static void _unwind__flush_access(struct maps *maps) static void _unwind__flush_access(struct maps *maps)
{ {
unw_flush_cache(maps->addr_space, 0, 0); unw_flush_cache(maps__addr_space(maps), 0, 0);
} }
static void _unwind__finish_access(struct maps *maps) static void _unwind__finish_access(struct maps *maps)
{ {
unw_destroy_addr_space(maps->addr_space); unw_destroy_addr_space(maps__addr_space(maps));
} }
static int get_entries(struct unwind_info *ui, unwind_entry_cb_t cb, static int get_entries(struct unwind_info *ui, unwind_entry_cb_t cb,
...@@ -709,7 +711,7 @@ static int get_entries(struct unwind_info *ui, unwind_entry_cb_t cb, ...@@ -709,7 +711,7 @@ static int get_entries(struct unwind_info *ui, unwind_entry_cb_t cb,
*/ */
if (max_stack - 1 > 0) { if (max_stack - 1 > 0) {
WARN_ONCE(!ui->thread, "WARNING: ui->thread is NULL"); WARN_ONCE(!ui->thread, "WARNING: ui->thread is NULL");
addr_space = ui->thread->maps->addr_space; addr_space = maps__addr_space(ui->thread->maps);
if (addr_space == NULL) if (addr_space == NULL)
return -1; return -1;
...@@ -759,7 +761,7 @@ static int _unwind__get_entries(unwind_entry_cb_t cb, void *arg, ...@@ -759,7 +761,7 @@ static int _unwind__get_entries(unwind_entry_cb_t cb, void *arg,
struct unwind_info ui = { struct unwind_info ui = {
.sample = data, .sample = data,
.thread = thread, .thread = thread,
.machine = thread->maps->machine, .machine = maps__machine(thread->maps),
.best_effort = best_effort .best_effort = best_effort
}; };
......
...@@ -22,12 +22,13 @@ int unwind__prepare_access(struct maps *maps, struct map *map, bool *initialized ...@@ -22,12 +22,13 @@ int unwind__prepare_access(struct maps *maps, struct map *map, bool *initialized
const char *arch; const char *arch;
enum dso_type dso_type; enum dso_type dso_type;
struct unwind_libunwind_ops *ops = local_unwind_libunwind_ops; struct unwind_libunwind_ops *ops = local_unwind_libunwind_ops;
struct machine *machine;
int err; int err;
if (!dwarf_callchain_users) if (!dwarf_callchain_users)
return 0; return 0;
if (maps->addr_space) { if (maps__addr_space(maps)) {
pr_debug("unwind: thread map already set, dso=%s\n", pr_debug("unwind: thread map already set, dso=%s\n",
map->dso->name); map->dso->name);
if (initialized) if (initialized)
...@@ -35,15 +36,16 @@ int unwind__prepare_access(struct maps *maps, struct map *map, bool *initialized ...@@ -35,15 +36,16 @@ int unwind__prepare_access(struct maps *maps, struct map *map, bool *initialized
return 0; return 0;
} }
machine = maps__machine(maps);
/* env->arch is NULL for live-mode (i.e. perf top) */ /* env->arch is NULL for live-mode (i.e. perf top) */
if (!maps->machine->env || !maps->machine->env->arch) if (!machine->env || !machine->env->arch)
goto out_register; goto out_register;
dso_type = dso__type(map->dso, maps->machine); dso_type = dso__type(map->dso, machine);
if (dso_type == DSO__TYPE_UNKNOWN) if (dso_type == DSO__TYPE_UNKNOWN)
return 0; return 0;
arch = perf_env__arch(maps->machine->env); arch = perf_env__arch(machine->env);
if (!strcmp(arch, "x86")) { if (!strcmp(arch, "x86")) {
if (dso_type != DSO__TYPE_64BIT) if (dso_type != DSO__TYPE_64BIT)
...@@ -60,7 +62,7 @@ int unwind__prepare_access(struct maps *maps, struct map *map, bool *initialized ...@@ -60,7 +62,7 @@ int unwind__prepare_access(struct maps *maps, struct map *map, bool *initialized
out_register: out_register:
unwind__register_ops(maps, ops); unwind__register_ops(maps, ops);
err = maps->unwind_libunwind_ops->prepare_access(maps); err = maps__unwind_libunwind_ops(maps)->prepare_access(maps);
if (initialized) if (initialized)
*initialized = err ? false : true; *initialized = err ? false : true;
return err; return err;
...@@ -68,14 +70,18 @@ int unwind__prepare_access(struct maps *maps, struct map *map, bool *initialized ...@@ -68,14 +70,18 @@ int unwind__prepare_access(struct maps *maps, struct map *map, bool *initialized
void unwind__flush_access(struct maps *maps) void unwind__flush_access(struct maps *maps)
{ {
if (maps->unwind_libunwind_ops) const struct unwind_libunwind_ops *ops = maps__unwind_libunwind_ops(maps);
maps->unwind_libunwind_ops->flush_access(maps);
if (ops)
ops->flush_access(maps);
} }
void unwind__finish_access(struct maps *maps) void unwind__finish_access(struct maps *maps)
{ {
if (maps->unwind_libunwind_ops) const struct unwind_libunwind_ops *ops = maps__unwind_libunwind_ops(maps);
maps->unwind_libunwind_ops->finish_access(maps);
if (ops)
ops->finish_access(maps);
} }
int unwind__get_entries(unwind_entry_cb_t cb, void *arg, int unwind__get_entries(unwind_entry_cb_t cb, void *arg,
...@@ -83,8 +89,9 @@ int unwind__get_entries(unwind_entry_cb_t cb, void *arg, ...@@ -83,8 +89,9 @@ int unwind__get_entries(unwind_entry_cb_t cb, void *arg,
struct perf_sample *data, int max_stack, struct perf_sample *data, int max_stack,
bool best_effort) bool best_effort)
{ {
if (thread->maps->unwind_libunwind_ops) const struct unwind_libunwind_ops *ops = maps__unwind_libunwind_ops(thread->maps);
return thread->maps->unwind_libunwind_ops->get_entries(cb, arg, thread, data,
max_stack, best_effort); if (ops)
return ops->get_entries(cb, arg, thread, data, max_stack, best_effort);
return 0; return 0;
} }
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