This patch prepares code before enabling the clang -target bpf.
The clang compiler does not like #include <stdint.h> when
using '-target bpf' it will fail with:
fatal error: 'gnu/stubs-32.h' file not found
This is because using clang -target bpf, then clang will have '__bpf__'
defined instead of '__x86_64__' hence the gnu/stubs-32.h include
attempt as /usr/include/gnu/stubs.h contains, on x86_64:
#if !defined __x86_64__
# include <gnu/stubs-32.h>
#endif
#if defined __x86_64__ && defined __LP64__
# include <gnu/stubs-64.h>
#endif
#if defined __x86_64__ && defined __ILP32__
# include <gnu/stubs-x32.h>
#endif
This can be worked around by installing the 32-bit version of
glibc-devel.i686 on your distribution.
But the BPF programs does not really need to include stdint.h,
if converting:
uint64_t -> __u64
uint32_t -> __u32
uint16_t -> __u16
uint8_t -> __u8
This patch does this type syntax conversion.
The build of a ebpf files had an issue for system like Debian
because they don't have a asm/types.h in the include path if the
architecture is not defined which is the case due to target bpf.
This results in:
clang-5.0 -Wall -Iinclude -O2 \
-D__KERNEL__ -D__ASM_SYSREG_H \
-target bpf -S -emit-llvm vlan_filter.c -o vlan_filter.ll
In file included from vlan_filter.c:19:
In file included from include/linux/bpf.h:11:
/usr/include/linux/types.h:5:10: fatal error: 'asm/types.h' file not
found
#include <asm/types.h>
^~~~~~~~~~~~~
1 error generated.
Makefile:523: recipe for target 'vlan_filter.bpf' failed
This patch fixes the issue by adding a include path setting the
architecture to the one of the builder.
Signed-off-by: Jesper Dangaard Brouer <netoptimizer@brouer.com>
Sidned-off-by: Eric Leblond <eric@regit.org>
all: lb.bpf filter.bpf bypass_filter.bpf xdp_filter.bpf vlan_filter.bpf
%.bpf: %.c
- ${CC} -Wall $(BPF_CFLAGS) -O2 -D__KERNEL__ -D__ASM_SYSREG_H -emit-llvm -c $< -o - | ${LLC} -march=bpf -filetype=obj -o $@
+ ${CC} -Wall $(BPF_CFLAGS) -O2 -I/usr/include/$(build_cpu)-$(build_os)/ -D__KERNEL__ -D__ASM_SYSREG_H -emit-llvm -c $< -o - | ${LLC} -march=bpf -filetype=obj -o $@
CLEANFILES = *.bpf
* 02110-1301, USA.
*/
-#include <stdint.h>
#include <stddef.h>
#include <linux/bpf.h>
} __attribute__((__aligned__(8)));
struct pair {
- uint64_t time;
- uint64_t packets;
- uint64_t bytes;
+ __u64 time;
+ __u64 packets;
+ __u64 bytes;
} __attribute__((__aligned__(8)));
struct bpf_map_def SEC("maps") flow_table_v4 = {
*/
static __always_inline int ipv4_filter(struct __sk_buff *skb)
{
- uint32_t nhoff, verlen;
+ __u32 nhoff, verlen;
struct flowv4_keys tuple;
struct pair *value;
- uint16_t port;
+ __u16 port;
nhoff = skb->cb[0];
#if 0
if ((tuple.port16[0] == 22) || (tuple.port16[1] == 22))
{
- uint16_t sp = tuple.port16[0];
- //uint16_t dp = tuple.port16[1];
+ __u16 sp = tuple.port16[0];
+ //__u16 dp = tuple.port16[1];
char fmt[] = "Parsed SSH flow: %u %d -> %u\n";
bpf_trace_printk(fmt, sizeof(fmt), tuple.src, sp, tuple.dst);
}
if (value) {
#if 0
{
- uint16_t sp = tuple.port16[0];
- //uint16_t dp = tuple.port16[1];
+ __u16 sp = tuple.port16[0];
+ //__u16 dp = tuple.port16[1];
char bfmt[] = "Found flow: %u %d -> %u\n";
bpf_trace_printk(bfmt, sizeof(bfmt), tuple.src, sp, tuple.dst);
}
*/
static __always_inline int ipv6_filter(struct __sk_buff *skb)
{
- uint32_t nhoff;
- uint8_t nhdr;
+ __u32 nhoff;
+ __u8 nhdr;
struct flowv6_keys tuple;
struct pair *value;
- uint16_t port;
+ __u16 port;
nhoff = skb->cb[0];
char __license[] SEC("license") = "GPL";
-uint32_t __version SEC("version") = LINUX_VERSION_CODE;
+__u32 __version SEC("version") = LINUX_VERSION_CODE;
* 02110-1301, USA.
*/
-#include <stdint.h>
#include <stddef.h>
#include <linux/bpf.h>
char __license[] SEC("license") = "GPL";
-uint32_t __version SEC("version") = LINUX_VERSION_CODE;
+__u32 __version SEC("version") = LINUX_VERSION_CODE;
* From: http://www.azillionmonkeys.com/qed/hash.html
*/
-#define get16bits(d) (*((const uint16_t *) (d)))
+#define get16bits(d) (*((const __u16 *) (d)))
static __always_inline
-uint32_t SuperFastHash (const char *data, int len, uint32_t initval) {
- uint32_t hash = initval;
- uint32_t tmp;
+__u32 SuperFastHash (const char *data, int len, __u32 initval) {
+ __u32 hash = initval;
+ __u32 tmp;
int rem;
if (len <= 0 || data == NULL) return 0;
hash += get16bits (data);
tmp = (get16bits (data+2) << 11) ^ hash;
hash = (hash << 16) ^ tmp;
- data += 2*sizeof (uint16_t);
+ data += 2*sizeof (__u16);
hash += hash >> 11;
}
switch (rem) {
case 3: hash += get16bits (data);
hash ^= hash << 16;
- hash ^= ((signed char)data[sizeof (uint16_t)]) << 18;
+ hash ^= ((signed char)data[sizeof (__u16)]) << 18;
hash += hash >> 11;
break;
case 2: hash += get16bits (data);
* 02110-1301, USA.
*/
-#include <stdint.h>
#include <stddef.h>
#include <linux/bpf.h>
static __always_inline int ipv4_hash(struct __sk_buff *skb)
{
- uint32_t nhoff;
- uint32_t src, dst;
+ __u32 nhoff;
+ __u32 src, dst;
nhoff = skb->cb[0];
src = load_word(skb, nhoff + offsetof(struct iphdr, saddr));
static __always_inline int ipv6_hash(struct __sk_buff *skb)
{
- uint32_t nhoff;
- uint32_t src, dst, hash;
+ __u32 nhoff;
+ __u32 src, dst, hash;
nhoff = skb->cb[0];
hash = 0;
/* libbpf needs version section to check sync of eBPF code and kernel
* but socket filter don't need it */
-uint32_t __version __section("version") = LINUX_VERSION_CODE;
+__u32 __version __section("version") = LINUX_VERSION_CODE;
* 02110-1301, USA.
*/
-#include <stdint.h>
#include <stddef.h>
#include <linux/bpf.h>
#define LINUX_VERSION_CODE 263682
int SEC("filter") hashfilter(struct __sk_buff *skb) {
- uint16_t vlan_id = skb->vlan_tci & 0x0fff;
+ __u16 vlan_id = skb->vlan_tci & 0x0fff;
/* accept VLAN 2 and 4 and drop the rest */
switch (vlan_id) {
case 2:
char __license[] SEC("license") = "GPL";
-uint32_t __version SEC("version") = LINUX_VERSION_CODE;
+__u32 __version SEC("version") = LINUX_VERSION_CODE;
*/
#define KBUILD_MODNAME "foo"
-#include <stdint.h>
-#include <string.h>
#include <stddef.h>
#include <linux/bpf.h>
} __attribute__((__aligned__(8)));
struct pair {
- uint64_t time;
- uint64_t packets;
- uint64_t bytes;
+ __u64 time;
+ __u64 packets;
+ __u64 bytes;
} __attribute__((__aligned__(8)));
struct bpf_map_def SEC("maps") flow_table_v4 = {
};
static __always_inline int get_sport(void *trans_data, void *data_end,
- uint8_t protocol)
+ __u8 protocol)
{
struct tcphdr *th;
struct udphdr *uh;
}
static __always_inline int get_dport(void *trans_data, void *data_end,
- uint8_t protocol)
+ __u8 protocol)
{
struct tcphdr *th;
struct udphdr *uh;
int sport;
struct flowv4_keys tuple;
struct pair *value;
- uint32_t key0 = 0;
+ __u32 key0 = 0;
#if BUILD_CPUMAP
- uint32_t cpu_dest;
- uint32_t *cpu_max = bpf_map_lookup_elem(&cpus_count, &key0);
- uint32_t *cpu_selected;
- uint32_t cpu_hash;
+ __u32 cpu_dest;
+ __u32 *cpu_max = bpf_map_lookup_elem(&cpus_count, &key0);
+ __u32 *cpu_selected;
+ __u32 cpu_hash;
#endif
int *iface_peer;
int tx_port = 0;
if ((void *)(iph + 1) > data_end)
return XDP_PASS;
- tuple.ip_proto = (uint32_t) iph->protocol;
+ tuple.ip_proto = (__u32) iph->protocol;
tuple.src = iph->saddr;
tuple.dst = iph->daddr;
if (sport == -1)
return XDP_PASS;
- tuple.port16[0] = (uint16_t)sport;
- tuple.port16[1] = (uint16_t)dport;
+ tuple.port16[0] = (__u16)sport;
+ tuple.port16[1] = (__u16)dport;
value = bpf_map_lookup_elem(&flow_table_v4, &tuple);
#if 0
{
int sport;
struct flowv6_keys tuple;
struct pair *value;
- uint32_t key0 = 0;
+ __u32 key0 = 0;
#if BUILD_CPUMAP
- uint32_t cpu_dest;
+ __u32 cpu_dest;
int *cpu_max = bpf_map_lookup_elem(&cpus_count, &key0);
- uint32_t *cpu_selected;
- uint32_t cpu_hash;
+ __u32 *cpu_selected;
+ __u32 cpu_hash;
#endif
int tx_port = 0;
int *iface_peer;
void *data = (void *)(long)ctx->data;
struct ethhdr *eth = data;
int rc = XDP_PASS;
- uint16_t h_proto;
- uint64_t nh_off;
+ __u16 h_proto;
+ __u64 nh_off;
nh_off = sizeof(*eth);
if (data + nh_off > data_end)
char __license[] SEC("license") = "GPL";
-uint32_t __version SEC("version") = LINUX_VERSION_CODE;
+__u32 __version SEC("version") = LINUX_VERSION_CODE;