> -----Original Messages-----
> From: "Emil Tsalapatis" <[email protected]>
> Send time:Wednesday, 10/06/2026 02:21:49
> To: "Eduard Zingerman" <[email protected]>, "Nuoqi Gui"
> <[email protected]>, "Alexei Starovoitov" <[email protected]>,
> "Daniel Borkmann" <[email protected]>, "Andrii Nakryiko"
> <[email protected]>
> Cc: "Kumar Kartikeya Dwivedi" <[email protected]>, "John Fastabend"
> <[email protected]>, "Martin KaFai Lau" <[email protected]>, "Song
> Liu" <[email protected]>, "Yonghong Song" <[email protected]>, "Jiri
> Olsa" <[email protected]>, "Shuah Khan" <[email protected]>,
> [email protected], [email protected],
> [email protected]
> Subject: Re: [PATCH bpf-next 1/2] bpf: Reject scalar addition from untrusted
> memory
>
> On Tue Jun 9, 2026 at 1:01 PM EDT, Eduard Zingerman wrote:
> > On Tue, 2026-06-09 at 22:55 +0800, Nuoqi Gui wrote:
> >> scalar += rdonly_untrusted_mem reaches adjust_ptr_min_max_vals() with the
> >> pointer as the source register. The untrusted PTR_TO_MEM case returns there
> >> without updating the scalar destination, leaving stale verifier state.
> >>
> >> Reject that addition before the early return. Pointer += scalar remains
> >> handled by the existing untrusted-memory rule.
> >>
> >> Fixes: f2362a57aeff ("bpf: allow void* cast using bpf_rdonly_cast()")
> >> Signed-off-by: Nuoqi Gui <[email protected]>
> >> ---
> >> kernel/bpf/verifier.c | 8 ++++++++
> >> 1 file changed, 8 insertions(+)
> >>
> >> diff --git a/kernel/bpf/verifier.c b/kernel/bpf/verifier.c
> >> index c8d980fdd709..c6b350f9585a 100644
> >> --- a/kernel/bpf/verifier.c
> >> +++ b/kernel/bpf/verifier.c
> >> @@ -14823,6 +14823,14 @@ static int adjust_reg_min_max_vals(struct
> >> bpf_verifier_env *env,
> >> * This is legal, but we have to reverse our
> >> * src/dest handling in computing the range
> >> */
> >> + if (opcode == BPF_ADD &&
> >> + base_type(src_reg->type) == PTR_TO_MEM &&
> >> + (src_reg->type & PTR_UNTRUSTED)) {
> >> + verbose(env, "R%d tried to add from %s
> >> to scalar\n",
> >> + insn->dst_reg,
> >> + reg_type_str(env,
> >> src_reg->type));
> >> + return -EACCES;
> >> + }
> >> err = mark_chain_precision(env, insn->dst_reg);
> >> if (err)
> >> return err;
> >
> > Should the fix be like this:
> >
> > diff --git a/kernel/bpf/verifier.c b/kernel/bpf/verifier.c
> > index 7d27ba396d32..9c85dd680a46 100644
> > --- a/kernel/bpf/verifier.c
> > +++ b/kernel/bpf/verifier.c
> > @@ -13593,8 +13593,10 @@ static int adjust_ptr_min_max_vals(struct
> > bpf_verifier_env *env,
> > * Accesses to untrusted PTR_TO_MEM are done through probe
> > * instructions, hence no need to track offsets.
> > */
> > - if (base_type(ptr_reg->type) == PTR_TO_MEM && (ptr_reg->type &
> > PTR_UNTRUSTED))
> > + if (base_type(ptr_reg->type) == PTR_TO_MEM && (ptr_reg->type &
> > PTR_UNTRUSTED)) {
> > + *dst_reg = *ptr_reg;
> > return 0;
> > + }
> >
> > switch (base_type(ptr_reg->type)) {
> > case PTR_TO_CTX:
> >
> > Instead?
>
> Seconded, AFAICT there is no reason to fail verification since we just
> the pointer's value to the scalar. Whether the operation makes sense is
> up to the programmer to decide.
Thanks.
Agreed, rejecting scalar += pointer is not the right semantic fix. The bug is
that the untrusted PTR_TO_MEM early return leaves dst state stale.
I'll send v2 moving the fix into adjust_ptr_min_max_vals().