From: sashiko-bot@kernel.org
To: "Arnaldo Carvalho de Melo" <acme@kernel.org>
Cc: linux-perf-users@vger.kernel.org
Subject: Re: [PATCH 17/29] perf tools: Bounds check perf_event_attr fields against attr.size before printing
Date: Sun, 24 May 2026 04:01:54 +0000 [thread overview]
Message-ID: <20260524040154.D68C91F000E9@smtp.kernel.org> (raw)
In-Reply-To: <20260524032709.1080771-18-acme@kernel.org>
Thank you for your contribution! Sashiko AI review found 2 potential issue(s) to consider:
New issues:
- [High] The newly added fallback logic in `syscall_arg__scnprintf_augmented_perf_event_attr` blindly copies 64 bytes from the payload, causing an out-of-bounds read if the actual payload is smaller.
Pre-existing issues:
- [High] `perf_event_attr__fprintf()` trusts the user-controlled `attr->size` field for bounds checking without validating it against the actual buffer limits in the `perf trace` path.
--
commit 45122ffcfc4c42d20283944fff0ce0ea20c80919
Author: Arnaldo Carvalho de Melo <acme@redhat.com>
perf tools: Bounds check perf_event_attr fields against attr.size before printing
perf_event_attr__fprintf() accessed all struct fields unconditionally,
but attrs from older perf.data files or BPF-captured syscall payloads
may have a smaller size than the current struct. Fields beyond the
recorded size contain uninitialized or zero-filled data.
> diff --git a/tools/perf/trace/beauty/perf_event_open.c b/tools/perf/trace/beauty/perf_event_open.c
> index c1c7445dcff99..6315b46bcdf02 100644
> --- a/tools/perf/trace/beauty/perf_event_open.c
> +++ b/tools/perf/trace/beauty/perf_event_open.c
[ ... ]
> @@ -80,7 +81,27 @@ static size_t perf_event_attr___scnprintf(struct perf_event_attr *attr, char *bf
>
> static size_t syscall_arg__scnprintf_augmented_perf_event_attr(struct syscall_arg *arg, char *bf, size_t size)
> {
> - return perf_event_attr___scnprintf((void *)arg->augmented.args->value, bf, size,
> + struct perf_event_attr *attr = (void *)arg->augmented.args->value;
> + struct perf_event_attr local_attr;
> +
> + /*
> + * augmented_raw_syscalls.bpf.c (shipped with perf) copies
> + * PERF_ATTR_SIZE_VER0 bytes when the tracee passes size=0,
> + * but leaves the size field as 0. The payload size is
> + * guaranteed by perf's own BPF program, not externally
> + * controllable. Copy to a local so we can fix up size
> + * without writing to the potentially read-only augmented
> + * args buffer.
> + */
> + if (!attr->size) {
> + memcpy(&local_attr, attr, PERF_ATTR_SIZE_VER0);
[Severity: High]
Can this memcpy() read out of bounds if the actual payload is smaller than
PERF_ATTR_SIZE_VER0?
If a tracee uses a TOCTOU race to pass a small payload size (like 8 bytes)
to the first read, but modifies attr->size to 0 before the second read, the
user-space payload might only be 8 bytes long.
Since the BPF program sys_enter_perf_event_open dynamically sizes its output
based on the tracee's attr->size using bpf_perf_event_output(), could blindly
copying 64 bytes here read past the end of the perf ring buffer and cause a
crash?
> + memset((void *)&local_attr + PERF_ATTR_SIZE_VER0, 0,
> + sizeof(local_attr) - PERF_ATTR_SIZE_VER0);
> + local_attr.size = PERF_ATTR_SIZE_VER0;
> + attr = &local_attr;
> + }
> +
> + return perf_event_attr___scnprintf(attr, bf, size,
> trace__show_zeros(arg->trace));
> }
[ ... ]
> diff --git a/tools/perf/util/perf_event_attr_fprintf.c b/tools/perf/util/perf_event_attr_fprintf.c
> index 741c3d657a8b6..3933639d76c54 100644
> --- a/tools/perf/util/perf_event_attr_fprintf.c
> +++ b/tools/perf/util/perf_event_attr_fprintf.c
[ ... ]
> @@ -275,24 +275,56 @@ static void __p_config_id(struct perf_pmu *pmu, char *buf, size_t size, u32 type
[ ... ]
> int perf_event_attr__fprintf(FILE *fp, struct perf_event_attr *attr,
> attr__fprintf_f attr__fprintf, void *priv)
> {
> struct perf_pmu *pmu = perf_pmus__find_by_type(attr->type);
> + /*
> + * size == 0 means ABI0 — the producer didn't set attr.size.
> + * perf_event__fprintf_attr() may pass the raw mmap'd event
> + * before the local copy, so default to PERF_ATTR_SIZE_VER0
> + * (the ABI0 footprint) to avoid reading past the attr into
> + * the ID array that follows it in HEADER_ATTR events.
> + */
> + u32 attr_size = attr->size ?: PERF_ATTR_SIZE_VER0;
> char buf[BUF_SIZE];
> int ret = 0;
>
> - if (!pmu && (attr->type == PERF_TYPE_HARDWARE || attr->type == PERF_TYPE_HW_CACHE)) {
> + /*
> + * Cap to what we understand: all callers store the attr in a
> + * buffer of sizeof(*attr) bytes (perf.data read path copies
> + * min(attr.size, sizeof), BPF augmented path copies into a
> + * fixed-size value[] array). A spoofed attr->size larger
> + * than sizeof would cause PRINT_ATTRn to read past the
> + * actual buffer.
> + */
> + if (attr_size > sizeof(*attr))
> + attr_size = sizeof(*attr);
[Severity: High]
This is a pre-existing issue, but does capping attr_size to sizeof(*attr)
fully prevent out-of-bounds reads in the perf trace path?
The comment mentions the BPF augmented path copies into a fixed-size value[]
array, but augmented__output actually trims the emitted ring buffer event to
exactly the copied size (len + size).
If a tracee uses a TOCTOU race to supply a small payload while spoofing
attr->size to 136, this function will use 136 as the size. Will the PRINT_ATTRn
macros then read up to 136 bytes from a much smaller payload, resulting in
an out-of-bounds read?
> +
> + if (!pmu && attr_size >= offsetof(struct perf_event_attr, config) + sizeof(attr->config) &&
> + (attr->type == PERF_TYPE_HARDWARE || attr->type == PERF_TYPE_HW_CACHE)) {
--
Sashiko AI review · https://sashiko.dev/#/patchset/20260524032709.1080771-1-acme@kernel.org?part=17
next prev parent reply other threads:[~2026-05-24 4:01 UTC|newest]
Thread overview: 46+ messages / expand[flat|nested] mbox.gz Atom feed top
2026-05-24 3:26 [PATCHES v2 00/29] perf: Harden perf.data parsing against crafted/corrupted files Arnaldo Carvalho de Melo
2026-05-24 3:26 ` [PATCH 01/29] perf session: Add minimum event size and alignment validation Arnaldo Carvalho de Melo
2026-05-24 4:13 ` sashiko-bot
2026-05-24 3:26 ` [PATCH 02/29] perf session: Bounds-check one_mmap event pointer in peek_event Arnaldo Carvalho de Melo
2026-05-24 4:03 ` sashiko-bot
2026-05-24 3:26 ` [PATCH 03/29] perf tools: Fix event_contains() macro to verify full field extent Arnaldo Carvalho de Melo
2026-05-24 3:26 ` [PATCH 04/29] perf zstd: Fix compression error path in zstd_compress_stream_to_records() Arnaldo Carvalho de Melo
2026-05-24 4:06 ` sashiko-bot
2026-05-24 3:26 ` [PATCH 05/29] perf zstd: Fix multi-iteration decompression and error handling Arnaldo Carvalho de Melo
2026-05-24 3:26 ` [PATCH 06/29] perf session: Fix PERF_RECORD_READ swap and dump for variable-length events Arnaldo Carvalho de Melo
2026-05-24 3:26 ` [PATCH 07/29] perf session: Fix swap_sample_id_all() crash on crafted events Arnaldo Carvalho de Melo
2026-05-24 3:26 ` [PATCH 08/29] perf session: Add validated swap infrastructure with null-termination checks Arnaldo Carvalho de Melo
2026-05-24 4:05 ` sashiko-bot
2026-05-24 3:26 ` [PATCH 09/29] perf session: Use bounded copy for PERF_RECORD_TIME_CONV Arnaldo Carvalho de Melo
2026-05-24 3:26 ` [PATCH 10/29] perf session: Validate HEADER_ATTR attr.size before swapping Arnaldo Carvalho de Melo
2026-05-24 4:08 ` sashiko-bot
2026-05-24 3:26 ` [PATCH 11/29] perf session: Validate nr fields against event size on both swap and common paths Arnaldo Carvalho de Melo
2026-05-24 3:26 ` [PATCH 12/29] perf header: Byte-swap build ID event pid and bounds check section entries Arnaldo Carvalho de Melo
2026-05-24 4:08 ` sashiko-bot
2026-05-24 3:26 ` [PATCH 13/29] perf cpumap: Reject RANGE_CPUS with start_cpu > end_cpu Arnaldo Carvalho de Melo
2026-05-24 4:04 ` sashiko-bot
2026-05-24 3:26 ` [PATCH 14/29] perf auxtrace: Harden auxtrace_error event handling Arnaldo Carvalho de Melo
2026-05-24 3:26 ` [PATCH 15/29] perf session: Add byte-swap and bounds check for PERF_RECORD_BPF_METADATA events Arnaldo Carvalho de Melo
2026-05-24 4:13 ` sashiko-bot
2026-05-24 3:26 ` [PATCH 16/29] perf header: Validate null-termination in PERF_RECORD_EVENT_UPDATE string fields Arnaldo Carvalho de Melo
2026-05-24 3:26 ` [PATCH 17/29] perf tools: Bounds check perf_event_attr fields against attr.size before printing Arnaldo Carvalho de Melo
2026-05-24 4:01 ` sashiko-bot [this message]
2026-05-24 3:26 ` [PATCH 18/29] perf header: Propagate feature section processing errors Arnaldo Carvalho de Melo
2026-05-24 3:26 ` [PATCH 19/29] perf header: Validate f_attr.ids section before use in perf_session__read_header() Arnaldo Carvalho de Melo
2026-05-24 3:26 ` [PATCH 20/29] perf header: Validate feature section size and add read path bounds checking Arnaldo Carvalho de Melo
2026-05-24 4:37 ` sashiko-bot
2026-05-24 3:26 ` [PATCH 21/29] perf header: Sanity check HEADER_EVENT_DESC attr.size before swap Arnaldo Carvalho de Melo
2026-05-24 3:26 ` [PATCH 22/29] perf header: Validate bitmap size before allocating in do_read_bitmap() Arnaldo Carvalho de Melo
2026-05-24 3:26 ` [PATCH 23/29] perf session: Add byte-swap handler for PERF_RECORD_COMPRESSED2 Arnaldo Carvalho de Melo
2026-05-24 3:26 ` [PATCH 24/29] perf tools: Harden compressed event processing Arnaldo Carvalho de Melo
2026-05-24 4:35 ` sashiko-bot
2026-05-24 3:26 ` [PATCH 25/29] perf session: Check for decompression buffer size overflow Arnaldo Carvalho de Melo
2026-05-24 3:27 ` [PATCH 26/29] perf session: Bound nr_cpus_avail and validate sample CPU Arnaldo Carvalho de Melo
2026-05-24 6:23 ` sashiko-bot
2026-05-24 3:27 ` [PATCH 27/29] perf kwork: Bounds check work->cpu before indexing cpus_runtime[] Arnaldo Carvalho de Melo
2026-05-24 3:27 ` [PATCH 28/29] perf session: Snapshot event->header.size in process_user_event() Arnaldo Carvalho de Melo
2026-05-24 4:31 ` sashiko-bot
2026-05-24 3:27 ` [PATCH 29/29] perf test: Add truncated perf.data robustness test Arnaldo Carvalho de Melo
-- strict thread matches above, loose matches on Subject: below --
2026-05-25 1:05 [PATCHES v3 00/29] perf: Harden perf.data parsing against crafted/corrupted files Arnaldo Carvalho de Melo
2026-05-25 1:05 ` [PATCH 17/29] perf tools: Bounds check perf_event_attr fields against attr.size before printing Arnaldo Carvalho de Melo
2026-05-25 1:59 ` sashiko-bot
2026-05-26 21:17 [PATCHES v4 00/29] perf: Harden perf.data parsing against crafted/corrupted files Arnaldo Carvalho de Melo
2026-05-26 21:17 ` [PATCH 17/29] perf tools: Bounds check perf_event_attr fields against attr.size before printing Arnaldo Carvalho de Melo
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=20260524040154.D68C91F000E9@smtp.kernel.org \
--to=sashiko-bot@kernel.org \
--cc=acme@kernel.org \
--cc=linux-perf-users@vger.kernel.org \
--cc=sashiko-reviews@lists.linux.dev \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox