Linux Perf Users
 help / color / mirror / Atom feed
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

  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