Commit 28b93c64 authored by Andrii Nakryiko's avatar Andrii Nakryiko Committed by Alexei Starovoitov

libbpf: Clean up and improve CO-RE reloc logging

Add logging of local/target type kind (struct/union/typedef/etc). Preserve
unresolved root type ID (for cases of typedef). Improve the format of CO-RE
reloc spec output format to contain only relevant and succinct info.
Signed-off-by: default avatarAndrii Nakryiko <andriin@fb.com>
Signed-off-by: default avatarAlexei Starovoitov <ast@kernel.org>
Link: https://lore.kernel.org/bpf/20200818223921.2911963-3-andriin@fb.com
parent 81ba0889
...@@ -1131,14 +1131,14 @@ static int btf_ext_setup_line_info(struct btf_ext *btf_ext) ...@@ -1131,14 +1131,14 @@ static int btf_ext_setup_line_info(struct btf_ext *btf_ext)
return btf_ext_setup_info(btf_ext, &param); return btf_ext_setup_info(btf_ext, &param);
} }
static int btf_ext_setup_field_reloc(struct btf_ext *btf_ext) static int btf_ext_setup_core_relos(struct btf_ext *btf_ext)
{ {
struct btf_ext_sec_setup_param param = { struct btf_ext_sec_setup_param param = {
.off = btf_ext->hdr->field_reloc_off, .off = btf_ext->hdr->core_relo_off,
.len = btf_ext->hdr->field_reloc_len, .len = btf_ext->hdr->core_relo_len,
.min_rec_size = sizeof(struct bpf_field_reloc), .min_rec_size = sizeof(struct bpf_core_relo),
.ext_info = &btf_ext->field_reloc_info, .ext_info = &btf_ext->core_relo_info,
.desc = "field_reloc", .desc = "core_relo",
}; };
return btf_ext_setup_info(btf_ext, &param); return btf_ext_setup_info(btf_ext, &param);
...@@ -1217,10 +1217,9 @@ struct btf_ext *btf_ext__new(__u8 *data, __u32 size) ...@@ -1217,10 +1217,9 @@ struct btf_ext *btf_ext__new(__u8 *data, __u32 size)
if (err) if (err)
goto done; goto done;
if (btf_ext->hdr->hdr_len < if (btf_ext->hdr->hdr_len < offsetofend(struct btf_ext_header, core_relo_len))
offsetofend(struct btf_ext_header, field_reloc_len))
goto done; goto done;
err = btf_ext_setup_field_reloc(btf_ext); err = btf_ext_setup_core_relos(btf_ext);
if (err) if (err)
goto done; goto done;
......
...@@ -24,44 +24,6 @@ struct btf_type; ...@@ -24,44 +24,6 @@ struct btf_type;
struct bpf_object; struct bpf_object;
/*
* The .BTF.ext ELF section layout defined as
* struct btf_ext_header
* func_info subsection
*
* The func_info subsection layout:
* record size for struct bpf_func_info in the func_info subsection
* struct btf_sec_func_info for section #1
* a list of bpf_func_info records for section #1
* where struct bpf_func_info mimics one in include/uapi/linux/bpf.h
* but may not be identical
* struct btf_sec_func_info for section #2
* a list of bpf_func_info records for section #2
* ......
*
* Note that the bpf_func_info record size in .BTF.ext may not
* be the same as the one defined in include/uapi/linux/bpf.h.
* The loader should ensure that record_size meets minimum
* requirement and pass the record as is to the kernel. The
* kernel will handle the func_info properly based on its contents.
*/
struct btf_ext_header {
__u16 magic;
__u8 version;
__u8 flags;
__u32 hdr_len;
/* All offsets are in bytes relative to the end of this header */
__u32 func_info_off;
__u32 func_info_len;
__u32 line_info_off;
__u32 line_info_len;
/* optional part of .BTF.ext header */
__u32 field_reloc_off;
__u32 field_reloc_len;
};
LIBBPF_API void btf__free(struct btf *btf); LIBBPF_API void btf__free(struct btf *btf);
LIBBPF_API struct btf *btf__new(const void *data, __u32 size); LIBBPF_API struct btf *btf__new(const void *data, __u32 size);
LIBBPF_API struct btf *btf__parse(const char *path, struct btf_ext **btf_ext); LIBBPF_API struct btf *btf__parse(const char *path, struct btf_ext **btf_ext);
......
...@@ -2524,7 +2524,7 @@ static int bpf_object__load_vmlinux_btf(struct bpf_object *obj) ...@@ -2524,7 +2524,7 @@ static int bpf_object__load_vmlinux_btf(struct bpf_object *obj)
int err; int err;
/* CO-RE relocations need kernel BTF */ /* CO-RE relocations need kernel BTF */
if (obj->btf_ext && obj->btf_ext->field_reloc_info.len) if (obj->btf_ext && obj->btf_ext->core_relo_info.len)
need_vmlinux_btf = true; need_vmlinux_btf = true;
bpf_object__for_each_program(prog, obj) { bpf_object__for_each_program(prog, obj) {
...@@ -4074,6 +4074,10 @@ struct bpf_core_spec { ...@@ -4074,6 +4074,10 @@ struct bpf_core_spec {
const struct btf *btf; const struct btf *btf;
/* high-level spec: named fields and array indices only */ /* high-level spec: named fields and array indices only */
struct bpf_core_accessor spec[BPF_CORE_SPEC_MAX_LEN]; struct bpf_core_accessor spec[BPF_CORE_SPEC_MAX_LEN];
/* original unresolved (no skip_mods_or_typedefs) root type ID */
__u32 root_type_id;
/* CO-RE relocation kind */
enum bpf_core_relo_kind relo_kind;
/* high-level spec length */ /* high-level spec length */
int len; int len;
/* raw, low-level spec: 1-to-1 with accessor spec string */ /* raw, low-level spec: 1-to-1 with accessor spec string */
...@@ -4104,8 +4108,36 @@ static bool is_flex_arr(const struct btf *btf, ...@@ -4104,8 +4108,36 @@ static bool is_flex_arr(const struct btf *btf,
return acc->idx == btf_vlen(t) - 1; return acc->idx == btf_vlen(t) - 1;
} }
static const char *core_relo_kind_str(enum bpf_core_relo_kind kind)
{
switch (kind) {
case BPF_FIELD_BYTE_OFFSET: return "byte_off";
case BPF_FIELD_BYTE_SIZE: return "byte_sz";
case BPF_FIELD_EXISTS: return "field_exists";
case BPF_FIELD_SIGNED: return "signed";
case BPF_FIELD_LSHIFT_U64: return "lshift_u64";
case BPF_FIELD_RSHIFT_U64: return "rshift_u64";
default: return "unknown";
}
}
static bool core_relo_is_field_based(enum bpf_core_relo_kind kind)
{
switch (kind) {
case BPF_FIELD_BYTE_OFFSET:
case BPF_FIELD_BYTE_SIZE:
case BPF_FIELD_EXISTS:
case BPF_FIELD_SIGNED:
case BPF_FIELD_LSHIFT_U64:
case BPF_FIELD_RSHIFT_U64:
return true;
default:
return false;
}
}
/* /*
* Turn bpf_field_reloc into a low- and high-level spec representation, * Turn bpf_core_relo into a low- and high-level spec representation,
* validating correctness along the way, as well as calculating resulting * validating correctness along the way, as well as calculating resulting
* field bit offset, specified by accessor string. Low-level spec captures * field bit offset, specified by accessor string. Low-level spec captures
* every single level of nestedness, including traversing anonymous * every single level of nestedness, including traversing anonymous
...@@ -4135,9 +4167,10 @@ static bool is_flex_arr(const struct btf *btf, ...@@ -4135,9 +4167,10 @@ static bool is_flex_arr(const struct btf *btf,
* - array element #3 access (corresponds to '3' in low-level spec). * - array element #3 access (corresponds to '3' in low-level spec).
* *
*/ */
static int bpf_core_spec_parse(const struct btf *btf, static int bpf_core_parse_spec(const struct btf *btf,
__u32 type_id, __u32 type_id,
const char *spec_str, const char *spec_str,
enum bpf_core_relo_kind relo_kind,
struct bpf_core_spec *spec) struct bpf_core_spec *spec)
{ {
int access_idx, parsed_len, i; int access_idx, parsed_len, i;
...@@ -4152,6 +4185,8 @@ static int bpf_core_spec_parse(const struct btf *btf, ...@@ -4152,6 +4185,8 @@ static int bpf_core_spec_parse(const struct btf *btf,
memset(spec, 0, sizeof(*spec)); memset(spec, 0, sizeof(*spec));
spec->btf = btf; spec->btf = btf;
spec->root_type_id = type_id;
spec->relo_kind = relo_kind;
/* parse spec_str="0:1:2:3:4" into array raw_spec=[0, 1, 2, 3, 4] */ /* parse spec_str="0:1:2:3:4" into array raw_spec=[0, 1, 2, 3, 4] */
while (*spec_str) { while (*spec_str) {
...@@ -4178,6 +4213,9 @@ static int bpf_core_spec_parse(const struct btf *btf, ...@@ -4178,6 +4213,9 @@ static int bpf_core_spec_parse(const struct btf *btf,
spec->spec[0].idx = access_idx; spec->spec[0].idx = access_idx;
spec->len++; spec->len++;
if (!core_relo_is_field_based(relo_kind))
return -EINVAL;
sz = btf__resolve_size(btf, id); sz = btf__resolve_size(btf, id);
if (sz < 0) if (sz < 0)
return sz; return sz;
...@@ -4285,17 +4323,17 @@ static struct ids_vec *bpf_core_find_cands(const struct btf *local_btf, ...@@ -4285,17 +4323,17 @@ static struct ids_vec *bpf_core_find_cands(const struct btf *local_btf,
const struct btf *targ_btf) const struct btf *targ_btf)
{ {
size_t local_essent_len, targ_essent_len; size_t local_essent_len, targ_essent_len;
const char *local_name, *targ_name; const char *local_name, *targ_name, *targ_kind;
const struct btf_type *t; const struct btf_type *t, *local_t;
struct ids_vec *cand_ids; struct ids_vec *cand_ids;
__u32 *new_ids; __u32 *new_ids;
int i, err, n; int i, err, n;
t = btf__type_by_id(local_btf, local_type_id); local_t = btf__type_by_id(local_btf, local_type_id);
if (!t) if (!local_t)
return ERR_PTR(-EINVAL); return ERR_PTR(-EINVAL);
local_name = btf__name_by_offset(local_btf, t->name_off); local_name = btf__name_by_offset(local_btf, local_t->name_off);
if (str_is_empty(local_name)) if (str_is_empty(local_name))
return ERR_PTR(-EINVAL); return ERR_PTR(-EINVAL);
local_essent_len = bpf_core_essential_name_len(local_name); local_essent_len = bpf_core_essential_name_len(local_name);
...@@ -4310,6 +4348,7 @@ static struct ids_vec *bpf_core_find_cands(const struct btf *local_btf, ...@@ -4310,6 +4348,7 @@ static struct ids_vec *bpf_core_find_cands(const struct btf *local_btf,
targ_name = btf__name_by_offset(targ_btf, t->name_off); targ_name = btf__name_by_offset(targ_btf, t->name_off);
if (str_is_empty(targ_name)) if (str_is_empty(targ_name))
continue; continue;
targ_kind = btf_kind_str(t);
t = skip_mods_and_typedefs(targ_btf, i, NULL); t = skip_mods_and_typedefs(targ_btf, i, NULL);
if (!btf_is_composite(t) && !btf_is_array(t)) if (!btf_is_composite(t) && !btf_is_array(t))
...@@ -4320,8 +4359,9 @@ static struct ids_vec *bpf_core_find_cands(const struct btf *local_btf, ...@@ -4320,8 +4359,9 @@ static struct ids_vec *bpf_core_find_cands(const struct btf *local_btf,
continue; continue;
if (strncmp(local_name, targ_name, local_essent_len) == 0) { if (strncmp(local_name, targ_name, local_essent_len) == 0) {
pr_debug("[%d] %s: found candidate [%d] %s\n", pr_debug("CO-RE relocating [%d] %s %s: found target candidate [%d] %s %s\n",
local_type_id, local_name, i, targ_name); local_type_id, btf_kind_str(local_t),
local_name, i, targ_kind, targ_name);
new_ids = reallocarray(cand_ids->data, new_ids = reallocarray(cand_ids->data,
cand_ids->len + 1, cand_ids->len + 1,
sizeof(*cand_ids->data)); sizeof(*cand_ids->data));
...@@ -4510,6 +4550,8 @@ static int bpf_core_spec_match(struct bpf_core_spec *local_spec, ...@@ -4510,6 +4550,8 @@ static int bpf_core_spec_match(struct bpf_core_spec *local_spec,
memset(targ_spec, 0, sizeof(*targ_spec)); memset(targ_spec, 0, sizeof(*targ_spec));
targ_spec->btf = targ_btf; targ_spec->btf = targ_btf;
targ_spec->root_type_id = targ_id;
targ_spec->relo_kind = local_spec->relo_kind;
local_acc = &local_spec->spec[0]; local_acc = &local_spec->spec[0];
targ_acc = &targ_spec->spec[0]; targ_acc = &targ_spec->spec[0];
...@@ -4570,7 +4612,7 @@ static int bpf_core_spec_match(struct bpf_core_spec *local_spec, ...@@ -4570,7 +4612,7 @@ static int bpf_core_spec_match(struct bpf_core_spec *local_spec,
} }
static int bpf_core_calc_field_relo(const struct bpf_program *prog, static int bpf_core_calc_field_relo(const struct bpf_program *prog,
const struct bpf_field_reloc *relo, const struct bpf_core_relo *relo,
const struct bpf_core_spec *spec, const struct bpf_core_spec *spec,
__u32 *val, bool *validate) __u32 *val, bool *validate)
{ {
...@@ -4691,7 +4733,7 @@ static int bpf_core_calc_field_relo(const struct bpf_program *prog, ...@@ -4691,7 +4733,7 @@ static int bpf_core_calc_field_relo(const struct bpf_program *prog,
* 2. rX += <imm> (arithmetic operations with immediate operand); * 2. rX += <imm> (arithmetic operations with immediate operand);
*/ */
static int bpf_core_reloc_insn(struct bpf_program *prog, static int bpf_core_reloc_insn(struct bpf_program *prog,
const struct bpf_field_reloc *relo, const struct bpf_core_relo *relo,
int relo_idx, int relo_idx,
const struct bpf_core_spec *local_spec, const struct bpf_core_spec *local_spec,
const struct bpf_core_spec *targ_spec) const struct bpf_core_spec *targ_spec)
...@@ -4795,25 +4837,30 @@ static void bpf_core_dump_spec(int level, const struct bpf_core_spec *spec) ...@@ -4795,25 +4837,30 @@ static void bpf_core_dump_spec(int level, const struct bpf_core_spec *spec)
__u32 type_id; __u32 type_id;
int i; int i;
type_id = spec->spec[0].type_id; type_id = spec->root_type_id;
t = btf__type_by_id(spec->btf, type_id); t = btf__type_by_id(spec->btf, type_id);
s = btf__name_by_offset(spec->btf, t->name_off); s = btf__name_by_offset(spec->btf, t->name_off);
libbpf_print(level, "[%u] %s + ", type_id, s);
for (i = 0; i < spec->raw_len; i++) libbpf_print(level, "[%u] %s %s", type_id, btf_kind_str(t), str_is_empty(s) ? "<anon>" : s);
libbpf_print(level, "%d%s", spec->raw_spec[i],
i == spec->raw_len - 1 ? " => " : ":");
libbpf_print(level, "%u.%u @ &x",
spec->bit_offset / 8, spec->bit_offset % 8);
if (core_relo_is_field_based(spec->relo_kind)) {
for (i = 0; i < spec->len; i++) { for (i = 0; i < spec->len; i++) {
if (spec->spec[i].name) if (spec->spec[i].name)
libbpf_print(level, ".%s", spec->spec[i].name); libbpf_print(level, ".%s", spec->spec[i].name);
else else if (i > 0 || spec->spec[i].idx > 0)
libbpf_print(level, "[%u]", spec->spec[i].idx); libbpf_print(level, "[%u]", spec->spec[i].idx);
} }
libbpf_print(level, " (");
for (i = 0; i < spec->raw_len; i++)
libbpf_print(level, "%s%d", i == 0 ? "" : ":", spec->raw_spec[i]);
if (spec->bit_offset % 8)
libbpf_print(level, " @ offset %u.%u)",
spec->bit_offset / 8, spec->bit_offset % 8);
else
libbpf_print(level, " @ offset %u)", spec->bit_offset / 8);
}
} }
static size_t bpf_core_hash_fn(const void *key, void *ctx) static size_t bpf_core_hash_fn(const void *key, void *ctx)
...@@ -4877,12 +4924,12 @@ static void *u32_as_hash_key(__u32 x) ...@@ -4877,12 +4924,12 @@ static void *u32_as_hash_key(__u32 x)
* CPU-wise compared to prebuilding a map from all local type names to * CPU-wise compared to prebuilding a map from all local type names to
* a list of candidate type names. It's also sped up by caching resolved * a list of candidate type names. It's also sped up by caching resolved
* list of matching candidates per each local "root" type ID, that has at * list of matching candidates per each local "root" type ID, that has at
* least one bpf_field_reloc associated with it. This list is shared * least one bpf_core_relo associated with it. This list is shared
* between multiple relocations for the same type ID and is updated as some * between multiple relocations for the same type ID and is updated as some
* of the candidates are pruned due to structural incompatibility. * of the candidates are pruned due to structural incompatibility.
*/ */
static int bpf_core_reloc_field(struct bpf_program *prog, static int bpf_core_reloc_field(struct bpf_program *prog,
const struct bpf_field_reloc *relo, const struct bpf_core_relo *relo,
int relo_idx, int relo_idx,
const struct btf *local_btf, const struct btf *local_btf,
const struct btf *targ_btf, const struct btf *targ_btf,
...@@ -4891,8 +4938,8 @@ static int bpf_core_reloc_field(struct bpf_program *prog, ...@@ -4891,8 +4938,8 @@ static int bpf_core_reloc_field(struct bpf_program *prog,
const char *prog_name = bpf_program__title(prog, false); const char *prog_name = bpf_program__title(prog, false);
struct bpf_core_spec local_spec, cand_spec, targ_spec; struct bpf_core_spec local_spec, cand_spec, targ_spec;
const void *type_key = u32_as_hash_key(relo->type_id); const void *type_key = u32_as_hash_key(relo->type_id);
const struct btf_type *local_type, *cand_type; const struct btf_type *local_type;
const char *local_name, *cand_name; const char *local_name;
struct ids_vec *cand_ids; struct ids_vec *cand_ids;
__u32 local_id, cand_id; __u32 local_id, cand_id;
const char *spec_str; const char *spec_str;
...@@ -4911,24 +4958,24 @@ static int bpf_core_reloc_field(struct bpf_program *prog, ...@@ -4911,24 +4958,24 @@ static int bpf_core_reloc_field(struct bpf_program *prog,
if (str_is_empty(spec_str)) if (str_is_empty(spec_str))
return -EINVAL; return -EINVAL;
err = bpf_core_spec_parse(local_btf, local_id, spec_str, &local_spec); err = bpf_core_parse_spec(local_btf, local_id, spec_str, relo->kind, &local_spec);
if (err) { if (err) {
pr_warn("prog '%s': relo #%d: parsing [%d] %s + %s failed: %d\n", pr_warn("prog '%s': relo #%d: parsing [%d] %s %s + %s failed: %d\n",
prog_name, relo_idx, local_id, local_name, spec_str, prog_name, relo_idx, local_id, btf_kind_str(local_type),
err); local_name, spec_str, err);
return -EINVAL; return -EINVAL;
} }
pr_debug("prog '%s': relo #%d: kind %d, spec is ", prog_name, relo_idx, pr_debug("prog '%s': relo #%d: kind <%s> (%d), spec is ", prog_name,
relo->kind); relo_idx, core_relo_kind_str(relo->kind), relo->kind);
bpf_core_dump_spec(LIBBPF_DEBUG, &local_spec); bpf_core_dump_spec(LIBBPF_DEBUG, &local_spec);
libbpf_print(LIBBPF_DEBUG, "\n"); libbpf_print(LIBBPF_DEBUG, "\n");
if (!hashmap__find(cand_cache, type_key, (void **)&cand_ids)) { if (!hashmap__find(cand_cache, type_key, (void **)&cand_ids)) {
cand_ids = bpf_core_find_cands(local_btf, local_id, targ_btf); cand_ids = bpf_core_find_cands(local_btf, local_id, targ_btf);
if (IS_ERR(cand_ids)) { if (IS_ERR(cand_ids)) {
pr_warn("prog '%s': relo #%d: target candidate search failed for [%d] %s: %ld", pr_warn("prog '%s': relo #%d: target candidate search failed for [%d] %s %s: %ld",
prog_name, relo_idx, local_id, local_name, prog_name, relo_idx, local_id, btf_kind_str(local_type), local_name,
PTR_ERR(cand_ids)); PTR_ERR(cand_ids));
return PTR_ERR(cand_ids); return PTR_ERR(cand_ids);
} }
...@@ -4941,20 +4988,20 @@ static int bpf_core_reloc_field(struct bpf_program *prog, ...@@ -4941,20 +4988,20 @@ static int bpf_core_reloc_field(struct bpf_program *prog,
for (i = 0, j = 0; i < cand_ids->len; i++) { for (i = 0, j = 0; i < cand_ids->len; i++) {
cand_id = cand_ids->data[i]; cand_id = cand_ids->data[i];
cand_type = btf__type_by_id(targ_btf, cand_id); err = bpf_core_spec_match(&local_spec, targ_btf, cand_id, &cand_spec);
cand_name = btf__name_by_offset(targ_btf, cand_type->name_off);
err = bpf_core_spec_match(&local_spec, targ_btf,
cand_id, &cand_spec);
pr_debug("prog '%s': relo #%d: matching candidate #%d %s against spec ",
prog_name, relo_idx, i, cand_name);
bpf_core_dump_spec(LIBBPF_DEBUG, &cand_spec);
libbpf_print(LIBBPF_DEBUG, ": %d\n", err);
if (err < 0) { if (err < 0) {
pr_warn("prog '%s': relo #%d: matching error: %d\n", pr_warn("prog '%s': relo #%d: error matching candidate #%d ",
prog_name, relo_idx, err); prog_name, relo_idx, i);
bpf_core_dump_spec(LIBBPF_WARN, &cand_spec);
libbpf_print(LIBBPF_WARN, ": %d\n", err);
return err; return err;
} }
pr_debug("prog '%s': relo #%d: %s candidate #%d ", prog_name,
relo_idx, err == 0 ? "non-matching" : "matching", i);
bpf_core_dump_spec(LIBBPF_DEBUG, &cand_spec);
libbpf_print(LIBBPF_DEBUG, "\n");
if (err == 0) if (err == 0)
continue; continue;
...@@ -4996,8 +5043,8 @@ static int bpf_core_reloc_field(struct bpf_program *prog, ...@@ -4996,8 +5043,8 @@ static int bpf_core_reloc_field(struct bpf_program *prog,
* to a specific instruction number in its log. * to a specific instruction number in its log.
*/ */
if (j == 0) if (j == 0)
pr_debug("prog '%s': relo #%d: no matching targets found for [%d] %s + %s\n", pr_debug("prog '%s': relo #%d: no matching targets found\n",
prog_name, relo_idx, local_id, local_name, spec_str); prog_name, relo_idx);
/* bpf_core_reloc_insn should know how to handle missing targ_spec */ /* bpf_core_reloc_insn should know how to handle missing targ_spec */
err = bpf_core_reloc_insn(prog, relo, relo_idx, &local_spec, err = bpf_core_reloc_insn(prog, relo, relo_idx, &local_spec,
...@@ -5012,10 +5059,10 @@ static int bpf_core_reloc_field(struct bpf_program *prog, ...@@ -5012,10 +5059,10 @@ static int bpf_core_reloc_field(struct bpf_program *prog,
} }
static int static int
bpf_core_reloc_fields(struct bpf_object *obj, const char *targ_btf_path) bpf_object__relocate_core(struct bpf_object *obj, const char *targ_btf_path)
{ {
const struct btf_ext_info_sec *sec; const struct btf_ext_info_sec *sec;
const struct bpf_field_reloc *rec; const struct bpf_core_relo *rec;
const struct btf_ext_info *seg; const struct btf_ext_info *seg;
struct hashmap_entry *entry; struct hashmap_entry *entry;
struct hashmap *cand_cache = NULL; struct hashmap *cand_cache = NULL;
...@@ -5024,6 +5071,9 @@ bpf_core_reloc_fields(struct bpf_object *obj, const char *targ_btf_path) ...@@ -5024,6 +5071,9 @@ bpf_core_reloc_fields(struct bpf_object *obj, const char *targ_btf_path)
const char *sec_name; const char *sec_name;
int i, err = 0; int i, err = 0;
if (obj->btf_ext->core_relo_info.len == 0)
return 0;
if (targ_btf_path) if (targ_btf_path)
targ_btf = btf__parse_elf(targ_btf_path, NULL); targ_btf = btf__parse_elf(targ_btf_path, NULL);
else else
...@@ -5039,7 +5089,7 @@ bpf_core_reloc_fields(struct bpf_object *obj, const char *targ_btf_path) ...@@ -5039,7 +5089,7 @@ bpf_core_reloc_fields(struct bpf_object *obj, const char *targ_btf_path)
goto out; goto out;
} }
seg = &obj->btf_ext->field_reloc_info; seg = &obj->btf_ext->core_relo_info;
for_each_btf_ext_sec(seg, sec) { for_each_btf_ext_sec(seg, sec) {
sec_name = btf__name_by_offset(obj->btf, sec->sec_name_off); sec_name = btf__name_by_offset(obj->btf, sec->sec_name_off);
if (str_is_empty(sec_name)) { if (str_is_empty(sec_name)) {
...@@ -5087,17 +5137,6 @@ bpf_core_reloc_fields(struct bpf_object *obj, const char *targ_btf_path) ...@@ -5087,17 +5137,6 @@ bpf_core_reloc_fields(struct bpf_object *obj, const char *targ_btf_path)
return err; return err;
} }
static int
bpf_object__relocate_core(struct bpf_object *obj, const char *targ_btf_path)
{
int err = 0;
if (obj->btf_ext->field_reloc_info.len)
err = bpf_core_reloc_fields(obj, targ_btf_path);
return err;
}
static int static int
bpf_program__reloc_text(struct bpf_program *prog, struct bpf_object *obj, bpf_program__reloc_text(struct bpf_program *prog, struct bpf_object *obj,
struct reloc_desc *relo) struct reloc_desc *relo)
......
...@@ -138,6 +138,44 @@ struct btf_ext_info { ...@@ -138,6 +138,44 @@ struct btf_ext_info {
i < (sec)->num_info; \ i < (sec)->num_info; \
i++, rec = (void *)rec + (seg)->rec_size) i++, rec = (void *)rec + (seg)->rec_size)
/*
* The .BTF.ext ELF section layout defined as
* struct btf_ext_header
* func_info subsection
*
* The func_info subsection layout:
* record size for struct bpf_func_info in the func_info subsection
* struct btf_sec_func_info for section #1
* a list of bpf_func_info records for section #1
* where struct bpf_func_info mimics one in include/uapi/linux/bpf.h
* but may not be identical
* struct btf_sec_func_info for section #2
* a list of bpf_func_info records for section #2
* ......
*
* Note that the bpf_func_info record size in .BTF.ext may not
* be the same as the one defined in include/uapi/linux/bpf.h.
* The loader should ensure that record_size meets minimum
* requirement and pass the record as is to the kernel. The
* kernel will handle the func_info properly based on its contents.
*/
struct btf_ext_header {
__u16 magic;
__u8 version;
__u8 flags;
__u32 hdr_len;
/* All offsets are in bytes relative to the end of this header */
__u32 func_info_off;
__u32 func_info_len;
__u32 line_info_off;
__u32 line_info_len;
/* optional part of .BTF.ext header */
__u32 core_relo_off;
__u32 core_relo_len;
};
struct btf_ext { struct btf_ext {
union { union {
struct btf_ext_header *hdr; struct btf_ext_header *hdr;
...@@ -145,7 +183,7 @@ struct btf_ext { ...@@ -145,7 +183,7 @@ struct btf_ext {
}; };
struct btf_ext_info func_info; struct btf_ext_info func_info;
struct btf_ext_info line_info; struct btf_ext_info line_info;
struct btf_ext_info field_reloc_info; struct btf_ext_info core_relo_info;
__u32 data_size; __u32 data_size;
}; };
...@@ -170,32 +208,34 @@ struct bpf_line_info_min { ...@@ -170,32 +208,34 @@ struct bpf_line_info_min {
__u32 line_col; __u32 line_col;
}; };
/* bpf_field_info_kind encodes which aspect of captured field has to be /* bpf_core_relo_kind encodes which aspect of captured field/type/enum value
* adjusted by relocations. Currently supported values are: * has to be adjusted by relocations.
* - BPF_FIELD_BYTE_OFFSET: field offset (in bytes);
* - BPF_FIELD_EXISTS: field existence (1, if field exists; 0, otherwise);
*/ */
enum bpf_field_info_kind { enum bpf_core_relo_kind {
BPF_FIELD_BYTE_OFFSET = 0, /* field byte offset */ BPF_FIELD_BYTE_OFFSET = 0, /* field byte offset */
BPF_FIELD_BYTE_SIZE = 1, BPF_FIELD_BYTE_SIZE = 1, /* field size in bytes */
BPF_FIELD_EXISTS = 2, /* field existence in target kernel */ BPF_FIELD_EXISTS = 2, /* field existence in target kernel */
BPF_FIELD_SIGNED = 3, BPF_FIELD_SIGNED = 3, /* field signedness (0 - unsigned, 1 - signed) */
BPF_FIELD_LSHIFT_U64 = 4, BPF_FIELD_LSHIFT_U64 = 4, /* bitfield-specific left bitshift */
BPF_FIELD_RSHIFT_U64 = 5, BPF_FIELD_RSHIFT_U64 = 5, /* bitfield-specific right bitshift */
}; };
/* The minimum bpf_field_reloc checked by the loader /* The minimum bpf_core_relo checked by the loader
* *
* Field relocation captures the following data: * CO-RE relocation captures the following data:
* - insn_off - instruction offset (in bytes) within a BPF program that needs * - insn_off - instruction offset (in bytes) within a BPF program that needs
* its insn->imm field to be relocated with actual field info; * its insn->imm field to be relocated with actual field info;
* - type_id - BTF type ID of the "root" (containing) entity of a relocatable * - type_id - BTF type ID of the "root" (containing) entity of a relocatable
* field; * type or field;
* - access_str_off - offset into corresponding .BTF string section. String * - access_str_off - offset into corresponding .BTF string section. String
* itself encodes an accessed field using a sequence of field and array * interpretation depends on specific relocation kind:
* indicies, separated by colon (:). It's conceptually very close to LLVM's * - for field-based relocations, string encodes an accessed field using
* getelementptr ([0]) instruction's arguments for identifying offset to * a sequence of field and array indices, separated by colon (:). It's
* a field. * conceptually very close to LLVM's getelementptr ([0]) instruction's
* arguments for identifying offset to a field.
* - for type-based relocations, strings is expected to be just "0";
* - for enum value-based relocations, string contains an index of enum
* value within its enum type;
* *
* Example to provide a better feel. * Example to provide a better feel.
* *
...@@ -226,11 +266,11 @@ enum bpf_field_info_kind { ...@@ -226,11 +266,11 @@ enum bpf_field_info_kind {
* *
* [0] https://llvm.org/docs/LangRef.html#getelementptr-instruction * [0] https://llvm.org/docs/LangRef.html#getelementptr-instruction
*/ */
struct bpf_field_reloc { struct bpf_core_relo {
__u32 insn_off; __u32 insn_off;
__u32 type_id; __u32 type_id;
__u32 access_str_off; __u32 access_str_off;
enum bpf_field_info_kind kind; enum bpf_core_relo_kind kind;
}; };
#endif /* __LIBBPF_LIBBPF_INTERNAL_H */ #endif /* __LIBBPF_LIBBPF_INTERNAL_H */
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