forked from Minki/linux
f3515b5d0b
To overcome bugs as described and fixed in 89087c456f
("bpf: Fix
values type used in test_maps"), provide a generic BPF_DECLARE_PERCPU()
and bpf_percpu() accessor macro for all percpu map values used in
tests.
Declaring variables works as follows (also works for structs):
BPF_DECLARE_PERCPU(uint32_t, my_value);
They can then be accessed normally as uint32_t type through:
bpf_percpu(my_value, <cpu_nr>)
For example:
bpf_percpu(my_value, 0)++;
Implicitly, we make sure that the passed type is allocated and aligned
by gcc at least on a 8-byte boundary, so that it works together with
the map lookup/update syscall for percpu maps. We use it as a usage
example in test_maps, so that others are free to adapt this into their
code when necessary.
Signed-off-by: Daniel Borkmann <daniel@iogearbox.net>
Acked-by: Alexei Starovoitov <ast@kernel.org>
Signed-off-by: David S. Miller <davem@davemloft.net>
65 lines
1.4 KiB
C
65 lines
1.4 KiB
C
#ifndef __BPF_UTIL__
|
|
#define __BPF_UTIL__
|
|
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <errno.h>
|
|
|
|
#include <asm/byteorder.h>
|
|
|
|
#if __BYTE_ORDER == __LITTLE_ENDIAN
|
|
# define __bpf_ntohs(x) __builtin_bswap16(x)
|
|
# define __bpf_htons(x) __builtin_bswap16(x)
|
|
#elif __BYTE_ORDER == __BIG_ENDIAN
|
|
# define __bpf_ntohs(x) (x)
|
|
# define __bpf_htons(x) (x)
|
|
#else
|
|
# error "Fix your __BYTE_ORDER?!"
|
|
#endif
|
|
|
|
#define bpf_htons(x) \
|
|
(__builtin_constant_p(x) ? \
|
|
__constant_htons(x) : __bpf_htons(x))
|
|
#define bpf_ntohs(x) \
|
|
(__builtin_constant_p(x) ? \
|
|
__constant_ntohs(x) : __bpf_ntohs(x))
|
|
|
|
static inline unsigned int bpf_num_possible_cpus(void)
|
|
{
|
|
static const char *fcpu = "/sys/devices/system/cpu/possible";
|
|
unsigned int start, end, possible_cpus = 0;
|
|
char buff[128];
|
|
FILE *fp;
|
|
|
|
fp = fopen(fcpu, "r");
|
|
if (!fp) {
|
|
printf("Failed to open %s: '%s'!\n", fcpu, strerror(errno));
|
|
exit(1);
|
|
}
|
|
|
|
while (fgets(buff, sizeof(buff), fp)) {
|
|
if (sscanf(buff, "%u-%u", &start, &end) == 2) {
|
|
possible_cpus = start == 0 ? end + 1 : 0;
|
|
break;
|
|
}
|
|
}
|
|
|
|
fclose(fp);
|
|
if (!possible_cpus) {
|
|
printf("Failed to retrieve # possible CPUs!\n");
|
|
exit(1);
|
|
}
|
|
|
|
return possible_cpus;
|
|
}
|
|
|
|
#define __bpf_percpu_val_align __attribute__((__aligned__(8)))
|
|
|
|
#define BPF_DECLARE_PERCPU(type, name) \
|
|
struct { type v; /* padding */ } __bpf_percpu_val_align \
|
|
name[bpf_num_possible_cpus()]
|
|
#define bpf_percpu(name, cpu) name[(cpu)].v
|
|
|
|
#endif /* __BPF_UTIL__ */
|