All of lore.kernel.org
 help / color / mirror / Atom feed
From: sashiko-bot@kernel.org
To: "Jens Remus" <jremus@linux.ibm.com>
Cc: bpf@vger.kernel.org
Subject: Re: [PATCH v16 03/20] unwind_user/sframe: Store .sframe section data in per-mm maple tree
Date: Thu, 21 May 2026 15:04:26 +0000	[thread overview]
Message-ID: <20260521150427.8BE671F000E9@smtp.kernel.org> (raw)
In-Reply-To: <20260521142546.3908498-4-jremus@linux.ibm.com>

Thank you for your contribution! Sashiko AI review found 5 potential issue(s) to consider:
- [High] Concurrent `sframe_remove_section` calls lead to a TOCTOU race causing SRCU double-queueing and memory leaks.
- [High] The `sframe_mt` maple tree is initialized without `MT_FLAGS_USE_RCU`, causing use-after-free during lockless traversal.
- [High] `sframe_free_mm()` bypasses SRCU and synchronously frees objects, racing with concurrent SRCU readers.
- [Low] Mixing `scoped_guard()` with goto-based error handling violates subsystem cleanup rules.
- [Low] `current_has_sframe()` dereferences `current->mm` without including `<linux/sched.h>`, causing incomplete type errors.
--

commit 4b0aee91d63748e3221127b74ad789be7e3d2c6b
Author: Josh Poimboeuf <jpoimboe@kernel.org>
Subject: unwind_user/sframe: Store .sframe section data in per-mm maple tree

Associate an .sframe section with its mm by adding it to a per-mm maple
tree which is indexed by the corresponding text address range.  A single
.sframe section can be associated with multiple text ranges.

> diff --git a/include/linux/sframe.h b/include/linux/sframe.h
> index 0642595534f95..38047760e252e 100644
> --- a/include/linux/sframe.h
> +++ b/include/linux/sframe.h
> @@ -19,18 +24,31 @@ struct sframe_section {
>  	signed char	fp_off;
>  };
>  
> +#define INIT_MM_SFRAME .sframe_mt = MTREE_INIT(sframe_mt, 0),

[Severity: High]
Does this maple tree initialization need the MT_FLAGS_USE_RCU flag?

Since sframe_remove_section() traverses the tree locklessly using
mt_for_each(), omitting the RCU flag means internal tree nodes split or
merged by concurrent updates could be freed immediately via
kmem_cache_free(), potentially causing a use-after-free for the
lockless readers.

> +extern void sframe_free_mm(struct mm_struct *mm);
> +
>  extern int sframe_add_section(unsigned long sframe_start, unsigned long sframe_end,
>  			      unsigned long text_start, unsigned long text_end);
>  extern int sframe_remove_section(unsigned long sframe_addr);
>  
> +static inline bool current_has_sframe(void)
> +{
> +	struct mm_struct *mm = current->mm;
> +
> +	return mm && !mtree_empty(&mm->sframe_mt);
> +}

[Severity: Low]
Will this compile cleanly in all contexts?

The header dereferences current->mm, but <linux/sched.h> doesn't appear
to be included here. Files including <linux/sframe.h> without also
including sched.h might fail to compile with incomplete type errors.

> diff --git a/kernel/unwind/sframe.c b/kernel/unwind/sframe.c
> index d24e9d4f8bef7..46dba3cb016df 100644
> --- a/kernel/unwind/sframe.c
> +++ b/kernel/unwind/sframe.c
[ ... ]
> @@ -122,15 +125,74 @@ int sframe_add_section(unsigned long sframe_start, unsigned long sframe_end,
>  	if (ret)
>  		goto err_free;
>  
> -	/* TODO nowhere to store it yet - just free it and return an error */
> -	ret = -ENOSYS;
> +	ret = mtree_insert_range(sframe_mt, sec->text_start, sec->text_end - 1,
> +				 sec, GFP_KERNEL_ACCOUNT);
> +	if (ret) {
> +		dbg("mtree_insert_range failed: text=%lx-%lx\n",
> +		    sec->text_start, sec->text_end);
> +		goto err_free;
> +	}
> +
> +	return 0;
>  
>  err_free:
>  	free_section(sec);
>  	return ret;
>  }

[Severity: Low]
This isn't a bug, but does mixing goto-based error handling (err_free)
with the scoped_guard() used earlier in this function violate the
cleanup subsystem guidelines?

The standard suggests managing the sec pointer via __free(kfree) instead
of using goto labels to avoid confusing ownership semantics.

> +static int __sframe_remove_section(struct mm_struct *mm,
> +				   struct sframe_section *sec)
> +{
> +	if (!mtree_erase(&mm->sframe_mt, sec->text_start)) {
> +		dbg("mtree_erase failed: text=%lx\n", sec->text_start);
> +		return -EINVAL;
> +	}
> +
> +	call_srcu(&sframe_srcu, &sec->rcu, sframe_free_srcu);
> +
> +	return 0;
> +}

[Severity: High]
Could this lockless tree access result in corrupting the SRCU callback
list and leaking memory?

If a concurrent thread inserts a new section at the same text_start
address before mtree_erase() is called, mtree_erase() will successfully
remove and return the new section.

Because the return value of mtree_erase() is not checked against sec,
the code proceeds to call call_srcu() on the original sec pointer. This
could double-queue the old section's RCU head and permanently leak the
newly inserted section.

[ ... ]
> +void sframe_free_mm(struct mm_struct *mm)
> +{
> +	struct sframe_section *sec;
> +	unsigned long index = 0;
> +
> +	if (!mm)
> +		return;
> +
> +	mt_for_each(&mm->sframe_mt, sec, index, ULONG_MAX)
> +		free_section(sec);
> +
> +	mtree_destroy(&mm->sframe_mt);
>  }

[Severity: High]
Does bypassing the SRCU grace period here cause a use-after-free for
concurrent readers?

If an unwinder accesses a section under sframe_srcu and then drops its
mm reference, triggering teardown, the synchronous free_section()
(which calls kfree directly) will free the section while the unwinder
is still relying on SRCU to keep it alive. Should this teardown path
use call_srcu() as well?

-- 
Sashiko AI review · https://sashiko.dev/#/patchset/20260521142546.3908498-1-jremus@linux.ibm.com?part=3

  reply	other threads:[~2026-05-21 15:04 UTC|newest]

Thread overview: 43+ messages / expand[flat|nested]  mbox.gz  Atom feed  top
2026-05-21 14:25 [PATCH v16 00/20] unwind_deferred: Implement sframe handling Jens Remus
2026-05-21 14:25 ` [PATCH v16 01/20] unwind_user: Add generic and arch-specific headers to MAINTAINERS Jens Remus
2026-05-21 14:25 ` [PATCH v16 02/20] unwind_user/sframe: Add support for reading .sframe headers Jens Remus
2026-05-21 14:51   ` sashiko-bot
2026-05-22 10:11     ` Jens Remus
2026-05-27 20:09   ` Steven Rostedt
2026-05-21 14:25 ` [PATCH v16 03/20] unwind_user/sframe: Store .sframe section data in per-mm maple tree Jens Remus
2026-05-21 15:04   ` sashiko-bot [this message]
2026-05-27 20:20   ` Steven Rostedt
2026-05-21 14:25 ` [PATCH v16 04/20] x86/uaccess: Add unsafe_copy_from_user() implementation Jens Remus
2026-05-21 14:25 ` [PATCH v16 05/20] unwind_user/sframe: Add support for reading .sframe contents Jens Remus
2026-05-21 15:18   ` sashiko-bot
2026-05-22  9:26     ` Jens Remus
2026-05-27 19:49       ` Steven Rostedt
2026-05-21 14:25 ` [PATCH v16 06/20] unwind_user/sframe: Detect .sframe sections in executables Jens Remus
2026-05-21 14:25 ` [PATCH v16 07/20] unwind_user/sframe: Wire up unwind_user to sframe Jens Remus
2026-05-21 14:53   ` sashiko-bot
2026-05-22  9:55     ` Jens Remus
2026-05-21 14:25 ` [PATCH v16 08/20] unwind_user: Stop when reaching an outermost frame Jens Remus
2026-05-21 14:47   ` sashiko-bot
2026-05-22 10:12     ` Jens Remus
2026-05-21 14:25 ` [PATCH v16 09/20] unwind_user/sframe: Add support for outermost frame indication Jens Remus
2026-05-21 14:25 ` [PATCH v16 10/20] unwind_user/sframe: Remove .sframe section on detected corruption Jens Remus
2026-05-21 15:19   ` sashiko-bot
2026-05-22 10:03     ` Jens Remus
2026-05-21 14:25 ` [PATCH v16 11/20] unwind_user/sframe: Show file name in debug output Jens Remus
2026-05-21 15:05   ` sashiko-bot
2026-05-22  9:58     ` Jens Remus
2026-05-21 14:25 ` [PATCH v16 12/20] unwind_user/sframe: Add .sframe validation option Jens Remus
2026-05-21 15:02   ` sashiko-bot
2026-05-22 10:08     ` Jens Remus
2026-05-21 14:25 ` [PATCH v16 13/20] unwind_user: Enable archs that pass RA in a register Jens Remus
2026-05-21 14:25 ` [PATCH v16 14/20] unwind_user: Flexible FP/RA recovery rules Jens Remus
2026-05-21 14:25 ` [PATCH v16 15/20] unwind_user: Flexible CFA " Jens Remus
2026-05-21 14:25 ` [PATCH v16 16/20] unwind_user/sframe: Add support for SFrame V3 flexible FDEs Jens Remus
2026-05-21 15:14   ` sashiko-bot
2026-05-22 10:15     ` Jens Remus
2026-05-21 14:25 ` [PATCH v16 17/20] unwind_user/sframe: Separate reading of FRE from reading of FRE data words Jens Remus
2026-05-21 14:25 ` [PATCH v16 18/20] unwind_user/sframe: Duplicate registered .sframe section data on clone/fork Jens Remus
2026-05-21 15:37   ` sashiko-bot
2026-05-21 14:25 ` [PATCH v16 19/20] unwind_user/sframe/x86: Enable sframe unwinding on x86 Jens Remus
2026-05-21 14:25 ` [PATCH v16 20/20] unwind_user/sframe: Add prctl() interface for registering .sframe sections Jens Remus
2026-05-21 15:23   ` sashiko-bot

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=20260521150427.8BE671F000E9@smtp.kernel.org \
    --to=sashiko-bot@kernel.org \
    --cc=bpf@vger.kernel.org \
    --cc=jremus@linux.ibm.com \
    --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 an external index of several public inboxes,
see mirroring instructions on how to clone and mirror
all data and code used by this external index.