]> git.ipfire.org Git - thirdparty/linux.git/commitdiff
selftests/bpf: add tests for PTR_TO_FLOW_KEYS offset bounds
authorNuoqi Gui <gnq25@mails.tsinghua.edu.cn>
Sat, 6 Jun 2026 10:50:38 +0000 (18:50 +0800)
committerAlexei Starovoitov <ast@kernel.org>
Sat, 6 Jun 2026 23:46:53 +0000 (16:46 -0700)
Add verifier tests covering pointer arithmetic on a PTR_TO_FLOW_KEYS
register. This covers the bpf-next regression where an out-of-bounds
constant offset introduced as flow_keys += K and then dereferenced at
insn->off 0 was accepted, while the equivalent flow_keys + K direct offset
was rejected.

The tests check that in-bounds constant arithmetic on the keys pointer is
still accepted, out-of-bounds constant arithmetic is rejected for both read
and write, and a truly varying offset from bpf_get_prandom_u32() remains
rejected by the existing PTR_TO_FLOW_KEYS pointer arithmetic rules.

Signed-off-by: Nuoqi Gui <gnq25@mails.tsinghua.edu.cn>
Acked-by: Eduard Zingerman <eddyz87@gmail.com>
Link: https://lore.kernel.org/r/20260606-c3-01-v3-v3-2-97c51f592f15@mails.tsinghua.edu.cn
Signed-off-by: Alexei Starovoitov <ast@kernel.org>
tools/testing/selftests/bpf/prog_tests/verifier.c
tools/testing/selftests/bpf/progs/verifier_flow_keys.c [new file with mode: 0644]

index 89779d897abae0ff7ba6131247dedeb8a7b65c9f..8a3d69e2453c3a0ddc2d0d523f26dac48fb09b40 100644 (file)
@@ -38,6 +38,7 @@
 #include "verifier_div0.skel.h"
 #include "verifier_div_mod_bounds.skel.h"
 #include "verifier_div_overflow.skel.h"
+#include "verifier_flow_keys.skel.h"
 #include "verifier_global_subprogs.skel.h"
 #include "verifier_global_ptr_args.skel.h"
 #include "verifier_gotol.skel.h"
@@ -190,6 +191,7 @@ void test_verifier_direct_stack_access_wraparound(void) { RUN(verifier_direct_st
 void test_verifier_div0(void)                 { RUN(verifier_div0); }
 void test_verifier_div_mod_bounds(void)       { RUN(verifier_div_mod_bounds); }
 void test_verifier_div_overflow(void)         { RUN(verifier_div_overflow); }
+void test_verifier_flow_keys(void)            { RUN(verifier_flow_keys); }
 void test_verifier_global_subprogs(void)      { RUN(verifier_global_subprogs); }
 void test_verifier_global_ptr_args(void)      { RUN(verifier_global_ptr_args); }
 void test_verifier_gotol(void)                { RUN(verifier_gotol); }
diff --git a/tools/testing/selftests/bpf/progs/verifier_flow_keys.c b/tools/testing/selftests/bpf/progs/verifier_flow_keys.c
new file mode 100644 (file)
index 0000000..d780a36
--- /dev/null
@@ -0,0 +1,97 @@
+// SPDX-License-Identifier: GPL-2.0
+/* Bounds checks for PTR_TO_FLOW_KEYS pointer arithmetic. */
+
+#include "vmlinux.h"
+#include <bpf/bpf_helpers.h>
+#include "bpf_misc.h"
+
+/* sizeof(struct bpf_flow_keys) is well under 4096, so +0x1000 is OOB. */
+
+SEC("flow_dissector")
+__description("flow_keys: in-bounds constant pointer arithmetic accepted")
+__success
+__naked void flow_keys_const_inbounds(void)
+{
+       asm volatile ("                                 \
+       r1 = *(u64 *)(r1 + %[flow_keys]);               \
+       r1 += 8;                                        \
+       r0 = *(u64 *)(r1 + 0);                          \
+       r0 = 0;                                         \
+       exit;                                           \
+"      :
+       : __imm_const(flow_keys, offsetof(struct __sk_buff, flow_keys))
+       : __clobber_all);
+}
+
+SEC("flow_dissector")
+__description("flow_keys: OOB via constant pointer arithmetic rejected")
+__failure __msg("invalid access to flow keys off=4096 size=8")
+__naked void flow_keys_const_oob_read(void)
+{
+       asm volatile ("                                 \
+       r1 = *(u64 *)(r1 + %[flow_keys]);               \
+       r1 += 4096;                                     \
+       r0 = *(u64 *)(r1 + 0);                          \
+       r0 = 0;                                         \
+       exit;                                           \
+"      :
+       : __imm_const(flow_keys, offsetof(struct __sk_buff, flow_keys))
+       : __clobber_all);
+}
+
+SEC("flow_dissector")
+__description("flow_keys: OOB write via constant pointer arithmetic rejected")
+__failure __msg("invalid access to flow keys off=4096 size=8")
+__naked void flow_keys_const_oob_write(void)
+{
+       asm volatile ("                                 \
+       r1 = *(u64 *)(r1 + %[flow_keys]);               \
+       r1 += 4096;                                     \
+       r2 = 0;                                         \
+       *(u64 *)(r1 + 0) = r2;                          \
+       r0 = 0;                                         \
+       exit;                                           \
+"      :
+       : __imm_const(flow_keys, offsetof(struct __sk_buff, flow_keys))
+       : __clobber_all);
+}
+
+/* Equivalent OOB expressed directly in insn->off; this form was always
+ * rejected and is kept to show both forms now share one diagnostic.
+ */
+SEC("flow_dissector")
+__description("flow_keys: OOB via insn->off rejected")
+__failure __msg("invalid access to flow keys off=4096 size=8")
+__naked void flow_keys_insn_off_oob(void)
+{
+       asm volatile ("                                 \
+       r1 = *(u64 *)(r1 + %[flow_keys]);               \
+       r0 = *(u64 *)(r1 + 4096);                       \
+       r0 = 0;                                         \
+       exit;                                           \
+"      :
+       : __imm_const(flow_keys, offsetof(struct __sk_buff, flow_keys))
+       : __clobber_all);
+}
+
+SEC("flow_dissector")
+__description("flow_keys: variable pointer arithmetic rejected")
+__failure __msg("R1 pointer arithmetic on flow_keys prohibited")
+__naked void flow_keys_var_read(void)
+{
+       asm volatile ("                                 \
+       r6 = r1;                                        \
+       call %[bpf_get_prandom_u32];                    \
+       r0 &= 0xFFFF;                                   \
+       r1 = *(u64 *)(r6 + %[flow_keys]);               \
+       r1 += r0;                                       \
+       r0 = *(u64 *)(r1 + 0);                          \
+       r0 = 0;                                         \
+       exit;                                           \
+"      :
+       : __imm_const(flow_keys, offsetof(struct __sk_buff, flow_keys)),
+         __imm(bpf_get_prandom_u32)
+       : __clobber_all);
+}
+
+char _license[] SEC("license") = "GPL";