[<prev] [next>] [<thread-prev] [thread-next>] [day] [month] [year] [list]
Message-ID: <aAvDStpWjwC7grya@google.com>
Date: Fri, 25 Apr 2025 10:15:54 -0700
From: Namhyung Kim <namhyung@...nel.org>
To: Ian Rogers <irogers@...gle.com>
Cc: Peter Zijlstra <peterz@...radead.org>, Ingo Molnar <mingo@...hat.com>,
Arnaldo Carvalho de Melo <acme@...nel.org>,
Mark Rutland <mark.rutland@....com>,
Alexander Shishkin <alexander.shishkin@...ux.intel.com>,
Jiri Olsa <jolsa@...nel.org>,
Adrian Hunter <adrian.hunter@...el.com>,
Kan Liang <kan.liang@...ux.intel.com>,
Athira Rajeev <atrajeev@...ux.ibm.com>,
Kajol Jain <kjain@...ux.ibm.com>, Li Huafei <lihuafei1@...wei.com>,
"Steinar H. Gunderson" <sesse@...gle.com>,
James Clark <james.clark@...aro.org>,
Stephen Brennan <stephen.s.brennan@...cle.com>,
Andi Kleen <ak@...ux.intel.com>, Dmitry Vyukov <dvyukov@...gle.com>,
Zhongqiu Han <quic_zhonhan@...cinc.com>,
Yicong Yang <yangyicong@...ilicon.com>,
Krzysztof Łopatowski <krzysztof.m.lopatowski@...il.com>,
"Dr. David Alan Gilbert" <linux@...blig.org>,
Zixian Cai <fzczx123@...il.com>,
Steve Clevenger <scclevenger@...amperecomputing.com>,
Thomas Falcon <thomas.falcon@...el.com>,
Martin Liska <martin.liska@....com>,
Martin Liška <m.liska@...link.cz>,
Song Liu <song@...nel.org>, linux-perf-users@...r.kernel.org,
linux-kernel@...r.kernel.org
Subject: Re: [PATCH v2 7/8] perf dso: Move build_id to dso_id
Hi Ian,
On Thu, Apr 24, 2025 at 12:58:30PM -0700, Ian Rogers wrote:
> The dso_id previously contained the major, minor, inode and inode
> generation information from a mmap2 event - the inode generation would
> be zero when reading from /proc/pid/maps. The build_id was in the
> dso. With build ID mmap2 events these fields wouldn't be initialized
> which would largely mean the special empty case where any dso would
> match for equality. This isn't desirable as if a dso is replaced we
> want the comparison to yield a difference.
>
> To support detecting the difference between DSOs based on build_id,
> move the build_id out of the DSO and into the dso_id. The dso_id is
> also stored in the DSO so nothing is lost. Capture in the dso_id what
> parts have been initialized and rename dso_id__inject to
> dso_id__improve_id so that it is clear the dso_id is being improved
> upon with additional information. With the build_id in the dso_id, use
> memcmp to compare for equality.
>
> Signed-off-by: Ian Rogers <irogers@...gle.com>
> ---
> tools/perf/builtin-buildid-list.c | 2 +-
> tools/perf/builtin-inject.c | 32 ++++---
> tools/perf/builtin-report.c | 11 ++-
> tools/perf/include/perf/perf_dlfilter.h | 2 +-
> tools/perf/tests/symbols.c | 4 +-
> tools/perf/util/build-id.c | 4 +-
> tools/perf/util/dso.c | 109 +++++++++++++-----------
> tools/perf/util/dso.h | 75 ++++++++--------
> tools/perf/util/dsos.c | 18 ++--
> tools/perf/util/machine.c | 28 +++---
> tools/perf/util/map.c | 13 ++-
> tools/perf/util/map.h | 5 +-
> tools/perf/util/sort.c | 27 +++---
> tools/perf/util/synthetic-events.c | 18 ++--
> 14 files changed, 194 insertions(+), 154 deletions(-)
>
> diff --git a/tools/perf/builtin-buildid-list.c b/tools/perf/builtin-buildid-list.c
> index ba8ba0303920..151cd84b6dfe 100644
> --- a/tools/perf/builtin-buildid-list.c
> +++ b/tools/perf/builtin-buildid-list.c
> @@ -31,7 +31,7 @@ static int buildid__map_cb(struct map *map, void *arg __maybe_unused)
>
> memset(bid_buf, 0, sizeof(bid_buf));
> if (dso__has_build_id(dso))
> - build_id__snprintf(dso__bid_const(dso), bid_buf, sizeof(bid_buf));
> + build_id__snprintf(dso__bid(dso), bid_buf, sizeof(bid_buf));
Can you please split this kind of interface changes?
> printf("%s %16" PRIx64 " %16" PRIx64, bid_buf, map__start(map), map__end(map));
> if (dso_long_name != NULL)
> printf(" %s", dso_long_name);
> diff --git a/tools/perf/builtin-inject.c b/tools/perf/builtin-inject.c
> index 11e49cafa3af..e74a3a70ff6f 100644
> --- a/tools/perf/builtin-inject.c
> +++ b/tools/perf/builtin-inject.c
> @@ -587,15 +587,17 @@ static int perf_event__repipe_mmap2(const struct perf_tool *tool,
> struct perf_sample *sample,
> struct machine *machine)
> {
> - struct dso_id id;
> - struct dso_id *dso_id = NULL;
> + struct dso_id id = dso_id_empty;
>
> - if (!(event->header.misc & PERF_RECORD_MISC_MMAP_BUILD_ID)) {
> + if (event->header.misc & PERF_RECORD_MISC_MMAP_BUILD_ID) {
> + build_id__init(&id.build_id, event->mmap2.build_id, event->mmap2.build_id_size);
> + } else {
> id.maj = event->mmap2.maj;
> id.min = event->mmap2.min;
> id.ino = event->mmap2.ino;
> id.ino_generation = event->mmap2.ino_generation;
> - dso_id = &id;
> + id.mmap2_valid = true;
> + id.mmap2_ino_generation_valid = true;
> }
>
> return perf_event__repipe_common_mmap(
> @@ -603,7 +605,7 @@ static int perf_event__repipe_mmap2(const struct perf_tool *tool,
> event->mmap2.pid, event->mmap2.tid,
> event->mmap2.start, event->mmap2.len, event->mmap2.pgoff,
> event->mmap2.flags, event->mmap2.prot,
> - event->mmap2.filename, dso_id,
> + event->mmap2.filename, &id,
> perf_event__process_mmap2);
> }
>
> @@ -671,19 +673,20 @@ static int perf_event__repipe_tracing_data(struct perf_session *session,
> static int dso__read_build_id(struct dso *dso)
> {
> struct nscookie nsc;
> + struct build_id bid;
>
> if (dso__has_build_id(dso))
> return 0;
>
> mutex_lock(dso__lock(dso));
> nsinfo__mountns_enter(dso__nsinfo(dso), &nsc);
> - if (filename__read_build_id(dso__long_name(dso), dso__bid(dso)) > 0)
> - dso__set_has_build_id(dso);
> + if (filename__read_build_id(dso__long_name(dso), &bid) > 0)
> + dso__set_build_id(dso, &bid);
> else if (dso__nsinfo(dso)) {
> char *new_name = dso__filename_with_chroot(dso, dso__long_name(dso));
>
> - if (new_name && filename__read_build_id(new_name, dso__bid(dso)) > 0)
> - dso__set_has_build_id(dso);
> + if (new_name && filename__read_build_id(new_name, &bid) > 0)
> + dso__set_build_id(dso, &bid);
> free(new_name);
> }
> nsinfo__mountns_exit(&nsc);
> @@ -732,10 +735,11 @@ static bool perf_inject__lookup_known_build_id(struct perf_inject *inject,
> struct dso *dso)
> {
> struct str_node *pos;
> - int bid_len;
> + struct build_id bid;
>
> strlist__for_each_entry(pos, inject->known_build_ids) {
> const char *build_id, *dso_name;
> + int bid_len;
>
> build_id = skip_spaces(pos->s);
> dso_name = strchr(build_id, ' ');
> @@ -744,11 +748,11 @@ static bool perf_inject__lookup_known_build_id(struct perf_inject *inject,
> if (strcmp(dso__long_name(dso), dso_name))
> continue;
> for (int ix = 0; 2 * ix + 1 < bid_len; ++ix) {
> - dso__bid(dso)->data[ix] = (hex(build_id[2 * ix]) << 4 |
> - hex(build_id[2 * ix + 1]));
> + bid.data[ix] = (hex(build_id[2 * ix]) << 4 |
> + hex(build_id[2 * ix + 1]));
> }
> - dso__bid(dso)->size = bid_len / 2;
> - dso__set_has_build_id(dso);
> + bid.size = bid_len / 2;
> + dso__set_build_id(dso, &bid);
> return true;
> }
> return false;
> diff --git a/tools/perf/builtin-report.c b/tools/perf/builtin-report.c
> index f0299c7ee025..98b1f73c28da 100644
> --- a/tools/perf/builtin-report.c
> +++ b/tools/perf/builtin-report.c
> @@ -858,17 +858,24 @@ static int maps__fprintf_task_cb(struct map *map, void *data)
> struct maps__fprintf_task_args *args = data;
> const struct dso *dso = map__dso(map);
> u32 prot = map__prot(map);
> + const struct dso_id *dso_id = dso__id_const(dso);
> int ret;
> + char buf[SBUILD_ID_SIZE];
> +
> + if (dso_id->mmap2_valid)
> + snprintf(buf, sizeof(buf), "%" PRIu64, dso_id->ino);
> + else
> + build_id__snprintf(&dso_id->build_id, buf, sizeof(buf));
>
> ret = fprintf(args->fp,
> - "%*s %" PRIx64 "-%" PRIx64 " %c%c%c%c %08" PRIx64 " %" PRIu64 " %s\n",
> + "%*s %" PRIx64 "-%" PRIx64 " %c%c%c%c %08" PRIx64 " %s %s\n",
> args->indent, "", map__start(map), map__end(map),
> prot & PROT_READ ? 'r' : '-',
> prot & PROT_WRITE ? 'w' : '-',
> prot & PROT_EXEC ? 'x' : '-',
> map__flags(map) ? 's' : 'p',
> map__pgoff(map),
> - dso__id_const(dso)->ino, dso__name(dso));
> + buf, dso__name(dso));
>
> if (ret < 0)
> return ret;
> diff --git a/tools/perf/include/perf/perf_dlfilter.h b/tools/perf/include/perf/perf_dlfilter.h
> index 16fc4568ac53..2d3540ed3c58 100644
> --- a/tools/perf/include/perf/perf_dlfilter.h
> +++ b/tools/perf/include/perf/perf_dlfilter.h
> @@ -87,7 +87,7 @@ struct perf_dlfilter_al {
> __u8 is_64_bit; /* Only valid if dso is not NULL */
> __u8 is_kernel_ip; /* True if in kernel space */
> __u32 buildid_size;
> - __u8 *buildid;
> + const __u8 *buildid;
> /* Below members are only populated by resolve_ip() */
> __u8 filtered; /* True if this sample event will be filtered out */
> const char *comm;
> diff --git a/tools/perf/tests/symbols.c b/tools/perf/tests/symbols.c
> index ee20a366f32f..b07fdf831868 100644
> --- a/tools/perf/tests/symbols.c
> +++ b/tools/perf/tests/symbols.c
> @@ -96,8 +96,8 @@ static int create_map(struct test_info *ti, char *filename, struct map **map_p)
> dso__put(dso);
>
> /* Create a dummy map at 0x100000 */
> - *map_p = map__new(ti->machine, 0x100000, 0xffffffff, 0, NULL,
> - PROT_EXEC, 0, NULL, filename, ti->thread);
> + *map_p = map__new(ti->machine, 0x100000, 0xffffffff, 0, &dso_id_empty,
> + PROT_EXEC, /*flags=*/0, filename, ti->thread);
> if (!*map_p) {
> pr_debug("Failed to create map!");
> return TEST_FAIL;
> diff --git a/tools/perf/util/build-id.c b/tools/perf/util/build-id.c
> index 3386fa8e1e7e..b8d784120f20 100644
> --- a/tools/perf/util/build-id.c
> +++ b/tools/perf/util/build-id.c
> @@ -251,7 +251,7 @@ char *__dso__build_id_filename(const struct dso *dso, char *bf, size_t size,
> if (!dso__has_build_id(dso))
> return NULL;
>
> - build_id__snprintf(dso__bid_const(dso), sbuild_id, sizeof(sbuild_id));
> + build_id__snprintf(dso__bid(dso), sbuild_id, sizeof(sbuild_id));
> linkname = build_id_cache__linkname(sbuild_id, NULL, 0);
> if (!linkname)
> return NULL;
> @@ -334,7 +334,7 @@ static int machine__write_buildid_table_cb(struct dso *dso, void *data)
> }
>
> in_kernel = dso__kernel(dso) || is_kernel_module(name, PERF_RECORD_MISC_CPUMODE_UNKNOWN);
> - return write_buildid(name, name_len, dso__bid(dso), args->machine->pid,
> + return write_buildid(name, name_len, &dso__id(dso)->build_id, args->machine->pid,
> in_kernel ? args->kmisc : args->umisc, args->fd);
> }
>
> diff --git a/tools/perf/util/dso.c b/tools/perf/util/dso.c
> index 776506b93b8b..493750142de7 100644
> --- a/tools/perf/util/dso.c
> +++ b/tools/perf/util/dso.c
> @@ -217,7 +217,7 @@ int dso__read_binary_type_filename(const struct dso *dso,
> break;
> }
>
> - build_id__snprintf(dso__bid_const(dso), build_id_hex, sizeof(build_id_hex));
> + build_id__snprintf(dso__bid(dso), build_id_hex, sizeof(build_id_hex));
> len = __symbol__join_symfs(filename, size, "/usr/lib/debug/.build-id/");
> snprintf(filename + len, size - len, "%.2s/%s.debug",
> build_id_hex, build_id_hex + 2);
> @@ -1379,64 +1379,76 @@ static void dso__set_long_name_id(struct dso *dso, const char *name, bool name_a
>
> static int __dso_id__cmp(const struct dso_id *a, const struct dso_id *b)
> {
> - if (a->maj > b->maj) return -1;
> - if (a->maj < b->maj) return 1;
> + if (a->mmap2_valid && b->mmap2_valid) {
> + if (a->maj > b->maj) return -1;
> + if (a->maj < b->maj) return 1;
>
> - if (a->min > b->min) return -1;
> - if (a->min < b->min) return 1;
> + if (a->min > b->min) return -1;
> + if (a->min < b->min) return 1;
>
> - if (a->ino > b->ino) return -1;
> - if (a->ino < b->ino) return 1;
> -
> - /*
> - * Synthesized MMAP events have zero ino_generation, avoid comparing
> - * them with MMAP events with actual ino_generation.
> - *
> - * I found it harmful because the mismatch resulted in a new
> - * dso that did not have a build ID whereas the original dso did have a
> - * build ID. The build ID was essential because the object was not found
> - * otherwise. - Adrian
> - */
> - if (a->ino_generation && b->ino_generation) {
> + if (a->ino > b->ino) return -1;
> + if (a->ino < b->ino) return 1;
> + }
> + if (a->mmap2_ino_generation_valid && b->mmap2_ino_generation_valid) {
> if (a->ino_generation > b->ino_generation) return -1;
> if (a->ino_generation < b->ino_generation) return 1;
> }
> -
> + if (build_id__is_defined(&a->build_id) && build_id__is_defined(&b->build_id)) {
> + if (a->build_id.size != b->build_id.size)
> + return a->build_id.size < b->build_id.size ? -1 : 1;
> + return memcmp(a->build_id.data, b->build_id.data, a->build_id.size);
> + }
> return 0;
> }
>
> -bool dso_id__empty(const struct dso_id *id)
> -{
> - if (!id)
> - return true;
> -
> - return !id->maj && !id->min && !id->ino && !id->ino_generation;
> -}
> +const struct dso_id dso_id_empty = {
> + {
> + .maj = 0,
> + .min = 0,
> + .ino = 0,
> + .ino_generation = 0,
> + },
> + .mmap2_valid = false,
> + .mmap2_ino_generation_valid = false,
> + {
> + .size = 0,
> + }
> +};
>
> -void __dso__inject_id(struct dso *dso, const struct dso_id *id)
> +void __dso__improve_id(struct dso *dso, const struct dso_id *id)
> {
> struct dsos *dsos = dso__dsos(dso);
> struct dso_id *dso_id = dso__id(dso);
> + bool changed = false;
>
> /* dsos write lock held by caller. */
>
> - dso_id->maj = id->maj;
> - dso_id->min = id->min;
> - dso_id->ino = id->ino;
> - dso_id->ino_generation = id->ino_generation;
> -
> - if (dsos)
> + if (id->mmap2_valid && !dso_id->mmap2_valid) {
> + dso_id->maj = id->maj;
> + dso_id->min = id->min;
> + dso_id->ino = id->ino;
> + dso_id->mmap2_valid = true;
> + changed = true;
> + }
> + if (id->mmap2_ino_generation_valid && !dso_id->mmap2_ino_generation_valid) {
> + dso_id->ino_generation = id->ino_generation;
> + dso_id->mmap2_ino_generation_valid = true;
> + changed = true;
> + }
> + if (build_id__is_defined(&id->build_id) && !build_id__is_defined(&dso_id->build_id)) {
> + dso_id->build_id = id->build_id;
> + changed = true;
> + }
> + if (changed && dsos)
> dsos->sorted = false;
> }
>
> int dso_id__cmp(const struct dso_id *a, const struct dso_id *b)
> {
> - /*
> - * The second is always dso->id, so zeroes if not set, assume passing
> - * NULL for a means a zeroed id
> - */
> - if (dso_id__empty(a) || dso_id__empty(b))
> + if (a == &dso_id_empty || b == &dso_id_empty) {
> + /* There is no valid data to compare so the comparison always returns identical. */
> return 0;
> + }
>
> return __dso_id__cmp(a, b);
> }
> @@ -1533,7 +1545,6 @@ struct dso *dso__new_id(const char *name, const struct dso_id *id)
> dso->loaded = 0;
> dso->rel = 0;
> dso->sorted_by_name = 0;
> - dso->has_build_id = 0;
> dso->has_srcline = 1;
> dso->a2l_fails = 1;
> dso->kernel = DSO_SPACE__USER;
> @@ -1638,15 +1649,14 @@ int dso__swap_init(struct dso *dso, unsigned char eidata)
> return 0;
> }
>
> -void dso__set_build_id(struct dso *dso, struct build_id *bid)
> +void dso__set_build_id(struct dso *dso, const struct build_id *bid)
> {
> - RC_CHK_ACCESS(dso)->bid = *bid;
> - RC_CHK_ACCESS(dso)->has_build_id = 1;
> + dso__id(dso)->build_id = *bid;
> }
>
> -bool dso__build_id_equal(const struct dso *dso, struct build_id *bid)
> +bool dso__build_id_equal(const struct dso *dso, const struct build_id *bid)
> {
> - const struct build_id *dso_bid = dso__bid_const(dso);
> + const struct build_id *dso_bid = dso__bid(dso);
>
> if (dso_bid->size > bid->size && dso_bid->size == BUILD_ID_SIZE) {
> /*
> @@ -1665,18 +1675,20 @@ bool dso__build_id_equal(const struct dso *dso, struct build_id *bid)
> void dso__read_running_kernel_build_id(struct dso *dso, struct machine *machine)
> {
> char path[PATH_MAX];
> + struct build_id bid;
>
> if (machine__is_default_guest(machine))
> return;
> sprintf(path, "%s/sys/kernel/notes", machine->root_dir);
> - if (sysfs__read_build_id(path, dso__bid(dso)) == 0)
> - dso__set_has_build_id(dso);
> + sysfs__read_build_id(path, &bid);
> + dso__set_build_id(dso, &bid);
Why not check the return value anymore?
Thanks,
Namhyung
> }
>
> int dso__kernel_module_get_build_id(struct dso *dso,
> const char *root_dir)
> {
> char filename[PATH_MAX];
> + struct build_id bid;
> /*
> * kernel module short names are of the form "[module]" and
> * we need just "module" here.
> @@ -1687,9 +1699,8 @@ int dso__kernel_module_get_build_id(struct dso *dso,
> "%s/sys/module/%.*s/notes/.note.gnu.build-id",
> root_dir, (int)strlen(name) - 1, name);
>
> - if (sysfs__read_build_id(filename, dso__bid(dso)) == 0)
> - dso__set_has_build_id(dso);
> -
> + sysfs__read_build_id(filename, &bid);
> + dso__set_build_id(dso, &bid);
> return 0;
> }
>
> diff --git a/tools/perf/util/dso.h b/tools/perf/util/dso.h
> index c87564471f9b..3457d713d3c5 100644
> --- a/tools/perf/util/dso.h
> +++ b/tools/perf/util/dso.h
> @@ -185,14 +185,33 @@ enum dso_load_errno {
> #define DSO__DATA_CACHE_SIZE 4096
> #define DSO__DATA_CACHE_MASK ~(DSO__DATA_CACHE_SIZE - 1)
>
> -/*
> - * Data about backing storage DSO, comes from PERF_RECORD_MMAP2 meta events
> +/**
> + * struct dso_id
> + *
> + * Data about backing storage DSO, comes from PERF_RECORD_MMAP2 meta events,
> + * reading from /proc/pid/maps or synthesis of build_ids from DSOs. Possibly
> + * incomplete at any particular use.
> */
> struct dso_id {
> - u32 maj;
> - u32 min;
> - u64 ino;
> - u64 ino_generation;
> + /* Data related to the mmap2 event or read from /proc/pid/maps. */
> + struct {
> + u32 maj;
> + u32 min;
> + u64 ino;
> + u64 ino_generation;
> + };
> + /** @mmap2_valid: Are the maj, min and ino fields valid? */
> + bool mmap2_valid;
> + /**
> + * @mmap2_ino_generation_valid: Is the ino_generation valid? Generally
> + * false for /proc/pid/maps mmap event.
> + */
> + bool mmap2_ino_generation_valid;
> + /**
> + * @build_id: A possibly populated build_id. build_id__is_defined checks
> + * whether it is populated.
> + */
> + struct build_id build_id;
> };
>
> struct dso_cache {
> @@ -243,7 +262,6 @@ DECLARE_RC_STRUCT(dso) {
> u64 addr;
> struct symbol *symbol;
> } last_find_result;
> - struct build_id bid;
> u64 text_offset;
> u64 text_end;
> const char *short_name;
> @@ -276,7 +294,6 @@ DECLARE_RC_STRUCT(dso) {
> enum dso_swap_type needs_swap:2;
> bool is_kmod:1;
> u8 adjust_symbols:1;
> - u8 has_build_id:1;
> u8 header_build_id:1;
> u8 has_srcline:1;
> u8 hit:1;
> @@ -292,6 +309,9 @@ DECLARE_RC_STRUCT(dso) {
> };
>
> extern struct mutex _dso__data_open_lock;
> +extern const struct dso_id dso_id_empty;
> +
> +int dso_id__cmp(const struct dso_id *a, const struct dso_id *b);
>
> /* dso__for_each_symbol - iterate over the symbols of given type
> *
> @@ -362,31 +382,11 @@ static inline void dso__set_auxtrace_cache(struct dso *dso, struct auxtrace_cach
> RC_CHK_ACCESS(dso)->auxtrace_cache = cache;
> }
>
> -static inline struct build_id *dso__bid(struct dso *dso)
> -{
> - return &RC_CHK_ACCESS(dso)->bid;
> -}
> -
> -static inline const struct build_id *dso__bid_const(const struct dso *dso)
> -{
> - return &RC_CHK_ACCESS(dso)->bid;
> -}
> -
> static inline struct dso_bpf_prog *dso__bpf_prog(struct dso *dso)
> {
> return &RC_CHK_ACCESS(dso)->bpf_prog;
> }
>
> -static inline bool dso__has_build_id(const struct dso *dso)
> -{
> - return RC_CHK_ACCESS(dso)->has_build_id;
> -}
> -
> -static inline void dso__set_has_build_id(struct dso *dso)
> -{
> - RC_CHK_ACCESS(dso)->has_build_id = true;
> -}
> -
> static inline bool dso__has_srcline(const struct dso *dso)
> {
> return RC_CHK_ACCESS(dso)->has_srcline;
> @@ -462,6 +462,16 @@ static inline const struct dso_id *dso__id_const(const struct dso *dso)
> return &RC_CHK_ACCESS(dso)->id;
> }
>
> +static inline const struct build_id *dso__bid(const struct dso *dso)
> +{
> + return &dso__id_const(dso)->build_id;
> +}
> +
> +static inline bool dso__has_build_id(const struct dso *dso)
> +{
> + return build_id__is_defined(dso__bid(dso));
> +}
> +
> static inline struct rb_root_cached *dso__inlined_nodes(struct dso *dso)
> {
> return &RC_CHK_ACCESS(dso)->inlined_nodes;
> @@ -699,9 +709,6 @@ static inline void dso__set_text_offset(struct dso *dso, u64 val)
> RC_CHK_ACCESS(dso)->text_offset = val;
> }
>
> -int dso_id__cmp(const struct dso_id *a, const struct dso_id *b);
> -bool dso_id__empty(const struct dso_id *id);
> -
> struct dso *dso__new_id(const char *name, const struct dso_id *id);
> struct dso *dso__new(const char *name);
> void dso__delete(struct dso *dso);
> @@ -709,7 +716,7 @@ void dso__delete(struct dso *dso);
> int dso__cmp_id(struct dso *a, struct dso *b);
> void dso__set_short_name(struct dso *dso, const char *name, bool name_allocated);
> void dso__set_long_name(struct dso *dso, const char *name, bool name_allocated);
> -void __dso__inject_id(struct dso *dso, const struct dso_id *id);
> +void __dso__improve_id(struct dso *dso, const struct dso_id *id);
>
> int dso__name_len(const struct dso *dso);
>
> @@ -739,8 +746,8 @@ void dso__sort_by_name(struct dso *dso);
>
> int dso__swap_init(struct dso *dso, unsigned char eidata);
>
> -void dso__set_build_id(struct dso *dso, struct build_id *bid);
> -bool dso__build_id_equal(const struct dso *dso, struct build_id *bid);
> +void dso__set_build_id(struct dso *dso, const struct build_id *bid);
> +bool dso__build_id_equal(const struct dso *dso, const struct build_id *bid);
> void dso__read_running_kernel_build_id(struct dso *dso,
> struct machine *machine);
> int dso__kernel_module_get_build_id(struct dso *dso, const char *root_dir);
> diff --git a/tools/perf/util/dsos.c b/tools/perf/util/dsos.c
> index b2172632b3cd..ad919ed28ba4 100644
> --- a/tools/perf/util/dsos.c
> +++ b/tools/perf/util/dsos.c
> @@ -72,6 +72,7 @@ static int dsos__read_build_ids_cb(struct dso *dso, void *data)
> {
> struct dsos__read_build_ids_cb_args *args = data;
> struct nscookie nsc;
> + struct build_id bid;
>
> if (args->with_hits && !dso__hit(dso) && !dso__is_vdso(dso))
> return 0;
> @@ -80,15 +81,15 @@ static int dsos__read_build_ids_cb(struct dso *dso, void *data)
> return 0;
> }
> nsinfo__mountns_enter(dso__nsinfo(dso), &nsc);
> - if (filename__read_build_id(dso__long_name(dso), dso__bid(dso)) > 0) {
> + if (filename__read_build_id(dso__long_name(dso), &bid) > 0) {
> + dso__set_build_id(dso, &bid);
> args->have_build_id = true;
> - dso__set_has_build_id(dso);
> } else if (errno == ENOENT && dso__nsinfo(dso)) {
> char *new_name = dso__filename_with_chroot(dso, dso__long_name(dso));
>
> - if (new_name && filename__read_build_id(new_name, dso__bid(dso)) > 0) {
> + if (new_name && filename__read_build_id(new_name, &bid) > 0) {
> + dso__set_build_id(dso, &bid);
> args->have_build_id = true;
> - dso__set_has_build_id(dso);
> }
> free(new_name);
> }
> @@ -284,7 +285,7 @@ struct dso *dsos__find(struct dsos *dsos, const char *name, bool cmp_short)
> struct dso *res;
>
> down_read(&dsos->lock);
> - res = __dsos__find_id(dsos, name, NULL, cmp_short, /*write_locked=*/false);
> + res = __dsos__find_id(dsos, name, &dso_id_empty, cmp_short, /*write_locked=*/false);
> up_read(&dsos->lock);
> return res;
> }
> @@ -341,8 +342,8 @@ static struct dso *__dsos__findnew_id(struct dsos *dsos, const char *name, const
> {
> struct dso *dso = __dsos__find_id(dsos, name, id, false, /*write_locked=*/true);
>
> - if (dso && dso_id__empty(dso__id(dso)) && !dso_id__empty(id))
> - __dso__inject_id(dso, id);
> + if (dso)
> + __dso__improve_id(dso, id);
>
> return dso ? dso : __dsos__addnew_id(dsos, name, id);
> }
> @@ -433,7 +434,8 @@ struct dso *dsos__findnew_module_dso(struct dsos *dsos,
>
> down_write(&dsos->lock);
>
> - dso = __dsos__find_id(dsos, m->name, NULL, /*cmp_short=*/true, /*write_locked=*/true);
> + dso = __dsos__find_id(dsos, m->name, &dso_id_empty, /*cmp_short=*/true,
> + /*write_locked=*/true);
> if (dso) {
> up_write(&dsos->lock);
> return dso;
> diff --git a/tools/perf/util/machine.c b/tools/perf/util/machine.c
> index b048165b10c1..04062148a9ec 100644
> --- a/tools/perf/util/machine.c
> +++ b/tools/perf/util/machine.c
> @@ -1696,21 +1696,21 @@ int machine__process_mmap2_event(struct machine *machine,
> {
> struct thread *thread;
> struct map *map;
> - struct dso_id dso_id = {
> - .maj = event->mmap2.maj,
> - .min = event->mmap2.min,
> - .ino = event->mmap2.ino,
> - .ino_generation = event->mmap2.ino_generation,
> - };
> - struct build_id __bid, *bid = NULL;
> + struct dso_id dso_id = dso_id_empty;
> int ret = 0;
>
> if (dump_trace)
> perf_event__fprintf_mmap2(event, stdout);
>
> if (event->header.misc & PERF_RECORD_MISC_MMAP_BUILD_ID) {
> - bid = &__bid;
> - build_id__init(bid, event->mmap2.build_id, event->mmap2.build_id_size);
> + build_id__init(&dso_id.build_id, event->mmap2.build_id, event->mmap2.build_id_size);
> + } else {
> + dso_id.maj = event->mmap2.maj;
> + dso_id.min = event->mmap2.min;
> + dso_id.ino = event->mmap2.ino;
> + dso_id.ino_generation = event->mmap2.ino_generation;
> + dso_id.mmap2_valid = true;
> + dso_id.mmap2_ino_generation_valid = true;
> }
>
> if (sample->cpumode == PERF_RECORD_MISC_GUEST_KERNEL ||
> @@ -1722,7 +1722,7 @@ int machine__process_mmap2_event(struct machine *machine,
> };
>
> strlcpy(xm.name, event->mmap2.filename, KMAP_NAME_LEN);
> - ret = machine__process_kernel_mmap_event(machine, &xm, bid);
> + ret = machine__process_kernel_mmap_event(machine, &xm, &dso_id.build_id);
> if (ret < 0)
> goto out_problem;
> return 0;
> @@ -1736,7 +1736,7 @@ int machine__process_mmap2_event(struct machine *machine,
> map = map__new(machine, event->mmap2.start,
> event->mmap2.len, event->mmap2.pgoff,
> &dso_id, event->mmap2.prot,
> - event->mmap2.flags, bid,
> + event->mmap2.flags,
> event->mmap2.filename, thread);
>
> if (map == NULL)
> @@ -1794,8 +1794,8 @@ int machine__process_mmap_event(struct machine *machine, union perf_event *event
> prot = PROT_EXEC;
>
> map = map__new(machine, event->mmap.start,
> - event->mmap.len, event->mmap.pgoff,
> - NULL, prot, 0, NULL, event->mmap.filename, thread);
> + event->mmap.len, event->mmap.pgoff,
> + &dso_id_empty, prot, /*flags=*/0, event->mmap.filename, thread);
>
> if (map == NULL)
> goto out_problem_map;
> @@ -3157,7 +3157,7 @@ struct dso *machine__findnew_dso_id(struct machine *machine, const char *filenam
>
> struct dso *machine__findnew_dso(struct machine *machine, const char *filename)
> {
> - return machine__findnew_dso_id(machine, filename, NULL);
> + return machine__findnew_dso_id(machine, filename, &dso_id_empty);
> }
>
> char *machine__resolve_kernel_addr(void *vmachine, unsigned long long *addrp, char **modp)
> diff --git a/tools/perf/util/map.c b/tools/perf/util/map.c
> index 8858520c16f4..41cdddc987ee 100644
> --- a/tools/perf/util/map.c
> +++ b/tools/perf/util/map.c
> @@ -120,8 +120,8 @@ static void map__init(struct map *map, u64 start, u64 end, u64 pgoff,
> }
>
> struct map *map__new(struct machine *machine, u64 start, u64 len,
> - u64 pgoff, struct dso_id *id,
> - u32 prot, u32 flags, struct build_id *bid,
> + u64 pgoff, const struct dso_id *id,
> + u32 prot, u32 flags,
> char *filename, struct thread *thread)
> {
> struct map *result;
> @@ -132,7 +132,7 @@ struct map *map__new(struct machine *machine, u64 start, u64 len,
> map = zalloc(sizeof(*map));
> if (ADD_RC_CHK(result, map)) {
> char newfilename[PATH_MAX];
> - struct dso *dso, *header_bid_dso;
> + struct dso *dso;
> int anon, no_dso, vdso, android;
>
> android = is_android_lib(filename);
> @@ -189,16 +189,15 @@ struct map *map__new(struct machine *machine, u64 start, u64 len,
> dso__set_nsinfo(dso, nsi);
> mutex_unlock(dso__lock(dso));
>
> - if (build_id__is_defined(bid)) {
> - dso__set_build_id(dso, bid);
> - } else {
> + if (!build_id__is_defined(&id->build_id)) {
> /*
> * If the mmap event had no build ID, search for an existing dso from the
> * build ID header by name. Otherwise only the dso loaded at the time of
> * reading the header will have the build ID set and all future mmaps will
> * have it missing.
> */
> - header_bid_dso = dsos__find(&machine->dsos, filename, false);
> + struct dso *header_bid_dso = dsos__find(&machine->dsos, filename, false);
> +
> if (header_bid_dso && dso__header_build_id(header_bid_dso)) {
> dso__set_build_id(dso, dso__bid(header_bid_dso));
> dso__set_header_build_id(dso, 1);
> diff --git a/tools/perf/util/map.h b/tools/perf/util/map.h
> index 768501eec70e..979b3e11b9bc 100644
> --- a/tools/perf/util/map.h
> +++ b/tools/perf/util/map.h
> @@ -173,11 +173,10 @@ struct thread;
> __map__for_each_symbol_by_name(map, sym_name, (pos), idx)
>
> struct dso_id;
> -struct build_id;
>
> struct map *map__new(struct machine *machine, u64 start, u64 len,
> - u64 pgoff, struct dso_id *id, u32 prot, u32 flags,
> - struct build_id *bid, char *filename, struct thread *thread);
> + u64 pgoff, const struct dso_id *id, u32 prot, u32 flags,
> + char *filename, struct thread *thread);
> struct map *map__new2(u64 start, struct dso *dso);
> void map__delete(struct map *map);
> struct map *map__clone(struct map *map);
> diff --git a/tools/perf/util/sort.c b/tools/perf/util/sort.c
> index 594b75ca95bf..d20b980d5052 100644
> --- a/tools/perf/util/sort.c
> +++ b/tools/perf/util/sort.c
> @@ -1709,22 +1709,27 @@ sort__dcacheline_cmp(struct hist_entry *left, struct hist_entry *right)
> if (rc)
> return rc;
> /*
> - * Addresses with no major/minor numbers are assumed to be
> + * Addresses with no major/minor numbers or build ID are assumed to be
> * anonymous in userspace. Sort those on pid then address.
> *
> * The kernel and non-zero major/minor mapped areas are
> * assumed to be unity mapped. Sort those on address.
> */
> + if (left->cpumode != PERF_RECORD_MISC_KERNEL && (map__flags(l_map) & MAP_SHARED) == 0) {
> + const struct dso_id *dso_id = dso__id_const(l_dso);
>
> - if ((left->cpumode != PERF_RECORD_MISC_KERNEL) &&
> - (!(map__flags(l_map) & MAP_SHARED)) && !dso__id(l_dso)->maj && !dso__id(l_dso)->min &&
> - !dso__id(l_dso)->ino && !dso__id(l_dso)->ino_generation) {
> - /* userspace anonymous */
> + if (!dso_id->mmap2_valid)
> + dso_id = dso__id_const(r_dso);
>
> - if (thread__pid(left->thread) > thread__pid(right->thread))
> - return -1;
> - if (thread__pid(left->thread) < thread__pid(right->thread))
> - return 1;
> + if (!build_id__is_defined(&dso_id->build_id) &&
> + (!dso_id->mmap2_valid || (dso_id->maj == 0 && dso_id->min == 0))) {
> + /* userspace anonymous */
> +
> + if (thread__pid(left->thread) > thread__pid(right->thread))
> + return -1;
> + if (thread__pid(left->thread) < thread__pid(right->thread))
> + return 1;
> + }
> }
>
> addr:
> @@ -1749,6 +1754,7 @@ static int hist_entry__dcacheline_snprintf(struct hist_entry *he, char *bf,
> if (he->mem_info) {
> struct map *map = mem_info__daddr(he->mem_info)->ms.map;
> struct dso *dso = map ? map__dso(map) : NULL;
> + const struct dso_id *dso_id = dso ? dso__id_const(dso) : &dso_id_empty;
>
> addr = cl_address(mem_info__daddr(he->mem_info)->al_addr, chk_double_cl);
> ms = &mem_info__daddr(he->mem_info)->ms;
> @@ -1757,8 +1763,7 @@ static int hist_entry__dcacheline_snprintf(struct hist_entry *he, char *bf,
> if ((he->cpumode != PERF_RECORD_MISC_KERNEL) &&
> map && !(map__prot(map) & PROT_EXEC) &&
> (map__flags(map) & MAP_SHARED) &&
> - (dso__id(dso)->maj || dso__id(dso)->min || dso__id(dso)->ino ||
> - dso__id(dso)->ino_generation))
> + (!dso_id->mmap2_valid || (dso_id->maj == 0 && dso_id->min == 0)))
> level = 's';
> else if (!map)
> level = 'X';
> diff --git a/tools/perf/util/synthetic-events.c b/tools/perf/util/synthetic-events.c
> index 68bb7c5fe1b1..8fb2ea544d3a 100644
> --- a/tools/perf/util/synthetic-events.c
> +++ b/tools/perf/util/synthetic-events.c
> @@ -372,7 +372,7 @@ static void perf_record_mmap2__read_build_id(struct perf_record_mmap2 *event,
> struct nsinfo *nsi;
> struct nscookie nc;
> struct dso *dso = NULL;
> - struct dso_id id;
> + struct dso_id dso_id = dso_id_empty;
> int rc;
>
> if (is_kernel) {
> @@ -380,12 +380,18 @@ static void perf_record_mmap2__read_build_id(struct perf_record_mmap2 *event,
> goto out;
> }
>
> - id.maj = event->maj;
> - id.min = event->min;
> - id.ino = event->ino;
> - id.ino_generation = event->ino_generation;
> + if (event->header.misc & PERF_RECORD_MISC_MMAP_BUILD_ID) {
> + build_id__init(&dso_id.build_id, event->build_id, event->build_id_size);
> + } else {
> + dso_id.maj = event->maj;
> + dso_id.min = event->min;
> + dso_id.ino = event->ino;
> + dso_id.ino_generation = event->ino_generation;
> + dso_id.mmap2_valid = true;
> + dso_id.mmap2_ino_generation_valid = true;
> + };
>
> - dso = dsos__findnew_id(&machine->dsos, event->filename, &id);
> + dso = dsos__findnew_id(&machine->dsos, event->filename, &dso_id);
> if (dso && dso__has_build_id(dso)) {
> bid = *dso__bid(dso);
> rc = 0;
> --
> 2.49.0.850.g28803427d3-goog
>
Powered by blists - more mailing lists