Skip to content

Commit 0cc114d

Browse files
D-Wytheanakryiko
authored andcommitted
libbpf: Fix error when st-prefix_ops and ops from differ btf
When a module registers a struct_ops, the struct_ops type and its corresponding map_value type ("bpf_struct_ops_") may reside in different btf objects, here are four possible case: +--------+---------------+-------------+---------------------------------+ | |bpf_struct_ops_| xxx_ops | | +--------+---------------+-------------+---------------------------------+ | case 0 | btf_vmlinux | btf_vmlinux | be used and reg only in vmlinux | +--------+---------------+-------------+---------------------------------+ | case 1 | btf_vmlinux | mod_btf | INVALID | +--------+---------------+-------------+---------------------------------+ | case 2 | mod_btf | btf_vmlinux | reg in mod but be used both in | | | | | vmlinux and mod. | +--------+---------------+-------------+---------------------------------+ | case 3 | mod_btf | mod_btf | be used and reg only in mod | +--------+---------------+-------------+---------------------------------+ Currently we figure out the mod_btf by searching with the struct_ops type, which makes it impossible to figure out the mod_btf when the struct_ops type is in btf_vmlinux while it's corresponding map_value type is in mod_btf (case 2). The fix is to use the corresponding map_value type ("bpf_struct_ops_") as the lookup anchor instead of the struct_ops type to figure out the `btf` and `mod_btf` via find_ksym_btf_id(), and then we can locate the kern_type_id via btf__find_by_name_kind() with the `btf` we just obtained from find_ksym_btf_id(). With this change the lookup obtains the correct btf and mod_btf for case 2, preserves correct behavior for other valid cases, and still fails as expected for the invalid scenario (case 1). Fixes: 590a008 ("bpf: libbpf: Add STRUCT_OPS support") Signed-off-by: D. Wythe <[email protected]> Signed-off-by: Andrii Nakryiko <[email protected]> Acked-by: Andrii Nakryiko <[email protected]> Acked-by: Martin KaFai Lau <[email protected]> Link: https://lore.kernel.org/bpf/[email protected]
1 parent 991e555 commit 0cc114d

File tree

1 file changed

+17
-19
lines changed

1 file changed

+17
-19
lines changed

tools/lib/bpf/libbpf.c

Lines changed: 17 additions & 19 deletions
Original file line numberDiff line numberDiff line change
@@ -1018,35 +1018,33 @@ find_struct_ops_kern_types(struct bpf_object *obj, const char *tname_raw,
10181018
const struct btf_member *kern_data_member;
10191019
struct btf *btf = NULL;
10201020
__s32 kern_vtype_id, kern_type_id;
1021-
char tname[256];
1021+
char tname[192], stname[256];
10221022
__u32 i;
10231023

10241024
snprintf(tname, sizeof(tname), "%.*s",
10251025
(int)bpf_core_essential_name_len(tname_raw), tname_raw);
10261026

1027-
kern_type_id = find_ksym_btf_id(obj, tname, BTF_KIND_STRUCT,
1028-
&btf, mod_btf);
1029-
if (kern_type_id < 0) {
1030-
pr_warn("struct_ops init_kern: struct %s is not found in kernel BTF\n",
1031-
tname);
1032-
return kern_type_id;
1033-
}
1034-
kern_type = btf__type_by_id(btf, kern_type_id);
1027+
snprintf(stname, sizeof(stname), "%s%s", STRUCT_OPS_VALUE_PREFIX, tname);
10351028

1036-
/* Find the corresponding "map_value" type that will be used
1037-
* in map_update(BPF_MAP_TYPE_STRUCT_OPS). For example,
1038-
* find "struct bpf_struct_ops_tcp_congestion_ops" from the
1039-
* btf_vmlinux.
1029+
/* Look for the corresponding "map_value" type that will be used
1030+
* in map_update(BPF_MAP_TYPE_STRUCT_OPS) first, figure out the btf
1031+
* and the mod_btf.
1032+
* For example, find "struct bpf_struct_ops_tcp_congestion_ops".
10401033
*/
1041-
kern_vtype_id = find_btf_by_prefix_kind(btf, STRUCT_OPS_VALUE_PREFIX,
1042-
tname, BTF_KIND_STRUCT);
1034+
kern_vtype_id = find_ksym_btf_id(obj, stname, BTF_KIND_STRUCT, &btf, mod_btf);
10431035
if (kern_vtype_id < 0) {
1044-
pr_warn("struct_ops init_kern: struct %s%s is not found in kernel BTF\n",
1045-
STRUCT_OPS_VALUE_PREFIX, tname);
1036+
pr_warn("struct_ops init_kern: struct %s is not found in kernel BTF\n", stname);
10461037
return kern_vtype_id;
10471038
}
10481039
kern_vtype = btf__type_by_id(btf, kern_vtype_id);
10491040

1041+
kern_type_id = btf__find_by_name_kind(btf, tname, BTF_KIND_STRUCT);
1042+
if (kern_type_id < 0) {
1043+
pr_warn("struct_ops init_kern: struct %s is not found in kernel BTF\n", tname);
1044+
return kern_type_id;
1045+
}
1046+
kern_type = btf__type_by_id(btf, kern_type_id);
1047+
10501048
/* Find "struct tcp_congestion_ops" from
10511049
* struct bpf_struct_ops_tcp_congestion_ops {
10521050
* [ ... ]
@@ -1059,8 +1057,8 @@ find_struct_ops_kern_types(struct bpf_object *obj, const char *tname_raw,
10591057
break;
10601058
}
10611059
if (i == btf_vlen(kern_vtype)) {
1062-
pr_warn("struct_ops init_kern: struct %s data is not found in struct %s%s\n",
1063-
tname, STRUCT_OPS_VALUE_PREFIX, tname);
1060+
pr_warn("struct_ops init_kern: struct %s data is not found in struct %s\n",
1061+
tname, stname);
10641062
return -EINVAL;
10651063
}
10661064

0 commit comments

Comments
 (0)