diff options
author | Alexei Starovoitov <ast@plumgrid.com> | 2014-12-01 18:06:39 -0500 |
---|---|---|
committer | David S. Miller <davem@davemloft.net> | 2014-12-06 00:47:34 -0500 |
commit | fbe3310840c65f3cf97dd90d23e177d061c376f2 (patch) | |
tree | ee28163a6c53e0131fd2d3d626d02b0610eaed2b /samples/bpf/sockex2_user.c | |
parent | a80857822b0c2ed608c93504bd3687b78f20c619 (diff) |
samples: bpf: large eBPF program in C
sockex2_kern.c is purposefully large eBPF program in C.
llvm compiles ~200 lines of C code into ~300 eBPF instructions.
It's similar to __skb_flow_dissect() to demonstrate that complex packet parsing
can be done by eBPF.
Then it uses (struct flow_keys)->dst IP address (or hash of ipv6 dst) to keep
stats of number of packets per IP.
User space loads eBPF program, attaches it to loopback interface and prints
dest_ip->#packets stats every second.
Usage:
$sudo samples/bpf/sockex2
ip 127.0.0.1 count 19
ip 127.0.0.1 count 178115
ip 127.0.0.1 count 369437
ip 127.0.0.1 count 559841
ip 127.0.0.1 count 750539
Signed-off-by: Alexei Starovoitov <ast@plumgrid.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
Diffstat (limited to 'samples/bpf/sockex2_user.c')
-rw-r--r-- | samples/bpf/sockex2_user.c | 44 |
1 files changed, 44 insertions, 0 deletions
diff --git a/samples/bpf/sockex2_user.c b/samples/bpf/sockex2_user.c new file mode 100644 index 000000000000..d2d5f5a790d3 --- /dev/null +++ b/samples/bpf/sockex2_user.c | |||
@@ -0,0 +1,44 @@ | |||
1 | #include <stdio.h> | ||
2 | #include <assert.h> | ||
3 | #include <linux/bpf.h> | ||
4 | #include "libbpf.h" | ||
5 | #include "bpf_load.h" | ||
6 | #include <unistd.h> | ||
7 | #include <arpa/inet.h> | ||
8 | |||
9 | int main(int ac, char **argv) | ||
10 | { | ||
11 | char filename[256]; | ||
12 | FILE *f; | ||
13 | int i, sock; | ||
14 | |||
15 | snprintf(filename, sizeof(filename), "%s_kern.o", argv[0]); | ||
16 | |||
17 | if (load_bpf_file(filename)) { | ||
18 | printf("%s", bpf_log_buf); | ||
19 | return 1; | ||
20 | } | ||
21 | |||
22 | sock = open_raw_sock("lo"); | ||
23 | |||
24 | assert(setsockopt(sock, SOL_SOCKET, SO_ATTACH_BPF, prog_fd, | ||
25 | sizeof(prog_fd[0])) == 0); | ||
26 | |||
27 | f = popen("ping -c5 localhost", "r"); | ||
28 | (void) f; | ||
29 | |||
30 | for (i = 0; i < 5; i++) { | ||
31 | int key = 0, next_key; | ||
32 | long long value; | ||
33 | |||
34 | while (bpf_get_next_key(map_fd[0], &key, &next_key) == 0) { | ||
35 | bpf_lookup_elem(map_fd[0], &next_key, &value); | ||
36 | printf("ip %s count %lld\n", | ||
37 | inet_ntoa((struct in_addr){htonl(next_key)}), | ||
38 | value); | ||
39 | key = next_key; | ||
40 | } | ||
41 | sleep(1); | ||
42 | } | ||
43 | return 0; | ||
44 | } | ||