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
test__fail(void)42 void test__fail(void) { /* for network_helpers.c */ }
43
libbpf_print_fn(enum libbpf_print_level level,const char * format,va_list args)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
sig_handler(int sig)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
get_xdp_feature(const char * arg)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
get_xdp_feature_str(void)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
parse_arg(int key,char * arg,struct argp_state * state)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
set_env_default(void)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
dut_echo_thread(void * arg)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
dut_run_echo_thread(pthread_t * t,int * sockfd)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
dut_attach_xdp_prog(struct xdp_features * skel,int flags)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
recv_msg(int sockfd,void * buf,size_t bufsize,void * val,size_t val_size)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
dut_run(struct xdp_features * skel)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
tester_collect_detected_cap(struct xdp_features * skel,unsigned int dut_stats)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
send_and_recv_msg(int sockfd,enum test_commands cmd,void * val,size_t val_size)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
send_echo_msg(void)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
tester_run(struct xdp_features * skel)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
main(int argc,char ** argv)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