Commit 47a71c1f authored by Andrii Nakryiko's avatar Andrii Nakryiko Committed by Daniel Borkmann

bpf: Add log_true_size output field to return necessary log buffer size

Add output-only log_true_size and btf_log_true_size field to
BPF_PROG_LOAD and BPF_BTF_LOAD commands, respectively. It will return
the size of log buffer necessary to fit in all the log contents at
specified log_level. This is very useful for BPF loader libraries like
libbpf to be able to size log buffer correctly, but could be used by
users directly, if necessary, as well.

This patch plumbs all this through the code, taking into account actual
bpf_attr size provided by user to determine if these new fields are
expected by users. And if they are, set them from kernel on return.

We refactory btf_parse() function to accommodate this, moving attr and
uattr handling inside it. The rest is very straightforward code, which
is split from the logging accounting changes in the previous patch to
make it simpler to review logic vs UAPI changes.
Signed-off-by: default avatarAndrii Nakryiko <andrii@kernel.org>
Signed-off-by: default avatarDaniel Borkmann <daniel@iogearbox.net>
Acked-by: default avatarLorenz Bauer <lmb@isovalent.com>
Link: https://lore.kernel.org/bpf/20230406234205.323208-13-andrii@kernel.org
parent fa1c7d5c
...@@ -2175,7 +2175,7 @@ int bpf_check_uarg_tail_zero(bpfptr_t uaddr, size_t expected_size, ...@@ -2175,7 +2175,7 @@ int bpf_check_uarg_tail_zero(bpfptr_t uaddr, size_t expected_size,
size_t actual_size); size_t actual_size);
/* verify correctness of eBPF program */ /* verify correctness of eBPF program */
int bpf_check(struct bpf_prog **fp, union bpf_attr *attr, bpfptr_t uattr); int bpf_check(struct bpf_prog **fp, union bpf_attr *attr, bpfptr_t uattr, u32 uattr_size);
#ifndef CONFIG_BPF_JIT_ALWAYS_ON #ifndef CONFIG_BPF_JIT_ALWAYS_ON
void bpf_patch_call_args(struct bpf_insn *insn, u32 stack_depth); void bpf_patch_call_args(struct bpf_insn *insn, u32 stack_depth);
......
...@@ -125,7 +125,7 @@ extern const struct file_operations btf_fops; ...@@ -125,7 +125,7 @@ extern const struct file_operations btf_fops;
void btf_get(struct btf *btf); void btf_get(struct btf *btf);
void btf_put(struct btf *btf); void btf_put(struct btf *btf);
int btf_new_fd(const union bpf_attr *attr, bpfptr_t uattr); int btf_new_fd(const union bpf_attr *attr, bpfptr_t uattr, u32 uattr_sz);
struct btf *btf_get_by_fd(int fd); struct btf *btf_get_by_fd(int fd);
int btf_get_info_by_fd(const struct btf *btf, int btf_get_info_by_fd(const struct btf *btf,
const union bpf_attr *attr, const union bpf_attr *attr,
......
...@@ -1407,6 +1407,11 @@ union bpf_attr { ...@@ -1407,6 +1407,11 @@ union bpf_attr {
__aligned_u64 fd_array; /* array of FDs */ __aligned_u64 fd_array; /* array of FDs */
__aligned_u64 core_relos; __aligned_u64 core_relos;
__u32 core_relo_rec_size; /* sizeof(struct bpf_core_relo) */ __u32 core_relo_rec_size; /* sizeof(struct bpf_core_relo) */
/* output: actual total log contents size (including termintaing zero).
* It could be both larger than original log_size (if log was
* truncated), or smaller (if log buffer wasn't filled completely).
*/
__u32 log_true_size;
}; };
struct { /* anonymous struct used by BPF_OBJ_* commands */ struct { /* anonymous struct used by BPF_OBJ_* commands */
...@@ -1492,6 +1497,11 @@ union bpf_attr { ...@@ -1492,6 +1497,11 @@ union bpf_attr {
__u32 btf_size; __u32 btf_size;
__u32 btf_log_size; __u32 btf_log_size;
__u32 btf_log_level; __u32 btf_log_level;
/* output: actual total log contents size (including termintaing zero).
* It could be both larger than original log_size (if log was
* truncated), or smaller (if log buffer wasn't filled completely).
*/
__u32 btf_log_true_size;
}; };
struct { struct {
......
...@@ -5504,9 +5504,10 @@ static int btf_check_type_tags(struct btf_verifier_env *env, ...@@ -5504,9 +5504,10 @@ static int btf_check_type_tags(struct btf_verifier_env *env,
return 0; return 0;
} }
static struct btf *btf_parse(bpfptr_t btf_data, u32 btf_data_size, static struct btf *btf_parse(const union bpf_attr *attr, bpfptr_t uattr, u32 uattr_size)
u32 log_level, char __user *log_ubuf, u32 log_size)
{ {
bpfptr_t btf_data = make_bpfptr(attr->btf, uattr.is_kernel);
char __user *log_ubuf = u64_to_user_ptr(attr->btf_log_buf);
struct btf_struct_metas *struct_meta_tab; struct btf_struct_metas *struct_meta_tab;
struct btf_verifier_env *env = NULL; struct btf_verifier_env *env = NULL;
struct bpf_verifier_log *log; struct bpf_verifier_log *log;
...@@ -5514,7 +5515,7 @@ static struct btf *btf_parse(bpfptr_t btf_data, u32 btf_data_size, ...@@ -5514,7 +5515,7 @@ static struct btf *btf_parse(bpfptr_t btf_data, u32 btf_data_size,
u8 *data; u8 *data;
int err; int err;
if (btf_data_size > BTF_MAX_SIZE) if (attr->btf_size > BTF_MAX_SIZE)
return ERR_PTR(-E2BIG); return ERR_PTR(-E2BIG);
env = kzalloc(sizeof(*env), GFP_KERNEL | __GFP_NOWARN); env = kzalloc(sizeof(*env), GFP_KERNEL | __GFP_NOWARN);
...@@ -5522,13 +5523,13 @@ static struct btf *btf_parse(bpfptr_t btf_data, u32 btf_data_size, ...@@ -5522,13 +5523,13 @@ static struct btf *btf_parse(bpfptr_t btf_data, u32 btf_data_size,
return ERR_PTR(-ENOMEM); return ERR_PTR(-ENOMEM);
log = &env->log; log = &env->log;
if (log_level || log_ubuf || log_size) { if (attr->btf_log_level || log_ubuf || attr->btf_log_size) {
/* user requested verbose verifier output /* user requested verbose verifier output
* and supplied buffer to store the verification trace * and supplied buffer to store the verification trace
*/ */
log->level = log_level; log->level = attr->btf_log_level;
log->ubuf = log_ubuf; log->ubuf = log_ubuf;
log->len_total = log_size; log->len_total = attr->btf_log_size;
/* log attributes have to be sane */ /* log attributes have to be sane */
if (!bpf_verifier_log_attr_valid(log)) { if (!bpf_verifier_log_attr_valid(log)) {
...@@ -5544,16 +5545,16 @@ static struct btf *btf_parse(bpfptr_t btf_data, u32 btf_data_size, ...@@ -5544,16 +5545,16 @@ static struct btf *btf_parse(bpfptr_t btf_data, u32 btf_data_size,
} }
env->btf = btf; env->btf = btf;
data = kvmalloc(btf_data_size, GFP_KERNEL | __GFP_NOWARN); data = kvmalloc(attr->btf_size, GFP_KERNEL | __GFP_NOWARN);
if (!data) { if (!data) {
err = -ENOMEM; err = -ENOMEM;
goto errout; goto errout;
} }
btf->data = data; btf->data = data;
btf->data_size = btf_data_size; btf->data_size = attr->btf_size;
if (copy_from_bpfptr(data, btf_data, btf_data_size)) { if (copy_from_bpfptr(data, btf_data, attr->btf_size)) {
err = -EFAULT; err = -EFAULT;
goto errout; goto errout;
} }
...@@ -5594,6 +5595,12 @@ static struct btf *btf_parse(bpfptr_t btf_data, u32 btf_data_size, ...@@ -5594,6 +5595,12 @@ static struct btf *btf_parse(bpfptr_t btf_data, u32 btf_data_size,
} }
bpf_vlog_finalize(log); bpf_vlog_finalize(log);
if (uattr_size >= offsetofend(union bpf_attr, btf_log_true_size) &&
copy_to_bpfptr_offset(uattr, offsetof(union bpf_attr, btf_log_true_size),
&log->len_max, sizeof(log->len_max))) {
err = -EFAULT;
goto errout_meta;
}
if (bpf_vlog_truncated(log)) { if (bpf_vlog_truncated(log)) {
err = -ENOSPC; err = -ENOSPC;
goto errout_meta; goto errout_meta;
...@@ -7218,15 +7225,12 @@ static int __btf_new_fd(struct btf *btf) ...@@ -7218,15 +7225,12 @@ static int __btf_new_fd(struct btf *btf)
return anon_inode_getfd("btf", &btf_fops, btf, O_RDONLY | O_CLOEXEC); return anon_inode_getfd("btf", &btf_fops, btf, O_RDONLY | O_CLOEXEC);
} }
int btf_new_fd(const union bpf_attr *attr, bpfptr_t uattr) int btf_new_fd(const union bpf_attr *attr, bpfptr_t uattr, u32 uattr_size)
{ {
struct btf *btf; struct btf *btf;
int ret; int ret;
btf = btf_parse(make_bpfptr(attr->btf, uattr.is_kernel), btf = btf_parse(attr, uattr, uattr_size);
attr->btf_size, attr->btf_log_level,
u64_to_user_ptr(attr->btf_log_buf),
attr->btf_log_size);
if (IS_ERR(btf)) if (IS_ERR(btf))
return PTR_ERR(btf); return PTR_ERR(btf);
......
...@@ -2501,9 +2501,9 @@ static bool is_perfmon_prog_type(enum bpf_prog_type prog_type) ...@@ -2501,9 +2501,9 @@ static bool is_perfmon_prog_type(enum bpf_prog_type prog_type)
} }
/* last field in 'union bpf_attr' used by this command */ /* last field in 'union bpf_attr' used by this command */
#define BPF_PROG_LOAD_LAST_FIELD core_relo_rec_size #define BPF_PROG_LOAD_LAST_FIELD log_true_size
static int bpf_prog_load(union bpf_attr *attr, bpfptr_t uattr) static int bpf_prog_load(union bpf_attr *attr, bpfptr_t uattr, u32 uattr_size)
{ {
enum bpf_prog_type type = attr->prog_type; enum bpf_prog_type type = attr->prog_type;
struct bpf_prog *prog, *dst_prog = NULL; struct bpf_prog *prog, *dst_prog = NULL;
...@@ -2653,7 +2653,7 @@ static int bpf_prog_load(union bpf_attr *attr, bpfptr_t uattr) ...@@ -2653,7 +2653,7 @@ static int bpf_prog_load(union bpf_attr *attr, bpfptr_t uattr)
goto free_prog_sec; goto free_prog_sec;
/* run eBPF verifier */ /* run eBPF verifier */
err = bpf_check(&prog, attr, uattr); err = bpf_check(&prog, attr, uattr, uattr_size);
if (err < 0) if (err < 0)
goto free_used_maps; goto free_used_maps;
...@@ -4371,9 +4371,9 @@ static int bpf_obj_get_info_by_fd(const union bpf_attr *attr, ...@@ -4371,9 +4371,9 @@ static int bpf_obj_get_info_by_fd(const union bpf_attr *attr,
return err; return err;
} }
#define BPF_BTF_LOAD_LAST_FIELD btf_log_level #define BPF_BTF_LOAD_LAST_FIELD btf_log_true_size
static int bpf_btf_load(const union bpf_attr *attr, bpfptr_t uattr) static int bpf_btf_load(const union bpf_attr *attr, bpfptr_t uattr, __u32 uattr_size)
{ {
if (CHECK_ATTR(BPF_BTF_LOAD)) if (CHECK_ATTR(BPF_BTF_LOAD))
return -EINVAL; return -EINVAL;
...@@ -4381,7 +4381,7 @@ static int bpf_btf_load(const union bpf_attr *attr, bpfptr_t uattr) ...@@ -4381,7 +4381,7 @@ static int bpf_btf_load(const union bpf_attr *attr, bpfptr_t uattr)
if (!bpf_capable()) if (!bpf_capable())
return -EPERM; return -EPERM;
return btf_new_fd(attr, uattr); return btf_new_fd(attr, uattr, uattr_size);
} }
#define BPF_BTF_GET_FD_BY_ID_LAST_FIELD btf_id #define BPF_BTF_GET_FD_BY_ID_LAST_FIELD btf_id
...@@ -5059,7 +5059,7 @@ static int __sys_bpf(int cmd, bpfptr_t uattr, unsigned int size) ...@@ -5059,7 +5059,7 @@ static int __sys_bpf(int cmd, bpfptr_t uattr, unsigned int size)
err = map_freeze(&attr); err = map_freeze(&attr);
break; break;
case BPF_PROG_LOAD: case BPF_PROG_LOAD:
err = bpf_prog_load(&attr, uattr); err = bpf_prog_load(&attr, uattr, size);
break; break;
case BPF_OBJ_PIN: case BPF_OBJ_PIN:
err = bpf_obj_pin(&attr); err = bpf_obj_pin(&attr);
...@@ -5104,7 +5104,7 @@ static int __sys_bpf(int cmd, bpfptr_t uattr, unsigned int size) ...@@ -5104,7 +5104,7 @@ static int __sys_bpf(int cmd, bpfptr_t uattr, unsigned int size)
err = bpf_raw_tracepoint_open(&attr); err = bpf_raw_tracepoint_open(&attr);
break; break;
case BPF_BTF_LOAD: case BPF_BTF_LOAD:
err = bpf_btf_load(&attr, uattr); err = bpf_btf_load(&attr, uattr, size);
break; break;
case BPF_BTF_GET_FD_BY_ID: case BPF_BTF_GET_FD_BY_ID:
err = bpf_btf_get_fd_by_id(&attr); err = bpf_btf_get_fd_by_id(&attr);
......
...@@ -18694,7 +18694,7 @@ struct btf *bpf_get_btf_vmlinux(void) ...@@ -18694,7 +18694,7 @@ struct btf *bpf_get_btf_vmlinux(void)
return btf_vmlinux; return btf_vmlinux;
} }
int bpf_check(struct bpf_prog **prog, union bpf_attr *attr, bpfptr_t uattr) int bpf_check(struct bpf_prog **prog, union bpf_attr *attr, bpfptr_t uattr, __u32 uattr_size)
{ {
u64 start_time = ktime_get_ns(); u64 start_time = ktime_get_ns();
struct bpf_verifier_env *env; struct bpf_verifier_env *env;
...@@ -18861,6 +18861,12 @@ int bpf_check(struct bpf_prog **prog, union bpf_attr *attr, bpfptr_t uattr) ...@@ -18861,6 +18861,12 @@ int bpf_check(struct bpf_prog **prog, union bpf_attr *attr, bpfptr_t uattr)
env->prog->aux->verified_insns = env->insn_processed; env->prog->aux->verified_insns = env->insn_processed;
bpf_vlog_finalize(log); bpf_vlog_finalize(log);
if (uattr_size >= offsetofend(union bpf_attr, log_true_size) &&
copy_to_bpfptr_offset(uattr, offsetof(union bpf_attr, log_true_size),
&log->len_max, sizeof(log->len_max))) {
ret = -EFAULT;
goto err_release_maps;
}
if (bpf_vlog_truncated(log)) if (bpf_vlog_truncated(log))
ret = -ENOSPC; ret = -ENOSPC;
if (log->level && log->level != BPF_LOG_KERNEL && !log->ubuf) if (log->level && log->level != BPF_LOG_KERNEL && !log->ubuf)
......
...@@ -1407,6 +1407,11 @@ union bpf_attr { ...@@ -1407,6 +1407,11 @@ union bpf_attr {
__aligned_u64 fd_array; /* array of FDs */ __aligned_u64 fd_array; /* array of FDs */
__aligned_u64 core_relos; __aligned_u64 core_relos;
__u32 core_relo_rec_size; /* sizeof(struct bpf_core_relo) */ __u32 core_relo_rec_size; /* sizeof(struct bpf_core_relo) */
/* output: actual total log contents size (including termintaing zero).
* It could be both larger than original log_size (if log was
* truncated), or smaller (if log buffer wasn't filled completely).
*/
__u32 log_true_size;
}; };
struct { /* anonymous struct used by BPF_OBJ_* commands */ struct { /* anonymous struct used by BPF_OBJ_* commands */
...@@ -1492,6 +1497,11 @@ union bpf_attr { ...@@ -1492,6 +1497,11 @@ union bpf_attr {
__u32 btf_size; __u32 btf_size;
__u32 btf_log_size; __u32 btf_log_size;
__u32 btf_log_level; __u32 btf_log_level;
/* output: actual total log contents size (including termintaing zero).
* It could be both larger than original log_size (if log was
* truncated), or smaller (if log buffer wasn't filled completely).
*/
__u32 btf_log_true_size;
}; };
struct { struct {
...@@ -1513,7 +1523,7 @@ union bpf_attr { ...@@ -1513,7 +1523,7 @@ union bpf_attr {
struct { /* struct used by BPF_LINK_CREATE command */ struct { /* struct used by BPF_LINK_CREATE command */
union { union {
__u32 prog_fd; /* eBPF program to attach */ __u32 prog_fd; /* eBPF program to attach */
__u32 map_fd; /* eBPF struct_ops to attach */ __u32 map_fd; /* struct_ops to attach */
}; };
union { union {
__u32 target_fd; /* object to attach to */ __u32 target_fd; /* object to attach to */
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment