bpf: Introduce function-by-function verification
New llvm and old llvm with libbpf help produce BTF that distinguish global and static functions. Unlike arguments of static function the arguments of global functions cannot be removed or optimized away by llvm. The compiler has to use exactly the arguments specified in a function prototype. The argument type information allows the verifier validate each global function independently. For now only supported argument types are pointer to context and scalars. In the future pointers to structures, sizes, pointer to packet data can be supported as well. Consider the following example: static int f1(int ...) { ... } int f3(int b); int f2(int a) { f1(a) + f3(a); } int f3(int b) { ... } int main(...) { f1(...) + f2(...) + f3(...); } The verifier will start its safety checks from the first global function f2(). It will recursively descend into f1() because it's static. Then it will check that arguments match for the f3() invocation inside f2(). It will not descend into f3(). It will finish f2() that has to be successfully verified for all possible values of 'a'. Then it will proceed with f3(). That function also has to be safe for all possible values of 'b'. Then it will start subprog 0 (which is main() function). It will recursively descend into f1() and will skip full check of f2() and f3(), since they are global. The order of processing global functions doesn't affect safety, since all global functions must be proven safe based on their arguments only. Such function by function verification can drastically improve speed of the verification and reduce complexity. Note that the stack limit of 512 still applies to the call chain regardless whether functions were static or global. The nested level of 8 also still applies. The same recursion prevention checks are in place as well. The type information and static/global kind is preserved after the verification hence in the above example global function f2() and f3() can be replaced later by equivalent functions with the same types that are loaded and verified later without affecting safety of this main() program. Such replacement (re-linking) of global functions is a subject of future patches. Signed-off-by: Alexei Starovoitov <ast@kernel.org> Signed-off-by: Daniel Borkmann <daniel@iogearbox.net> Acked-by: Song Liu <songliubraving@fb.com> Link: https://lore.kernel.org/bpf/20200110064124.1760511-3-ast@kernel.org
This commit is contained in:
parent
2d3eb67f64
commit
51c39bb1d5
@ -566,6 +566,7 @@ static inline void bpf_dispatcher_change_prog(struct bpf_dispatcher *d,
|
|||||||
#endif
|
#endif
|
||||||
|
|
||||||
struct bpf_func_info_aux {
|
struct bpf_func_info_aux {
|
||||||
|
u16 linkage;
|
||||||
bool unreliable;
|
bool unreliable;
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -1081,7 +1082,11 @@ int btf_distill_func_proto(struct bpf_verifier_log *log,
|
|||||||
const char *func_name,
|
const char *func_name,
|
||||||
struct btf_func_model *m);
|
struct btf_func_model *m);
|
||||||
|
|
||||||
int btf_check_func_arg_match(struct bpf_verifier_env *env, int subprog);
|
struct bpf_reg_state;
|
||||||
|
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);
|
||||||
|
|
||||||
struct bpf_prog *bpf_prog_by_id(u32 id);
|
struct bpf_prog *bpf_prog_by_id(u32 id);
|
||||||
|
|
||||||
|
@ -304,11 +304,13 @@ struct bpf_insn_aux_data {
|
|||||||
u64 map_key_state; /* constant (32 bit) key tracking for maps */
|
u64 map_key_state; /* constant (32 bit) key tracking for maps */
|
||||||
int ctx_field_size; /* the ctx field size for load insn, maybe 0 */
|
int ctx_field_size; /* the ctx field size for load insn, maybe 0 */
|
||||||
int sanitize_stack_off; /* stack slot to be cleared */
|
int sanitize_stack_off; /* stack slot to be cleared */
|
||||||
bool seen; /* this insn was processed by the verifier */
|
u32 seen; /* this insn was processed by the verifier at env->pass_cnt */
|
||||||
bool zext_dst; /* this insn zero extends dst reg */
|
bool zext_dst; /* this insn zero extends dst reg */
|
||||||
u8 alu_state; /* used in combination with alu_limit */
|
u8 alu_state; /* used in combination with alu_limit */
|
||||||
bool prune_point;
|
|
||||||
|
/* below fields are initialized once */
|
||||||
unsigned int orig_idx; /* original instruction index */
|
unsigned int orig_idx; /* original instruction index */
|
||||||
|
bool prune_point;
|
||||||
};
|
};
|
||||||
|
|
||||||
#define MAX_USED_MAPS 64 /* max number of maps accessed by one eBPF program */
|
#define MAX_USED_MAPS 64 /* max number of maps accessed by one eBPF program */
|
||||||
@ -379,6 +381,7 @@ struct bpf_verifier_env {
|
|||||||
int *insn_stack;
|
int *insn_stack;
|
||||||
int cur_stack;
|
int cur_stack;
|
||||||
} cfg;
|
} cfg;
|
||||||
|
u32 pass_cnt; /* number of times do_check() was called */
|
||||||
u32 subprog_cnt;
|
u32 subprog_cnt;
|
||||||
/* number of instructions analyzed by the verifier */
|
/* number of instructions analyzed by the verifier */
|
||||||
u32 prev_insn_processed, insn_processed;
|
u32 prev_insn_processed, insn_processed;
|
||||||
@ -428,4 +431,7 @@ bpf_prog_offload_replace_insn(struct bpf_verifier_env *env, u32 off,
|
|||||||
void
|
void
|
||||||
bpf_prog_offload_remove_insns(struct bpf_verifier_env *env, u32 off, u32 cnt);
|
bpf_prog_offload_remove_insns(struct bpf_verifier_env *env, u32 off, u32 cnt);
|
||||||
|
|
||||||
|
int check_ctx_reg(struct bpf_verifier_env *env,
|
||||||
|
const struct bpf_reg_state *reg, int regno);
|
||||||
|
|
||||||
#endif /* _LINUX_BPF_VERIFIER_H */
|
#endif /* _LINUX_BPF_VERIFIER_H */
|
||||||
|
@ -146,6 +146,12 @@ enum {
|
|||||||
BTF_VAR_GLOBAL_EXTERN = 2,
|
BTF_VAR_GLOBAL_EXTERN = 2,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
enum btf_func_linkage {
|
||||||
|
BTF_FUNC_STATIC = 0,
|
||||||
|
BTF_FUNC_GLOBAL = 1,
|
||||||
|
BTF_FUNC_EXTERN = 2,
|
||||||
|
};
|
||||||
|
|
||||||
/* BTF_KIND_VAR is followed by a single "struct btf_var" to describe
|
/* BTF_KIND_VAR is followed by a single "struct btf_var" to describe
|
||||||
* additional information related to the variable such as its linkage.
|
* additional information related to the variable such as its linkage.
|
||||||
*/
|
*/
|
||||||
|
175
kernel/bpf/btf.c
175
kernel/bpf/btf.c
@ -2651,8 +2651,8 @@ static s32 btf_func_check_meta(struct btf_verifier_env *env,
|
|||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (btf_type_vlen(t)) {
|
if (btf_type_vlen(t) > BTF_FUNC_GLOBAL) {
|
||||||
btf_verifier_log_type(env, t, "vlen != 0");
|
btf_verifier_log_type(env, t, "Invalid func linkage");
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -3506,7 +3506,8 @@ static u8 bpf_ctx_convert_map[] = {
|
|||||||
|
|
||||||
static const struct btf_member *
|
static const struct btf_member *
|
||||||
btf_get_prog_ctx_type(struct bpf_verifier_log *log, struct btf *btf,
|
btf_get_prog_ctx_type(struct bpf_verifier_log *log, struct btf *btf,
|
||||||
const struct btf_type *t, enum bpf_prog_type prog_type)
|
const struct btf_type *t, enum bpf_prog_type prog_type,
|
||||||
|
int arg)
|
||||||
{
|
{
|
||||||
const struct btf_type *conv_struct;
|
const struct btf_type *conv_struct;
|
||||||
const struct btf_type *ctx_struct;
|
const struct btf_type *ctx_struct;
|
||||||
@ -3527,12 +3528,13 @@ btf_get_prog_ctx_type(struct bpf_verifier_log *log, struct btf *btf,
|
|||||||
* is not supported yet.
|
* is not supported yet.
|
||||||
* BPF_PROG_TYPE_RAW_TRACEPOINT is fine.
|
* BPF_PROG_TYPE_RAW_TRACEPOINT is fine.
|
||||||
*/
|
*/
|
||||||
bpf_log(log, "BPF program ctx type is not a struct\n");
|
if (log->level & BPF_LOG_LEVEL)
|
||||||
|
bpf_log(log, "arg#%d type is not a struct\n", arg);
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
tname = btf_name_by_offset(btf, t->name_off);
|
tname = btf_name_by_offset(btf, t->name_off);
|
||||||
if (!tname) {
|
if (!tname) {
|
||||||
bpf_log(log, "BPF program ctx struct doesn't have a name\n");
|
bpf_log(log, "arg#%d struct doesn't have a name\n", arg);
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
/* prog_type is valid bpf program type. No need for bounds check. */
|
/* prog_type is valid bpf program type. No need for bounds check. */
|
||||||
@ -3565,11 +3567,12 @@ btf_get_prog_ctx_type(struct bpf_verifier_log *log, struct btf *btf,
|
|||||||
static int btf_translate_to_vmlinux(struct bpf_verifier_log *log,
|
static int btf_translate_to_vmlinux(struct bpf_verifier_log *log,
|
||||||
struct btf *btf,
|
struct btf *btf,
|
||||||
const struct btf_type *t,
|
const struct btf_type *t,
|
||||||
enum bpf_prog_type prog_type)
|
enum bpf_prog_type prog_type,
|
||||||
|
int arg)
|
||||||
{
|
{
|
||||||
const struct btf_member *prog_ctx_type, *kern_ctx_type;
|
const struct btf_member *prog_ctx_type, *kern_ctx_type;
|
||||||
|
|
||||||
prog_ctx_type = btf_get_prog_ctx_type(log, btf, t, prog_type);
|
prog_ctx_type = btf_get_prog_ctx_type(log, btf, t, prog_type, arg);
|
||||||
if (!prog_ctx_type)
|
if (!prog_ctx_type)
|
||||||
return -ENOENT;
|
return -ENOENT;
|
||||||
kern_ctx_type = prog_ctx_type + 1;
|
kern_ctx_type = prog_ctx_type + 1;
|
||||||
@ -3731,7 +3734,7 @@ bool btf_ctx_access(int off, int size, enum bpf_access_type type,
|
|||||||
info->reg_type = PTR_TO_BTF_ID;
|
info->reg_type = PTR_TO_BTF_ID;
|
||||||
|
|
||||||
if (tgt_prog) {
|
if (tgt_prog) {
|
||||||
ret = btf_translate_to_vmlinux(log, btf, t, tgt_prog->type);
|
ret = btf_translate_to_vmlinux(log, btf, t, tgt_prog->type, arg);
|
||||||
if (ret > 0) {
|
if (ret > 0) {
|
||||||
info->btf_id = ret;
|
info->btf_id = ret;
|
||||||
return true;
|
return true;
|
||||||
@ -4112,11 +4115,16 @@ int btf_distill_func_proto(struct bpf_verifier_log *log,
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
int btf_check_func_arg_match(struct bpf_verifier_env *env, int subprog)
|
/* Compare BTF of a function with given bpf_reg_state.
|
||||||
|
* Returns:
|
||||||
|
* EFAULT - there is a verifier bug. Abort verification.
|
||||||
|
* EINVAL - there is a type mismatch or BTF is not available.
|
||||||
|
* 0 - BTF matches with what bpf_reg_state expects.
|
||||||
|
* Only PTR_TO_CTX and SCALAR_VALUE states are recognized.
|
||||||
|
*/
|
||||||
|
int btf_check_func_arg_match(struct bpf_verifier_env *env, int subprog,
|
||||||
|
struct bpf_reg_state *reg)
|
||||||
{
|
{
|
||||||
struct bpf_verifier_state *st = env->cur_state;
|
|
||||||
struct bpf_func_state *func = st->frame[st->curframe];
|
|
||||||
struct bpf_reg_state *reg = func->regs;
|
|
||||||
struct bpf_verifier_log *log = &env->log;
|
struct bpf_verifier_log *log = &env->log;
|
||||||
struct bpf_prog *prog = env->prog;
|
struct bpf_prog *prog = env->prog;
|
||||||
struct btf *btf = prog->aux->btf;
|
struct btf *btf = prog->aux->btf;
|
||||||
@ -4126,27 +4134,30 @@ int btf_check_func_arg_match(struct bpf_verifier_env *env, int subprog)
|
|||||||
const char *tname;
|
const char *tname;
|
||||||
|
|
||||||
if (!prog->aux->func_info)
|
if (!prog->aux->func_info)
|
||||||
return 0;
|
return -EINVAL;
|
||||||
|
|
||||||
btf_id = prog->aux->func_info[subprog].type_id;
|
btf_id = prog->aux->func_info[subprog].type_id;
|
||||||
if (!btf_id)
|
if (!btf_id)
|
||||||
return 0;
|
return -EFAULT;
|
||||||
|
|
||||||
if (prog->aux->func_info_aux[subprog].unreliable)
|
if (prog->aux->func_info_aux[subprog].unreliable)
|
||||||
return 0;
|
return -EINVAL;
|
||||||
|
|
||||||
t = btf_type_by_id(btf, btf_id);
|
t = btf_type_by_id(btf, btf_id);
|
||||||
if (!t || !btf_type_is_func(t)) {
|
if (!t || !btf_type_is_func(t)) {
|
||||||
bpf_log(log, "BTF of subprog %d doesn't point to KIND_FUNC\n",
|
/* These checks were already done by the verifier while loading
|
||||||
|
* struct bpf_func_info
|
||||||
|
*/
|
||||||
|
bpf_log(log, "BTF of func#%d doesn't point to KIND_FUNC\n",
|
||||||
subprog);
|
subprog);
|
||||||
return -EINVAL;
|
return -EFAULT;
|
||||||
}
|
}
|
||||||
tname = btf_name_by_offset(btf, t->name_off);
|
tname = btf_name_by_offset(btf, t->name_off);
|
||||||
|
|
||||||
t = btf_type_by_id(btf, t->type);
|
t = btf_type_by_id(btf, t->type);
|
||||||
if (!t || !btf_type_is_func_proto(t)) {
|
if (!t || !btf_type_is_func_proto(t)) {
|
||||||
bpf_log(log, "Invalid type of func %s\n", tname);
|
bpf_log(log, "Invalid BTF of func %s\n", tname);
|
||||||
return -EINVAL;
|
return -EFAULT;
|
||||||
}
|
}
|
||||||
args = (const struct btf_param *)(t + 1);
|
args = (const struct btf_param *)(t + 1);
|
||||||
nargs = btf_type_vlen(t);
|
nargs = btf_type_vlen(t);
|
||||||
@ -4172,25 +4183,127 @@ int btf_check_func_arg_match(struct bpf_verifier_env *env, int subprog)
|
|||||||
bpf_log(log, "R%d is not a pointer\n", i + 1);
|
bpf_log(log, "R%d is not a pointer\n", i + 1);
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
/* If program is passing PTR_TO_CTX into subprogram
|
/* If function expects ctx type in BTF check that caller
|
||||||
* check that BTF type matches.
|
* is passing PTR_TO_CTX.
|
||||||
*/
|
*/
|
||||||
if (reg[i + 1].type == PTR_TO_CTX &&
|
if (btf_get_prog_ctx_type(log, btf, t, prog->type, i)) {
|
||||||
!btf_get_prog_ctx_type(log, btf, t, prog->type))
|
if (reg[i + 1].type != PTR_TO_CTX) {
|
||||||
goto out;
|
bpf_log(log,
|
||||||
/* All other pointers are ok */
|
"arg#%d expected pointer to ctx, but got %s\n",
|
||||||
continue;
|
i, btf_kind_str[BTF_INFO_KIND(t->info)]);
|
||||||
|
goto out;
|
||||||
|
}
|
||||||
|
if (check_ctx_reg(env, ®[i + 1], i + 1))
|
||||||
|
goto out;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
bpf_log(log, "Unrecognized argument type %s\n",
|
bpf_log(log, "Unrecognized arg#%d type %s\n",
|
||||||
btf_kind_str[BTF_INFO_KIND(t->info)]);
|
i, btf_kind_str[BTF_INFO_KIND(t->info)]);
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
return 0;
|
return 0;
|
||||||
out:
|
out:
|
||||||
/* LLVM optimizations can remove arguments from static functions. */
|
/* Compiler optimizations can remove arguments from static functions
|
||||||
bpf_log(log,
|
* or mismatched type can be passed into a global function.
|
||||||
"Type info disagrees with actual arguments due to compiler optimizations\n");
|
* In such cases mark the function as unreliable from BTF point of view.
|
||||||
|
*/
|
||||||
prog->aux->func_info_aux[subprog].unreliable = true;
|
prog->aux->func_info_aux[subprog].unreliable = true;
|
||||||
|
return -EINVAL;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Convert BTF of a function into bpf_reg_state if possible
|
||||||
|
* Returns:
|
||||||
|
* EFAULT - there is a verifier bug. Abort verification.
|
||||||
|
* EINVAL - cannot convert BTF.
|
||||||
|
* 0 - Successfully converted BTF into bpf_reg_state
|
||||||
|
* (either PTR_TO_CTX or SCALAR_VALUE).
|
||||||
|
*/
|
||||||
|
int btf_prepare_func_args(struct bpf_verifier_env *env, int subprog,
|
||||||
|
struct bpf_reg_state *reg)
|
||||||
|
{
|
||||||
|
struct bpf_verifier_log *log = &env->log;
|
||||||
|
struct bpf_prog *prog = env->prog;
|
||||||
|
struct btf *btf = prog->aux->btf;
|
||||||
|
const struct btf_param *args;
|
||||||
|
const struct btf_type *t;
|
||||||
|
u32 i, nargs, btf_id;
|
||||||
|
const char *tname;
|
||||||
|
|
||||||
|
if (!prog->aux->func_info ||
|
||||||
|
prog->aux->func_info_aux[subprog].linkage != BTF_FUNC_GLOBAL) {
|
||||||
|
bpf_log(log, "Verifier bug\n");
|
||||||
|
return -EFAULT;
|
||||||
|
}
|
||||||
|
|
||||||
|
btf_id = prog->aux->func_info[subprog].type_id;
|
||||||
|
if (!btf_id) {
|
||||||
|
bpf_log(log, "Global functions need valid BTF\n");
|
||||||
|
return -EFAULT;
|
||||||
|
}
|
||||||
|
|
||||||
|
t = btf_type_by_id(btf, btf_id);
|
||||||
|
if (!t || !btf_type_is_func(t)) {
|
||||||
|
/* These checks were already done by the verifier while loading
|
||||||
|
* struct bpf_func_info
|
||||||
|
*/
|
||||||
|
bpf_log(log, "BTF of func#%d doesn't point to KIND_FUNC\n",
|
||||||
|
subprog);
|
||||||
|
return -EFAULT;
|
||||||
|
}
|
||||||
|
tname = btf_name_by_offset(btf, t->name_off);
|
||||||
|
|
||||||
|
if (log->level & BPF_LOG_LEVEL)
|
||||||
|
bpf_log(log, "Validating %s() func#%d...\n",
|
||||||
|
tname, subprog);
|
||||||
|
|
||||||
|
if (prog->aux->func_info_aux[subprog].unreliable) {
|
||||||
|
bpf_log(log, "Verifier bug in function %s()\n", tname);
|
||||||
|
return -EFAULT;
|
||||||
|
}
|
||||||
|
|
||||||
|
t = btf_type_by_id(btf, t->type);
|
||||||
|
if (!t || !btf_type_is_func_proto(t)) {
|
||||||
|
bpf_log(log, "Invalid type of function %s()\n", tname);
|
||||||
|
return -EFAULT;
|
||||||
|
}
|
||||||
|
args = (const struct btf_param *)(t + 1);
|
||||||
|
nargs = btf_type_vlen(t);
|
||||||
|
if (nargs > 5) {
|
||||||
|
bpf_log(log, "Global function %s() with %d > 5 args. Buggy compiler.\n",
|
||||||
|
tname, nargs);
|
||||||
|
return -EINVAL;
|
||||||
|
}
|
||||||
|
/* check that function returns int */
|
||||||
|
t = btf_type_by_id(btf, t->type);
|
||||||
|
while (btf_type_is_modifier(t))
|
||||||
|
t = btf_type_by_id(btf, t->type);
|
||||||
|
if (!btf_type_is_int(t) && !btf_type_is_enum(t)) {
|
||||||
|
bpf_log(log,
|
||||||
|
"Global function %s() doesn't return scalar. Only those are supported.\n",
|
||||||
|
tname);
|
||||||
|
return -EINVAL;
|
||||||
|
}
|
||||||
|
/* Convert BTF function arguments into verifier types.
|
||||||
|
* Only PTR_TO_CTX and SCALAR are supported atm.
|
||||||
|
*/
|
||||||
|
for (i = 0; i < nargs; i++) {
|
||||||
|
t = btf_type_by_id(btf, args[i].type);
|
||||||
|
while (btf_type_is_modifier(t))
|
||||||
|
t = btf_type_by_id(btf, t->type);
|
||||||
|
if (btf_type_is_int(t) || btf_type_is_enum(t)) {
|
||||||
|
reg[i + 1].type = SCALAR_VALUE;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (btf_type_is_ptr(t) &&
|
||||||
|
btf_get_prog_ctx_type(log, btf, t, prog->type, i)) {
|
||||||
|
reg[i + 1].type = PTR_TO_CTX;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
bpf_log(log, "Arg#%d type %s in %s() is not supported yet.\n",
|
||||||
|
i, btf_kind_str[BTF_INFO_KIND(t->info)], tname);
|
||||||
|
return -EINVAL;
|
||||||
|
}
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1122,10 +1122,6 @@ static void init_reg_state(struct bpf_verifier_env *env,
|
|||||||
regs[BPF_REG_FP].type = PTR_TO_STACK;
|
regs[BPF_REG_FP].type = PTR_TO_STACK;
|
||||||
mark_reg_known_zero(env, regs, BPF_REG_FP);
|
mark_reg_known_zero(env, regs, BPF_REG_FP);
|
||||||
regs[BPF_REG_FP].frameno = state->frameno;
|
regs[BPF_REG_FP].frameno = state->frameno;
|
||||||
|
|
||||||
/* 1st arg to a function */
|
|
||||||
regs[BPF_REG_1].type = PTR_TO_CTX;
|
|
||||||
mark_reg_known_zero(env, regs, BPF_REG_1);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#define BPF_MAIN_FUNC (-1)
|
#define BPF_MAIN_FUNC (-1)
|
||||||
@ -2739,8 +2735,8 @@ static int get_callee_stack_depth(struct bpf_verifier_env *env,
|
|||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
static int check_ctx_reg(struct bpf_verifier_env *env,
|
int check_ctx_reg(struct bpf_verifier_env *env,
|
||||||
const struct bpf_reg_state *reg, int regno)
|
const struct bpf_reg_state *reg, int regno)
|
||||||
{
|
{
|
||||||
/* Access to ctx or passing it to a helper is only allowed in
|
/* Access to ctx or passing it to a helper is only allowed in
|
||||||
* its original, unmodified form.
|
* its original, unmodified form.
|
||||||
@ -3956,12 +3952,26 @@ static int release_reference(struct bpf_verifier_env *env,
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void clear_caller_saved_regs(struct bpf_verifier_env *env,
|
||||||
|
struct bpf_reg_state *regs)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
|
||||||
|
/* after the call registers r0 - r5 were scratched */
|
||||||
|
for (i = 0; i < CALLER_SAVED_REGS; i++) {
|
||||||
|
mark_reg_not_init(env, regs, caller_saved[i]);
|
||||||
|
check_reg_arg(env, caller_saved[i], DST_OP_NO_MARK);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
static int check_func_call(struct bpf_verifier_env *env, struct bpf_insn *insn,
|
static int check_func_call(struct bpf_verifier_env *env, struct bpf_insn *insn,
|
||||||
int *insn_idx)
|
int *insn_idx)
|
||||||
{
|
{
|
||||||
struct bpf_verifier_state *state = env->cur_state;
|
struct bpf_verifier_state *state = env->cur_state;
|
||||||
|
struct bpf_func_info_aux *func_info_aux;
|
||||||
struct bpf_func_state *caller, *callee;
|
struct bpf_func_state *caller, *callee;
|
||||||
int i, err, subprog, target_insn;
|
int i, err, subprog, target_insn;
|
||||||
|
bool is_global = false;
|
||||||
|
|
||||||
if (state->curframe + 1 >= MAX_CALL_FRAMES) {
|
if (state->curframe + 1 >= MAX_CALL_FRAMES) {
|
||||||
verbose(env, "the call stack of %d frames is too deep\n",
|
verbose(env, "the call stack of %d frames is too deep\n",
|
||||||
@ -3984,6 +3994,32 @@ static int check_func_call(struct bpf_verifier_env *env, struct bpf_insn *insn,
|
|||||||
return -EFAULT;
|
return -EFAULT;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func_info_aux = env->prog->aux->func_info_aux;
|
||||||
|
if (func_info_aux)
|
||||||
|
is_global = func_info_aux[subprog].linkage == BTF_FUNC_GLOBAL;
|
||||||
|
err = btf_check_func_arg_match(env, subprog, caller->regs);
|
||||||
|
if (err == -EFAULT)
|
||||||
|
return err;
|
||||||
|
if (is_global) {
|
||||||
|
if (err) {
|
||||||
|
verbose(env, "Caller passes invalid args into func#%d\n",
|
||||||
|
subprog);
|
||||||
|
return err;
|
||||||
|
} else {
|
||||||
|
if (env->log.level & BPF_LOG_LEVEL)
|
||||||
|
verbose(env,
|
||||||
|
"Func#%d is global and valid. Skipping.\n",
|
||||||
|
subprog);
|
||||||
|
clear_caller_saved_regs(env, caller->regs);
|
||||||
|
|
||||||
|
/* All global functions return SCALAR_VALUE */
|
||||||
|
mark_reg_unknown(env, caller->regs, BPF_REG_0);
|
||||||
|
|
||||||
|
/* continue with next insn after call */
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
callee = kzalloc(sizeof(*callee), GFP_KERNEL);
|
callee = kzalloc(sizeof(*callee), GFP_KERNEL);
|
||||||
if (!callee)
|
if (!callee)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
@ -4010,18 +4046,11 @@ static int check_func_call(struct bpf_verifier_env *env, struct bpf_insn *insn,
|
|||||||
for (i = BPF_REG_1; i <= BPF_REG_5; i++)
|
for (i = BPF_REG_1; i <= BPF_REG_5; i++)
|
||||||
callee->regs[i] = caller->regs[i];
|
callee->regs[i] = caller->regs[i];
|
||||||
|
|
||||||
/* after the call registers r0 - r5 were scratched */
|
clear_caller_saved_regs(env, caller->regs);
|
||||||
for (i = 0; i < CALLER_SAVED_REGS; i++) {
|
|
||||||
mark_reg_not_init(env, caller->regs, caller_saved[i]);
|
|
||||||
check_reg_arg(env, caller_saved[i], DST_OP_NO_MARK);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* only increment it after check_reg_arg() finished */
|
/* only increment it after check_reg_arg() finished */
|
||||||
state->curframe++;
|
state->curframe++;
|
||||||
|
|
||||||
if (btf_check_func_arg_match(env, subprog))
|
|
||||||
return -EINVAL;
|
|
||||||
|
|
||||||
/* and go analyze first insn of the callee */
|
/* and go analyze first insn of the callee */
|
||||||
*insn_idx = target_insn;
|
*insn_idx = target_insn;
|
||||||
|
|
||||||
@ -6771,12 +6800,13 @@ static int check_btf_func(struct bpf_verifier_env *env,
|
|||||||
|
|
||||||
/* check type_id */
|
/* check type_id */
|
||||||
type = btf_type_by_id(btf, krecord[i].type_id);
|
type = btf_type_by_id(btf, krecord[i].type_id);
|
||||||
if (!type || BTF_INFO_KIND(type->info) != BTF_KIND_FUNC) {
|
if (!type || !btf_type_is_func(type)) {
|
||||||
verbose(env, "invalid type id %d in func info",
|
verbose(env, "invalid type id %d in func info",
|
||||||
krecord[i].type_id);
|
krecord[i].type_id);
|
||||||
ret = -EINVAL;
|
ret = -EINVAL;
|
||||||
goto err_free;
|
goto err_free;
|
||||||
}
|
}
|
||||||
|
info_aux[i].linkage = BTF_INFO_VLEN(type->info);
|
||||||
prev_offset = krecord[i].insn_off;
|
prev_offset = krecord[i].insn_off;
|
||||||
urecord += urec_size;
|
urecord += urec_size;
|
||||||
}
|
}
|
||||||
@ -7756,35 +7786,13 @@ static bool reg_type_mismatch(enum bpf_reg_type src, enum bpf_reg_type prev)
|
|||||||
|
|
||||||
static int do_check(struct bpf_verifier_env *env)
|
static int do_check(struct bpf_verifier_env *env)
|
||||||
{
|
{
|
||||||
struct bpf_verifier_state *state;
|
struct bpf_verifier_state *state = env->cur_state;
|
||||||
struct bpf_insn *insns = env->prog->insnsi;
|
struct bpf_insn *insns = env->prog->insnsi;
|
||||||
struct bpf_reg_state *regs;
|
struct bpf_reg_state *regs;
|
||||||
int insn_cnt = env->prog->len;
|
int insn_cnt = env->prog->len;
|
||||||
bool do_print_state = false;
|
bool do_print_state = false;
|
||||||
int prev_insn_idx = -1;
|
int prev_insn_idx = -1;
|
||||||
|
|
||||||
env->prev_linfo = NULL;
|
|
||||||
|
|
||||||
state = kzalloc(sizeof(struct bpf_verifier_state), GFP_KERNEL);
|
|
||||||
if (!state)
|
|
||||||
return -ENOMEM;
|
|
||||||
state->curframe = 0;
|
|
||||||
state->speculative = false;
|
|
||||||
state->branches = 1;
|
|
||||||
state->frame[0] = kzalloc(sizeof(struct bpf_func_state), GFP_KERNEL);
|
|
||||||
if (!state->frame[0]) {
|
|
||||||
kfree(state);
|
|
||||||
return -ENOMEM;
|
|
||||||
}
|
|
||||||
env->cur_state = state;
|
|
||||||
init_func_state(env, state->frame[0],
|
|
||||||
BPF_MAIN_FUNC /* callsite */,
|
|
||||||
0 /* frameno */,
|
|
||||||
0 /* subprogno, zero == main subprog */);
|
|
||||||
|
|
||||||
if (btf_check_func_arg_match(env, 0))
|
|
||||||
return -EINVAL;
|
|
||||||
|
|
||||||
for (;;) {
|
for (;;) {
|
||||||
struct bpf_insn *insn;
|
struct bpf_insn *insn;
|
||||||
u8 class;
|
u8 class;
|
||||||
@ -7862,7 +7870,7 @@ static int do_check(struct bpf_verifier_env *env)
|
|||||||
}
|
}
|
||||||
|
|
||||||
regs = cur_regs(env);
|
regs = cur_regs(env);
|
||||||
env->insn_aux_data[env->insn_idx].seen = true;
|
env->insn_aux_data[env->insn_idx].seen = env->pass_cnt;
|
||||||
prev_insn_idx = env->insn_idx;
|
prev_insn_idx = env->insn_idx;
|
||||||
|
|
||||||
if (class == BPF_ALU || class == BPF_ALU64) {
|
if (class == BPF_ALU || class == BPF_ALU64) {
|
||||||
@ -8082,7 +8090,7 @@ process_bpf_exit:
|
|||||||
return err;
|
return err;
|
||||||
|
|
||||||
env->insn_idx++;
|
env->insn_idx++;
|
||||||
env->insn_aux_data[env->insn_idx].seen = true;
|
env->insn_aux_data[env->insn_idx].seen = env->pass_cnt;
|
||||||
} else {
|
} else {
|
||||||
verbose(env, "invalid BPF_LD mode\n");
|
verbose(env, "invalid BPF_LD mode\n");
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
@ -8095,7 +8103,6 @@ process_bpf_exit:
|
|||||||
env->insn_idx++;
|
env->insn_idx++;
|
||||||
}
|
}
|
||||||
|
|
||||||
env->prog->aux->stack_depth = env->subprog_info[0].stack_depth;
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -8372,7 +8379,7 @@ static int adjust_insn_aux_data(struct bpf_verifier_env *env,
|
|||||||
memcpy(new_data + off + cnt - 1, old_data + off,
|
memcpy(new_data + off + cnt - 1, old_data + off,
|
||||||
sizeof(struct bpf_insn_aux_data) * (prog_len - off - cnt + 1));
|
sizeof(struct bpf_insn_aux_data) * (prog_len - off - cnt + 1));
|
||||||
for (i = off; i < off + cnt - 1; i++) {
|
for (i = off; i < off + cnt - 1; i++) {
|
||||||
new_data[i].seen = true;
|
new_data[i].seen = env->pass_cnt;
|
||||||
new_data[i].zext_dst = insn_has_def32(env, insn + i);
|
new_data[i].zext_dst = insn_has_def32(env, insn + i);
|
||||||
}
|
}
|
||||||
env->insn_aux_data = new_data;
|
env->insn_aux_data = new_data;
|
||||||
@ -9484,6 +9491,7 @@ static void free_states(struct bpf_verifier_env *env)
|
|||||||
kfree(sl);
|
kfree(sl);
|
||||||
sl = sln;
|
sl = sln;
|
||||||
}
|
}
|
||||||
|
env->free_list = NULL;
|
||||||
|
|
||||||
if (!env->explored_states)
|
if (!env->explored_states)
|
||||||
return;
|
return;
|
||||||
@ -9497,11 +9505,159 @@ static void free_states(struct bpf_verifier_env *env)
|
|||||||
kfree(sl);
|
kfree(sl);
|
||||||
sl = sln;
|
sl = sln;
|
||||||
}
|
}
|
||||||
|
env->explored_states[i] = NULL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* The verifier is using insn_aux_data[] to store temporary data during
|
||||||
|
* verification and to store information for passes that run after the
|
||||||
|
* verification like dead code sanitization. do_check_common() for subprogram N
|
||||||
|
* may analyze many other subprograms. sanitize_insn_aux_data() clears all
|
||||||
|
* temporary data after do_check_common() finds that subprogram N cannot be
|
||||||
|
* verified independently. pass_cnt counts the number of times
|
||||||
|
* do_check_common() was run and insn->aux->seen tells the pass number
|
||||||
|
* insn_aux_data was touched. These variables are compared to clear temporary
|
||||||
|
* data from failed pass. For testing and experiments do_check_common() can be
|
||||||
|
* run multiple times even when prior attempt to verify is unsuccessful.
|
||||||
|
*/
|
||||||
|
static void sanitize_insn_aux_data(struct bpf_verifier_env *env)
|
||||||
|
{
|
||||||
|
struct bpf_insn *insn = env->prog->insnsi;
|
||||||
|
struct bpf_insn_aux_data *aux;
|
||||||
|
int i, class;
|
||||||
|
|
||||||
|
for (i = 0; i < env->prog->len; i++) {
|
||||||
|
class = BPF_CLASS(insn[i].code);
|
||||||
|
if (class != BPF_LDX && class != BPF_STX)
|
||||||
|
continue;
|
||||||
|
aux = &env->insn_aux_data[i];
|
||||||
|
if (aux->seen != env->pass_cnt)
|
||||||
|
continue;
|
||||||
|
memset(aux, 0, offsetof(typeof(*aux), orig_idx));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static int do_check_common(struct bpf_verifier_env *env, int subprog)
|
||||||
|
{
|
||||||
|
struct bpf_verifier_state *state;
|
||||||
|
struct bpf_reg_state *regs;
|
||||||
|
int ret, i;
|
||||||
|
|
||||||
|
env->prev_linfo = NULL;
|
||||||
|
env->pass_cnt++;
|
||||||
|
|
||||||
|
state = kzalloc(sizeof(struct bpf_verifier_state), GFP_KERNEL);
|
||||||
|
if (!state)
|
||||||
|
return -ENOMEM;
|
||||||
|
state->curframe = 0;
|
||||||
|
state->speculative = false;
|
||||||
|
state->branches = 1;
|
||||||
|
state->frame[0] = kzalloc(sizeof(struct bpf_func_state), GFP_KERNEL);
|
||||||
|
if (!state->frame[0]) {
|
||||||
|
kfree(state);
|
||||||
|
return -ENOMEM;
|
||||||
|
}
|
||||||
|
env->cur_state = state;
|
||||||
|
init_func_state(env, state->frame[0],
|
||||||
|
BPF_MAIN_FUNC /* callsite */,
|
||||||
|
0 /* frameno */,
|
||||||
|
subprog);
|
||||||
|
|
||||||
|
regs = state->frame[state->curframe]->regs;
|
||||||
|
if (subprog) {
|
||||||
|
ret = btf_prepare_func_args(env, subprog, regs);
|
||||||
|
if (ret)
|
||||||
|
goto out;
|
||||||
|
for (i = BPF_REG_1; i <= BPF_REG_5; i++) {
|
||||||
|
if (regs[i].type == PTR_TO_CTX)
|
||||||
|
mark_reg_known_zero(env, regs, i);
|
||||||
|
else if (regs[i].type == SCALAR_VALUE)
|
||||||
|
mark_reg_unknown(env, regs, i);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
/* 1st arg to a function */
|
||||||
|
regs[BPF_REG_1].type = PTR_TO_CTX;
|
||||||
|
mark_reg_known_zero(env, regs, BPF_REG_1);
|
||||||
|
ret = btf_check_func_arg_match(env, subprog, regs);
|
||||||
|
if (ret == -EFAULT)
|
||||||
|
/* unlikely verifier bug. abort.
|
||||||
|
* ret == 0 and ret < 0 are sadly acceptable for
|
||||||
|
* main() function due to backward compatibility.
|
||||||
|
* Like socket filter program may be written as:
|
||||||
|
* int bpf_prog(struct pt_regs *ctx)
|
||||||
|
* and never dereference that ctx in the program.
|
||||||
|
* 'struct pt_regs' is a type mismatch for socket
|
||||||
|
* filter that should be using 'struct __sk_buff'.
|
||||||
|
*/
|
||||||
|
goto out;
|
||||||
}
|
}
|
||||||
|
|
||||||
kvfree(env->explored_states);
|
ret = do_check(env);
|
||||||
|
out:
|
||||||
|
free_verifier_state(env->cur_state, true);
|
||||||
|
env->cur_state = NULL;
|
||||||
|
while (!pop_stack(env, NULL, NULL));
|
||||||
|
free_states(env);
|
||||||
|
if (ret)
|
||||||
|
/* clean aux data in case subprog was rejected */
|
||||||
|
sanitize_insn_aux_data(env);
|
||||||
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Verify all global functions in a BPF program one by one based on their BTF.
|
||||||
|
* All global functions must pass verification. Otherwise the whole program is rejected.
|
||||||
|
* Consider:
|
||||||
|
* int bar(int);
|
||||||
|
* int foo(int f)
|
||||||
|
* {
|
||||||
|
* return bar(f);
|
||||||
|
* }
|
||||||
|
* int bar(int b)
|
||||||
|
* {
|
||||||
|
* ...
|
||||||
|
* }
|
||||||
|
* foo() will be verified first for R1=any_scalar_value. During verification it
|
||||||
|
* will be assumed that bar() already verified successfully and call to bar()
|
||||||
|
* from foo() will be checked for type match only. Later bar() will be verified
|
||||||
|
* independently to check that it's safe for R1=any_scalar_value.
|
||||||
|
*/
|
||||||
|
static int do_check_subprogs(struct bpf_verifier_env *env)
|
||||||
|
{
|
||||||
|
struct bpf_prog_aux *aux = env->prog->aux;
|
||||||
|
int i, ret;
|
||||||
|
|
||||||
|
if (!aux->func_info)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
for (i = 1; i < env->subprog_cnt; i++) {
|
||||||
|
if (aux->func_info_aux[i].linkage != BTF_FUNC_GLOBAL)
|
||||||
|
continue;
|
||||||
|
env->insn_idx = env->subprog_info[i].start;
|
||||||
|
WARN_ON_ONCE(env->insn_idx == 0);
|
||||||
|
ret = do_check_common(env, i);
|
||||||
|
if (ret) {
|
||||||
|
return ret;
|
||||||
|
} else if (env->log.level & BPF_LOG_LEVEL) {
|
||||||
|
verbose(env,
|
||||||
|
"Func#%d is safe for any args that match its prototype\n",
|
||||||
|
i);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int do_check_main(struct bpf_verifier_env *env)
|
||||||
|
{
|
||||||
|
int ret;
|
||||||
|
|
||||||
|
env->insn_idx = 0;
|
||||||
|
ret = do_check_common(env, 0);
|
||||||
|
if (!ret)
|
||||||
|
env->prog->aux->stack_depth = env->subprog_info[0].stack_depth;
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
static void print_verification_stats(struct bpf_verifier_env *env)
|
static void print_verification_stats(struct bpf_verifier_env *env)
|
||||||
{
|
{
|
||||||
int i;
|
int i;
|
||||||
@ -9849,18 +10005,14 @@ int bpf_check(struct bpf_prog **prog, union bpf_attr *attr,
|
|||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
goto skip_full_check;
|
goto skip_full_check;
|
||||||
|
|
||||||
ret = do_check(env);
|
ret = do_check_subprogs(env);
|
||||||
if (env->cur_state) {
|
ret = ret ?: do_check_main(env);
|
||||||
free_verifier_state(env->cur_state, true);
|
|
||||||
env->cur_state = NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (ret == 0 && bpf_prog_is_dev_bound(env->prog->aux))
|
if (ret == 0 && bpf_prog_is_dev_bound(env->prog->aux))
|
||||||
ret = bpf_prog_offload_finalize(env);
|
ret = bpf_prog_offload_finalize(env);
|
||||||
|
|
||||||
skip_full_check:
|
skip_full_check:
|
||||||
while (!pop_stack(env, NULL, NULL));
|
kvfree(env->explored_states);
|
||||||
free_states(env);
|
|
||||||
|
|
||||||
if (ret == 0)
|
if (ret == 0)
|
||||||
ret = check_max_stack_depth(env);
|
ret = check_max_stack_depth(env);
|
||||||
|
Loading…
Reference in New Issue
Block a user