]> git.baikalelectronics.ru Git - kernel.git/commitdiff
bpf: Introduce dynamic program extensions
authorAlexei Starovoitov <ast@kernel.org>
Tue, 21 Jan 2020 00:53:46 +0000 (16:53 -0800)
committerDaniel Borkmann <daniel@iogearbox.net>
Wed, 22 Jan 2020 22:04:52 +0000 (23:04 +0100)
Introduce dynamic program extensions. The users can load additional BPF
functions and replace global functions in previously loaded BPF programs while
these programs are executing.

Global functions are verified individually by the verifier based on their types only.
Hence the global function in the new program which types match older function can
safely replace that corresponding function.

This new function/program is called 'an extension' of old program. At load time
the verifier uses (attach_prog_fd, attach_btf_id) pair to identify the function
to be replaced. The BPF program type is derived from the target program into
extension program. Technically bpf_verifier_ops is copied from target program.
The BPF_PROG_TYPE_EXT program type is a placeholder. It has empty verifier_ops.
The extension program can call the same bpf helper functions as target program.
Single BPF_PROG_TYPE_EXT type is used to extend XDP, SKB and all other program
types. The verifier allows only one level of replacement. Meaning that the
extension program cannot recursively extend an extension. That also means that
the maximum stack size is increasing from 512 to 1024 bytes and maximum
function nesting level from 8 to 16. The programs don't always consume that
much. The stack usage is determined by the number of on-stack variables used by
the program. The verifier could have enforced 512 limit for combined original
plus extension program, but it makes for difficult user experience. The main
use case for extensions is to provide generic mechanism to plug external
programs into policy program or function call chaining.

BPF trampoline is used to track both fentry/fexit and program extensions
because both are using the same nop slot at the beginning of every BPF
function. Attaching fentry/fexit to a function that was replaced is not
allowed. The opposite is true as well. Replacing a function that currently
being analyzed with fentry/fexit is not allowed. The executable page allocated
by BPF trampoline is not used by program extensions. This inefficiency will be
optimized in future patches.

Function by function verification of global function supports scalars and
pointer to context only. Hence program extensions are supported for such class
of global functions only. In the future the verifier will be extended with
support to pointers to structures, arrays with sizes, etc.

Signed-off-by: Alexei Starovoitov <ast@kernel.org>
Signed-off-by: Daniel Borkmann <daniel@iogearbox.net>
Acked-by: John Fastabend <john.fastabend@gmail.com>
Acked-by: Andrii Nakryiko <andriin@fb.com>
Acked-by: Toke Høiland-Jørgensen <toke@redhat.com>
Link: https://lore.kernel.org/bpf/20200121005348.2769920-2-ast@kernel.org
include/linux/bpf.h
include/linux/bpf_types.h
include/linux/btf.h
include/uapi/linux/bpf.h
kernel/bpf/btf.c
kernel/bpf/syscall.c
kernel/bpf/trampoline.c
kernel/bpf/verifier.c

