1 // SPDX-License-Identifier: GPL-2.0
2 /*
3 * Tests for attaching, detaching, and replacing flow_dissector BPF program.
4 */
5
6 #define _GNU_SOURCE
7 #include <errno.h>
8 #include <fcntl.h>
9 #include <sched.h>
10 #include <stdbool.h>
11 #include <sys/stat.h>
12 #include <unistd.h>
13
14 #include <linux/bpf.h>
15 #include <bpf/bpf.h>
16
17 #include "test_progs.h"
18
19 static int init_net = -1;
20
query_attached_prog_id(int netns)21 static __u32 query_attached_prog_id(int netns)
22 {
23 __u32 prog_ids[1] = {};
24 __u32 prog_cnt = ARRAY_SIZE(prog_ids);
25 int err;
26
27 err = bpf_prog_query(netns, BPF_FLOW_DISSECTOR, 0, NULL,
28 prog_ids, &prog_cnt);
29 if (CHECK_FAIL(err)) {
30 perror("bpf_prog_query");
31 return 0;
32 }
33
34 return prog_cnt == 1 ? prog_ids[0] : 0;
35 }
36
prog_is_attached(int netns)37 static bool prog_is_attached(int netns)
38 {
39 return query_attached_prog_id(netns) > 0;
40 }
41
load_prog(enum bpf_prog_type type)42 static int load_prog(enum bpf_prog_type type)
43 {
44 struct bpf_insn prog[] = {
45 BPF_MOV64_IMM(BPF_REG_0, BPF_OK),
46 BPF_EXIT_INSN(),
47 };
48 int fd;
49
50 fd = bpf_load_program(type, prog, ARRAY_SIZE(prog), "GPL", 0, NULL, 0);
51 if (CHECK_FAIL(fd < 0))
52 perror("bpf_load_program");
53
54 return fd;
55 }
56
query_prog_id(int prog)57 static __u32 query_prog_id(int prog)
58 {
59 struct bpf_prog_info info = {};
60 __u32 info_len = sizeof(info);
61 int err;
62
63 err = bpf_obj_get_info_by_fd(prog, &info, &info_len);
64 if (CHECK_FAIL(err || info_len != sizeof(info))) {
65 perror("bpf_obj_get_info_by_fd");
66 return 0;
67 }
68
69 return info.id;
70 }
71
unshare_net(int old_net)72 static int unshare_net(int old_net)
73 {
74 int err, new_net;
75
76 err = unshare(CLONE_NEWNET);
77 if (CHECK_FAIL(err)) {
78 perror("unshare(CLONE_NEWNET)");
79 return -1;
80 }
81 new_net = open("/proc/self/ns/net", O_RDONLY);
82 if (CHECK_FAIL(new_net < 0)) {
83 perror("open(/proc/self/ns/net)");
84 setns(old_net, CLONE_NEWNET);
85 return -1;
86 }
87 return new_net;
88 }
89
test_prog_attach_prog_attach(int netns,int prog1,int prog2)90 static void test_prog_attach_prog_attach(int netns, int prog1, int prog2)
91 {
92 int err;
93
94 err = bpf_prog_attach(prog1, 0, BPF_FLOW_DISSECTOR, 0);
95 if (CHECK_FAIL(err)) {
96 perror("bpf_prog_attach(prog1)");
97 return;
98 }
99 CHECK_FAIL(query_attached_prog_id(netns) != query_prog_id(prog1));
100
101 /* Expect success when attaching a different program */
102 err = bpf_prog_attach(prog2, 0, BPF_FLOW_DISSECTOR, 0);
103 if (CHECK_FAIL(err)) {
104 perror("bpf_prog_attach(prog2) #1");
105 goto out_detach;
106 }
107 CHECK_FAIL(query_attached_prog_id(netns) != query_prog_id(prog2));
108
109 /* Expect failure when attaching the same program twice */
110 err = bpf_prog_attach(prog2, 0, BPF_FLOW_DISSECTOR, 0);
111 if (CHECK_FAIL(!err || errno != EINVAL))
112 perror("bpf_prog_attach(prog2) #2");
113 CHECK_FAIL(query_attached_prog_id(netns) != query_prog_id(prog2));
114
115 out_detach:
116 err = bpf_prog_detach2(prog2, 0, BPF_FLOW_DISSECTOR);
117 if (CHECK_FAIL(err))
118 perror("bpf_prog_detach");
119 CHECK_FAIL(prog_is_attached(netns));
120 }
121
test_link_create_link_create(int netns,int prog1,int prog2)122 static void test_link_create_link_create(int netns, int prog1, int prog2)
123 {
124 DECLARE_LIBBPF_OPTS(bpf_link_create_opts, opts);
125 int link1, link2;
126
127 link1 = bpf_link_create(prog1, netns, BPF_FLOW_DISSECTOR, &opts);
128 if (CHECK_FAIL(link < 0)) {
129 perror("bpf_link_create(prog1)");
130 return;
131 }
132 CHECK_FAIL(query_attached_prog_id(netns) != query_prog_id(prog1));
133
134 /* Expect failure creating link when another link exists */
135 errno = 0;
136 link2 = bpf_link_create(prog2, netns, BPF_FLOW_DISSECTOR, &opts);
137 if (CHECK_FAIL(link2 != -1 || errno != E2BIG))
138 perror("bpf_prog_attach(prog2) expected E2BIG");
139 if (link2 != -1)
140 close(link2);
141 CHECK_FAIL(query_attached_prog_id(netns) != query_prog_id(prog1));
142
143 close(link1);
144 CHECK_FAIL(prog_is_attached(netns));
145 }
146
test_prog_attach_link_create(int netns,int prog1,int prog2)147 static void test_prog_attach_link_create(int netns, int prog1, int prog2)
148 {
149 DECLARE_LIBBPF_OPTS(bpf_link_create_opts, opts);
150 int err, link;
151
152 err = bpf_prog_attach(prog1, 0, BPF_FLOW_DISSECTOR, 0);
153 if (CHECK_FAIL(err)) {
154 perror("bpf_prog_attach(prog1)");
155 return;
156 }
157 CHECK_FAIL(query_attached_prog_id(netns) != query_prog_id(prog1));
158
159 /* Expect failure creating link when prog attached */
160 errno = 0;
161 link = bpf_link_create(prog2, netns, BPF_FLOW_DISSECTOR, &opts);
162 if (CHECK_FAIL(link != -1 || errno != EEXIST))
163 perror("bpf_link_create(prog2) expected EEXIST");
164 if (link != -1)
165 close(link);
166 CHECK_FAIL(query_attached_prog_id(netns) != query_prog_id(prog1));
167
168 err = bpf_prog_detach2(prog1, 0, BPF_FLOW_DISSECTOR);
169 if (CHECK_FAIL(err))
170 perror("bpf_prog_detach");
171 CHECK_FAIL(prog_is_attached(netns));
172 }
173
test_link_create_prog_attach(int netns,int prog1,int prog2)174 static void test_link_create_prog_attach(int netns, int prog1, int prog2)
175 {
176 DECLARE_LIBBPF_OPTS(bpf_link_create_opts, opts);
177 int err, link;
178
179 link = bpf_link_create(prog1, netns, BPF_FLOW_DISSECTOR, &opts);
180 if (CHECK_FAIL(link < 0)) {
181 perror("bpf_link_create(prog1)");
182 return;
183 }
184 CHECK_FAIL(query_attached_prog_id(netns) != query_prog_id(prog1));
185
186 /* Expect failure attaching prog when link exists */
187 errno = 0;
188 err = bpf_prog_attach(prog2, 0, BPF_FLOW_DISSECTOR, 0);
189 if (CHECK_FAIL(!err || errno != EEXIST))
190 perror("bpf_prog_attach(prog2) expected EEXIST");
191 CHECK_FAIL(query_attached_prog_id(netns) != query_prog_id(prog1));
192
193 close(link);
194 CHECK_FAIL(prog_is_attached(netns));
195 }
196
test_link_create_prog_detach(int netns,int prog1,int prog2)197 static void test_link_create_prog_detach(int netns, int prog1, int prog2)
198 {
199 DECLARE_LIBBPF_OPTS(bpf_link_create_opts, opts);
200 int err, link;
201
202 link = bpf_link_create(prog1, netns, BPF_FLOW_DISSECTOR, &opts);
203 if (CHECK_FAIL(link < 0)) {
204 perror("bpf_link_create(prog1)");
205 return;
206 }
207 CHECK_FAIL(query_attached_prog_id(netns) != query_prog_id(prog1));
208
209 /* Expect failure detaching prog when link exists */
210 errno = 0;
211 err = bpf_prog_detach2(prog1, 0, BPF_FLOW_DISSECTOR);
212 if (CHECK_FAIL(!err || errno != EINVAL))
213 perror("bpf_prog_detach expected EINVAL");
214 CHECK_FAIL(query_attached_prog_id(netns) != query_prog_id(prog1));
215
216 close(link);
217 CHECK_FAIL(prog_is_attached(netns));
218 }
219
test_prog_attach_detach_query(int netns,int prog1,int prog2)220 static void test_prog_attach_detach_query(int netns, int prog1, int prog2)
221 {
222 int err;
223
224 err = bpf_prog_attach(prog1, 0, BPF_FLOW_DISSECTOR, 0);
225 if (CHECK_FAIL(err)) {
226 perror("bpf_prog_attach(prog1)");
227 return;
228 }
229 CHECK_FAIL(query_attached_prog_id(netns) != query_prog_id(prog1));
230
231 err = bpf_prog_detach2(prog1, 0, BPF_FLOW_DISSECTOR);
232 if (CHECK_FAIL(err)) {
233 perror("bpf_prog_detach");
234 return;
235 }
236
237 /* Expect no prog attached after successful detach */
238 CHECK_FAIL(prog_is_attached(netns));
239 }
240
test_link_create_close_query(int netns,int prog1,int prog2)241 static void test_link_create_close_query(int netns, int prog1, int prog2)
242 {
243 DECLARE_LIBBPF_OPTS(bpf_link_create_opts, opts);
244 int link;
245
246 link = bpf_link_create(prog1, netns, BPF_FLOW_DISSECTOR, &opts);
247 if (CHECK_FAIL(link < 0)) {
248 perror("bpf_link_create(prog1)");
249 return;
250 }
251 CHECK_FAIL(query_attached_prog_id(netns) != query_prog_id(prog1));
252
253 close(link);
254 /* Expect no prog attached after closing last link FD */
255 CHECK_FAIL(prog_is_attached(netns));
256 }
257
test_link_update_no_old_prog(int netns,int prog1,int prog2)258 static void test_link_update_no_old_prog(int netns, int prog1, int prog2)
259 {
260 DECLARE_LIBBPF_OPTS(bpf_link_create_opts, create_opts);
261 DECLARE_LIBBPF_OPTS(bpf_link_update_opts, update_opts);
262 int err, link;
263
264 link = bpf_link_create(prog1, netns, BPF_FLOW_DISSECTOR, &create_opts);
265 if (CHECK_FAIL(link < 0)) {
266 perror("bpf_link_create(prog1)");
267 return;
268 }
269 CHECK_FAIL(query_attached_prog_id(netns) != query_prog_id(prog1));
270
271 /* Expect success replacing the prog when old prog not specified */
272 update_opts.flags = 0;
273 update_opts.old_prog_fd = 0;
274 err = bpf_link_update(link, prog2, &update_opts);
275 if (CHECK_FAIL(err))
276 perror("bpf_link_update");
277 CHECK_FAIL(query_attached_prog_id(netns) != query_prog_id(prog2));
278
279 close(link);
280 CHECK_FAIL(prog_is_attached(netns));
281 }
282
test_link_update_replace_old_prog(int netns,int prog1,int prog2)283 static void test_link_update_replace_old_prog(int netns, int prog1, int prog2)
284 {
285 DECLARE_LIBBPF_OPTS(bpf_link_create_opts, create_opts);
286 DECLARE_LIBBPF_OPTS(bpf_link_update_opts, update_opts);
287 int err, link;
288
289 link = bpf_link_create(prog1, netns, BPF_FLOW_DISSECTOR, &create_opts);
290 if (CHECK_FAIL(link < 0)) {
291 perror("bpf_link_create(prog1)");
292 return;
293 }
294 CHECK_FAIL(query_attached_prog_id(netns) != query_prog_id(prog1));
295
296 /* Expect success F_REPLACE and old prog specified to succeed */
297 update_opts.flags = BPF_F_REPLACE;
298 update_opts.old_prog_fd = prog1;
299 err = bpf_link_update(link, prog2, &update_opts);
300 if (CHECK_FAIL(err))
301 perror("bpf_link_update");
302 CHECK_FAIL(query_attached_prog_id(netns) != query_prog_id(prog2));
303
304 close(link);
305 CHECK_FAIL(prog_is_attached(netns));
306 }
307
test_link_update_same_prog(int netns,int prog1,int prog2)308 static void test_link_update_same_prog(int netns, int prog1, int prog2)
309 {
310 DECLARE_LIBBPF_OPTS(bpf_link_create_opts, create_opts);
311 DECLARE_LIBBPF_OPTS(bpf_link_update_opts, update_opts);
312 int err, link;
313
314 link = bpf_link_create(prog1, netns, BPF_FLOW_DISSECTOR, &create_opts);
315 if (CHECK_FAIL(link < 0)) {
316 perror("bpf_link_create(prog1)");
317 return;
318 }
319 CHECK_FAIL(query_attached_prog_id(netns) != query_prog_id(prog1));
320
321 /* Expect success updating the prog with the same one */
322 update_opts.flags = 0;
323 update_opts.old_prog_fd = 0;
324 err = bpf_link_update(link, prog1, &update_opts);
325 if (CHECK_FAIL(err))
326 perror("bpf_link_update");
327 CHECK_FAIL(query_attached_prog_id(netns) != query_prog_id(prog1));
328
329 close(link);
330 CHECK_FAIL(prog_is_attached(netns));
331 }
332
test_link_update_invalid_opts(int netns,int prog1,int prog2)333 static void test_link_update_invalid_opts(int netns, int prog1, int prog2)
334 {
335 DECLARE_LIBBPF_OPTS(bpf_link_create_opts, create_opts);
336 DECLARE_LIBBPF_OPTS(bpf_link_update_opts, update_opts);
337 int err, link;
338
339 link = bpf_link_create(prog1, netns, BPF_FLOW_DISSECTOR, &create_opts);
340 if (CHECK_FAIL(link < 0)) {
341 perror("bpf_link_create(prog1)");
342 return;
343 }
344 CHECK_FAIL(query_attached_prog_id(netns) != query_prog_id(prog1));
345
346 /* Expect update to fail w/ old prog FD but w/o F_REPLACE*/
347 errno = 0;
348 update_opts.flags = 0;
349 update_opts.old_prog_fd = prog1;
350 err = bpf_link_update(link, prog2, &update_opts);
351 if (CHECK_FAIL(!err || errno != EINVAL)) {
352 perror("bpf_link_update expected EINVAL");
353 goto out_close;
354 }
355 CHECK_FAIL(query_attached_prog_id(netns) != query_prog_id(prog1));
356
357 /* Expect update to fail on old prog FD mismatch */
358 errno = 0;
359 update_opts.flags = BPF_F_REPLACE;
360 update_opts.old_prog_fd = prog2;
361 err = bpf_link_update(link, prog2, &update_opts);
362 if (CHECK_FAIL(!err || errno != EPERM)) {
363 perror("bpf_link_update expected EPERM");
364 goto out_close;
365 }
366 CHECK_FAIL(query_attached_prog_id(netns) != query_prog_id(prog1));
367
368 /* Expect update to fail for invalid old prog FD */
369 errno = 0;
370 update_opts.flags = BPF_F_REPLACE;
371 update_opts.old_prog_fd = -1;
372 err = bpf_link_update(link, prog2, &update_opts);
373 if (CHECK_FAIL(!err || errno != EBADF)) {
374 perror("bpf_link_update expected EBADF");
375 goto out_close;
376 }
377 CHECK_FAIL(query_attached_prog_id(netns) != query_prog_id(prog1));
378
379 /* Expect update to fail with invalid flags */
380 errno = 0;
381 update_opts.flags = BPF_F_ALLOW_MULTI;
382 update_opts.old_prog_fd = 0;
383 err = bpf_link_update(link, prog2, &update_opts);
384 if (CHECK_FAIL(!err || errno != EINVAL))
385 perror("bpf_link_update expected EINVAL");
386 CHECK_FAIL(query_attached_prog_id(netns) != query_prog_id(prog1));
387
388 out_close:
389 close(link);
390 CHECK_FAIL(prog_is_attached(netns));
391 }
392
test_link_update_invalid_prog(int netns,int prog1,int prog2)393 static void test_link_update_invalid_prog(int netns, int prog1, int prog2)
394 {
395 DECLARE_LIBBPF_OPTS(bpf_link_create_opts, create_opts);
396 DECLARE_LIBBPF_OPTS(bpf_link_update_opts, update_opts);
397 int err, link, prog3;
398
399 link = bpf_link_create(prog1, netns, BPF_FLOW_DISSECTOR, &create_opts);
400 if (CHECK_FAIL(link < 0)) {
401 perror("bpf_link_create(prog1)");
402 return;
403 }
404 CHECK_FAIL(query_attached_prog_id(netns) != query_prog_id(prog1));
405
406 /* Expect failure when new prog FD is not valid */
407 errno = 0;
408 update_opts.flags = 0;
409 update_opts.old_prog_fd = 0;
410 err = bpf_link_update(link, -1, &update_opts);
411 if (CHECK_FAIL(!err || errno != EBADF)) {
412 perror("bpf_link_update expected EINVAL");
413 goto out_close_link;
414 }
415 CHECK_FAIL(query_attached_prog_id(netns) != query_prog_id(prog1));
416
417 prog3 = load_prog(BPF_PROG_TYPE_SOCKET_FILTER);
418 if (prog3 < 0)
419 goto out_close_link;
420
421 /* Expect failure when new prog FD type doesn't match */
422 errno = 0;
423 update_opts.flags = 0;
424 update_opts.old_prog_fd = 0;
425 err = bpf_link_update(link, prog3, &update_opts);
426 if (CHECK_FAIL(!err || errno != EINVAL))
427 perror("bpf_link_update expected EINVAL");
428 CHECK_FAIL(query_attached_prog_id(netns) != query_prog_id(prog1));
429
430 close(prog3);
431 out_close_link:
432 close(link);
433 CHECK_FAIL(prog_is_attached(netns));
434 }
435
test_link_update_netns_gone(int netns,int prog1,int prog2)436 static void test_link_update_netns_gone(int netns, int prog1, int prog2)
437 {
438 DECLARE_LIBBPF_OPTS(bpf_link_create_opts, create_opts);
439 DECLARE_LIBBPF_OPTS(bpf_link_update_opts, update_opts);
440 int err, link, old_net;
441
442 old_net = netns;
443 netns = unshare_net(old_net);
444 if (netns < 0)
445 return;
446
447 link = bpf_link_create(prog1, netns, BPF_FLOW_DISSECTOR, &create_opts);
448 if (CHECK_FAIL(link < 0)) {
449 perror("bpf_link_create(prog1)");
450 return;
451 }
452 CHECK_FAIL(query_attached_prog_id(netns) != query_prog_id(prog1));
453
454 close(netns);
455 err = setns(old_net, CLONE_NEWNET);
456 if (CHECK_FAIL(err)) {
457 perror("setns(CLONE_NEWNET)");
458 close(link);
459 return;
460 }
461
462 /* Expect failure when netns destroyed */
463 errno = 0;
464 update_opts.flags = 0;
465 update_opts.old_prog_fd = 0;
466 err = bpf_link_update(link, prog2, &update_opts);
467 if (CHECK_FAIL(!err || errno != ENOLINK))
468 perror("bpf_link_update");
469
470 close(link);
471 }
472
test_link_get_info(int netns,int prog1,int prog2)473 static void test_link_get_info(int netns, int prog1, int prog2)
474 {
475 DECLARE_LIBBPF_OPTS(bpf_link_create_opts, create_opts);
476 DECLARE_LIBBPF_OPTS(bpf_link_update_opts, update_opts);
477 struct bpf_link_info info = {};
478 struct stat netns_stat = {};
479 __u32 info_len, link_id;
480 int err, link, old_net;
481
482 old_net = netns;
483 netns = unshare_net(old_net);
484 if (netns < 0)
485 return;
486
487 err = fstat(netns, &netns_stat);
488 if (CHECK_FAIL(err)) {
489 perror("stat(netns)");
490 goto out_resetns;
491 }
492
493 link = bpf_link_create(prog1, netns, BPF_FLOW_DISSECTOR, &create_opts);
494 if (CHECK_FAIL(link < 0)) {
495 perror("bpf_link_create(prog1)");
496 goto out_resetns;
497 }
498
499 info_len = sizeof(info);
500 err = bpf_obj_get_info_by_fd(link, &info, &info_len);
501 if (CHECK_FAIL(err)) {
502 perror("bpf_obj_get_info");
503 goto out_unlink;
504 }
505 CHECK_FAIL(info_len != sizeof(info));
506
507 /* Expect link info to be sane and match prog and netns details */
508 CHECK_FAIL(info.type != BPF_LINK_TYPE_NETNS);
509 CHECK_FAIL(info.id == 0);
510 CHECK_FAIL(info.prog_id != query_prog_id(prog1));
511 CHECK_FAIL(info.netns.netns_ino != netns_stat.st_ino);
512 CHECK_FAIL(info.netns.attach_type != BPF_FLOW_DISSECTOR);
513
514 update_opts.flags = 0;
515 update_opts.old_prog_fd = 0;
516 err = bpf_link_update(link, prog2, &update_opts);
517 if (CHECK_FAIL(err)) {
518 perror("bpf_link_update(prog2)");
519 goto out_unlink;
520 }
521
522 link_id = info.id;
523 info_len = sizeof(info);
524 err = bpf_obj_get_info_by_fd(link, &info, &info_len);
525 if (CHECK_FAIL(err)) {
526 perror("bpf_obj_get_info");
527 goto out_unlink;
528 }
529 CHECK_FAIL(info_len != sizeof(info));
530
531 /* Expect no info change after update except in prog id */
532 CHECK_FAIL(info.type != BPF_LINK_TYPE_NETNS);
533 CHECK_FAIL(info.id != link_id);
534 CHECK_FAIL(info.prog_id != query_prog_id(prog2));
535 CHECK_FAIL(info.netns.netns_ino != netns_stat.st_ino);
536 CHECK_FAIL(info.netns.attach_type != BPF_FLOW_DISSECTOR);
537
538 /* Leave netns link is attached to and close last FD to it */
539 err = setns(old_net, CLONE_NEWNET);
540 if (CHECK_FAIL(err)) {
541 perror("setns(NEWNET)");
542 goto out_unlink;
543 }
544 close(netns);
545 old_net = -1;
546 netns = -1;
547
548 info_len = sizeof(info);
549 err = bpf_obj_get_info_by_fd(link, &info, &info_len);
550 if (CHECK_FAIL(err)) {
551 perror("bpf_obj_get_info");
552 goto out_unlink;
553 }
554 CHECK_FAIL(info_len != sizeof(info));
555
556 /* Expect netns_ino to change to 0 */
557 CHECK_FAIL(info.type != BPF_LINK_TYPE_NETNS);
558 CHECK_FAIL(info.id != link_id);
559 CHECK_FAIL(info.prog_id != query_prog_id(prog2));
560 CHECK_FAIL(info.netns.netns_ino != 0);
561 CHECK_FAIL(info.netns.attach_type != BPF_FLOW_DISSECTOR);
562
563 out_unlink:
564 close(link);
565 out_resetns:
566 if (old_net != -1)
567 setns(old_net, CLONE_NEWNET);
568 if (netns != -1)
569 close(netns);
570 }
571
run_tests(int netns)572 static void run_tests(int netns)
573 {
574 struct test {
575 const char *test_name;
576 void (*test_func)(int netns, int prog1, int prog2);
577 } tests[] = {
578 { "prog attach, prog attach",
579 test_prog_attach_prog_attach },
580 { "link create, link create",
581 test_link_create_link_create },
582 { "prog attach, link create",
583 test_prog_attach_link_create },
584 { "link create, prog attach",
585 test_link_create_prog_attach },
586 { "link create, prog detach",
587 test_link_create_prog_detach },
588 { "prog attach, detach, query",
589 test_prog_attach_detach_query },
590 { "link create, close, query",
591 test_link_create_close_query },
592 { "link update no old prog",
593 test_link_update_no_old_prog },
594 { "link update with replace old prog",
595 test_link_update_replace_old_prog },
596 { "link update with same prog",
597 test_link_update_same_prog },
598 { "link update invalid opts",
599 test_link_update_invalid_opts },
600 { "link update invalid prog",
601 test_link_update_invalid_prog },
602 { "link update netns gone",
603 test_link_update_netns_gone },
604 { "link get info",
605 test_link_get_info },
606 };
607 int i, progs[2] = { -1, -1 };
608 char test_name[80];
609
610 for (i = 0; i < ARRAY_SIZE(progs); i++) {
611 progs[i] = load_prog(BPF_PROG_TYPE_FLOW_DISSECTOR);
612 if (progs[i] < 0)
613 goto out_close;
614 }
615
616 for (i = 0; i < ARRAY_SIZE(tests); i++) {
617 snprintf(test_name, sizeof(test_name),
618 "flow dissector %s%s",
619 tests[i].test_name,
620 netns == init_net ? " (init_net)" : "");
621 if (test__start_subtest(test_name))
622 tests[i].test_func(netns, progs[0], progs[1]);
623 }
624 out_close:
625 for (i = 0; i < ARRAY_SIZE(progs); i++) {
626 if (progs[i] != -1)
627 CHECK_FAIL(close(progs[i]));
628 }
629 }
630
test_flow_dissector_reattach(void)631 void test_flow_dissector_reattach(void)
632 {
633 int err, new_net, saved_net;
634
635 saved_net = open("/proc/self/ns/net", O_RDONLY);
636 if (CHECK_FAIL(saved_net < 0)) {
637 perror("open(/proc/self/ns/net");
638 return;
639 }
640
641 init_net = open("/proc/1/ns/net", O_RDONLY);
642 if (CHECK_FAIL(init_net < 0)) {
643 perror("open(/proc/1/ns/net)");
644 goto out_close;
645 }
646
647 err = setns(init_net, CLONE_NEWNET);
648 if (CHECK_FAIL(err)) {
649 perror("setns(/proc/1/ns/net)");
650 goto out_close;
651 }
652
653 if (prog_is_attached(init_net)) {
654 test__skip();
655 printf("Can't test with flow dissector attached to init_net\n");
656 goto out_setns;
657 }
658
659 /* First run tests in root network namespace */
660 run_tests(init_net);
661
662 /* Then repeat tests in a non-root namespace */
663 new_net = unshare_net(init_net);
664 if (new_net < 0)
665 goto out_setns;
666 run_tests(new_net);
667 close(new_net);
668
669 out_setns:
670 /* Move back to netns we started in. */
671 err = setns(saved_net, CLONE_NEWNET);
672 if (CHECK_FAIL(err))
673 perror("setns(/proc/self/ns/net)");
674
675 out_close:
676 close(init_net);
677 close(saved_net);
678 }
679