Commit d10ef2b8 authored by Andrii Nakryiko's avatar Andrii Nakryiko Committed by Alexei Starovoitov

libbpf: Unify low-level BPF_PROG_LOAD APIs into bpf_prog_load()

Add a new unified OPTS-based low-level API for program loading,
bpf_prog_load() ([0]).  bpf_prog_load() accepts few "mandatory"
parameters as input arguments (program type, name, license,
instructions) and all the other optional (as in not required to specify
for all types of BPF programs) fields into struct bpf_prog_load_opts.

This makes all the other non-extensible APIs variant for BPF_PROG_LOAD
obsolete and they are slated for deprecation in libbpf v0.7:
  - bpf_load_program();
  - bpf_load_program_xattr();
  - bpf_verify_program().

Implementation-wise, internal helper libbpf__bpf_prog_load is refactored
to become a public bpf_prog_load() API. struct bpf_prog_load_params used
internally is replaced by public struct bpf_prog_load_opts.

Unfortunately, while conceptually all this is pretty straightforward,
the biggest complication comes from the already existing bpf_prog_load()
*high-level* API, which has nothing to do with BPF_PROG_LOAD command.

We try really hard to have a new API named bpf_prog_load(), though,
because it maps naturally to BPF_PROG_LOAD command.

For that, we rename old bpf_prog_load() into bpf_prog_load_deprecated()
and mark it as COMPAT_VERSION() for shared library users compiled
against old version of libbpf. Statically linked users and shared lib
users compiled against new version of libbpf headers will get "rerouted"
to bpf_prog_deprecated() through a macro helper that decides whether to
use new or old bpf_prog_load() based on number of input arguments (see
___libbpf_overload in libbpf_common.h).

To test that existing
bpf_prog_load()-using code compiles and works as expected, I've compiled
and ran selftests as is. I had to remove (locally) selftest/bpf/Makefile
-Dbpf_prog_load=bpf_prog_test_load hack because it was conflicting with
the macro-based overload approach. I don't expect anyone else to do
something like this in practice, though. This is testing-specific way to
replace bpf_prog_load() calls with special testing variant of it, which
adds extra prog_flags value. After testing I kept this selftests hack,
but ensured that we use a new bpf_prog_load_deprecated name for this.

This patch also marks bpf_prog_load() and bpf_prog_load_xattr() as deprecated.
bpf_object interface has to be used for working with struct bpf_program.
Libbpf doesn't support loading just a bpf_program.

The silver lining is that when we get to libbpf 1.0 all these
complication will be gone and we'll have one clean bpf_prog_load()
low-level API with no backwards compatibility hackery surrounding it.

  [0] Closes: https://github.com/libbpf/libbpf/issues/284Signed-off-by: default avatarAndrii Nakryiko <andrii@kernel.org>
