bpf: Add log_true_size output field to return necessary log buffer size
authorAndrii Nakryiko <andrii@kernel.org>
Thu, 6 Apr 2023 23:41:58 +0000 (16:41 -0700)
committerDaniel Borkmann <daniel@iogearbox.net>
Tue, 11 Apr 2023 16:05:43 +0000 (18:05 +0200)
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: Andrii Nakryiko <andrii@kernel.org>
Signed-off-by: Daniel Borkmann <daniel@iogearbox.net>
Acked-by: Lorenz Bauer <lmb@isovalent.com>
Link: https://lore.kernel.org/bpf/20230406234205.323208-13-andrii@kernel.org
include/linux/bpf.h
include/linux/btf.h
include/uapi/linux/bpf.h
kernel/bpf/btf.c
kernel/bpf/syscall.c
kernel/bpf/verifier.c
tools/include/uapi/linux/bpf.h

index 002a811..2c6095b 100644 (file)
@@ -2175,7 +2175,7 @@ int bpf_check_uarg_tail_zero(bpfptr_t uaddr, size_t expected_size,
                             size_t actual_size);
 
 /* 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
 void bpf_patch_call_args(struct bpf_insn *insn, u32 stack_depth);
index d53b10c..4952501 100644 (file)
@@ -125,7 +125,7 @@ extern const struct file_operations btf_fops;
 
 void btf_get(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);
 int btf_get_info_by_fd(const struct btf *btf,
                       const union bpf_attr *attr,
index e3d3b51..3823100 100644 (file)
@@ -1407,6 +1407,11 @@ union bpf_attr {
                __aligned_u64   fd_array;       /* array of FDs */
                __aligned_u64   core_relos;
                __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 */
@@ -1492,6 +1497,11 @@ union bpf_attr {
                __u32           btf_size;
                __u32           btf_log_size;
                __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 {
index 5aa540e..0748cf4 100644 (file)
@@ -5504,9 +5504,10 @@ static int btf_check_type_tags(struct btf_verifier_env *env,
        return 0;
 }
 
-static struct btf *btf_parse(bpfptr_t btf_data, u32 btf_data_size,
-                            u32 log_level, char __user *log_ubuf, u32 log_size)
+static struct btf *btf_parse(const union bpf_attr *attr, bpfptr_t uattr, u32 uattr_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_verifier_env *env = NULL;
        struct bpf_verifier_log *log;
@@ -5514,7 +5515,7 @@ static struct btf *btf_parse(bpfptr_t btf_data, u32 btf_data_size,
        u8 *data;
        int err;
 
-       if (btf_data_size > BTF_MAX_SIZE)
+       if (attr->btf_size > BTF_MAX_SIZE)
                return ERR_PTR(-E2BIG);
 
        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,
                return ERR_PTR(-ENOMEM);
 
        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
                 * and supplied buffer to store the verification trace
                 */
-               log->level = log_level;
+               log->level = attr->btf_log_level;
                log->ubuf = log_ubuf;
-               log->len_total = log_size;
+               log->len_total = attr->btf_log_size;
 
                /* log attributes have to be sane */
                if (!bpf_verifier_log_attr_valid(log)) {
@@ -5544,16 +5545,16 @@ static struct btf *btf_parse(bpfptr_t btf_data, u32 btf_data_size,
        }
        env->btf = btf;
 
-       data = kvmalloc(btf_data_size, GFP_KERNEL | __GFP_NOWARN);
+       data = kvmalloc(attr->btf_size, GFP_KERNEL | __GFP_NOWARN);
        if (!data) {
                err = -ENOMEM;
                goto errout;
        }
 
        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;
                goto errout;
        }
@@ -5594,6 +5595,12 @@ static struct btf *btf_parse(bpfptr_t btf_data, u32 btf_data_size,
        }
 
        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)) {
                err = -ENOSPC;
                goto errout_meta;
@@ -7218,15 +7225,12 @@ static int __btf_new_fd(struct btf *btf)
        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;
        int ret;
 
-       btf = btf_parse(make_bpfptr(attr->btf, uattr.is_kernel),
-                       attr->btf_size, attr->btf_log_level,
-                       u64_to_user_ptr(attr->btf_log_buf),
-                       attr->btf_log_size);
+       btf = btf_parse(attr, uattr, uattr_size);
        if (IS_ERR(btf))
                return PTR_ERR(btf);
 
index e18ac7f..6d57550 100644 (file)
@@ -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 */
-#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;
        struct bpf_prog *prog, *dst_prog = NULL;
@@ -2653,7 +2653,7 @@ static int bpf_prog_load(union bpf_attr *attr, bpfptr_t uattr)
                goto free_prog_sec;
 
        /* run eBPF verifier */
-       err = bpf_check(&prog, attr, uattr);
+       err = bpf_check(&prog, attr, uattr, uattr_size);
        if (err < 0)
                goto free_used_maps;
 
@@ -4371,9 +4371,9 @@ static int bpf_obj_get_info_by_fd(const union bpf_attr *attr,
        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))
                return -EINVAL;
@@ -4381,7 +4381,7 @@ static int bpf_btf_load(const union bpf_attr *attr, bpfptr_t uattr)
        if (!bpf_capable())
                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
@@ -5059,7 +5059,7 @@ static int __sys_bpf(int cmd, bpfptr_t uattr, unsigned int size)
                err = map_freeze(&attr);
                break;
        case BPF_PROG_LOAD:
-               err = bpf_prog_load(&attr, uattr);
+               err = bpf_prog_load(&attr, uattr, size);
                break;
        case BPF_OBJ_PIN:
                err = bpf_obj_pin(&attr);
@@ -5104,7 +5104,7 @@ static int __sys_bpf(int cmd, bpfptr_t uattr, unsigned int size)
                err = bpf_raw_tracepoint_open(&attr);
                break;
        case BPF_BTF_LOAD:
-               err = bpf_btf_load(&attr, uattr);
+               err = bpf_btf_load(&attr, uattr, size);
                break;
        case BPF_BTF_GET_FD_BY_ID:
                err = bpf_btf_get_fd_by_id(&attr);
index a98cbc0..308e7ab 100644 (file)
@@ -18694,7 +18694,7 @@ struct btf *bpf_get_btf_vmlinux(void)
        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();
        struct bpf_verifier_env *env;
@@ -18861,6 +18861,12 @@ skip_full_check:
        env->prog->aux->verified_insns = env->insn_processed;
 
        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))
                ret = -ENOSPC;
        if (log->level && log->level != BPF_LOG_KERNEL && !log->ubuf)
index d6c5a02..3823100 100644 (file)
@@ -1407,6 +1407,11 @@ union bpf_attr {
                __aligned_u64   fd_array;       /* array of FDs */
                __aligned_u64   core_relos;
                __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 */
@@ -1492,6 +1497,11 @@ union bpf_attr {
                __u32           btf_size;
                __u32           btf_log_size;
                __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 {
@@ -1513,7 +1523,7 @@ union bpf_attr {
        struct { /* struct used by BPF_LINK_CREATE command */
                union {
                        __u32           prog_fd;        /* eBPF program to attach */
-                       __u32           map_fd;         /* eBPF struct_ops to attach */
+                       __u32           map_fd;         /* struct_ops to attach */
                };
                union {
                        __u32           target_fd;      /* object to attach to */