index 8e3b8f4ad18359c1cce380e0a71b3e9cf3dd4f6c..05d16615054c2e53d193e290c1b0a1101ffd9494 100644 (file)
@@ -465,7 +465,8 @@ void notrace __bpf_prog_exit(struct bpf_prog *prog, u64 start);
 enum bpf_tramp_prog_type {
        BPF_TRAMP_FENTRY,
        BPF_TRAMP_FEXIT,
-       BPF_TRAMP_MAX
+       BPF_TRAMP_MAX,
+       BPF_TRAMP_REPLACE, /* more than MAX */
 };
 
 struct bpf_trampoline {
@@ -480,6 +481,11 @@ struct bpf_trampoline {
                void *addr;
                bool ftrace_managed;
        } func;
+       /* if !NULL this is BPF_PROG_TYPE_EXT program that extends another BPF
+        * program by replacing one of its functions. func.addr is the address
+        * of the function it replaced.
+        */
+       struct bpf_prog *extension_prog;
        /* list of BPF programs using this trampoline */
        struct hlist_head progs_hlist[BPF_TRAMP_MAX];
        /* Number of attached programs. A counter per kind. */
@@ -1107,6 +1113,8 @@ int btf_check_func_arg_match(struct bpf_verifier_env *env, int subprog,
                             struct bpf_reg_state *regs);
 int btf_prepare_func_args(struct bpf_verifier_env *env, int subprog,
                          struct bpf_reg_state *reg);
+int btf_check_type_match(struct bpf_verifier_env *env, struct bpf_prog *prog,
+                        struct btf *btf, const struct btf_type *t);
 
 struct bpf_prog *bpf_prog_by_id(u32 id);
 
index 9f326e6ef8857fc17fd6bd50bc4d14e7d32ddb9b..c81d4ece79a499d92a2b38276943f10c91821327 100644 (file)
@@ -68,6 +68,8 @@ BPF_PROG_TYPE(BPF_PROG_TYPE_SK_REUSEPORT, sk_reuseport,
 #if defined(CONFIG_BPF_JIT)
 BPF_PROG_TYPE(BPF_PROG_TYPE_STRUCT_OPS, bpf_struct_ops,
              void *, void *)
+BPF_PROG_TYPE(BPF_PROG_TYPE_EXT, bpf_extension,
+             void *, void *)
 #endif
 
 BPF_MAP_TYPE(BPF_MAP_TYPE_ARRAY, array_map_ops)
index 881e9b76ef49dc1bc64a96f78fc85077b25d9a41..5c1ea99b480fa7164766454be9871b7d247d0d40 100644 (file)
@@ -107,6 +107,11 @@ static inline u16 btf_type_vlen(const struct btf_type *t)
        return BTF_INFO_VLEN(t->info);
 }
 
+static inline u16 btf_func_linkage(const struct btf_type *t)
+{
+       return BTF_INFO_VLEN(t->info);
+}
+
 static inline bool btf_type_kflag(const struct btf_type *t)
 {
        return BTF_INFO_KFLAG(t->info);
index 033d90a2282dac71ec1c1bf62dfacbe603830411..e81628eb059c8ba8f95b4359d6796020d50d2338 100644 (file)
@@ -180,6 +180,7 @@ enum bpf_prog_type {
        BPF_PROG_TYPE_CGROUP_SOCKOPT,
        BPF_PROG_TYPE_TRACING,
        BPF_PROG_TYPE_STRUCT_OPS,
+       BPF_PROG_TYPE_EXT,
 };
 
 enum bpf_attach_type {
index 832b5d7fd892ff5fcf9851ef7b15197111282260..32963b6d5a9c4e29272d8846eba5f3a171700427 100644 (file)
@@ -276,6 +276,11 @@ static const char * const btf_kind_str[NR_BTF_KINDS] = {
        [BTF_KIND_DATASEC]      = "DATASEC",
 };
 
+static const char *btf_type_str(const struct btf_type *t)
+{
+       return btf_kind_str[BTF_INFO_KIND(t->info)];
+}
+
 struct btf_kind_operations {
        s32 (*check_meta)(struct btf_verifier_env *env,
                          const struct btf_type *t,
@@ -4115,6 +4120,148 @@ int btf_distill_func_proto(struct bpf_verifier_log *log,
        return 0;
 }
 
+/* Compare BTFs of two functions assuming only scalars and pointers to context.
+ * t1 points to BTF_KIND_FUNC in btf1
+ * t2 points to BTF_KIND_FUNC in btf2
+ * Returns:
+ * EINVAL - function prototype mismatch
+ * EFAULT - verifier bug
+ * 0 - 99% match. The last 1% is validated by the verifier.
+ */
+int btf_check_func_type_match(struct bpf_verifier_log *log,
+                             struct btf *btf1, const struct btf_type *t1,
+                             struct btf *btf2, const struct btf_type *t2)
+{
+       const struct btf_param *args1, *args2;
+       const char *fn1, *fn2, *s1, *s2;
+       u32 nargs1, nargs2, i;
+
+       fn1 = btf_name_by_offset(btf1, t1->name_off);
+       fn2 = btf_name_by_offset(btf2, t2->name_off);
+
+       if (btf_func_linkage(t1) != BTF_FUNC_GLOBAL) {
+               bpf_log(log, "%s() is not a global function\n", fn1);
+               return -EINVAL;
+       }
+       if (btf_func_linkage(t2) != BTF_FUNC_GLOBAL) {
+               bpf_log(log, "%s() is not a global function\n", fn2);
+               return -EINVAL;
+       }
+
+       t1 = btf_type_by_id(btf1, t1->type);
+       if (!t1 || !btf_type_is_func_proto(t1))
+               return -EFAULT;
+       t2 = btf_type_by_id(btf2, t2->type);
+       if (!t2 || !btf_type_is_func_proto(t2))
+               return -EFAULT;
+
+       args1 = (const struct btf_param *)(t1 + 1);
+       nargs1 = btf_type_vlen(t1);
+       args2 = (const struct btf_param *)(t2 + 1);
+       nargs2 = btf_type_vlen(t2);
+
+       if (nargs1 != nargs2) {
+               bpf_log(log, "%s() has %d args while %s() has %d args\n",
+                       fn1, nargs1, fn2, nargs2);
+               return -EINVAL;
+       }
+
+       t1 = btf_type_skip_modifiers(btf1, t1->type, NULL);
+       t2 = btf_type_skip_modifiers(btf2, t2->type, NULL);
+       if (t1->info != t2->info) {
+               bpf_log(log,
+                       "Return type %s of %s() doesn't match type %s of %s()\n",
+                       btf_type_str(t1), fn1,
+                       btf_type_str(t2), fn2);
+               return -EINVAL;
+       }
+
+       for (i = 0; i < nargs1; i++) {
+               t1 = btf_type_skip_modifiers(btf1, args1[i].type, NULL);
+               t2 = btf_type_skip_modifiers(btf2, args2[i].type, NULL);
+
+               if (t1->info != t2->info) {
+                       bpf_log(log, "arg%d in %s() is %s while %s() has %s\n",
+                               i, fn1, btf_type_str(t1),
+                               fn2, btf_type_str(t2));
+                       return -EINVAL;
+               }
+               if (btf_type_has_size(t1) && t1->size != t2->size) {
+                       bpf_log(log,
+                               "arg%d in %s() has size %d while %s() has %d\n",
+                               i, fn1, t1->size,
+                               fn2, t2->size);
+                       return -EINVAL;
+               }
+
+               /* global functions are validated with scalars and pointers
+                * to context only. And only global functions can be replaced.
+                * Hence type check only those types.
+                */
+               if (btf_type_is_int(t1) || btf_type_is_enum(t1))
+                       continue;
+               if (!btf_type_is_ptr(t1)) {
+                       bpf_log(log,
+                               "arg%d in %s() has unrecognized type\n",
+                               i, fn1);
+                       return -EINVAL;
+               }
+               t1 = btf_type_skip_modifiers(btf1, t1->type, NULL);
+               t2 = btf_type_skip_modifiers(btf2, t2->type, NULL);
+               if (!btf_type_is_struct(t1)) {
+                       bpf_log(log,
+                               "arg%d in %s() is not a pointer to context\n",
+                               i, fn1);
+                       return -EINVAL;
+               }
+               if (!btf_type_is_struct(t2)) {
+                       bpf_log(log,
+                               "arg%d in %s() is not a pointer to context\n",
+                               i, fn2);
+                       return -EINVAL;
+               }
+               /* This is an optional check to make program writing easier.
+                * Compare names of structs and report an error to the user.
+                * btf_prepare_func_args() already checked that t2 struct
+                * is a context type. btf_prepare_func_args() will check
+                * later that t1 struct is a context type as well.
+                */
+               s1 = btf_name_by_offset(btf1, t1->name_off);
+               s2 = btf_name_by_offset(btf2, t2->name_off);
+               if (strcmp(s1, s2)) {
+                       bpf_log(log,
+                               "arg%d %s(struct %s *) doesn't match %s(struct %s *)\n",
+                               i, fn1, s1, fn2, s2);
+                       return -EINVAL;
+               }
+       }
+       return 0;
+}
+
+/* Compare BTFs of given program with BTF of target program */
+int btf_check_type_match(struct bpf_verifier_env *env, struct bpf_prog *prog,
+                        struct btf *btf2, const struct btf_type *t2)
+{
+       struct btf *btf1 = prog->aux->btf;
+       const struct btf_type *t1;
+       u32 btf_id = 0;
+
+       if (!prog->aux->func_info) {
+               bpf_log(&env->log, "Program extension requires BTF\n");
+               return -EINVAL;
+       }
+
+       btf_id = prog->aux->func_info[0].type_id;
+       if (!btf_id)
+               return -EFAULT;
+
+       t1 = btf_type_by_id(btf1, btf_id);
+       if (!t1 || !btf_type_is_func(t1))
+               return -EFAULT;
+
+       return btf_check_func_type_match(&env->log, btf1, t1, btf2, t2);
+}
+
 /* Compare BTF of a function with given bpf_reg_state.
  * Returns:
  * EFAULT - there is a verifier bug. Abort verification.
@@ -4224,6 +4371,7 @@ int btf_prepare_func_args(struct bpf_verifier_env *env, int subprog,
 {
        struct bpf_verifier_log *log = &env->log;
        struct bpf_prog *prog = env->prog;
+       enum bpf_prog_type prog_type = prog->type;
        struct btf *btf = prog->aux->btf;
        const struct btf_param *args;
        const struct btf_type *t;
@@ -4261,6 +4409,8 @@ int btf_prepare_func_args(struct bpf_verifier_env *env, int subprog,
                bpf_log(log, "Verifier bug in function %s()\n", tname);
                return -EFAULT;
        }
+       if (prog_type == BPF_PROG_TYPE_EXT)
+               prog_type = prog->aux->linked_prog->type;
 
        t = btf_type_by_id(btf, t->type);
        if (!t || !btf_type_is_func_proto(t)) {
@@ -4296,7 +4446,7 @@ int btf_prepare_func_args(struct bpf_verifier_env *env, int subprog,
                        continue;
                }
                if (btf_type_is_ptr(t) &&
-                   btf_get_prog_ctx_type(log, btf, t, prog->type, i)) {
+                   btf_get_prog_ctx_type(log, btf, t, prog_type, i)) {
                        reg[i + 1].type = PTR_TO_CTX;
                        continue;
                }
index 9a840c57f6df7ee874e751dca0aa5245bed9a70f..a91ad518c050335be6c751296c42149752e7c2fe 100644 (file)
@@ -1932,13 +1932,15 @@ bpf_prog_load_check_attach(enum bpf_prog_type prog_type,
                switch (prog_type) {
                case BPF_PROG_TYPE_TRACING:
                case BPF_PROG_TYPE_STRUCT_OPS:
+               case BPF_PROG_TYPE_EXT:
                        break;
                default:
                        return -EINVAL;
                }
        }
 
-       if (prog_fd && prog_type != BPF_PROG_TYPE_TRACING)
+       if (prog_fd && prog_type != BPF_PROG_TYPE_TRACING &&
+           prog_type != BPF_PROG_TYPE_EXT)
                return -EINVAL;
 
        switch (prog_type) {
@@ -1981,6 +1983,10 @@ bpf_prog_load_check_attach(enum bpf_prog_type prog_type,
                default:
                        return -EINVAL;
                }
+       case BPF_PROG_TYPE_EXT:
+               if (expected_attach_type)
+                       return -EINVAL;
+               /* fallthrough */
        default:
                return 0;
        }
@@ -2183,7 +2189,8 @@ static int bpf_tracing_prog_attach(struct bpf_prog *prog)
        int tr_fd, err;
 
        if (prog->expected_attach_type != BPF_TRACE_FENTRY &&
-           prog->expected_attach_type != BPF_TRACE_FEXIT) {
+           prog->expected_attach_type != BPF_TRACE_FEXIT &&
+           prog->type != BPF_PROG_TYPE_EXT) {
                err = -EINVAL;
                goto out_put_prog;
        }
@@ -2250,12 +2257,14 @@ static int bpf_raw_tracepoint_open(const union bpf_attr *attr)
 
        if (prog->type != BPF_PROG_TYPE_RAW_TRACEPOINT &&
            prog->type != BPF_PROG_TYPE_TRACING &&
+           prog->type != BPF_PROG_TYPE_EXT &&
            prog->type != BPF_PROG_TYPE_RAW_TRACEPOINT_WRITABLE) {
                err = -EINVAL;
                goto out_put_prog;
        }
 
-       if (prog->type == BPF_PROG_TYPE_TRACING) {
+       if (prog->type == BPF_PROG_TYPE_TRACING ||
+           prog->type == BPF_PROG_TYPE_EXT) {
                if (attr->raw_tracepoint.name) {
                        /* The attach point for this category of programs
                         * should be specified via btf_id during program load.
index 7657ede7aee211623ffbc782d29da49f0686684c..eb64c245052b427720490c5cc79ddfead714f14b 100644 (file)
@@ -5,6 +5,12 @@
 #include <linux/filter.h>
 #include <linux/ftrace.h>
 
+/* dummy _ops. The verifier will operate on target program's ops. */
+const struct bpf_verifier_ops bpf_extension_verifier_ops = {
+};
+const struct bpf_prog_ops bpf_extension_prog_ops = {
+};
+
 /* btf_vmlinux has ~22k attachable functions. 1k htab is enough. */
 #define TRAMPOLINE_HASH_BITS 10
 #define TRAMPOLINE_TABLE_SIZE (1 << TRAMPOLINE_HASH_BITS)
@@ -194,8 +200,10 @@ static enum bpf_tramp_prog_type bpf_attach_type_to_tramp(enum bpf_attach_type t)
        switch (t) {
        case BPF_TRACE_FENTRY:
                return BPF_TRAMP_FENTRY;
-       default:
+       case BPF_TRACE_FEXIT:
                return BPF_TRAMP_FEXIT;
+       default:
+               return BPF_TRAMP_REPLACE;
        }
 }
 
@@ -204,12 +212,31 @@ int bpf_trampoline_link_prog(struct bpf_prog *prog)
        enum bpf_tramp_prog_type kind;
        struct bpf_trampoline *tr;
        int err = 0;
+       int cnt;
 
        tr = prog->aux->trampoline;
        kind = bpf_attach_type_to_tramp(prog->expected_attach_type);
        mutex_lock(&tr->mutex);
-       if (tr->progs_cnt[BPF_TRAMP_FENTRY] + tr->progs_cnt[BPF_TRAMP_FEXIT]
-           >= BPF_MAX_TRAMP_PROGS) {
+       if (tr->extension_prog) {
+               /* cannot attach fentry/fexit if extension prog is attached.
+                * cannot overwrite extension prog either.
+                */
+               err = -EBUSY;
+               goto out;
+       }
+       cnt = tr->progs_cnt[BPF_TRAMP_FENTRY] + tr->progs_cnt[BPF_TRAMP_FEXIT];
+       if (kind == BPF_TRAMP_REPLACE) {
+               /* Cannot attach extension if fentry/fexit are in use. */
+               if (cnt) {
+                       err = -EBUSY;
+                       goto out;
+               }
+               tr->extension_prog = prog;
+               err = bpf_arch_text_poke(tr->func.addr, BPF_MOD_JUMP, NULL,
+                                        prog->bpf_func);
+               goto out;
+       }
+       if (cnt >= BPF_MAX_TRAMP_PROGS) {
                err = -E2BIG;
                goto out;
        }
@@ -240,9 +267,17 @@ int bpf_trampoline_unlink_prog(struct bpf_prog *prog)
        tr = prog->aux->trampoline;
        kind = bpf_attach_type_to_tramp(prog->expected_attach_type);
        mutex_lock(&tr->mutex);
+       if (kind == BPF_TRAMP_REPLACE) {
+               WARN_ON_ONCE(!tr->extension_prog);
+               err = bpf_arch_text_poke(tr->func.addr, BPF_MOD_JUMP,
+                                        tr->extension_prog->bpf_func, NULL);
+               tr->extension_prog = NULL;
+               goto out;
+       }
        hlist_del(&prog->aux->tramp_hlist);
        tr->progs_cnt[kind]--;
        err = bpf_trampoline_update(prog->aux->trampoline);
+out:
        mutex_unlock(&tr->mutex);
        return err;
 }
index 6defbec9eb62bb123d150d1a1edaea56b6d298cd..9fe94f64f0ecc21de8741b6683e614e19cbb14bc 100644 (file)
@@ -9564,7 +9564,7 @@ static int do_check_common(struct bpf_verifier_env *env, int subprog)
                        subprog);
 
        regs = state->frame[state->curframe]->regs;
-       if (subprog) {
+       if (subprog || env->prog->type == BPF_PROG_TYPE_EXT) {
                ret = btf_prepare_func_args(env, subprog, regs);
                if (ret)
                        goto out;
@@ -9742,6 +9742,7 @@ static int check_struct_ops_btf_id(struct bpf_verifier_env *env)
 static int check_attach_btf_id(struct bpf_verifier_env *env)
 {
        struct bpf_prog *prog = env->prog;
+       bool prog_extension = prog->type == BPF_PROG_TYPE_EXT;
        struct bpf_prog *tgt_prog = prog->aux->linked_prog;
        u32 btf_id = prog->aux->attach_btf_id;
        const char prefix[] = "btf_trace_";
@@ -9757,7 +9758,7 @@ static int check_attach_btf_id(struct bpf_verifier_env *env)
        if (prog->type == BPF_PROG_TYPE_STRUCT_OPS)
                return check_struct_ops_btf_id(env);
 
-       if (prog->type != BPF_PROG_TYPE_TRACING)
+       if (prog->type != BPF_PROG_TYPE_TRACING && !prog_extension)
                return 0;
 
        if (!btf_id) {
@@ -9793,8 +9794,59 @@ static int check_attach_btf_id(struct bpf_verifier_env *env)
                        return -EINVAL;
                }
                conservative = aux->func_info_aux[subprog].unreliable;
+               if (prog_extension) {
+                       if (conservative) {
+                               verbose(env,
+                                       "Cannot replace static functions\n");
+                               return -EINVAL;
+                       }
+                       if (!prog->jit_requested) {
+                               verbose(env,
+                                       "Extension programs should be JITed\n");
+                               return -EINVAL;
+                       }
+                       env->ops = bpf_verifier_ops[tgt_prog->type];
+               }
+               if (!tgt_prog->jited) {
+                       verbose(env, "Can attach to only JITed progs\n");
+                       return -EINVAL;
+               }
+               if (tgt_prog->type == prog->type) {
+                       /* Cannot fentry/fexit another fentry/fexit program.
+                        * Cannot attach program extension to another extension.
+                        * It's ok to attach fentry/fexit to extension program.
+                        */
+                       verbose(env, "Cannot recursively attach\n");
+                       return -EINVAL;
+               }
+               if (tgt_prog->type == BPF_PROG_TYPE_TRACING &&
+                   prog_extension &&
+                   (tgt_prog->expected_attach_type == BPF_TRACE_FENTRY ||
+                    tgt_prog->expected_attach_type == BPF_TRACE_FEXIT)) {
+                       /* Program extensions can extend all program types
+                        * except fentry/fexit. The reason is the following.
+                        * The fentry/fexit programs are used for performance
+                        * analysis, stats and can be attached to any program
+                        * type except themselves. When extension program is
+                        * replacing XDP function it is necessary to allow
+                        * performance analysis of all functions. Both original
+                        * XDP program and its program extension. Hence
+                        * attaching fentry/fexit to BPF_PROG_TYPE_EXT is
+                        * allowed. If extending of fentry/fexit was allowed it
+                        * would be possible to create long call chain
+                        * fentry->extension->fentry->extension beyond
+                        * reasonable stack size. Hence extending fentry is not
+                        * allowed.
+                        */
+                       verbose(env, "Cannot extend fentry/fexit\n");
+                       return -EINVAL;
+               }
                key = ((u64)aux->id) << 32 | btf_id;
        } else {
+               if (prog_extension) {
+                       verbose(env, "Cannot replace kernel functions\n");
+                       return -EINVAL;
+               }
                key = btf_id;
        }
 
@@ -9832,6 +9884,10 @@ static int check_attach_btf_id(struct bpf_verifier_env *env)
                prog->aux->attach_func_proto = t;
                prog->aux->attach_btf_trace = true;
                return 0;
+       default:
+               if (!prog_extension)
+                       return -EINVAL;
+               /* fallthrough */
        case BPF_TRACE_FENTRY:
        case BPF_TRACE_FEXIT:
                if (!btf_type_is_func(t)) {
@@ -9839,6 +9895,9 @@ static int check_attach_btf_id(struct bpf_verifier_env *env)
                                btf_id);
                        return -EINVAL;
                }
+               if (prog_extension &&
+                   btf_check_type_match(env, prog, btf, t))
+                       return -EINVAL;
                t = btf_type_by_id(btf, t->type);
                if (!btf_type_is_func_proto(t))
                        return -EINVAL;
@@ -9862,18 +9921,6 @@ static int check_attach_btf_id(struct bpf_verifier_env *env)
                if (ret < 0)
                        goto out;
                if (tgt_prog) {
-                       if (!tgt_prog->jited) {
-                               /* for now */
-                               verbose(env, "Can trace only JITed BPF progs\n");
-                               ret = -EINVAL;
-                               goto out;
-                       }
-                       if (tgt_prog->type == BPF_PROG_TYPE_TRACING) {
-                               /* prevent cycles */
-                               verbose(env, "Cannot recursively attach\n");
-                               ret = -EINVAL;
-                               goto out;
-                       }
                        if (subprog == 0)
                                addr = (long) tgt_prog->bpf_func;
                        else
@@ -9895,8 +9942,6 @@ out:
                if (ret)
                        bpf_trampoline_put(tr);
                return ret;
-       default:
-               return -EINVAL;
        }
 }
 
@@ -9966,10 +10011,6 @@ int bpf_check(struct bpf_prog **prog, union bpf_attr *attr,
                goto skip_full_check;
        }
 
-       ret = check_attach_btf_id(env);
-       if (ret)
-               goto skip_full_check;
-
        env->strict_alignment = !!(attr->prog_flags & BPF_F_STRICT_ALIGNMENT);
        if (!IS_ENABLED(CONFIG_HAVE_EFFICIENT_UNALIGNED_ACCESS))
                env->strict_alignment = true;
@@ -10006,6 +10047,10 @@ int bpf_check(struct bpf_prog **prog, union bpf_attr *attr,
        if (ret < 0)
                goto skip_full_check;
 
+       ret = check_attach_btf_id(env);
+       if (ret)
+               goto skip_full_check;
+
        ret = check_cfg(env);
        if (ret < 0)
                goto skip_full_check;