]> git.baikalelectronics.ru Git - kernel.git/commit
bpf: Treat bpf_sk_lookup remote_port as a 2-byte field
authorJakub Sitnicki <jakub@cloudflare.com>
Sat, 19 Mar 2022 18:33:54 +0000 (19:33 +0100)
committerAlexei Starovoitov <ast@kernel.org>
Mon, 21 Mar 2022 01:58:59 +0000 (18:58 -0700)
commit70ca83d8b45ab73ed0ff202e84d6cfd1820b7db1
tree5920610b18ccf22f189e77624879323eeb32b726
parent1fe4edf08ee3ec8cea64adb62ccc25f6d45edd90
bpf: Treat bpf_sk_lookup remote_port as a 2-byte field

In commit 9f98f541ae0b ("bpf: Make remote_port field in struct
bpf_sk_lookup 16-bit wide") the remote_port field has been split up and
re-declared from u32 to be16.

However, the accompanying changes to the context access converter have not
been well thought through when it comes big-endian platforms.

Today 2-byte wide loads from offsetof(struct bpf_sk_lookup, remote_port)
are handled as narrow loads from a 4-byte wide field.

This by itself is not enough to create a problem, but when we combine

 1. 32-bit wide access to ->remote_port backed by a 16-wide wide load, with
 2. inherent difference between litte- and big-endian in how narrow loads
    need have to be handled (see bpf_ctx_narrow_access_offset),

we get inconsistent results for a 2-byte loads from &ctx->remote_port on LE
and BE architectures. This in turn makes BPF C code for the common case of
2-byte load from ctx->remote_port not portable.

To rectify it, inform the context access converter that remote_port is
2-byte wide field, and only 1-byte loads need to be treated as narrow
loads.

At the same time, we special-case the 4-byte load from &ctx->remote_port to
continue handling it the same way as do today, in order to keep the
existing BPF programs working.

Fixes: 9f98f541ae0b ("bpf: Make remote_port field in struct bpf_sk_lookup 16-bit wide")
Signed-off-by: Jakub Sitnicki <jakub@cloudflare.com>
Signed-off-by: Alexei Starovoitov <ast@kernel.org>
Acked-by: Martin KaFai Lau <kafai@fb.com>
Link: https://lore.kernel.org/bpf/20220319183356.233666-2-jakub@cloudflare.com
net/core/filter.c