]> www.infradead.org Git - users/dwmw2/linux.git/commitdiff
bpf: Resolve fext program type when checking map compatibility
authorToke Høiland-Jørgensen <toke@redhat.com>
Wed, 14 Dec 2022 23:02:53 +0000 (00:02 +0100)
committerMartin KaFai Lau <martin.lau@kernel.org>
Thu, 15 Dec 2022 05:30:27 +0000 (21:30 -0800)
The bpf_prog_map_compatible() check makes sure that BPF program types are
not mixed inside BPF map types that can contain programs (tail call maps,
cpumaps and devmaps). It does this by setting the fields of the map->owner
struct to the values of the first program being checked against, and
rejecting any subsequent programs if the values don't match.

One of the values being set in the map owner struct is the program type,
and since the code did not resolve the prog type for fext programs, the map
owner type would be set to PROG_TYPE_EXT and subsequent loading of programs
of the target type into the map would fail.

This bug is seen in particular for XDP programs that are loaded as
PROG_TYPE_EXT using libxdp; these cannot insert programs into devmaps and
cpumaps because the check fails as described above.

Fix the bug by resolving the fext program type to its target program type
as elsewhere in the verifier.

v3:
- Add Yonghong's ACK

Fixes: f45d5b6ce2e8 ("bpf: generalise tail call map compatibility check")
Acked-by: Yonghong Song <yhs@fb.com>
Signed-off-by: Toke Høiland-Jørgensen <toke@redhat.com>
Link: https://lore.kernel.org/r/20221214230254.790066-1-toke@redhat.com
Signed-off-by: Martin KaFai Lau <martin.lau@kernel.org>
kernel/bpf/core.c

index 7f98dec6e90fbb543c61a75ffef7b4a65f4dd656..b334f4ddc4d592dd229e8aeb4024984df5531521 100644 (file)
@@ -2092,6 +2092,7 @@ static unsigned int __bpf_prog_ret0_warn(const void *ctx,
 bool bpf_prog_map_compatible(struct bpf_map *map,
                             const struct bpf_prog *fp)
 {
+       enum bpf_prog_type prog_type = resolve_prog_type(fp);
        bool ret;
 
        if (fp->kprobe_override)
@@ -2102,12 +2103,12 @@ bool bpf_prog_map_compatible(struct bpf_map *map,
                /* There's no owner yet where we could check for
                 * compatibility.
                 */
-               map->owner.type  = fp->type;
+               map->owner.type  = prog_type;
                map->owner.jited = fp->jited;
                map->owner.xdp_has_frags = fp->aux->xdp_has_frags;
                ret = true;
        } else {
-               ret = map->owner.type  == fp->type &&
+               ret = map->owner.type  == prog_type &&
                      map->owner.jited == fp->jited &&
                      map->owner.xdp_has_frags == fp->aux->xdp_has_frags;
        }