On Thu, 2025-01-02 at 19:24 +0100, Jose E. Marchesi wrote:

[...]

> IMO the BPP selftest (and BPF programs in general) must not include host
> glibc headers at all, regardless of what BPF compiler is used.  The
> glibc headers installed in the host are tailored to some particular
> architecture, be it x86_64 or whatever, not necessarily compatible with
> what the compilers assume for the BPF target.
>
> This particular case shows the problem well: all the glibc headers
> included by that BPF selftest assume that `long' is 32 bits, not 64
> bits, because x86_64 is not defined.  This conflicts with both clang's
> and GCC's assumption that in BPF a `long' is 64 bits.  This may or may
> not be a problem, depending on whether the BPF program uses the stuff
> defined in the headers and how it uses it.  Had you be using an arm or
> sparc host instead of x86_64, you may be including macros and stuff that
> assume chars are unsigned.  But chars are signed in bpf.

This makes sense, but might cause some friction.
The following glibc headers are included directly from selftests:
- errno.h
- features.h
- inttypes.h
- limits.h
- netinet/in.h
- netinet/udp.h
- sched.h
- stdint.h
- stdlib.h
- string.h
- sys/socket.h
- sys/types.h
- time.h
- unistd.h

However, removing includes for these headers does not help the test in
question, because some linux UAPI headers include libc headers when exported:

    In file included from /usr/include/netinet/udp.h:51,
                     from progs/test_cls_redirect_dynptr.c:20:
    
/home/eddy/work/tmp/gccbpf/lib/gcc/bpf-unknown-none/15.0.0/include/stdint.h:43:24:
 error: conflicting types for ‘int64_t’; have ‘long int’
       43 | typedef __INT64_TYPE__ int64_t;
          |                        ^~~~~~~
    In file included from /usr/include/sys/types.h:155,
                     from /usr/include/bits/socket.h:29,
                     from /usr/include/sys/socket.h:33,
                     from /usr/include/linux/if.h:28,
                     from /usr/include/linux/icmp.h:23,
                     from progs/test_cls_redirect_dynptr.c:12:
    /usr/include/bits/stdint-intn.h:27:19: note: previous declaration of 
‘int64_t’ with type ‘int64_t’ {aka ‘long long int’}
       27 | typedef __int64_t int64_t;
          |                   ^~~~~~~

On my system (Fedora 41) the linux/{icmp,if}.h UAPI headers are
provided by kernel-headers package, sys/socket.h is provided by
glibc-devel package.

The UAPI headers have two modes depending whether __KERNEL__ is
defined. When used during kernel build the __KERNEL__ is defined and
there are no outside references. When exported for packages like
kernel-headers (via 'make headers' target) the __KERNEL__ is not
defined and there are some references to libc includes
(in fact, references to '#ifdef __KERNEL__' blocks are cut out during
 headers export).

E.g. here is a fragment of linux/if.h, when viewed from kernel source:

    #ifndef _LINUX_IF_H
    #define _LINUX_IF_H

    #include <linux/libc-compat.h>          /* for compatibility with glibc */
    #include <linux/types.h>            /* for "__kernel_caddr_t" et al */
    #include <linux/socket.h>           /* for "struct sockaddr" et al  */
    #include <linux/compiler.h>         /* for "__user" et al           */

    #ifndef __KERNEL__
    #include <sys/socket.h>                     /* for struct sockaddr.         
*/
    #endif

And here is the same fragment as part of the kernel-headers package
(/usr/include/linux/if.h):

    #ifndef _LINUX_IF_H
    #define _LINUX_IF_H

    #include <linux/libc-compat.h>          /* for compatibility with glibc */
    #include <linux/types.h>            /* for "__kernel_caddr_t" et al */
    #include <linux/socket.h>           /* for "struct sockaddr" et al  */
                /* for "__user" et al           */

    #include <sys/socket.h>                     /* for struct sockaddr.         
*/

As far as I understand, the idea right now is that BPF users can
install the kernel-headers package (or its equivalent) and start
hacking. If we declare that this is no longer a blessed way, people
would need to switch to packages like kernel-devel that provide full
set of kernel headers for use with dkms etc, e.g. for my system the
if.h would be located here:
/usr/src/kernels/6.12.6-200.fc41.x86_64/include/uapi/linux/if.h .

To me this seems logical, however potentially such change might have
implications for existing BPF code-base.

Reply via email to