Merge tag 'for-linus-5.9-ofs1' of git://git.kernel.org/pub/scm/linux/kernel/git/hubca...
[linux-2.6-microblaze.git] / tools / testing / selftests / bpf / test_socket_cookie.c
1 // SPDX-License-Identifier: GPL-2.0
2 // Copyright (c) 2018 Facebook
3
4 #include <string.h>
5 #include <unistd.h>
6
7 #include <arpa/inet.h>
8 #include <netinet/in.h>
9 #include <sys/types.h>
10 #include <sys/socket.h>
11
12 #include <bpf/bpf.h>
13 #include <bpf/libbpf.h>
14
15 #include "bpf_rlimit.h"
16 #include "cgroup_helpers.h"
17
18 #define CG_PATH                 "/foo"
19 #define SOCKET_COOKIE_PROG      "./socket_cookie_prog.o"
20
21 struct socket_cookie {
22         __u64 cookie_key;
23         __u32 cookie_value;
24 };
25
26 static int start_server(void)
27 {
28         struct sockaddr_in6 addr;
29         int fd;
30
31         fd = socket(AF_INET6, SOCK_STREAM, 0);
32         if (fd == -1) {
33                 log_err("Failed to create server socket");
34                 goto out;
35         }
36
37         memset(&addr, 0, sizeof(addr));
38         addr.sin6_family = AF_INET6;
39         addr.sin6_addr = in6addr_loopback;
40         addr.sin6_port = 0;
41
42         if (bind(fd, (const struct sockaddr *)&addr, sizeof(addr)) == -1) {
43                 log_err("Failed to bind server socket");
44                 goto close_out;
45         }
46
47         if (listen(fd, 128) == -1) {
48                 log_err("Failed to listen on server socket");
49                 goto close_out;
50         }
51
52         goto out;
53
54 close_out:
55         close(fd);
56         fd = -1;
57 out:
58         return fd;
59 }
60
61 static int connect_to_server(int server_fd)
62 {
63         struct sockaddr_storage addr;
64         socklen_t len = sizeof(addr);
65         int fd;
66
67         fd = socket(AF_INET6, SOCK_STREAM, 0);
68         if (fd == -1) {
69                 log_err("Failed to create client socket");
70                 goto out;
71         }
72
73         if (getsockname(server_fd, (struct sockaddr *)&addr, &len)) {
74                 log_err("Failed to get server addr");
75                 goto close_out;
76         }
77
78         if (connect(fd, (const struct sockaddr *)&addr, len) == -1) {
79                 log_err("Fail to connect to server");
80                 goto close_out;
81         }
82
83         goto out;
84
85 close_out:
86         close(fd);
87         fd = -1;
88 out:
89         return fd;
90 }
91
92 static int validate_map(struct bpf_map *map, int client_fd)
93 {
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;
100
101         if (!map) {
102                 log_err("Map not found in BPF object");
103                 goto err;
104         }
105
106         map_fd = bpf_map__fd(map);
107
108         err = bpf_map_lookup_elem(map_fd, &client_fd, &val);
109
110         err = getsockname(client_fd, (struct sockaddr *)&addr, &len);
111         if (err) {
112                 log_err("Can't get client local addr");
113                 goto out;
114         }
115
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;
121         }
122
123         goto out;
124 err:
125         err = -1;
126 out:
127         return err;
128 }
129
130 static int run_test(int cgfd)
131 {
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;
141
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;
146
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;
151         }
152
153         bpf_object__for_each_program(prog, pobj) {
154                 prog_name = bpf_program__title(prog, /*needs_copy*/ false);
155
156                 if (libbpf_attach_type_by_name(prog_name, &attach_type))
157                         goto err;
158
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;
164                 }
165         }
166
167         server_fd = start_server();
168         if (server_fd == -1)
169                 goto err;
170
171         client_fd = connect_to_server(server_fd);
172         if (client_fd == -1)
173                 goto err;
174
175         if (validate_map(bpf_map__next(NULL, pobj), client_fd))
176                 goto err;
177
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;
187 }
188
189 int main(int argc, char **argv)
190 {
191         int cgfd = -1;
192         int err = 0;
193
194         cgfd = cgroup_setup_and_join(CG_PATH);
195         if (cgfd < 0)
196                 goto err;
197
198         if (run_test(cgfd))
199                 goto err;
200
201         goto out;
202 err:
203         err = -1;
204 out:
205         close(cgfd);
206         cleanup_cgroup_environment();
207         return err;
208 }