Signed-off-by: default avatarAlexei Starovoitov <ast@kernel.org>
Link: https://lore.kernel.org/bpf/20211103220845.2676888-4-andrii@kernel.org
parent 45493cba
...@@ -28,6 +28,7 @@ ...@@ -28,6 +28,7 @@
#include <asm/unistd.h> #include <asm/unistd.h>
#include <errno.h> #include <errno.h>
#include <linux/bpf.h> #include <linux/bpf.h>
#include <limits.h>
#include "bpf.h" #include "bpf.h"
#include "libbpf.h" #include "libbpf.h"
#include "libbpf_internal.h" #include "libbpf_internal.h"
...@@ -254,58 +255,91 @@ alloc_zero_tailing_info(const void *orecord, __u32 cnt, ...@@ -254,58 +255,91 @@ alloc_zero_tailing_info(const void *orecord, __u32 cnt,
return info; return info;
} }
int libbpf__bpf_prog_load(const struct bpf_prog_load_params *load_attr) DEFAULT_VERSION(bpf_prog_load_v0_6_0, bpf_prog_load, LIBBPF_0.6.0)
int bpf_prog_load_v0_6_0(enum bpf_prog_type prog_type,
const char *prog_name, const char *license,
const struct bpf_insn *insns, size_t insn_cnt,
const struct bpf_prog_load_opts *opts)
{ {
void *finfo = NULL, *linfo = NULL; void *finfo = NULL, *linfo = NULL;
const char *func_info, *line_info;
__u32 log_size, log_level, attach_prog_fd, attach_btf_obj_fd;
__u32 func_info_rec_size, line_info_rec_size;
int fd, attempts;
union bpf_attr attr; union bpf_attr attr;
int fd; char *log_buf;
if (!load_attr->log_buf != !load_attr->log_buf_sz) if (!OPTS_VALID(opts, bpf_prog_load_opts))
return libbpf_err(-EINVAL); return libbpf_err(-EINVAL);
if (load_attr->log_level > (4 | 2 | 1) || (load_attr->log_level && !load_attr->log_buf)) attempts = OPTS_GET(opts, attempts, 0);
if (attempts < 0)
return libbpf_err(-EINVAL); return libbpf_err(-EINVAL);
if (attempts == 0)
attempts = PROG_LOAD_ATTEMPTS;
memset(&attr, 0, sizeof(attr)); memset(&attr, 0, sizeof(attr));
attr.prog_type = load_attr->prog_type;
attr.expected_attach_type = load_attr->expected_attach_type;
if (load_attr->attach_prog_fd) attr.prog_type = prog_type;
attr.attach_prog_fd = load_attr->attach_prog_fd; attr.expected_attach_type = OPTS_GET(opts, expected_attach_type, 0);
else
attr.attach_btf_obj_fd = load_attr->attach_btf_obj_fd;
attr.attach_btf_id = load_attr->attach_btf_id;
attr.prog_ifindex = load_attr->prog_ifindex; attr.prog_btf_fd = OPTS_GET(opts, prog_btf_fd, 0);
attr.kern_version = load_attr->kern_version; attr.prog_flags = OPTS_GET(opts, prog_flags, 0);
attr.prog_ifindex = OPTS_GET(opts, prog_ifindex, 0);
attr.kern_version = OPTS_GET(opts, kern_version, 0);
attr.insn_cnt = (__u32)load_attr->insn_cnt; if (prog_name)
attr.insns = ptr_to_u64(load_attr->insns); strncat(attr.prog_name, prog_name, sizeof(attr.prog_name) - 1);
attr.license = ptr_to_u64(load_attr->license); attr.license = ptr_to_u64(license);
attr.log_level = load_attr->log_level; if (insn_cnt > UINT_MAX)
if (attr.log_level) { return libbpf_err(-E2BIG);
attr.log_buf = ptr_to_u64(load_attr->log_buf);
attr.log_size = load_attr->log_buf_sz;
}
attr.prog_btf_fd = load_attr->prog_btf_fd; attr.insns = ptr_to_u64(insns);
attr.prog_flags = load_attr->prog_flags; attr.insn_cnt = (__u32)insn_cnt;
attr.func_info_rec_size = load_attr->func_info_rec_size; attach_prog_fd = OPTS_GET(opts, attach_prog_fd, 0);
attr.func_info_cnt = load_attr->func_info_cnt; attach_btf_obj_fd = OPTS_GET(opts, attach_btf_obj_fd, 0);
attr.func_info = ptr_to_u64(load_attr->func_info);
attr.line_info_rec_size = load_attr->line_info_rec_size; if (attach_prog_fd && attach_btf_obj_fd)
attr.line_info_cnt = load_attr->line_info_cnt; return libbpf_err(-EINVAL);
attr.line_info = ptr_to_u64(load_attr->line_info);
attr.fd_array = ptr_to_u64(load_attr->fd_array);
if (load_attr->name) attr.attach_btf_id = OPTS_GET(opts, attach_btf_id, 0);
memcpy(attr.prog_name, load_attr->name, if (attach_prog_fd)
min(strlen(load_attr->name), (size_t)BPF_OBJ_NAME_LEN - 1)); attr.attach_prog_fd = attach_prog_fd;
else
attr.attach_btf_obj_fd = attach_btf_obj_fd;
fd = sys_bpf_prog_load(&attr, sizeof(attr), PROG_LOAD_ATTEMPTS); log_buf = OPTS_GET(opts, log_buf, NULL);
log_size = OPTS_GET(opts, log_size, 0);
log_level = OPTS_GET(opts, log_level, 0);
if (!!log_buf != !!log_size)
return libbpf_err(-EINVAL);
if (log_level > (4 | 2 | 1))
return libbpf_err(-EINVAL);
if (log_level && !log_buf)
return libbpf_err(-EINVAL);
attr.log_level = log_level;
attr.log_buf = ptr_to_u64(log_buf);
attr.log_size = log_size;
func_info_rec_size = OPTS_GET(opts, func_info_rec_size, 0);
func_info = OPTS_GET(opts, func_info, NULL);
attr.func_info_rec_size = func_info_rec_size;
attr.func_info = ptr_to_u64(func_info);
attr.func_info_cnt = OPTS_GET(opts, func_info_cnt, 0);
line_info_rec_size = OPTS_GET(opts, line_info_rec_size, 0);
line_info = OPTS_GET(opts, line_info, NULL);
attr.line_info_rec_size = line_info_rec_size;
attr.line_info = ptr_to_u64(line_info);
attr.line_info_cnt = OPTS_GET(opts, line_info_cnt, 0);
attr.fd_array = ptr_to_u64(OPTS_GET(opts, fd_array, NULL));
fd = sys_bpf_prog_load(&attr, sizeof(attr), attempts);
if (fd >= 0) if (fd >= 0)
return fd; return fd;
...@@ -315,11 +349,11 @@ int libbpf__bpf_prog_load(const struct bpf_prog_load_params *load_attr) ...@@ -315,11 +349,11 @@ int libbpf__bpf_prog_load(const struct bpf_prog_load_params *load_attr)
*/ */
while (errno == E2BIG && (!finfo || !linfo)) { while (errno == E2BIG && (!finfo || !linfo)) {
if (!finfo && attr.func_info_cnt && if (!finfo && attr.func_info_cnt &&
attr.func_info_rec_size < load_attr->func_info_rec_size) { attr.func_info_rec_size < func_info_rec_size) {
/* try with corrected func info records */ /* try with corrected func info records */
finfo = alloc_zero_tailing_info(load_attr->func_info, finfo = alloc_zero_tailing_info(func_info,
load_attr->func_info_cnt, attr.func_info_cnt,
load_attr->func_info_rec_size, func_info_rec_size,
attr.func_info_rec_size); attr.func_info_rec_size);
if (!finfo) { if (!finfo) {
errno = E2BIG; errno = E2BIG;
...@@ -327,13 +361,12 @@ int libbpf__bpf_prog_load(const struct bpf_prog_load_params *load_attr) ...@@ -327,13 +361,12 @@ int libbpf__bpf_prog_load(const struct bpf_prog_load_params *load_attr)
} }
attr.func_info = ptr_to_u64(finfo); attr.func_info = ptr_to_u64(finfo);
attr.func_info_rec_size = load_attr->func_info_rec_size; attr.func_info_rec_size = func_info_rec_size;
} else if (!linfo && attr.line_info_cnt && } else if (!linfo && attr.line_info_cnt &&
attr.line_info_rec_size < attr.line_info_rec_size < line_info_rec_size) {
load_attr->line_info_rec_size) { linfo = alloc_zero_tailing_info(line_info,
linfo = alloc_zero_tailing_info(load_attr->line_info, attr.line_info_cnt,
load_attr->line_info_cnt, line_info_rec_size,
load_attr->line_info_rec_size,
attr.line_info_rec_size); attr.line_info_rec_size);
if (!linfo) { if (!linfo) {
errno = E2BIG; errno = E2BIG;
...@@ -341,26 +374,26 @@ int libbpf__bpf_prog_load(const struct bpf_prog_load_params *load_attr) ...@@ -341,26 +374,26 @@ int libbpf__bpf_prog_load(const struct bpf_prog_load_params *load_attr)
} }
attr.line_info = ptr_to_u64(linfo); attr.line_info = ptr_to_u64(linfo);
attr.line_info_rec_size = load_attr->line_info_rec_size; attr.line_info_rec_size = line_info_rec_size;
} else { } else {
break; break;
} }
fd = sys_bpf_prog_load(&attr, sizeof(attr), PROG_LOAD_ATTEMPTS); fd = sys_bpf_prog_load(&attr, sizeof(attr), attempts);
if (fd >= 0) if (fd >= 0)
goto done; goto done;
} }
if (load_attr->log_level || !load_attr->log_buf) if (log_level || !log_buf)
goto done; goto done;
/* Try again with log */ /* Try again with log */
attr.log_buf = ptr_to_u64(load_attr->log_buf); log_buf[0] = 0;
attr.log_size = load_attr->log_buf_sz; attr.log_buf = ptr_to_u64(log_buf);
attr.log_size = log_size;
attr.log_level = 1; attr.log_level = 1;
load_attr->log_buf[0] = 0;
fd = sys_bpf_prog_load(&attr, sizeof(attr), PROG_LOAD_ATTEMPTS); fd = sys_bpf_prog_load(&attr, sizeof(attr), attempts);
done: done:
/* free() doesn't affect errno, so we don't need to restore it */ /* free() doesn't affect errno, so we don't need to restore it */
free(finfo); free(finfo);
...@@ -371,14 +404,13 @@ int libbpf__bpf_prog_load(const struct bpf_prog_load_params *load_attr) ...@@ -371,14 +404,13 @@ int libbpf__bpf_prog_load(const struct bpf_prog_load_params *load_attr)
int bpf_load_program_xattr(const struct bpf_load_program_attr *load_attr, int bpf_load_program_xattr(const struct bpf_load_program_attr *load_attr,
char *log_buf, size_t log_buf_sz) char *log_buf, size_t log_buf_sz)
{ {
struct bpf_prog_load_params p = {}; LIBBPF_OPTS(bpf_prog_load_opts, p);
if (!load_attr || !log_buf != !log_buf_sz) if (!load_attr || !log_buf != !log_buf_sz)
return libbpf_err(-EINVAL); return libbpf_err(-EINVAL);
p.prog_type = load_attr->prog_type;
p.expected_attach_type = load_attr->expected_attach_type; p.expected_attach_type = load_attr->expected_attach_type;
switch (p.prog_type) { switch (load_attr->prog_type) {
case BPF_PROG_TYPE_STRUCT_OPS: case BPF_PROG_TYPE_STRUCT_OPS:
case BPF_PROG_TYPE_LSM: case BPF_PROG_TYPE_LSM:
p.attach_btf_id = load_attr->attach_btf_id; p.attach_btf_id = load_attr->attach_btf_id;
...@@ -392,12 +424,9 @@ int bpf_load_program_xattr(const struct bpf_load_program_attr *load_attr, ...@@ -392,12 +424,9 @@ int bpf_load_program_xattr(const struct bpf_load_program_attr *load_attr,
p.prog_ifindex = load_attr->prog_ifindex; p.prog_ifindex = load_attr->prog_ifindex;
p.kern_version = load_attr->kern_version; p.kern_version = load_attr->kern_version;
} }
p.insn_cnt = load_attr->insns_cnt;
p.insns = load_attr->insns;
p.license = load_attr->license;
p.log_level = load_attr->log_level; p.log_level = load_attr->log_level;
p.log_buf = log_buf; p.log_buf = log_buf;
p.log_buf_sz = log_buf_sz; p.log_size = log_buf_sz;
p.prog_btf_fd = load_attr->prog_btf_fd; p.prog_btf_fd = load_attr->prog_btf_fd;
p.func_info_rec_size = load_attr->func_info_rec_size; p.func_info_rec_size = load_attr->func_info_rec_size;
p.func_info_cnt = load_attr->func_info_cnt; p.func_info_cnt = load_attr->func_info_cnt;
...@@ -405,10 +434,10 @@ int bpf_load_program_xattr(const struct bpf_load_program_attr *load_attr, ...@@ -405,10 +434,10 @@ int bpf_load_program_xattr(const struct bpf_load_program_attr *load_attr,
p.line_info_rec_size = load_attr->line_info_rec_size; p.line_info_rec_size = load_attr->line_info_rec_size;
p.line_info_cnt = load_attr->line_info_cnt; p.line_info_cnt = load_attr->line_info_cnt;
p.line_info = load_attr->line_info; p.line_info = load_attr->line_info;
p.name = load_attr->name;
p.prog_flags = load_attr->prog_flags; p.prog_flags = load_attr->prog_flags;
return libbpf__bpf_prog_load(&p); return bpf_prog_load(load_attr->prog_type, load_attr->name, load_attr->license,
load_attr->insns, load_attr->insns_cnt, &p);
} }
int bpf_load_program(enum bpf_prog_type type, const struct bpf_insn *insns, int bpf_load_program(enum bpf_prog_type type, const struct bpf_insn *insns,
......
...@@ -72,6 +72,71 @@ LIBBPF_API int bpf_create_map_in_map(enum bpf_map_type map_type, ...@@ -72,6 +72,71 @@ LIBBPF_API int bpf_create_map_in_map(enum bpf_map_type map_type,
int inner_map_fd, int max_entries, int inner_map_fd, int max_entries,
__u32 map_flags); __u32 map_flags);
struct bpf_prog_load_opts {
size_t sz; /* size of this struct for forward/backward compatibility */
/* libbpf can retry BPF_PROG_LOAD command if bpf() syscall returns
* -EAGAIN. This field determines how many attempts libbpf has to
* make. If not specified, libbpf will use default value of 5.
*/
int attempts;
enum bpf_attach_type expected_attach_type;
__u32 prog_btf_fd;
__u32 prog_flags;
__u32 prog_ifindex;
__u32 kern_version;
__u32 attach_btf_id;
__u32 attach_prog_fd;
__u32 attach_btf_obj_fd;
const int *fd_array;
/* .BTF.ext func info data */
const void *func_info;
__u32 func_info_cnt;
__u32 func_info_rec_size;
/* .BTF.ext line info data */
const void *line_info;
__u32 line_info_cnt;
__u32 line_info_rec_size;
/* verifier log options */
__u32 log_level;
__u32 log_size;
char *log_buf;
};
#define bpf_prog_load_opts__last_field log_buf
LIBBPF_API int bpf_prog_load(enum bpf_prog_type prog_type,
const char *prog_name, const char *license,
const struct bpf_insn *insns, size_t insn_cnt,
const struct bpf_prog_load_opts *opts);
/* this "specialization" should go away in libbpf 1.0 */
LIBBPF_API int bpf_prog_load_v0_6_0(enum bpf_prog_type prog_type,
const char *prog_name, const char *license,
const struct bpf_insn *insns, size_t insn_cnt,
const struct bpf_prog_load_opts *opts);
/* This is an elaborate way to not conflict with deprecated bpf_prog_load()
* API, defined in libbpf.h. Once we hit libbpf 1.0, all this will be gone.
* With this approach, if someone is calling bpf_prog_load() with
* 4 arguments, they will use the deprecated API, which keeps backwards
* compatibility (both source code and binary). If bpf_prog_load() is called
* with 6 arguments, though, it gets redirected to __bpf_prog_load.
* So looking forward to libbpf 1.0 when this hack will be gone and
* __bpf_prog_load() will be called just bpf_prog_load().
*/
#ifndef bpf_prog_load
#define bpf_prog_load(...) ___libbpf_overload(___bpf_prog_load, __VA_ARGS__)
#define ___bpf_prog_load4(file, type, pobj, prog_fd) \
bpf_prog_load_deprecated(file, type, pobj, prog_fd)
#define ___bpf_prog_load6(prog_type, prog_name, license, insns, insn_cnt, opts) \
bpf_prog_load(prog_type, prog_name, license, insns, insn_cnt, opts)
#endif /* bpf_prog_load */
struct bpf_load_program_attr { struct bpf_load_program_attr {
enum bpf_prog_type prog_type; enum bpf_prog_type prog_type;
enum bpf_attach_type expected_attach_type; enum bpf_attach_type expected_attach_type;
...@@ -103,13 +168,15 @@ struct bpf_load_program_attr { ...@@ -103,13 +168,15 @@ struct bpf_load_program_attr {
/* Recommend log buffer size */ /* Recommend log buffer size */
#define BPF_LOG_BUF_SIZE (UINT32_MAX >> 8) /* verifier maximum in kernels <= 5.1 */ #define BPF_LOG_BUF_SIZE (UINT32_MAX >> 8) /* verifier maximum in kernels <= 5.1 */
LIBBPF_API int LIBBPF_DEPRECATED_SINCE(0, 7, "use bpf_prog_load() instead")
bpf_load_program_xattr(const struct bpf_load_program_attr *load_attr, LIBBPF_API int bpf_load_program_xattr(const struct bpf_load_program_attr *load_attr,
char *log_buf, size_t log_buf_sz); char *log_buf, size_t log_buf_sz);
LIBBPF_DEPRECATED_SINCE(0, 7, "use bpf_prog_load() instead")
LIBBPF_API int bpf_load_program(enum bpf_prog_type type, LIBBPF_API int bpf_load_program(enum bpf_prog_type type,
const struct bpf_insn *insns, size_t insns_cnt, const struct bpf_insn *insns, size_t insns_cnt,
const char *license, __u32 kern_version, const char *license, __u32 kern_version,
char *log_buf, size_t log_buf_sz); char *log_buf, size_t log_buf_sz);
LIBBPF_DEPRECATED_SINCE(0, 7, "use bpf_prog_load() instead")
LIBBPF_API int bpf_verify_program(enum bpf_prog_type type, LIBBPF_API int bpf_verify_program(enum bpf_prog_type type,
const struct bpf_insn *insns, const struct bpf_insn *insns,
size_t insns_cnt, __u32 prog_flags, size_t insns_cnt, __u32 prog_flags,
......
...@@ -3,6 +3,8 @@ ...@@ -3,6 +3,8 @@
#ifndef __BPF_GEN_INTERNAL_H #ifndef __BPF_GEN_INTERNAL_H
#define __BPF_GEN_INTERNAL_H #define __BPF_GEN_INTERNAL_H
#include "bpf.h"
struct ksym_relo_desc { struct ksym_relo_desc {
const char *name; const char *name;
int kind; int kind;
...@@ -50,8 +52,10 @@ int bpf_gen__finish(struct bpf_gen *gen); ...@@ -50,8 +52,10 @@ int bpf_gen__finish(struct bpf_gen *gen);
void bpf_gen__free(struct bpf_gen *gen); void bpf_gen__free(struct bpf_gen *gen);
void bpf_gen__load_btf(struct bpf_gen *gen, const void *raw_data, __u32 raw_size); void bpf_gen__load_btf(struct bpf_gen *gen, const void *raw_data, __u32 raw_size);
void bpf_gen__map_create(struct bpf_gen *gen, struct bpf_create_map_params *map_attr, int map_idx); void bpf_gen__map_create(struct bpf_gen *gen, struct bpf_create_map_params *map_attr, int map_idx);
struct bpf_prog_load_params; void bpf_gen__prog_load(struct bpf_gen *gen,
void bpf_gen__prog_load(struct bpf_gen *gen, struct bpf_prog_load_params *load_attr, int prog_idx); enum bpf_prog_type prog_type, const char *prog_name,
const char *license, struct bpf_insn *insns, size_t insn_cnt,
struct bpf_prog_load_opts *load_attr, int prog_idx);
void bpf_gen__map_update_elem(struct bpf_gen *gen, int map_idx, void *value, __u32 value_size); void bpf_gen__map_update_elem(struct bpf_gen *gen, int map_idx, void *value, __u32 value_size);
void bpf_gen__map_freeze(struct bpf_gen *gen, int map_idx); void bpf_gen__map_freeze(struct bpf_gen *gen, int map_idx);
void bpf_gen__record_attach_target(struct bpf_gen *gen, const char *name, enum bpf_attach_type type); void bpf_gen__record_attach_target(struct bpf_gen *gen, const char *name, enum bpf_attach_type type);
......
...@@ -901,27 +901,27 @@ static void cleanup_relos(struct bpf_gen *gen, int insns) ...@@ -901,27 +901,27 @@ static void cleanup_relos(struct bpf_gen *gen, int insns)
} }
void bpf_gen__prog_load(struct bpf_gen *gen, void bpf_gen__prog_load(struct bpf_gen *gen,
struct bpf_prog_load_params *load_attr, int prog_idx) enum bpf_prog_type prog_type, const char *prog_name,
const char *license, struct bpf_insn *insns, size_t insn_cnt,
struct bpf_prog_load_opts *load_attr, int prog_idx)
{ {
int attr_size = offsetofend(union bpf_attr, fd_array); int attr_size = offsetofend(union bpf_attr, fd_array);
int prog_load_attr, license, insns, func_info, line_info; int prog_load_attr, license_off, insns_off, func_info, line_info;
union bpf_attr attr; union bpf_attr attr;
memset(&attr, 0, attr_size); memset(&attr, 0, attr_size);
pr_debug("gen: prog_load: type %d insns_cnt %zd\n", pr_debug("gen: prog_load: type %d insns_cnt %zd\n", prog_type, insn_cnt);
load_attr->prog_type, load_attr->insn_cnt);
/* add license string to blob of bytes */ /* add license string to blob of bytes */
license = add_data(gen, load_attr->license, strlen(load_attr->license) + 1); license_off = add_data(gen, license, strlen(license) + 1);
/* add insns to blob of bytes */ /* add insns to blob of bytes */
insns = add_data(gen, load_attr->insns, insns_off = add_data(gen, insns, insn_cnt * sizeof(struct bpf_insn));
load_attr->insn_cnt * sizeof(struct bpf_insn));
attr.prog_type = load_attr->prog_type; attr.prog_type = prog_type;
attr.expected_attach_type = load_attr->expected_attach_type; attr.expected_attach_type = load_attr->expected_attach_type;
attr.attach_btf_id = load_attr->attach_btf_id; attr.attach_btf_id = load_attr->attach_btf_id;
attr.prog_ifindex = load_attr->prog_ifindex; attr.prog_ifindex = load_attr->prog_ifindex;
attr.kern_version = 0; attr.kern_version = 0;
attr.insn_cnt = (__u32)load_attr->insn_cnt; attr.insn_cnt = (__u32)insn_cnt;
attr.prog_flags = load_attr->prog_flags; attr.prog_flags = load_attr->prog_flags;
attr.func_info_rec_size = load_attr->func_info_rec_size; attr.func_info_rec_size = load_attr->func_info_rec_size;
...@@ -934,15 +934,15 @@ void bpf_gen__prog_load(struct bpf_gen *gen, ...@@ -934,15 +934,15 @@ void bpf_gen__prog_load(struct bpf_gen *gen,
line_info = add_data(gen, load_attr->line_info, line_info = add_data(gen, load_attr->line_info,
attr.line_info_cnt * attr.line_info_rec_size); attr.line_info_cnt * attr.line_info_rec_size);
memcpy(attr.prog_name, load_attr->name, memcpy(attr.prog_name, prog_name,
min((unsigned)strlen(load_attr->name), BPF_OBJ_NAME_LEN - 1)); min((unsigned)strlen(prog_name), BPF_OBJ_NAME_LEN - 1));
prog_load_attr = add_data(gen, &attr, attr_size); prog_load_attr = add_data(gen, &attr, attr_size);
/* populate union bpf_attr with a pointer to license */ /* populate union bpf_attr with a pointer to license */
emit_rel_store(gen, attr_field(prog_load_attr, license), license); emit_rel_store(gen, attr_field(prog_load_attr, license), license_off);
/* populate union bpf_attr with a pointer to instructions */ /* populate union bpf_attr with a pointer to instructions */
emit_rel_store(gen, attr_field(prog_load_attr, insns), insns); emit_rel_store(gen, attr_field(prog_load_attr, insns), insns_off);
/* populate union bpf_attr with a pointer to func_info */ /* populate union bpf_attr with a pointer to func_info */
emit_rel_store(gen, attr_field(prog_load_attr, func_info), func_info); emit_rel_store(gen, attr_field(prog_load_attr, func_info), func_info);
...@@ -974,12 +974,12 @@ void bpf_gen__prog_load(struct bpf_gen *gen, ...@@ -974,12 +974,12 @@ void bpf_gen__prog_load(struct bpf_gen *gen,
emit(gen, BPF_STX_MEM(BPF_W, BPF_REG_0, BPF_REG_7, emit(gen, BPF_STX_MEM(BPF_W, BPF_REG_0, BPF_REG_7,
offsetof(union bpf_attr, attach_btf_obj_fd))); offsetof(union bpf_attr, attach_btf_obj_fd)));
} }
emit_relos(gen, insns); emit_relos(gen, insns_off);
/* emit PROG_LOAD command */ /* emit PROG_LOAD command */
emit_sys_bpf(gen, BPF_PROG_LOAD, prog_load_attr, attr_size); emit_sys_bpf(gen, BPF_PROG_LOAD, prog_load_attr, attr_size);
debug_ret(gen, "prog_load %s insn_cnt %d", attr.prog_name, attr.insn_cnt); debug_ret(gen, "prog_load %s insn_cnt %d", attr.prog_name, attr.insn_cnt);
/* successful or not, close btf module FDs used in extern ksyms and attach_btf_obj_fd */ /* successful or not, close btf module FDs used in extern ksyms and attach_btf_obj_fd */
cleanup_relos(gen, insns); cleanup_relos(gen, insns_off);
if (gen->attach_kind) if (gen->attach_kind)
emit_sys_close_blob(gen, emit_sys_close_blob(gen,
attr_field(prog_load_attr, attach_btf_obj_fd)); attr_field(prog_load_attr, attach_btf_obj_fd));
......
...@@ -221,7 +221,7 @@ struct reloc_desc { ...@@ -221,7 +221,7 @@ struct reloc_desc {
struct bpf_sec_def; struct bpf_sec_def;
typedef int (*init_fn_t)(struct bpf_program *prog, long cookie); typedef int (*init_fn_t)(struct bpf_program *prog, long cookie);
typedef int (*preload_fn_t)(struct bpf_program *prog, struct bpf_prog_load_params *attr, long cookie); typedef int (*preload_fn_t)(struct bpf_program *prog, struct bpf_prog_load_opts *opts, long cookie);
typedef struct bpf_link *(*attach_fn_t)(const struct bpf_program *prog, long cookie); typedef struct bpf_link *(*attach_fn_t)(const struct bpf_program *prog, long cookie);
/* stored as sec_def->cookie for all libbpf-supported SEC()s */ /* stored as sec_def->cookie for all libbpf-supported SEC()s */
...@@ -6391,16 +6391,16 @@ static int libbpf_find_attach_btf_id(struct bpf_program *prog, const char *attac ...@@ -6391,16 +6391,16 @@ static int libbpf_find_attach_btf_id(struct bpf_program *prog, const char *attac
/* this is called as prog->sec_def->preload_fn for libbpf-supported sec_defs */ /* this is called as prog->sec_def->preload_fn for libbpf-supported sec_defs */
static int libbpf_preload_prog(struct bpf_program *prog, static int libbpf_preload_prog(struct bpf_program *prog,
struct bpf_prog_load_params *attr, long cookie) struct bpf_prog_load_opts *opts, long cookie)
{ {
enum sec_def_flags def = cookie; enum sec_def_flags def = cookie;
/* old kernels might not support specifying expected_attach_type */ /* old kernels might not support specifying expected_attach_type */
if ((def & SEC_EXP_ATTACH_OPT) && !kernel_supports(prog->obj, FEAT_EXP_ATTACH_TYPE)) if ((def & SEC_EXP_ATTACH_OPT) && !kernel_supports(prog->obj, FEAT_EXP_ATTACH_TYPE))
attr->expected_attach_type = 0; opts->expected_attach_type = 0;
if (def & SEC_SLEEPABLE) if (def & SEC_SLEEPABLE)
attr->prog_flags |= BPF_F_SLEEPABLE; opts->prog_flags |= BPF_F_SLEEPABLE;
if ((prog->type == BPF_PROG_TYPE_TRACING || if ((prog->type == BPF_PROG_TYPE_TRACING ||
prog->type == BPF_PROG_TYPE_LSM || prog->type == BPF_PROG_TYPE_LSM ||
...@@ -6419,11 +6419,11 @@ static int libbpf_preload_prog(struct bpf_program *prog, ...@@ -6419,11 +6419,11 @@ static int libbpf_preload_prog(struct bpf_program *prog,
/* but by now libbpf common logic is not utilizing /* but by now libbpf common logic is not utilizing
* prog->atach_btf_obj_fd/prog->attach_btf_id anymore because * prog->atach_btf_obj_fd/prog->attach_btf_id anymore because
* this callback is called after attrs were populated by * this callback is called after opts were populated by
* libbpf, so this callback has to update attr explicitly here * libbpf, so this callback has to update opts explicitly here
*/ */
attr->attach_btf_obj_fd = btf_obj_fd; opts->attach_btf_obj_fd = btf_obj_fd;
attr->attach_btf_id = btf_type_id; opts->attach_btf_id = btf_type_id;
} }
return 0; return 0;
} }
...@@ -6433,7 +6433,8 @@ static int bpf_object_load_prog_instance(struct bpf_object *obj, struct bpf_prog ...@@ -6433,7 +6433,8 @@ static int bpf_object_load_prog_instance(struct bpf_object *obj, struct bpf_prog
const char *license, __u32 kern_version, const char *license, __u32 kern_version,
int *prog_fd) int *prog_fd)
{ {
struct bpf_prog_load_params load_attr = {}; LIBBPF_OPTS(bpf_prog_load_opts, load_attr);
const char *prog_name = NULL;
char *cp, errmsg[STRERR_BUFSIZE]; char *cp, errmsg[STRERR_BUFSIZE];
size_t log_buf_size = 0; size_t log_buf_size = 0;
char *log_buf = NULL; char *log_buf = NULL;
...@@ -6452,13 +6453,9 @@ static int bpf_object_load_prog_instance(struct bpf_object *obj, struct bpf_prog ...@@ -6452,13 +6453,9 @@ static int bpf_object_load_prog_instance(struct bpf_object *obj, struct bpf_prog
if (!insns || !insns_cnt) if (!insns || !insns_cnt)
return -EINVAL; return -EINVAL;
load_attr.prog_type = prog->type;
load_attr.expected_attach_type = prog->expected_attach_type; load_attr.expected_attach_type = prog->expected_attach_type;
if (kernel_supports(obj, FEAT_PROG_NAME)) if (kernel_supports(obj, FEAT_PROG_NAME))
load_attr.name = prog->name; prog_name = prog->name;
load_attr.insns = insns;
load_attr.insn_cnt = insns_cnt;
load_attr.license = license;
load_attr.attach_btf_id = prog->attach_btf_id; load_attr.attach_btf_id = prog->attach_btf_id;
load_attr.attach_prog_fd = prog->attach_prog_fd; load_attr.attach_prog_fd = prog->attach_prog_fd;
load_attr.attach_btf_obj_fd = prog->attach_btf_obj_fd; load_attr.attach_btf_obj_fd = prog->attach_btf_obj_fd;
...@@ -6492,7 +6489,8 @@ static int bpf_object_load_prog_instance(struct bpf_object *obj, struct bpf_prog ...@@ -6492,7 +6489,8 @@ static int bpf_object_load_prog_instance(struct bpf_object *obj, struct bpf_prog
} }
if (obj->gen_loader) { if (obj->gen_loader) {
bpf_gen__prog_load(obj->gen_loader, &load_attr, bpf_gen__prog_load(obj->gen_loader, prog->type, prog->name,
license, insns, insns_cnt, &load_attr,
prog - obj->programs); prog - obj->programs);
*prog_fd = -1; *prog_fd = -1;
return 0; return 0;
...@@ -6507,8 +6505,8 @@ static int bpf_object_load_prog_instance(struct bpf_object *obj, struct bpf_prog ...@@ -6507,8 +6505,8 @@ static int bpf_object_load_prog_instance(struct bpf_object *obj, struct bpf_prog
} }
load_attr.log_buf = log_buf; load_attr.log_buf = log_buf;
load_attr.log_buf_sz = log_buf_size; load_attr.log_size = log_buf_size;
ret = libbpf__bpf_prog_load(&load_attr); ret = bpf_prog_load(prog->type, prog_name, license, insns, insns_cnt, &load_attr);
if (ret >= 0) { if (ret >= 0) {
if (log_buf && load_attr.log_level) if (log_buf && load_attr.log_level)
...@@ -6554,19 +6552,19 @@ static int bpf_object_load_prog_instance(struct bpf_object *obj, struct bpf_prog ...@@ -6554,19 +6552,19 @@ static int bpf_object_load_prog_instance(struct bpf_object *obj, struct bpf_prog
pr_warn("-- BEGIN DUMP LOG ---\n"); pr_warn("-- BEGIN DUMP LOG ---\n");
pr_warn("\n%s\n", log_buf); pr_warn("\n%s\n", log_buf);
pr_warn("-- END LOG --\n"); pr_warn("-- END LOG --\n");
} else if (load_attr.insn_cnt >= BPF_MAXINSNS) { } else if (insns_cnt >= BPF_MAXINSNS) {
pr_warn("Program too large (%zu insns), at most %d insns\n", pr_warn("Program too large (%d insns), at most %d insns\n",
load_attr.insn_cnt, BPF_MAXINSNS); insns_cnt, BPF_MAXINSNS);
ret = -LIBBPF_ERRNO__PROG2BIG; ret = -LIBBPF_ERRNO__PROG2BIG;
} else if (load_attr.prog_type != BPF_PROG_TYPE_KPROBE) { } else if (prog->type != BPF_PROG_TYPE_KPROBE) {
/* Wrong program type? */ /* Wrong program type? */
int fd; int fd;
load_attr.prog_type = BPF_PROG_TYPE_KPROBE;
load_attr.expected_attach_type = 0; load_attr.expected_attach_type = 0;
load_attr.log_buf = NULL; load_attr.log_buf = NULL;
load_attr.log_buf_sz = 0; load_attr.log_size = 0;
fd = libbpf__bpf_prog_load(&load_attr); fd = bpf_prog_load(BPF_PROG_TYPE_KPROBE, prog_name, license,
insns, insns_cnt, &load_attr);
if (fd >= 0) { if (fd >= 0) {
close(fd); close(fd);
ret = -LIBBPF_ERRNO__PROGTYPE; ret = -LIBBPF_ERRNO__PROGTYPE;
...@@ -9170,8 +9168,9 @@ long libbpf_get_error(const void *ptr) ...@@ -9170,8 +9168,9 @@ long libbpf_get_error(const void *ptr)
return -errno; return -errno;
} }
int bpf_prog_load(const char *file, enum bpf_prog_type type, COMPAT_VERSION(bpf_prog_load_deprecated, bpf_prog_load, LIBBPF_0.0.1)
struct bpf_object **pobj, int *prog_fd) int bpf_prog_load_deprecated(const char *file, enum bpf_prog_type type,
struct bpf_object **pobj, int *prog_fd)
{ {
struct bpf_prog_load_attr attr; struct bpf_prog_load_attr attr;
......
...@@ -676,8 +676,9 @@ struct bpf_prog_load_attr { ...@@ -676,8 +676,9 @@ struct bpf_prog_load_attr {
LIBBPF_API int bpf_prog_load_xattr(const struct bpf_prog_load_attr *attr, LIBBPF_API int bpf_prog_load_xattr(const struct bpf_prog_load_attr *attr,
struct bpf_object **pobj, int *prog_fd); struct bpf_object **pobj, int *prog_fd);
LIBBPF_API int bpf_prog_load(const char *file, enum bpf_prog_type type, LIBBPF_DEPRECATED_SINCE(0, 7, "use bpf_object__open() and bpf_object__load() instead")
struct bpf_object **pobj, int *prog_fd); LIBBPF_API int bpf_prog_load_deprecated(const char *file, enum bpf_prog_type type,
struct bpf_object **pobj, int *prog_fd);
/* XDP related API */ /* XDP related API */
struct xdp_link_info { struct xdp_link_info {
......
...@@ -395,6 +395,8 @@ LIBBPF_0.6.0 { ...@@ -395,6 +395,8 @@ LIBBPF_0.6.0 {
bpf_object__next_program; bpf_object__next_program;
bpf_object__prev_map; bpf_object__prev_map;
bpf_object__prev_program; bpf_object__prev_program;
bpf_prog_load_deprecated;
bpf_prog_load;
bpf_program__insn_cnt; bpf_program__insn_cnt;
bpf_program__insns; bpf_program__insns;
btf__add_btf; btf__add_btf;
......
...@@ -41,6 +41,18 @@ ...@@ -41,6 +41,18 @@
#define __LIBBPF_MARK_DEPRECATED_0_7(X) #define __LIBBPF_MARK_DEPRECATED_0_7(X)
#endif #endif
/* This set of internal macros allows to do "function overloading" based on
* number of arguments provided by used in backwards-compatible way during the
* transition to libbpf 1.0
* It's ugly but necessary evil that will be cleaned up when we get to 1.0.
* See bpf_prog_load() overload for example.
*/
#define ___libbpf_cat(A, B) A ## B
#define ___libbpf_select(NAME, NUM) ___libbpf_cat(NAME, NUM)
#define ___libbpf_nth(_1, _2, _3, _4, _5, _6, N, ...) N
#define ___libbpf_cnt(...) ___libbpf_nth(__VA_ARGS__, 6, 5, 4, 3, 2, 1)
#define ___libbpf_overload(NAME, ...) ___libbpf_select(NAME, ___libbpf_cnt(__VA_ARGS__))(__VA_ARGS__)
/* Helper macro to declare and initialize libbpf options struct /* Helper macro to declare and initialize libbpf options struct
* *
* This dance with uninitialized declaration, followed by memset to zero, * This dance with uninitialized declaration, followed by memset to zero,
......
...@@ -276,37 +276,6 @@ int parse_cpu_mask_file(const char *fcpu, bool **mask, int *mask_sz); ...@@ -276,37 +276,6 @@ int parse_cpu_mask_file(const char *fcpu, bool **mask, int *mask_sz);
int libbpf__load_raw_btf(const char *raw_types, size_t types_len, int libbpf__load_raw_btf(const char *raw_types, size_t types_len,
const char *str_sec, size_t str_len); const char *str_sec, size_t str_len);
struct bpf_prog_load_params {
enum bpf_prog_type prog_type;
enum bpf_attach_type expected_attach_type;
const char *name;
const struct bpf_insn *insns;
size_t insn_cnt;
const char *license;
__u32 kern_version;
__u32 attach_prog_fd;
__u32 attach_btf_obj_fd;
__u32 attach_btf_id;
__u32 prog_ifindex;
__u32 prog_btf_fd;
__u32 prog_flags;
__u32 func_info_rec_size;
const void *func_info;
__u32 func_info_cnt;
__u32 line_info_rec_size;
const void *line_info;
__u32 line_info_cnt;
__u32 log_level;
char *log_buf;
size_t log_buf_sz;
int *fd_array;
};
int libbpf__bpf_prog_load(const struct bpf_prog_load_params *load_attr);
struct bpf_create_map_params { struct bpf_create_map_params {
const char *name; const char *name;
enum bpf_map_type map_type; enum bpf_map_type map_type;
......
...@@ -24,7 +24,7 @@ SAN_CFLAGS ?= ...@@ -24,7 +24,7 @@ SAN_CFLAGS ?=
CFLAGS += -g -O0 -rdynamic -Wall $(GENFLAGS) $(SAN_CFLAGS) \ CFLAGS += -g -O0 -rdynamic -Wall $(GENFLAGS) $(SAN_CFLAGS) \
-I$(CURDIR) -I$(INCLUDE_DIR) -I$(GENDIR) -I$(LIBDIR) \ -I$(CURDIR) -I$(INCLUDE_DIR) -I$(GENDIR) -I$(LIBDIR) \
-I$(TOOLSINCDIR) -I$(APIDIR) -I$(OUTPUT) \ -I$(TOOLSINCDIR) -I$(APIDIR) -I$(OUTPUT) \
-Dbpf_prog_load=bpf_prog_test_load \ -Dbpf_prog_load_deprecated=bpf_prog_test_load \
-Dbpf_load_program=bpf_test_load_program -Dbpf_load_program=bpf_test_load_program
LDLIBS += -lcap -lelf -lz -lrt -lpthread LDLIBS += -lcap -lelf -lz -lrt -lpthread
......
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