]> git.ipfire.org Git - thirdparty/linux.git/commitdiff
bpf: Refactor to avoid redundant calculation of bpf_reg_state
authorYonghong Song <yonghong.song@linux.dev>
Thu, 23 Apr 2026 03:34:46 +0000 (20:34 -0700)
committerAlexei Starovoitov <ast@kernel.org>
Thu, 23 Apr 2026 14:54:50 +0000 (07:54 -0700)
In many cases, once a bpf_reg_state is defined, it can pass to
callee's. Otherwise, callee will need to get bpf_reg_state again
based on regno. More importantly, this is needed for later stack
arguments for kfuncs since the register state for stack arguments does
not have a corresponding regno. So it makes sense to pass reg state
for callee's.

The following is the only change to avoid compilation warning:
   static int sanitize_check_bounds(struct bpf_verifier_env *env,
                                   const struct bpf_insn *insn,
  -                                const struct bpf_reg_state *dst_reg)
  +                                struct bpf_reg_state *dst_reg)

Acked-by: Puranjay Mohan <puranjay@kernel.org>
Acked-by: Kumar Kartikeya Dwivedi <memxor@gmail.com>
Reviewed-by: Amery Hung <ameryhung@gmail.com>
Signed-off-by: Yonghong Song <yonghong.song@linux.dev>
Link: https://lore.kernel.org/r/20260423033446.2538321-1-yonghong.song@linux.dev
Signed-off-by: Alexei Starovoitov <ast@kernel.org>
kernel/bpf/verifier.c

index d123449f5552e7e460fd04a277a9678b45a1f86a..debae6133e4c28fd6f152dbdad84d849c659992f 100644 (file)
@@ -3908,7 +3908,7 @@ static int check_stack_write_fixed_off(struct bpf_verifier_env *env,
        return 0;
 }
 
-/* Write the stack: 'stack[ptr_regno + off] = value_regno'. 'ptr_regno' is
+/* Write the stack: 'stack[ptr_reg + off] = value_regno'. 'ptr_reg' is
  * known to contain a variable offset.
  * This function checks whether the write is permitted and conservatively
  * tracks the effects of the write, considering that each stack slot in the
@@ -3929,13 +3929,13 @@ static int check_stack_write_fixed_off(struct bpf_verifier_env *env,
 static int check_stack_write_var_off(struct bpf_verifier_env *env,
                                     /* func where register points to */
                                     struct bpf_func_state *state,
-                                    int ptr_regno, int off, int size,
+                                    struct bpf_reg_state *ptr_reg, int off, int size,
                                     int value_regno, int insn_idx)
 {
        struct bpf_func_state *cur; /* state of the current function */
        int min_off, max_off;
        int i, err;
-       struct bpf_reg_state *ptr_reg = NULL, *value_reg = NULL;
+       struct bpf_reg_state *value_reg = NULL;
        struct bpf_insn *insn = &env->prog->insnsi[insn_idx];
        bool writing_zero = false;
        /* set if the fact that we're writing a zero is used to let any
@@ -3944,7 +3944,6 @@ static int check_stack_write_var_off(struct bpf_verifier_env *env,
        bool zero_used = false;
 
        cur = env->cur_state->frame[env->cur_state->curframe];
-       ptr_reg = &cur->regs[ptr_regno];
        min_off = ptr_reg->smin_value + off;
        max_off = ptr_reg->smax_value + off + size;
        if (value_regno >= 0)
@@ -4241,7 +4240,7 @@ enum bpf_access_src {
        ACCESS_HELPER = 2,  /* the access is performed by a helper */
 };
 
