WIP FPC-III support
[linux/fpc-iii.git] / tools / testing / selftests / bpf / test_socket_cookie.c
blobca7ca87e91aa4d302ba5e6bc373683ccdfd6675e
1 // SPDX-License-Identifier: GPL-2.0
2 // Copyright (c) 2018 Facebook
4 #include <string.h>
5 #include <unistd.h>
7 #include <arpa/inet.h>
8 #include <netinet/in.h>
9 #include <sys/types.h>
10 #include <sys/socket.h>
12 #include <bpf/bpf.h>
13 #include <bpf/libbpf.h>
15 #include "bpf_rlimit.h"
16 #include "cgroup_helpers.h"
18 #define CG_PATH "/foo"
19 #define SOCKET_COOKIE_PROG "./socket_cookie_prog.o"
21 struct socket_cookie {
22 __u64 cookie_key;
23 __u32 cookie_value;
26 static int start_server(void)
28 struct sockaddr_in6 addr;
29 int fd;
31 fd = socket(AF_INET6, SOCK_STREAM, 0);
32 if (fd == -1) {
33 log_err("Failed to create server socket");
34 goto out;
37 memset(&addr, 0, sizeof(addr));
38 addr.sin6_family = AF_INET6;
39 addr.sin6_addr = in6addr_loopback;
40 addr.sin6_port = 0;
42 if (bind(fd, (const struct sockaddr *)&addr, sizeof(addr)) == -1) {
43 log_err("Failed to bind server socket");
44 goto close_out;
47 if (listen(fd, 128) == -1) {
48 log_err("Failed to listen on server socket");
49 goto close_out;
52 goto out;
54 close_out:
55 close(fd);
56 fd = -1;
57 out:
58 return fd;
61 static int connect_to_server(int server_fd)
63 struct sockaddr_storage addr;
64 socklen_t len = sizeof(addr);
65 int fd;
67 fd = socket(AF_INET6, SOCK_STREAM, 0);
68 if (fd == -1) {
69 log_err("Failed to create client socket");
70 goto out;
73 if (getsockname(server_fd, (struct sockaddr *)&addr, &len)) {
74 log_err("Failed to get server addr");
75 goto close_out;
78 if (connect(fd, (const struct sockaddr *)&addr, len) == -1) {
79 log_err("Fail to connect to server");
80 goto close_out;
83 goto out;
85 close_out:
86 close(fd);
87 fd = -1;
88 out:
89 return fd;
92 static int validate_map(struct bpf_map *map, int client_fd)
94 __u32 cookie_expected_value;
95 struct sockaddr_in6 addr;
96 socklen_t len = sizeof(addr);
97 struct socket_cookie val;
98 int err = 0;
99 int map_fd;
101 if (!map) {
102 log_err("Map not found in BPF object");
103 goto err;
106 map_fd = bpf_map__fd(map);
108 err = bpf_map_lookup_elem(map_fd, &client_fd, &val);
110 err = getsockname(client_fd, (struct sockaddr *)&addr, &len);
111 if (err) {
112 log_err("Can't get client local addr");
113 goto out;
116 cookie_expected_value = (ntohs(addr.sin6_port) << 8) | 0xFF;
117 if (val.cookie_value != cookie_expected_value) {
118 log_err("Unexpected value in map: %x != %x", val.cookie_value,
119 cookie_expected_value);
120 goto err;
123 goto out;
124 err:
125 err = -1;
126 out:
127 return err;
130 static int run_test(int cgfd)
132 enum bpf_attach_type attach_type;
133 struct bpf_prog_load_attr attr;
134 struct bpf_program *prog;
135 struct bpf_object *pobj;
136 const char *prog_name;
137 int server_fd = -1;
138 int client_fd = -1;
139 int prog_fd = -1;
140 int err = 0;
142 memset(&attr, 0, sizeof(attr));
143 attr.file = SOCKET_COOKIE_PROG;
144 attr.prog_type = BPF_PROG_TYPE_UNSPEC;
145 attr.prog_flags = BPF_F_TEST_RND_HI32;
147 err = bpf_prog_load_xattr(&attr, &pobj, &prog_fd);
148 if (err) {
149 log_err("Failed to load %s", attr.file);
150 goto out;
153 bpf_object__for_each_program(prog, pobj) {
154 prog_name = bpf_program__section_name(prog);
156 if (libbpf_attach_type_by_name(prog_name, &attach_type))
157 goto err;
159 err = bpf_prog_attach(bpf_program__fd(prog), cgfd, attach_type,
160 BPF_F_ALLOW_OVERRIDE);
161 if (err) {
162 log_err("Failed to attach prog %s", prog_name);
163 goto out;
167 server_fd = start_server();
168 if (server_fd == -1)
169 goto err;
171 client_fd = connect_to_server(server_fd);
172 if (client_fd == -1)
173 goto err;
175 if (validate_map(bpf_map__next(NULL, pobj), client_fd))
176 goto err;
178 goto out;
179 err:
180 err = -1;
181 out:
182 close(client_fd);
183 close(server_fd);
184 bpf_object__close(pobj);
185 printf("%s\n", err ? "FAILED" : "PASSED");
186 return err;
189 int main(int argc, char **argv)
191 int cgfd = -1;
192 int err = 0;
194 cgfd = cgroup_setup_and_join(CG_PATH);
195 if (cgfd < 0)
196 goto err;
198 if (run_test(cgfd))
199 goto err;
201 goto out;
202 err:
203 err = -1;
204 out:
205 close(cgfd);
206 cleanup_cgroup_environment();
207 return err;