xref: /openbmc/linux/tools/testing/selftests/bpf/xdp_features.c (revision 724ba6751532055db75992fc6ae21c3e322e94a7)
1 // SPDX-License-Identifier: GPL-2.0
2 #include <uapi/linux/bpf.h>
3 #include <uapi/linux/netdev.h>
4 #include <linux/if_link.h>
5 #include <signal.h>
6 #include <argp.h>
7 #include <net/if.h>
8 #include <sys/socket.h>
9 #include <netinet/in.h>
10 #include <netinet/tcp.h>
11 #include <unistd.h>
12 #include <arpa/inet.h>
13 #include <bpf/bpf.h>
14 #include <bpf/libbpf.h>
15 #include <pthread.h>
16 
17 #include <network_helpers.h>
18 
19 #include "xdp_features.skel.h"
20 #include "xdp_features.h"
21 
22 #define RED(str)	"\033[0;31m" str "\033[0m"
23 #define GREEN(str)	"\033[0;32m" str "\033[0m"
24 #define YELLOW(str)	"\033[0;33m" str "\033[0m"
25 
26 static struct env {
27 	bool verbosity;
28 	char ifname[IF_NAMESIZE];
29 	int ifindex;
30 	bool is_tester;
31 	struct {
32 		enum netdev_xdp_act drv_feature;
33 		enum xdp_action action;
34 	} feature;
35 	struct sockaddr_storage dut_ctrl_addr;
36 	struct sockaddr_storage dut_addr;
37 	struct sockaddr_storage tester_addr;
38 } env;
39 
40 #define BUFSIZE		128
41 
42 void test__fail(void) { /* for network_helpers.c */ }
43 
44 static int libbpf_print_fn(enum libbpf_print_level level,
45 			   const char *format, va_list args)
46 {
47 	if (level == LIBBPF_DEBUG && !env.verbosity)
48 		return 0;
49 	return vfprintf(stderr, format, args);
50 }
51 
52 static volatile bool exiting;
53 
54 static void sig_handler(int sig)
55 {
56 	exiting = true;
57 }
58 
59 const char *argp_program_version = "xdp-features 0.0";
60 const char argp_program_doc[] =
61 "XDP features detection application.\n"
62 "\n"
63 "XDP features application checks the XDP advertised features match detected ones.\n"
64 "\n"
65 "USAGE: ./xdp-features [-vt] [-f <xdp-feature>] [-D <dut-data-ip>] [-T <tester-data-ip>] [-C <dut-ctrl-ip>] <iface-name>\n"
66 "\n"
67 "dut-data-ip, tester-data-ip, dut-ctrl-ip: IPv6 or IPv4-mapped-IPv6 addresses;\n"
68 "\n"
69 "XDP features\n:"
70 "- XDP_PASS\n"
71 "- XDP_DROP\n"
72 "- XDP_ABORTED\n"
73 "- XDP_REDIRECT\n"
74 "- XDP_NDO_XMIT\n"
75 "- XDP_TX\n";
76 
77 static const struct argp_option opts[] = {
78 	{ "verbose", 'v', NULL, 0, "Verbose debug output" },
79 	{ "tester", 't', NULL, 0, "Tester mode" },
80 	{ "feature", 'f', "XDP-FEATURE", 0, "XDP feature to test" },
81 	{ "dut_data_ip", 'D', "DUT-DATA-IP", 0, "DUT IP data channel" },
82 	{ "dut_ctrl_ip", 'C', "DUT-CTRL-IP", 0, "DUT IP control channel" },
83 	{ "tester_data_ip", 'T', "TESTER-DATA-IP", 0, "Tester IP data channel" },
84 	{},
85 };
86 
87 static int get_xdp_feature(const char *arg)
88 {
89 	if (!strcmp(arg, "XDP_PASS")) {
90 		env.feature.action = XDP_PASS;
91 		env.feature.drv_feature = NETDEV_XDP_ACT_BASIC;
92 	} else if (!strcmp(arg, "XDP_DROP")) {
93 		env.feature.drv_feature = NETDEV_XDP_ACT_BASIC;
94 		env.feature.action = XDP_DROP;
95 	} else if (!strcmp(arg, "XDP_ABORTED")) {
96 		env.feature.drv_feature = NETDEV_XDP_ACT_BASIC;
97 		env.feature.action = XDP_ABORTED;
98 	} else if (!strcmp(arg, "XDP_TX")) {
99 		env.feature.drv_feature = NETDEV_XDP_ACT_BASIC;
100 		env.feature.action = XDP_TX;
101 	} else if (!strcmp(arg, "XDP_REDIRECT")) {
102 		env.feature.drv_feature = NETDEV_XDP_ACT_REDIRECT;
103 		env.feature.action = XDP_REDIRECT;
104 	} else if (!strcmp(arg, "XDP_NDO_XMIT")) {
105 		env.feature.drv_feature = NETDEV_XDP_ACT_NDO_XMIT;
106 	} else {
107 		return -EINVAL;
108 	}
109 
110 	return 0;
111 }
112 
113 static char *get_xdp_feature_str(void)
114 {
115 	switch (env.feature.action) {
116 	case XDP_PASS:
117 		return YELLOW("XDP_PASS");
118 	case XDP_DROP:
119 		return YELLOW("XDP_DROP");
120 	case XDP_ABORTED:
121 		return YELLOW("XDP_ABORTED");
122 	case XDP_TX:
123 		return YELLOW("XDP_TX");
124 	case XDP_REDIRECT:
125 		return YELLOW("XDP_REDIRECT");
126 	default:
127 		break;
128 	}
129 
130 	if (env.feature.drv_feature == NETDEV_XDP_ACT_NDO_XMIT)
131 		return YELLOW("XDP_NDO_XMIT");
132 
133 	return "";
134 }
135 
136 static error_t parse_arg(int key, char *arg, struct argp_state *state)
137 {
138 	switch (key) {
139 	case 'v':
140 		env.verbosity = true;
141 		break;
142 	case 't':
143 		env.is_tester = true;
144 		break;
145 	case 'f':
146 		if (get_xdp_feature(arg) < 0) {
147 			fprintf(stderr, "Invalid xdp feature: %s\n", arg);
148 			argp_usage(state);
149 			return ARGP_ERR_UNKNOWN;
150 		}
151 		break;
152 	case 'D':
153 		if (make_sockaddr(AF_INET6, arg, DUT_ECHO_PORT,
154 				  &env.dut_addr, NULL)) {
155 			fprintf(stderr,
156 				"Invalid address assigned to the Device Under Test: %s\n",
157 				arg);
158 			return ARGP_ERR_UNKNOWN;
159 		}
160 		break;
161 	case 'C':
162 		if (make_sockaddr(AF_INET6, arg, DUT_CTRL_PORT,
163 				  &env.dut_ctrl_addr, NULL)) {
164 			fprintf(stderr,
165 				"Invalid address assigned to the Device Under Test: %s\n",
166 				arg);
167 			return ARGP_ERR_UNKNOWN;
168 		}
169 		break;
170 	case 'T':
171 		if (make_sockaddr(AF_INET6, arg, 0, &env.tester_addr, NULL)) {
172 			fprintf(stderr,
173 				"Invalid address assigned to the Tester device: %s\n",
174 				arg);
175 			return ARGP_ERR_UNKNOWN;
176 		}
177 		break;
178 	case ARGP_KEY_ARG:
179 		errno = 0;
180 		if (strlen(arg) >= IF_NAMESIZE) {
181 			fprintf(stderr, "Invalid device name: %s\n", arg);
182 			argp_usage(state);
183 			return ARGP_ERR_UNKNOWN;
184 		}
185 
186 		env.ifindex = if_nametoindex(arg);
187 		if (!env.ifindex)
188 			env.ifindex = strtoul(arg, NULL, 0);
189 		if (!env.ifindex || !if_indextoname(env.ifindex, env.ifname)) {
190 			fprintf(stderr,
191 				"Bad interface index or name (%d): %s\n",
192 				errno, strerror(errno));
193 			argp_usage(state);
194 			return ARGP_ERR_UNKNOWN;
195 		}
196 		break;
197 	default:
198 		return ARGP_ERR_UNKNOWN;
199 	}
200 
201 	return 0;
202 }
203 
204 static const struct argp argp = {
205 	.options = opts,
206 	.parser = parse_arg,
207 	.doc = argp_program_doc,
208 };
209 
210 static void set_env_default(void)
211 {
212 	env.feature.drv_feature = NETDEV_XDP_ACT_NDO_XMIT;
213 	env.feature.action = -EINVAL;
214 	env.ifindex = -ENODEV;
215 	strcpy(env.ifname, "unknown");
216 	make_sockaddr(AF_INET6, "::ffff:127.0.0.1", DUT_CTRL_PORT,
217 		      &env.dut_ctrl_addr, NULL);
218 	make_sockaddr(AF_INET6, "::ffff:127.0.0.1", DUT_ECHO_PORT,
219 		      &env.dut_addr, NULL);
220 	make_sockaddr(AF_INET6, "::ffff:127.0.0.1", 0, &env.tester_addr, NULL);
221 }
222 
223 static void *dut_echo_thread(void *arg)
224 {
225 	unsigned char buf[sizeof(struct tlv_hdr)];
226 	int sockfd = *(int *)arg;
227 
228 	while (!exiting) {
229 		struct tlv_hdr *tlv = (struct tlv_hdr *)buf;
230 		struct sockaddr_storage addr;
231 		socklen_t addrlen;
232 		size_t n;
233 
234 		n = recvfrom(sockfd, buf, sizeof(buf), MSG_WAITALL,
235 			     (struct sockaddr *)&addr, &addrlen);
236 		if (n != ntohs(tlv->len))
237 			continue;
238 
239 		if (ntohs(tlv->type) != CMD_ECHO)
240 			continue;
241 
242 		sendto(sockfd, buf, sizeof(buf), MSG_NOSIGNAL | MSG_CONFIRM,
243 		       (struct sockaddr *)&addr, addrlen);
244 	}
245 
246 	pthread_exit((void *)0);
247 	close(sockfd);
248 
249 	return NULL;
250 }
251 
252 static int dut_run_echo_thread(pthread_t *t, int *sockfd)
253 {
254 	int err;
255 
256 	sockfd = start_reuseport_server(AF_INET6, SOCK_DGRAM, NULL,
257 					DUT_ECHO_PORT, 0, 1);
258 	if (!sockfd) {
259 		fprintf(stderr,
260 			"Failed creating data UDP socket on device %s\n",
261 			env.ifname);
262 		return -errno;
263 	}
264 
265 	/* start echo channel */
266 	err = pthread_create(t, NULL, dut_echo_thread, sockfd);
267 	if (err) {
268 		fprintf(stderr,
269 			"Failed creating data UDP thread on device %s: %s\n",
270 			env.ifname, strerror(-err));
271 		free_fds(sockfd, 1);
272 		return -EINVAL;
273 	}
274 
275 	return 0;
276 }
277 
278 static int dut_attach_xdp_prog(struct xdp_features *skel, int flags)
279 {
280 	enum xdp_action action = env.feature.action;
281 	struct bpf_program *prog;
282 	unsigned int key = 0;
283 	int err, fd = 0;
284 
285 	if (env.feature.drv_feature == NETDEV_XDP_ACT_NDO_XMIT) {
286 		struct bpf_devmap_val entry = {
287 			.ifindex = env.ifindex,
288 		};
289 
290 		err = bpf_map__update_elem(skel->maps.dev_map,
291 					   &key, sizeof(key),
292 					   &entry, sizeof(entry), 0);
293 		if (err < 0)
294 			return err;
295 
296 		fd = bpf_program__fd(skel->progs.xdp_do_redirect_cpumap);
297 		action = XDP_REDIRECT;
298 	}
299 
300 	switch (action) {
301 	case XDP_TX:
302 		prog = skel->progs.xdp_do_tx;
303 		break;
304 	case XDP_DROP:
305 		prog = skel->progs.xdp_do_drop;
306 		break;
307 	case XDP_ABORTED:
308 		prog = skel->progs.xdp_do_aborted;
309 		break;
310 	case XDP_PASS:
311 		prog = skel->progs.xdp_do_pass;
312 		break;
313 	case XDP_REDIRECT: {
314 		struct bpf_cpumap_val entry = {
315 			.qsize = 2048,
316 			.bpf_prog.fd = fd,
317 		};
318 
319 		err = bpf_map__update_elem(skel->maps.cpu_map,
320 					   &key, sizeof(key),
321 					   &entry, sizeof(entry), 0);
322 		if (err < 0)
323 			return err;
324 
325 		prog = skel->progs.xdp_do_redirect;
326 		break;
327 	}
328 	default:
329 		return -EINVAL;
330 	}
331 
332 	err = bpf_xdp_attach(env.ifindex, bpf_program__fd(prog), flags, NULL);
333 	if (err)
334 		fprintf(stderr, "Failed attaching XDP program to device %s\n",
335 			env.ifname);
336 	return err;
337 }
338 
339 static int recv_msg(int sockfd, void *buf, size_t bufsize, void *val,
340 		    size_t val_size)
341 {
342 	struct tlv_hdr *tlv = (struct tlv_hdr *)buf;
343 	size_t len;
344 
345 	len = recv(sockfd, buf, bufsize, 0);
346 	if (len != ntohs(tlv->len) || len < sizeof(*tlv))
347 		return -EINVAL;
348 
349 	if (val) {
350 		len -= sizeof(*tlv);
351 		if (len > val_size)
352 			return -ENOMEM;
353 
354 		memcpy(val, tlv->data, len);
355 	}
356 
357 	return 0;
358 }
359 
360 static int dut_run(struct xdp_features *skel)
361 {
362 	int flags = XDP_FLAGS_UPDATE_IF_NOEXIST | XDP_FLAGS_DRV_MODE;
363 	int state, err, *sockfd, ctrl_sockfd, echo_sockfd;
364 	struct sockaddr_storage ctrl_addr;
365 	pthread_t dut_thread;
366 	socklen_t addrlen;
367 
368 	sockfd = start_reuseport_server(AF_INET6, SOCK_STREAM, NULL,
369 					DUT_CTRL_PORT, 0, 1);
370 	if (!sockfd) {
371 		fprintf(stderr,
372 			"Failed creating control socket on device %s\n", env.ifname);
373 		return -errno;
374 	}
375 
376 	ctrl_sockfd = accept(*sockfd, (struct sockaddr *)&ctrl_addr, &addrlen);
377 	if (ctrl_sockfd < 0) {
378 		fprintf(stderr,
379 			"Failed accepting connections on device %s control socket\n",
380 			env.ifname);
381 		free_fds(sockfd, 1);
382 		return -errno;
383 	}
384 
385 	/* CTRL loop */
386 	while (!exiting) {
387 		unsigned char buf[BUFSIZE] = {};
388 		struct tlv_hdr *tlv = (struct tlv_hdr *)buf;
389 
390 		err = recv_msg(ctrl_sockfd, buf, BUFSIZE, NULL, 0);
391 		if (err)
392 			continue;
393 
394 		switch (ntohs(tlv->type)) {
395 		case CMD_START: {
396 			if (state == CMD_START)
397 				continue;
398 
399 			state = CMD_START;
400 			/* Load the XDP program on the DUT */
401 			err = dut_attach_xdp_prog(skel, flags);
402 			if (err)
403 				goto out;
404 
405 			err = dut_run_echo_thread(&dut_thread, &echo_sockfd);
406 			if (err < 0)
407 				goto out;
408 
409 			tlv->type = htons(CMD_ACK);
410 			tlv->len = htons(sizeof(*tlv));
411 			err = send(ctrl_sockfd, buf, sizeof(*tlv), 0);
412 			if (err < 0)
413 				goto end_thread;
414 			break;
415 		}
416 		case CMD_STOP:
417 			if (state != CMD_START)
418 				break;
419 
420 			state = CMD_STOP;
421 
422 			exiting = true;
423 			bpf_xdp_detach(env.ifindex, flags, NULL);
424 
425 			tlv->type = htons(CMD_ACK);
426 			tlv->len = htons(sizeof(*tlv));
427 			err = send(ctrl_sockfd, buf, sizeof(*tlv), 0);
428 			goto end_thread;
429 		case CMD_GET_XDP_CAP: {
430 			LIBBPF_OPTS(bpf_xdp_query_opts, opts);
431 			unsigned long long val;
432 			size_t n;
433 
434 			err = bpf_xdp_query(env.ifindex, XDP_FLAGS_DRV_MODE,
435 					    &opts);
436 			if (err) {
437 				fprintf(stderr,
438 					"Failed querying XDP cap for device %s\n",
439 					env.ifname);
440 				goto end_thread;
441 			}
442 
443 			tlv->type = htons(CMD_ACK);
444 			n = sizeof(*tlv) + sizeof(opts.feature_flags);
445 			tlv->len = htons(n);
446 
447 			val = htobe64(opts.feature_flags);
448 			memcpy(tlv->data, &val, sizeof(val));
449 
450 			err = send(ctrl_sockfd, buf, n, 0);
451 			if (err < 0)
452 				goto end_thread;
453 			break;
454 		}
455 		case CMD_GET_STATS: {
456 			unsigned int key = 0, val;
457 			size_t n;
458 
459 			err = bpf_map__lookup_elem(skel->maps.dut_stats,
460 						   &key, sizeof(key),
461 						   &val, sizeof(val), 0);
462 			if (err) {
463 				fprintf(stderr,
464 					"bpf_map_lookup_elem failed (%d)\n", err);
465 				goto end_thread;
466 			}
467 
468 			tlv->type = htons(CMD_ACK);
469 			n = sizeof(*tlv) + sizeof(val);
470 			tlv->len = htons(n);
471 
472 			val = htonl(val);
473 			memcpy(tlv->data, &val, sizeof(val));
474 
475 			err = send(ctrl_sockfd, buf, n, 0);
476 			if (err < 0)
477 				goto end_thread;
478 			break;
479 		}
480 		default:
481 			break;
482 		}
483 	}
484 
485 end_thread:
486 	pthread_join(dut_thread, NULL);
487 out:
488 	bpf_xdp_detach(env.ifindex, flags, NULL);
489 	close(ctrl_sockfd);
490 	free_fds(sockfd, 1);
491 
492 	return err;
493 }
494 
495 static bool tester_collect_detected_cap(struct xdp_features *skel,
496 					unsigned int dut_stats)
497 {
498 	unsigned int err, key = 0, val;
499 
500 	if (!dut_stats)
501 		return false;
502 
503 	err = bpf_map__lookup_elem(skel->maps.stats, &key, sizeof(key),
504 				   &val, sizeof(val), 0);
505 	if (err) {
506 		fprintf(stderr, "bpf_map_lookup_elem failed (%d)\n", err);
507 		return false;
508 	}
509 
510 	switch (env.feature.action) {
511 	case XDP_PASS:
512 	case XDP_TX:
513 	case XDP_REDIRECT:
514 		return val > 0;
515 	case XDP_DROP:
516 	case XDP_ABORTED:
517 		return val == 0;
518 	default:
519 		break;
520 	}
521 
522 	if (env.feature.drv_feature == NETDEV_XDP_ACT_NDO_XMIT)
523 		return val > 0;
524 
525 	return false;
526 }
527 
528 static int send_and_recv_msg(int sockfd, enum test_commands cmd, void *val,
529 			     size_t val_size)
530 {
531 	unsigned char buf[BUFSIZE] = {};
532 	struct tlv_hdr *tlv = (struct tlv_hdr *)buf;
533 	int err;
534 
535 	tlv->type = htons(cmd);
536 	tlv->len = htons(sizeof(*tlv));
537 
538 	err = send(sockfd, buf, sizeof(*tlv), 0);
539 	if (err < 0)
540 		return err;
541 
542 	err = recv_msg(sockfd, buf, BUFSIZE, val, val_size);
543 	if (err < 0)
544 		return err;
545 
546 	return ntohs(tlv->type) == CMD_ACK ? 0 : -EINVAL;
547 }
548 
549 static int send_echo_msg(void)
550 {
551 	unsigned char buf[sizeof(struct tlv_hdr)];
552 	struct tlv_hdr *tlv = (struct tlv_hdr *)buf;
553 	int sockfd, n;
554 
555 	sockfd = socket(AF_INET6, SOCK_DGRAM, 0);
556 	if (sockfd < 0) {
557 		fprintf(stderr,
558 			"Failed creating data UDP socket on device %s\n",
559 			env.ifname);
560 		return -errno;
561 	}
562 
563 	tlv->type = htons(CMD_ECHO);
564 	tlv->len = htons(sizeof(*tlv));
565 
566 	n = sendto(sockfd, buf, sizeof(*tlv), MSG_NOSIGNAL | MSG_CONFIRM,
567 		   (struct sockaddr *)&env.dut_addr, sizeof(env.dut_addr));
568 	close(sockfd);
569 
570 	return n == ntohs(tlv->len) ? 0 : -EINVAL;
571 }
572 
573 static int tester_run(struct xdp_features *skel)
574 {
575 	int flags = XDP_FLAGS_UPDATE_IF_NOEXIST | XDP_FLAGS_DRV_MODE;
576 	unsigned long long advertised_feature;
577 	struct bpf_program *prog;
578 	unsigned int stats;
579 	int i, err, sockfd;
580 	bool detected_cap;
581 
582 	sockfd = socket(AF_INET6, SOCK_STREAM, 0);
583 	if (sockfd < 0) {
584 		fprintf(stderr,
585 			"Failed creating tester service control socket\n");
586 		return -errno;
587 	}
588 
589 	if (settimeo(sockfd, 1000) < 0)
590 		return -EINVAL;
591 
592 	err = connect(sockfd, (struct sockaddr *)&env.dut_ctrl_addr,
593 		      sizeof(env.dut_ctrl_addr));
594 	if (err) {
595 		fprintf(stderr,
596 			"Failed connecting to the Device Under Test control socket\n");
597 		return -errno;
598 	}
599 
600 	err = send_and_recv_msg(sockfd, CMD_GET_XDP_CAP, &advertised_feature,
601 				sizeof(advertised_feature));
602 	if (err < 0) {
603 		close(sockfd);
604 		return err;
605 	}
606 
607 	advertised_feature = be64toh(advertised_feature);
608 
609 	if (env.feature.drv_feature == NETDEV_XDP_ACT_NDO_XMIT ||
610 	    env.feature.action == XDP_TX)
611 		prog = skel->progs.xdp_tester_check_tx;
612 	else
613 		prog = skel->progs.xdp_tester_check_rx;
614 
615 	err = bpf_xdp_attach(env.ifindex, bpf_program__fd(prog), flags, NULL);
616 	if (err) {
617 		fprintf(stderr, "Failed attaching XDP program to device %s\n",
618 			env.ifname);
619 		goto out;
620 	}
621 
622 	err = send_and_recv_msg(sockfd, CMD_START, NULL, 0);
623 	if (err)
624 		goto out;
625 
626 	for (i = 0; i < 10 && !exiting; i++) {
627 		err = send_echo_msg();
628 		if (err < 0)
629 			goto out;
630 
631 		sleep(1);
632 	}
633 
634 	err = send_and_recv_msg(sockfd, CMD_GET_STATS, &stats, sizeof(stats));
635 	if (err)
636 		goto out;
637 
638 	/* stop the test */
639 	err = send_and_recv_msg(sockfd, CMD_STOP, NULL, 0);
640 	/* send a new echo message to wake echo thread of the dut */
641 	send_echo_msg();
642 
643 	detected_cap = tester_collect_detected_cap(skel, ntohl(stats));
644 
645 	fprintf(stdout, "Feature %s: [%s][%s]\n", get_xdp_feature_str(),
646 		detected_cap ? GREEN("DETECTED") : RED("NOT DETECTED"),
647 		env.feature.drv_feature & advertised_feature ? GREEN("ADVERTISED")
648 							     : RED("NOT ADVERTISED"));
649 out:
650 	bpf_xdp_detach(env.ifindex, flags, NULL);
651 	close(sockfd);
652 	return err < 0 ? err : 0;
653 }
654 
655 int main(int argc, char **argv)
656 {
657 	struct xdp_features *skel;
658 	int err;
659 
660 	libbpf_set_strict_mode(LIBBPF_STRICT_ALL);
661 	libbpf_set_print(libbpf_print_fn);
662 
663 	signal(SIGINT, sig_handler);
664 	signal(SIGTERM, sig_handler);
665 
666 	set_env_default();
667 
668 	/* Parse command line arguments */
669 	err = argp_parse(&argp, argc, argv, 0, NULL, NULL);
670 	if (err)
671 		return err;
672 
673 	if (env.ifindex < 0) {
674 		fprintf(stderr, "Invalid device name %s\n", env.ifname);
675 		return -ENODEV;
676 	}
677 
678 	/* Load and verify BPF application */
679 	skel = xdp_features__open();
680 	if (!skel) {
681 		fprintf(stderr, "Failed to open and load BPF skeleton\n");
682 		return -EINVAL;
683 	}
684 
685 	skel->rodata->tester_addr =
686 		((struct sockaddr_in6 *)&env.tester_addr)->sin6_addr;
687 	skel->rodata->dut_addr =
688 		((struct sockaddr_in6 *)&env.dut_addr)->sin6_addr;
689 
690 	/* Load & verify BPF programs */
691 	err = xdp_features__load(skel);
692 	if (err) {
693 		fprintf(stderr, "Failed to load and verify BPF skeleton\n");
694 		goto cleanup;
695 	}
696 
697 	err = xdp_features__attach(skel);
698 	if (err) {
699 		fprintf(stderr, "Failed to attach BPF skeleton\n");
700 		goto cleanup;
701 	}
702 
703 	if (env.is_tester) {
704 		/* Tester */
705 		fprintf(stdout, "Starting tester service on device %s\n",
706 			env.ifname);
707 		err = tester_run(skel);
708 	} else {
709 		/* DUT */
710 		fprintf(stdout, "Starting test on device %s\n", env.ifname);
711 		err = dut_run(skel);
712 	}
713 
714 cleanup:
715 	xdp_features__destroy(skel);
716 
717 	return err < 0 ? -err : 0;
718 }
719