libbpf: fix offsetof() and container_of() to work with CO-RE

[ Upstream commit bdeeed3498 ]

It seems like __builtin_offset() doesn't preserve CO-RE field
relocations properly. So if offsetof() macro is defined through
__builtin_offset(), CO-RE-enabled BPF code using container_of() will be
subtly and silently broken.

To avoid this problem, redefine offsetof() and container_of() in the
form that works with CO-RE relocations more reliably.

Fixes: 5fbc220862 ("tools/libpf: Add offsetof/container_of macro in bpf_helpers.h")
Reported-by: Lennart Poettering <lennart@poettering.net>
Signed-off-by: Andrii Nakryiko <andrii@kernel.org>
Acked-by: Yonghong Song <yhs@fb.com>
Link: https://lore.kernel.org/r/20230509065502.2306180-1-andrii@kernel.org
Signed-off-by: Alexei Starovoitov <ast@kernel.org>
Signed-off-by: Sasha Levin <sashal@kernel.org>
This commit is contained in:
Andrii Nakryiko 2023-05-08 23:55:02 -07:00 committed by Greg Kroah-Hartman
parent 3e7ee33b95
commit 71754ee427

View File

@ -72,16 +72,21 @@
/* /*
* Helper macros to manipulate data structures * Helper macros to manipulate data structures
*/ */
#ifndef offsetof
#define offsetof(TYPE, MEMBER) ((unsigned long)&((TYPE *)0)->MEMBER) /* offsetof() definition that uses __builtin_offset() might not preserve field
#endif * offset CO-RE relocation properly, so force-redefine offsetof() using
#ifndef container_of * old-school approach which works with CO-RE correctly
*/
#undef offsetof
#define offsetof(type, member) ((unsigned long)&((type *)0)->member)
/* redefined container_of() to ensure we use the above offsetof() macro */
#undef container_of
#define container_of(ptr, type, member) \ #define container_of(ptr, type, member) \
({ \ ({ \
void *__mptr = (void *)(ptr); \ void *__mptr = (void *)(ptr); \
((type *)(__mptr - offsetof(type, member))); \ ((type *)(__mptr - offsetof(type, member))); \
}) })
#endif
/* /*
* Helper macro to throw a compilation error if __bpf_unreachable() gets * Helper macro to throw a compilation error if __bpf_unreachable() gets