-static int check_stack_range_initialized(struct bpf_verifier_env *env,
+static int check_stack_range_initialized(struct bpf_verifier_env *env, struct bpf_reg_state *reg,
                                         int regno, int off, int access_size,
                                         bool zero_size_allowed,
                                         enum bpf_access_type type,
@@ -4252,31 +4251,29 @@ static struct bpf_reg_state *reg_state(struct bpf_verifier_env *env, int regno)
        return cur_regs(env) + regno;
 }
 
-/* Read the stack at 'ptr_regno + off' and put the result into the register
+/* Read the stack at 'reg + off' and put the result into the register
  * 'dst_regno'.
- * 'off' includes the pointer register's fixed offset(i.e. 'ptr_regno.off'),
+ * 'off' includes the pointer register's fixed offset(i.e. 'reg->off'),
  * but not its variable offset.
  * 'size' is assumed to be <= reg size and the access is assumed to be aligned.
  *
  * As opposed to check_stack_read_fixed_off, this function doesn't deal with
  * filling registers (i.e. reads of spilled register cannot be detected when
  * the offset is not fixed). We conservatively mark 'dst_regno' as containing
- * SCALAR_VALUE. That's why we assert that the 'ptr_regno' has a variable
+ * SCALAR_VALUE. That's why we assert that the 'reg' has a variable
  * offset; for a fixed offset check_stack_read_fixed_off should be used
  * instead.
  */
-static int check_stack_read_var_off(struct bpf_verifier_env *env,
+static int check_stack_read_var_off(struct bpf_verifier_env *env, struct bpf_reg_state *reg,
                                    int ptr_regno, int off, int size, int dst_regno)
 {
-       /* The state of the source register. */
-       struct bpf_reg_state *reg = reg_state(env, ptr_regno);
        struct bpf_func_state *ptr_state = bpf_func(env, reg);
        int err;
        int min_off, max_off;
 
        /* Note that we pass a NULL meta, so raw access will not be permitted.
         */
-       err = check_stack_range_initialized(env, ptr_regno, off, size,
+       err = check_stack_range_initialized(env, reg, ptr_regno, off, size,
                                            false, BPF_READ, NULL);
        if (err)
                return err;
@@ -4298,10 +4295,9 @@ static int check_stack_read_var_off(struct bpf_verifier_env *env,
  * can be -1, meaning that the read value is not going to a register.
  */
 static int check_stack_read(struct bpf_verifier_env *env,
-                           int ptr_regno, int off, int size,
+                           struct bpf_reg_state *reg, int ptr_regno, int off, int size,
                            int dst_regno)
 {
-       struct bpf_reg_state *reg = reg_state(env, ptr_regno);
        struct bpf_func_state *state = bpf_func(env, reg);
        int err;
        /* Some accesses are only permitted with a static offset. */
@@ -4337,7 +4333,7 @@ static int check_stack_read(struct bpf_verifier_env *env,
                 * than fixed offset ones. Note that dst_regno >= 0 on this
                 * branch.
                 */
-               err = check_stack_read_var_off(env, ptr_regno, off, size,
+               err = check_stack_read_var_off(env, reg, ptr_regno, off, size,
                                               dst_regno);
        }
        return err;
@@ -4347,17 +4343,16 @@ static int check_stack_read(struct bpf_verifier_env *env,
 /* check_stack_write dispatches to check_stack_write_fixed_off or
  * check_stack_write_var_off.
  *
- * 'ptr_regno' is the register used as a pointer into the stack.
+ * 'reg' is the register used as a pointer into the stack.
  * 'value_regno' is the register whose value we're writing to the stack. It can
  * be -1, meaning that we're not writing from a register.
  *
  * The caller must ensure that the offset falls within the maximum stack size.
  */
 static int check_stack_write(struct bpf_verifier_env *env,
-                            int ptr_regno, int off, int size,
+                            struct bpf_reg_state *reg, int off, int size,
                             int value_regno, int insn_idx)
 {
-       struct bpf_reg_state *reg = reg_state(env, ptr_regno);
        struct bpf_func_state *state = bpf_func(env, reg);
        int err;
 
@@ -4370,16 +4365,15 @@ static int check_stack_write(struct bpf_verifier_env *env,
                 * than fixed offset ones.
                 */
                err = check_stack_write_var_off(env, state,
-                                               ptr_regno, off, size,
+                                               reg, off, size,
                                                value_regno, insn_idx);
        }
        return err;
 }
 
-static int check_map_access_type(struct bpf_verifier_env *env, u32 regno,
+static int check_map_access_type(struct bpf_verifier_env *env, struct bpf_reg_state *reg,
                                 int off, int size, enum bpf_access_type type)
 {
-       struct bpf_reg_state *reg = reg_state(env, regno);
        struct bpf_map *map = reg->map_ptr;
        u32 cap = bpf_map_flags_to_cap(map);
 
@@ -4399,17 +4393,15 @@ static int check_map_access_type(struct bpf_verifier_env *env, u32 regno,
 }
 
 /* check read/write into memory region (e.g., map value, ringbuf sample, etc) */
-static int __check_mem_access(struct bpf_verifier_env *env, int regno,
+static int __check_mem_access(struct bpf_verifier_env *env, struct bpf_reg_state *reg, int regno,
                              int off, int size, u32 mem_size,
                              bool zero_size_allowed)
 {
        bool size_ok = size > 0 || (size == 0 && zero_size_allowed);
-       struct bpf_reg_state *reg;
 
        if (off >= 0 && size_ok && (u64)off + size <= mem_size)
                return 0;
 
-       reg = &cur_regs(env)[regno];
        switch (reg->type) {
        case PTR_TO_MAP_KEY:
                verbose(env, "invalid access to map key, key_size=%d off=%d size=%d\n",
@@ -4439,13 +4431,10 @@ static int __check_mem_access(struct bpf_verifier_env *env, int regno,
 }
 
 /* check read/write into a memory region with possible variable offset */
-static int check_mem_region_access(struct bpf_verifier_env *env, u32 regno,
+static int check_mem_region_access(struct bpf_verifier_env *env, struct bpf_reg_state *reg, u32 regno,
                                   int off, int size, u32 mem_size,
                                   bool zero_size_allowed)
 {
-       struct bpf_verifier_state *vstate = env->cur_state;
-       struct bpf_func_state *state = vstate->frame[vstate->curframe];
-       struct bpf_reg_state *reg = &state->regs[regno];
        int err;
 
        /* We may have adjusted the register pointing to memory region, so we
@@ -4466,7 +4455,7 @@ static int check_mem_region_access(struct bpf_verifier_env *env, u32 regno,
                        regno);
                return -EACCES;
        }
-       err = __check_mem_access(env, regno, reg->smin_value + off, size,
+       err = __check_mem_access(env, reg, regno, reg->smin_value + off, size,
                                 mem_size, zero_size_allowed);
        if (err) {
                verbose(env, "R%d min value is outside of the allowed memory range\n",
@@ -4483,7 +4472,7 @@ static int check_mem_region_access(struct bpf_verifier_env *env, u32 regno,
                        regno);
                return -EACCES;
        }
-       err = __check_mem_access(env, regno, reg->umax_value + off, size,
+       err = __check_mem_access(env, reg, regno, reg->umax_value + off, size,
                                 mem_size, zero_size_allowed);
        if (err) {
                verbose(env, "R%d max value is outside of the allowed memory range\n",
@@ -4787,19 +4776,16 @@ static u32 map_mem_size(const struct bpf_map *map)
 }
 
 /* check read/write into a map element with possible variable offset */
-static int check_map_access(struct bpf_verifier_env *env, u32 regno,
+static int check_map_access(struct bpf_verifier_env *env, struct bpf_reg_state *reg, u32 regno,
                            int off, int size, bool zero_size_allowed,
                            enum bpf_access_src src)
 {
-       struct bpf_verifier_state *vstate = env->cur_state;
-       struct bpf_func_state *state = vstate->frame[vstate->curframe];
-       struct bpf_reg_state *reg = &state->regs[regno];
        struct bpf_map *map = reg->map_ptr;
        u32 mem_size = map_mem_size(map);
        struct btf_record *rec;
        int err, i;
 
-       err = check_mem_region_access(env, regno, off, size, mem_size, zero_size_allowed);
+       err = check_mem_region_access(env, reg, regno, off, size, mem_size, zero_size_allowed);
        if (err)
                return err;
 
@@ -4895,10 +4881,9 @@ static bool may_access_direct_pkt_data(struct bpf_verifier_env *env,
        }
 }
 
-static int check_packet_access(struct bpf_verifier_env *env, u32 regno, int off,
+static int check_packet_access(struct bpf_verifier_env *env, struct bpf_reg_state *reg, u32 regno, int off,
                               int size, bool zero_size_allowed)
 {
-       struct bpf_reg_state *reg = reg_state(env, regno);
        int err;
 
        if (reg->range < 0) {
@@ -4906,7 +4891,7 @@ static int check_packet_access(struct bpf_verifier_env *env, u32 regno, int off,
                return -EINVAL;
        }
 
-       err = check_mem_region_access(env, regno, off, size, reg->range, zero_size_allowed);
+       err = check_mem_region_access(env, reg, regno, off, size, reg->range, zero_size_allowed);
        if (err)
                return err;
 
@@ -4961,7 +4946,7 @@ static int __check_ctx_access(struct bpf_verifier_env *env, int insn_idx, int of
        return -EACCES;
 }
 
-static int check_ctx_access(struct bpf_verifier_env *env, int insn_idx, u32 regno,
+static int check_ctx_access(struct bpf_verifier_env *env, int insn_idx, struct bpf_reg_state *reg, u32 regno,
                            int off, int access_size, enum bpf_access_type t,
                            struct bpf_insn_access_aux *info)
 {
@@ -4971,12 +4956,10 @@ static int check_ctx_access(struct bpf_verifier_env *env, int insn_idx, u32 regn
         */
        bool var_off_ok = is_var_ctx_off_allowed(env->prog);
        bool fixed_off_ok = !env->ops->convert_ctx_access;
-       struct bpf_reg_state *regs = cur_regs(env);
-       struct bpf_reg_state *reg = regs + regno;
        int err;
 
        if (var_off_ok)
-               err = check_mem_region_access(env, regno, off, access_size, U16_MAX, false);
+               err = check_mem_region_access(env, reg, regno, off, access_size, U16_MAX, false);
        else
                err = __check_ptr_off_reg(env, reg, regno, fixed_off_ok);
        if (err)
@@ -5002,10 +4985,9 @@ static int check_flow_keys_access(struct bpf_verifier_env *env, int off,
 }
 
 static int check_sock_access(struct bpf_verifier_env *env, int insn_idx,
-                            u32 regno, int off, int size,
+                            struct bpf_reg_state *reg, u32 regno, int off, int size,
                             enum bpf_access_type t)
 {
-       struct bpf_reg_state *reg = reg_state(env, regno);
        struct bpf_insn_access_aux info = {};
        bool valid;
 
@@ -5971,12 +5953,11 @@ static bool type_is_trusted_or_null(struct bpf_verifier_env *env,
 }
 
 static int check_ptr_to_btf_access(struct bpf_verifier_env *env,
-                                  struct bpf_reg_state *regs,
+                                  struct bpf_reg_state *regs, struct bpf_reg_state *reg,
                                   int regno, int off, int size,
                                   enum bpf_access_type atype,
                                   int value_regno)
 {
-       struct bpf_reg_state *reg = regs + regno;
        const struct btf_type *t = btf_type_by_id(reg->btf, reg->btf_id);
        const char *tname = btf_name_by_offset(reg->btf, t->name_off);
        const char *field_name = NULL;
@@ -6128,12 +6109,11 @@ static int check_ptr_to_btf_access(struct bpf_verifier_env *env,
 }
 
 static int check_ptr_to_map_access(struct bpf_verifier_env *env,
-                                  struct bpf_reg_state *regs,
+                                  struct bpf_reg_state *regs, struct bpf_reg_state *reg,
                                   int regno, int off, int size,
                                   enum bpf_access_type atype,
                                   int value_regno)
 {
-       struct bpf_reg_state *reg = regs + regno;
        struct bpf_map *map = reg->map_ptr;
        struct bpf_reg_state map_reg;
        enum bpf_type_flag flag = 0;
@@ -6222,11 +6202,10 @@ static int check_stack_slot_within_bounds(struct bpf_verifier_env *env,
  * 'off' includes `regno->offset`, but not its dynamic part (if any).
  */
 static int check_stack_access_within_bounds(
-               struct bpf_verifier_env *env,
+               struct bpf_verifier_env *env, struct bpf_reg_state *reg,
                int regno, int off, int access_size,
                enum bpf_access_type type)
 {
-       struct bpf_reg_state *reg = reg_state(env, regno);
        struct bpf_func_state *state = bpf_func(env, reg);
        s64 min_off, max_off;
        int err;
@@ -6314,12 +6293,11 @@ static void add_scalar_to_reg(struct bpf_reg_state *dst_reg, s64 val)
  * if t==write && value_regno==-1, some unknown value is stored into memory
  * if t==read && value_regno==-1, don't care what we read from memory
  */
-static int check_mem_access(struct bpf_verifier_env *env, int insn_idx, u32 regno,
+static int check_mem_access(struct bpf_verifier_env *env, int insn_idx, struct bpf_reg_state *reg, u32 regno,
                            int off, int bpf_size, enum bpf_access_type t,
                            int value_regno, bool strict_alignment_once, bool is_ldsx)
 {
        struct bpf_reg_state *regs = cur_regs(env);
-       struct bpf_reg_state *reg = regs + regno;
        int size, err = 0;
 
        size = bpf_size_to_bytes(bpf_size);
@@ -6336,7 +6314,7 @@ static int check_mem_access(struct bpf_verifier_env *env, int insn_idx, u32 regn
                        return -EACCES;
                }
 
-               err = check_mem_region_access(env, regno, off, size,
+               err = check_mem_region_access(env, reg, regno, off, size,
                                              reg->map_ptr->key_size, false);
                if (err)
                        return err;
@@ -6350,10 +6328,10 @@ static int check_mem_access(struct bpf_verifier_env *env, int insn_idx, u32 regn
                        verbose(env, "R%d leaks addr into map\n", value_regno);
                        return -EACCES;
                }
-               err = check_map_access_type(env, regno, off, size, t);
+               err = check_map_access_type(env, reg, off, size, t);
                if (err)
                        return err;
-               err = check_map_access(env, regno, off, size, false, ACCESS_DIRECT);
+               err = check_map_access(env, reg, regno, off, size, false, ACCESS_DIRECT);
                if (err)
                        return err;
                if (tnum_is_const(reg->var_off))
@@ -6422,7 +6400,7 @@ static int check_mem_access(struct bpf_verifier_env *env, int insn_idx, u32 regn
                 * instructions, hence no need to check bounds in that case.
                 */
                if (!rdonly_untrusted)
-                       err = check_mem_region_access(env, regno, off, size,
+                       err = check_mem_region_access(env, reg, regno, off, size,
                                                      reg->mem_size, false);
                if (!err && value_regno >= 0 && (t == BPF_READ || rdonly_mem))
                        mark_reg_unknown(env, regs, value_regno);
@@ -6440,7 +6418,7 @@ static int check_mem_access(struct bpf_verifier_env *env, int insn_idx, u32 regn
                        return -EACCES;
                }
 
-               err = check_ctx_access(env, insn_idx, regno, off, size, t, &info);
+               err = check_ctx_access(env, insn_idx, reg, regno, off, size, t, &info);
                if (!err && t == BPF_READ && value_regno >= 0) {
                        /* ctx access returns either a scalar, or a
                         * PTR_TO_PACKET[_META,_END]. In the latter
@@ -6477,15 +6455,15 @@ static int check_mem_access(struct bpf_verifier_env *env, int insn_idx, u32 regn
 
        } else if (reg->type == PTR_TO_STACK) {
                /* Basic bounds checks. */
-               err = check_stack_access_within_bounds(env, regno, off, size, t);
+               err = check_stack_access_within_bounds(env, reg, regno, off, size, t);
                if (err)
                        return err;
 
                if (t == BPF_READ)
-                       err = check_stack_read(env, regno, off, size,
+                       err = check_stack_read(env, reg, regno, off, size,
                                               value_regno);
                else
-                       err = check_stack_write(env, regno, off, size,
+                       err = check_stack_write(env, reg, off, size,
                                                value_regno, insn_idx);
        } else if (reg_is_pkt_pointer(reg)) {
                if (t == BPF_WRITE && !may_access_direct_pkt_data(env, NULL, t)) {
@@ -6498,7 +6476,7 @@ static int check_mem_access(struct bpf_verifier_env *env, int insn_idx, u32 regn
                                value_regno);
                        return -EACCES;
                }
-               err = check_packet_access(env, regno, off, size, false);
+               err = check_packet_access(env, reg, regno, off, size, false);
                if (!err && t == BPF_READ && value_regno >= 0)
                        mark_reg_unknown(env, regs, value_regno);
        } else if (reg->type == PTR_TO_FLOW_KEYS) {
@@ -6518,7 +6496,7 @@ static int check_mem_access(struct bpf_verifier_env *env, int insn_idx, u32 regn
                                regno, reg_type_str(env, reg->type));
                        return -EACCES;
                }
-               err = check_sock_access(env, insn_idx, regno, off, size, t);
+               err = check_sock_access(env, insn_idx, reg, regno, off, size, t);
                if (!err && value_regno >= 0)
                        mark_reg_unknown(env, regs, value_regno);
        } else if (reg->type == PTR_TO_TP_BUFFER) {
@@ -6527,10 +6505,10 @@ static int check_mem_access(struct bpf_verifier_env *env, int insn_idx, u32 regn
                        mark_reg_unknown(env, regs, value_regno);
        } else if (base_type(reg->type) == PTR_TO_BTF_ID &&
                   !type_may_be_null(reg->type)) {
-               err = check_ptr_to_btf_access(env, regs, regno, off, size, t,
+               err = check_ptr_to_btf_access(env, regs, reg, regno, off, size, t,
                                              value_regno);
        } else if (reg->type == CONST_PTR_TO_MAP) {
-               err = check_ptr_to_map_access(env, regs, regno, off, size, t,
+               err = check_ptr_to_map_access(env, regs, reg, regno, off, size, t,
                                              value_regno);
        } else if (base_type(reg->type) == PTR_TO_BUF &&
                   !type_may_be_null(reg->type)) {
@@ -6599,7 +6577,7 @@ static int check_load_mem(struct bpf_verifier_env *env, struct bpf_insn *insn,
        /* Check if (src_reg + off) is readable. The state of dst_reg will be
         * updated by this call.
         */
-       err = check_mem_access(env, env->insn_idx, insn->src_reg, insn->off,
+       err = check_mem_access(env, env->insn_idx, regs + insn->src_reg, insn->src_reg, insn->off,
                               BPF_SIZE(insn->code), BPF_READ, insn->dst_reg,
                               strict_alignment_once, is_ldsx);
        err = err ?: save_aux_ptr_type(env, src_reg_type,
@@ -6629,7 +6607,7 @@ static int check_store_reg(struct bpf_verifier_env *env, struct bpf_insn *insn,
        dst_reg_type = regs[insn->dst_reg].type;
 
        /* Check if (dst_reg + off) is writeable. */
-       err = check_mem_access(env, env->insn_idx, insn->dst_reg, insn->off,
+       err = check_mem_access(env, env->insn_idx, regs + insn->dst_reg, insn->dst_reg, insn->off,
                               BPF_SIZE(insn->code), BPF_WRITE, insn->src_reg,
                               strict_alignment_once, false);
        err = err ?: save_aux_ptr_type(env, dst_reg_type, false);
@@ -6640,6 +6618,7 @@ static int check_store_reg(struct bpf_verifier_env *env, struct bpf_insn *insn,
 static int check_atomic_rmw(struct bpf_verifier_env *env,
                            struct bpf_insn *insn)
 {
+       struct bpf_reg_state *dst_reg;
        int load_reg;
        int err;
 
@@ -6701,13 +6680,15 @@ static int check_atomic_rmw(struct bpf_verifier_env *env,
                load_reg = -1;
        }
 
+       dst_reg = cur_regs(env) + insn->dst_reg;
+
        /* Check whether we can read the memory, with second call for fetch
         * case to simulate the register fill.
         */
-       err = check_mem_access(env, env->insn_idx, insn->dst_reg, insn->off,
+       err = check_mem_access(env, env->insn_idx, dst_reg, insn->dst_reg, insn->off,
                               BPF_SIZE(insn->code), BPF_READ, -1, true, false);
        if (!err && load_reg >= 0)
-               err = check_mem_access(env, env->insn_idx, insn->dst_reg,
+               err = check_mem_access(env, env->insn_idx, dst_reg, insn->dst_reg,
                                       insn->off, BPF_SIZE(insn->code),
                                       BPF_READ, load_reg, true, false);
        if (err)
@@ -6719,7 +6700,7 @@ static int check_atomic_rmw(struct bpf_verifier_env *env,
                        return err;
        }
        /* Check whether we can write into the same memory. */
-       err = check_mem_access(env, env->insn_idx, insn->dst_reg, insn->off,
+       err = check_mem_access(env, env->insn_idx, dst_reg, insn->dst_reg, insn->off,
                               BPF_SIZE(insn->code), BPF_WRITE, -1, true, false);
        if (err)
                return err;
@@ -6808,11 +6789,10 @@ static int check_atomic(struct bpf_verifier_env *env, struct bpf_insn *insn)
  * read offsets are marked as read.
  */
 static int check_stack_range_initialized(
-               struct bpf_verifier_env *env, int regno, int off,
+               struct bpf_verifier_env *env, struct bpf_reg_state *reg, int regno, int off,
                int access_size, bool zero_size_allowed,
                enum bpf_access_type type, struct bpf_call_arg_meta *meta)
 {
-       struct bpf_reg_state *reg = reg_state(env, regno);
        struct bpf_func_state *state = bpf_func(env, reg);
        int err, min_off, max_off, i, j, slot, spi;
        /* Some accesses can write anything into the stack, others are
@@ -6834,7 +6814,7 @@ static int check_stack_range_initialized(
                return -EACCES;
        }
 
-       err = check_stack_access_within_bounds(env, regno, off, access_size, type);
+       err = check_stack_access_within_bounds(env, reg, regno, off, access_size, type);
        if (err)
                return err;
 
@@ -6965,7 +6945,7 @@ static int check_helper_mem_access(struct bpf_verifier_env *env, int regno,
        switch (base_type(reg->type)) {
        case PTR_TO_PACKET:
        case PTR_TO_PACKET_META:
-               return check_packet_access(env, regno, 0, access_size,
+               return check_packet_access(env, reg, regno, 0, access_size,
                                           zero_size_allowed);
        case PTR_TO_MAP_KEY:
                if (access_type == BPF_WRITE) {
@@ -6973,12 +6953,12 @@ static int check_helper_mem_access(struct bpf_verifier_env *env, int regno,
                                reg_type_str(env, reg->type));
                        return -EACCES;
                }
-               return check_mem_region_access(env, regno, 0, access_size,
+               return check_mem_region_access(env, reg, regno, 0, access_size,
                                               reg->map_ptr->key_size, false);
        case PTR_TO_MAP_VALUE:
-               if (check_map_access_type(env, regno, 0, access_size, access_type))
+               if (check_map_access_type(env, reg, 0, access_size, access_type))
                        return -EACCES;
-               return check_map_access(env, regno, 0, access_size,
+               return check_map_access(env, reg, regno, 0, access_size,
                                        zero_size_allowed, ACCESS_HELPER);
        case PTR_TO_MEM:
                if (type_is_rdonly_mem(reg->type)) {
@@ -6988,7 +6968,7 @@ static int check_helper_mem_access(struct bpf_verifier_env *env, int regno,
                                return -EACCES;
                        }
                }
-               return check_mem_region_access(env, regno, 0,
+               return check_mem_region_access(env, reg, regno, 0,
                                               access_size, reg->mem_size,
                                               zero_size_allowed);
        case PTR_TO_BUF:
@@ -7008,16 +6988,16 @@ static int check_helper_mem_access(struct bpf_verifier_env *env, int regno,
                                           max_access);
        case PTR_TO_STACK:
                return check_stack_range_initialized(
-                               env,
+                               env, reg,
                                regno, 0, access_size,
                                zero_size_allowed, access_type, meta);
        case PTR_TO_BTF_ID:
-               return check_ptr_to_btf_access(env, regs, regno, 0,
+               return check_ptr_to_btf_access(env, regs, reg, regno, 0,
                                               access_size, BPF_READ, -1);
        case PTR_TO_CTX:
                /* Only permit reading or writing syscall context using helper calls. */
                if (is_var_ctx_off_allowed(env->prog)) {
-                       int err = check_mem_region_access(env, regno, 0, access_size, U16_MAX,
+                       int err = check_mem_region_access(env, reg, regno, 0, access_size, U16_MAX,
                                                          zero_size_allowed);
                        if (err)
                                return err;
@@ -7178,11 +7158,10 @@ enum {
  * env->cur_state->active_locks remembers which map value element or allocated
  * object got locked and clears it after bpf_spin_unlock.
  */
-static int process_spin_lock(struct bpf_verifier_env *env, int regno, int flags)
+static int process_spin_lock(struct bpf_verifier_env *env, struct bpf_reg_state *reg, int regno, int flags)
 {
        bool is_lock = flags & PROCESS_SPIN_LOCK, is_res_lock = flags & PROCESS_RES_LOCK;
        const char *lock_str = is_res_lock ? "bpf_res_spin" : "bpf_spin";
-       struct bpf_reg_state *reg = reg_state(env, regno);
        struct bpf_verifier_state *cur = env->cur_state;
        bool is_const = tnum_is_const(reg->var_off);
        bool is_irq = flags & PROCESS_LOCK_IRQ;
@@ -7295,11 +7274,10 @@ static int process_spin_lock(struct bpf_verifier_env *env, int regno, int flags)
 }
 
 /* Check if @regno is a pointer to a specific field in a map value */
-static int check_map_field_pointer(struct bpf_verifier_env *env, u32 regno,
+static int check_map_field_pointer(struct bpf_verifier_env *env, struct bpf_reg_state *reg, u32 regno,
                                   enum btf_field_type field_type,
                                   struct bpf_map_desc *map_desc)
 {
-       struct bpf_reg_state *reg = reg_state(env, regno);
        bool is_const = tnum_is_const(reg->var_off);
        struct bpf_map *map = reg->map_ptr;
        u64 val = reg->var_off.value;
@@ -7349,26 +7327,26 @@ static int check_map_field_pointer(struct bpf_verifier_env *env, u32 regno,
        return 0;
 }
 
-static int process_timer_func(struct bpf_verifier_env *env, int regno,
+static int process_timer_func(struct bpf_verifier_env *env, struct bpf_reg_state *reg, int regno,
                              struct bpf_map_desc *map)
 {
        if (IS_ENABLED(CONFIG_PREEMPT_RT)) {
                verbose(env, "bpf_timer cannot be used for PREEMPT_RT.\n");
                return -EOPNOTSUPP;
        }
-       return check_map_field_pointer(env, regno, BPF_TIMER, map);
+       return check_map_field_pointer(env, reg, regno, BPF_TIMER, map);
 }
 
-static int process_timer_helper(struct bpf_verifier_env *env, int regno,
+static int process_timer_helper(struct bpf_verifier_env *env, struct bpf_reg_state *reg, int regno,
                                struct bpf_call_arg_meta *meta)
 {
-       return process_timer_func(env, regno, &meta->map);
+       return process_timer_func(env, reg, regno, &meta->map);
 }
 
-static int process_timer_kfunc(struct bpf_verifier_env *env, int regno,
+static int process_timer_kfunc(struct bpf_verifier_env *env, struct bpf_reg_state *reg, int regno,
                               struct bpf_kfunc_call_arg_meta *meta)
 {
-       return process_timer_func(env, regno, &meta->map);
+       return process_timer_func(env, reg, regno, &meta->map);
 }
 
 static int process_kptr_func(struct bpf_verifier_env *env, int regno,
@@ -7433,10 +7411,9 @@ static int process_kptr_func(struct bpf_verifier_env *env, int regno,
  * use case. The second level is tracked using the upper bit of bpf_dynptr->size
  * and checked dynamically during runtime.
  */
-static int process_dynptr_func(struct bpf_verifier_env *env, int regno, int insn_idx,
+static int process_dynptr_func(struct bpf_verifier_env *env, struct bpf_reg_state *reg, int regno, int insn_idx,
                               enum bpf_arg_type arg_type, int clone_ref_obj_id)
 {
-       struct bpf_reg_state *reg = reg_state(env, regno);
        int err;
 
        if (reg->type != PTR_TO_STACK && reg->type != CONST_PTR_TO_DYNPTR) {
@@ -7470,7 +7447,7 @@ static int process_dynptr_func(struct bpf_verifier_env *env, int regno, int insn
 
                /* we write BPF_DW bits (8 bytes) at a time */
                for (i = 0; i < BPF_DYNPTR_SIZE; i += 8) {
-                       err = check_mem_access(env, insn_idx, regno,
+                       err = check_mem_access(env, insn_idx, reg, regno,
                                               i, BPF_DW, BPF_WRITE, -1, false, false);
                        if (err)
                                return err;
@@ -7540,10 +7517,9 @@ static bool is_kfunc_arg_iter(struct bpf_kfunc_call_arg_meta *meta, int arg_idx,
        return btf_param_match_suffix(meta->btf, arg, "__iter");
 }
 
-static int process_iter_arg(struct bpf_verifier_env *env, int regno, int insn_idx,
+static int process_iter_arg(struct bpf_verifier_env *env, struct bpf_reg_state *reg, int regno, int insn_idx,
                            struct bpf_kfunc_call_arg_meta *meta)
 {
-       struct bpf_reg_state *reg = reg_state(env, regno);
        const struct btf_type *t;
        int spi, err, i, nr_slots, btf_id;
 
@@ -7575,7 +7551,7 @@ static int process_iter_arg(struct bpf_verifier_env *env, int regno, int insn_id
                }
 
                for (i = 0; i < nr_slots * 8; i += BPF_REG_SIZE) {
-                       err = check_mem_access(env, insn_idx, regno,
+                       err = check_mem_access(env, insn_idx, reg, regno,
                                               i, BPF_DW, BPF_WRITE, -1, false, false);
                        if (err)
                                return err;
@@ -8014,12 +7990,11 @@ static const struct bpf_reg_types *compatible_reg_types[__BPF_ARG_TYPE_MAX] = {
        [ARG_PTR_TO_DYNPTR]             = &dynptr_types,
 };
 
-static int check_reg_type(struct bpf_verifier_env *env, u32 regno,
+static int check_reg_type(struct bpf_verifier_env *env, struct bpf_reg_state *reg, u32 regno,
                          enum bpf_arg_type arg_type,
                          const u32 *arg_btf_id,
                          struct bpf_call_arg_meta *meta)
 {
-       struct bpf_reg_state *reg = reg_state(env, regno);
        enum bpf_reg_type expected, type = reg->type;
        const struct bpf_reg_types *compatible;
        int i, j, err;
@@ -8362,7 +8337,7 @@ static int check_reg_const_str(struct bpf_verifier_env *env,
                return -EACCES;
        }
 
-       err = check_map_access(env, regno, 0,
+       err = check_map_access(env, reg, regno, 0,
                               map->value_size - reg->var_off.value, false,
                               ACCESS_HELPER);
        if (err)
@@ -8498,7 +8473,7 @@ static int check_func_arg(struct bpf_verifier_env *env, u32 arg,
            base_type(arg_type) == ARG_PTR_TO_SPIN_LOCK)
                arg_btf_id = fn->arg_btf_id[arg];
 
-       err = check_reg_type(env, regno, arg_type, arg_btf_id, meta);
+       err = check_reg_type(env, reg, regno, arg_type, arg_btf_id, meta);
        if (err)
                return err;
 
@@ -8636,11 +8611,11 @@ skip_type_check:
                        return -EACCES;
                }
                if (meta->func_id == BPF_FUNC_spin_lock) {
-                       err = process_spin_lock(env, regno, PROCESS_SPIN_LOCK);
+                       err = process_spin_lock(env, reg, regno, PROCESS_SPIN_LOCK);
                        if (err)
                                return err;
                } else if (meta->func_id == BPF_FUNC_spin_unlock) {
-                       err = process_spin_lock(env, regno, 0);
+                       err = process_spin_lock(env, reg, regno, 0);
                        if (err)
                                return err;
                } else {
@@ -8649,7 +8624,7 @@ skip_type_check:
                }
                break;
        case ARG_PTR_TO_TIMER:
-               err = process_timer_helper(env, regno, meta);
+               err = process_timer_helper(env, reg, regno, meta);
                if (err)
                        return err;
                break;
@@ -8684,7 +8659,7 @@ skip_type_check:
                                         true, meta);
                break;
        case ARG_PTR_TO_DYNPTR:
-               err = process_dynptr_func(env, regno, insn_idx, arg_type, 0);
+               err = process_dynptr_func(env, reg, regno, insn_idx, arg_type, 0);
                if (err)
                        return err;
                break;
@@ -9343,7 +9318,7 @@ static int btf_check_func_arg_match(struct bpf_verifier_env *env, int subprog,
                        if (ret)
                                return ret;
 
-                       ret = process_dynptr_func(env, regno, -1, arg->arg_type, 0);
+                       ret = process_dynptr_func(env, reg, regno, -1, arg->arg_type, 0);
                        if (ret)
                                return ret;
                } else if (base_type(arg->arg_type) == ARG_PTR_TO_BTF_ID) {
@@ -9354,7 +9329,7 @@ static int btf_check_func_arg_match(struct bpf_verifier_env *env, int subprog,
                                continue;
 
                        memset(&meta, 0, sizeof(meta)); /* leave func_id as zero */
-                       err = check_reg_type(env, regno, arg->arg_type, &arg->btf_id, &meta);
+                       err = check_reg_type(env, reg, regno, arg->arg_type, &arg->btf_id, &meta);
                        err = err ?: check_func_arg_reg_off(env, reg, regno, arg->arg_type);
                        if (err)
                                return err;
@@ -10312,18 +10287,18 @@ static int check_helper_call(struct bpf_verifier_env *env, struct bpf_insn *insn
        if (err)
                return err;
 
+       regs = cur_regs(env);
+
        /* Mark slots with STACK_MISC in case of raw mode, stack offset
         * is inferred from register state.
         */
        for (i = 0; i < meta.access_size; i++) {
-               err = check_mem_access(env, insn_idx, meta.regno, i, BPF_B,
+               err = check_mem_access(env, insn_idx, regs + meta.regno, meta.regno, i, BPF_B,
                                       BPF_WRITE, -1, false, false);
                if (err)
                        return err;
        }
 
-       regs = cur_regs(env);
-
        if (meta.release_regno) {
                err = -EINVAL;
                if (arg_type_is_dynptr(fn->arg_type[meta.release_regno - BPF_REG_1])) {
@@ -11327,11 +11302,10 @@ get_kfunc_ptr_arg_type(struct bpf_verifier_env *env,
                       struct bpf_kfunc_call_arg_meta *meta,
                       const struct btf_type *t, const struct btf_type *ref_t,
                       const char *ref_tname, const struct btf_param *args,
-                      int argno, int nargs)
+                      int argno, int nargs, struct bpf_reg_state *reg)
 {
        u32 regno = argno + 1;
        struct bpf_reg_state *regs = cur_regs(env);
-       struct bpf_reg_state *reg = &regs[regno];
        bool arg_mem_size = false;
 
        if (meta->func_id == special_kfunc_list[KF_bpf_cast_to_kern_ctx] ||
@@ -11498,10 +11472,9 @@ static int process_kf_arg_ptr_to_btf_id(struct bpf_verifier_env *env,
        return 0;
 }
 
-static int process_irq_flag(struct bpf_verifier_env *env, int regno,
+static int process_irq_flag(struct bpf_verifier_env *env, struct bpf_reg_state *reg, int regno,
                             struct bpf_kfunc_call_arg_meta *meta)
 {
-       struct bpf_reg_state *reg = reg_state(env, regno);
        int err, kfunc_class = IRQ_NATIVE_KFUNC;
        bool irq_save;
 
@@ -11526,7 +11499,7 @@ static int process_irq_flag(struct bpf_verifier_env *env, int regno,
                        return -EINVAL;
                }
 
-               err = check_mem_access(env, env->insn_idx, regno, 0, BPF_DW, BPF_WRITE, -1, false, false);
+               err = check_mem_access(env, env->insn_idx, reg, regno, 0, BPF_DW, BPF_WRITE, -1, false, false);
                if (err)
                        return err;
 
@@ -12114,7 +12087,7 @@ static int check_kfunc_args(struct bpf_verifier_env *env, struct bpf_kfunc_call_
                ref_t = btf_type_skip_modifiers(btf, t->type, &ref_id);
                ref_tname = btf_name_by_offset(btf, ref_t->name_off);
 
-               kf_arg_type = get_kfunc_ptr_arg_type(env, meta, t, ref_t, ref_tname, args, i, nargs);
+               kf_arg_type = get_kfunc_ptr_arg_type(env, meta, t, ref_t, ref_tname, args, i, nargs, reg);
                if (kf_arg_type < 0)
                        return kf_arg_type;
 
@@ -12276,7 +12249,7 @@ static int check_kfunc_args(struct bpf_verifier_env *env, struct bpf_kfunc_call_
                                }
                        }
 
-                       ret = process_dynptr_func(env, regno, insn_idx, dynptr_arg_type, clone_ref_obj_id);
+                       ret = process_dynptr_func(env, reg, regno, insn_idx, dynptr_arg_type, clone_ref_obj_id);
                        if (ret < 0)
                                return ret;
 
@@ -12301,7 +12274,7 @@ static int check_kfunc_args(struct bpf_verifier_env *env, struct bpf_kfunc_call_
                                        return -EINVAL;
                                }
                        }
-                       ret = process_iter_arg(env, regno, insn_idx, meta);
+                       ret = process_iter_arg(env, reg, regno, insn_idx, meta);
                        if (ret < 0)
                                return ret;
                        break;
@@ -12478,7 +12451,7 @@ static int check_kfunc_args(struct bpf_verifier_env *env, struct bpf_kfunc_call_
                                verbose(env, "arg#%d doesn't point to a map value\n", i);
                                return -EINVAL;
                        }
-                       ret = check_map_field_pointer(env, regno, BPF_WORKQUEUE, &meta->map);
+                       ret = check_map_field_pointer(env, reg, regno, BPF_WORKQUEUE, &meta->map);
                        if (ret < 0)
                                return ret;
                        break;
@@ -12487,7 +12460,7 @@ static int check_kfunc_args(struct bpf_verifier_env *env, struct bpf_kfunc_call_
                                verbose(env, "arg#%d doesn't point to a map value\n", i);
                                return -EINVAL;
                        }
-                       ret = process_timer_kfunc(env, regno, meta);
+                       ret = process_timer_kfunc(env, reg, regno, meta);
                        if (ret < 0)
                                return ret;
                        break;
@@ -12496,7 +12469,7 @@ static int check_kfunc_args(struct bpf_verifier_env *env, struct bpf_kfunc_call_
                                verbose(env, "arg#%d doesn't point to a map value\n", i);
                                return -EINVAL;
                        }
-                       ret = check_map_field_pointer(env, regno, BPF_TASK_WORK, &meta->map);
+                       ret = check_map_field_pointer(env, reg, regno, BPF_TASK_WORK, &meta->map);
                        if (ret < 0)
                                return ret;
                        break;
@@ -12505,7 +12478,7 @@ static int check_kfunc_args(struct bpf_verifier_env *env, struct bpf_kfunc_call_
                                verbose(env, "arg#%d doesn't point to an irq flag on stack\n", i);
                                return -EINVAL;
                        }
-                       ret = process_irq_flag(env, regno, meta);
+                       ret = process_irq_flag(env, reg, regno, meta);
                        if (ret < 0)
                                return ret;
                        break;
@@ -12526,7 +12499,7 @@ static int check_kfunc_args(struct bpf_verifier_env *env, struct bpf_kfunc_call_
                        if (meta->func_id == special_kfunc_list[KF_bpf_res_spin_lock_irqsave] ||
                            meta->func_id == special_kfunc_list[KF_bpf_res_spin_unlock_irqrestore])
                                flags |= PROCESS_LOCK_IRQ;
-                       ret = process_spin_lock(env, regno, flags);
+                       ret = process_spin_lock(env, reg, regno, flags);
                        if (ret < 0)
                                return ret;
                        break;
@@ -13660,7 +13633,7 @@ static int check_stack_access_for_ptr_arithmetic(
 
 static int sanitize_check_bounds(struct bpf_verifier_env *env,
                                 const struct bpf_insn *insn,
-                                const struct bpf_reg_state *dst_reg)
+                                struct bpf_reg_state *dst_reg)
 {
        u32 dst = insn->dst_reg;
 
@@ -13677,7 +13650,7 @@ static int sanitize_check_bounds(struct bpf_verifier_env *env,
                        return -EACCES;
                break;
        case PTR_TO_MAP_VALUE:
-               if (check_map_access(env, dst, 0, 1, false, ACCESS_HELPER)) {
+               if (check_map_access(env, dst_reg, dst, 0, 1, false, ACCESS_HELPER)) {
                        verbose(env, "R%d pointer arithmetic of map value goes out of range, "
                                "prohibited for !root\n", dst);
                        return -EACCES;
@@ -17563,7 +17536,7 @@ static int do_check_insn(struct bpf_verifier_env *env, bool *do_print_state)
 
                dst_reg_type = cur_regs(env)[insn->dst_reg].type;
 
-               err = check_mem_access(env, env->insn_idx, insn->dst_reg,
+               err = check_mem_access(env, env->insn_idx, cur_regs(env) + insn->dst_reg, insn->dst_reg,
                                       insn->off, BPF_SIZE(insn->code),
                                       BPF_WRITE, -1, false, false);
                if (err)