|  | // SPDX-License-Identifier: GPL-2.0 | 
|  |  | 
|  | /** | 
|  | * Test XDP bonding support | 
|  | * | 
|  | * Sets up two bonded veth pairs between two fresh namespaces | 
|  | * and verifies that XDP_TX program loaded on a bond device | 
|  | * are correctly loaded onto the slave devices and XDP_TX'd | 
|  | * packets are balanced using bonding. | 
|  | */ | 
|  |  | 
|  | #define _GNU_SOURCE | 
|  | #include <sched.h> | 
|  | #include <net/if.h> | 
|  | #include <linux/if_link.h> | 
|  | #include "test_progs.h" | 
|  | #include "network_helpers.h" | 
|  | #include <linux/if_bonding.h> | 
|  | #include <linux/limits.h> | 
|  | #include <linux/udp.h> | 
|  |  | 
|  | #include "xdp_dummy.skel.h" | 
|  | #include "xdp_redirect_multi_kern.skel.h" | 
|  | #include "xdp_tx.skel.h" | 
|  |  | 
|  | #define BOND1_MAC {0x00, 0x11, 0x22, 0x33, 0x44, 0x55} | 
|  | #define BOND1_MAC_STR "00:11:22:33:44:55" | 
|  | #define BOND2_MAC {0x00, 0x22, 0x33, 0x44, 0x55, 0x66} | 
|  | #define BOND2_MAC_STR "00:22:33:44:55:66" | 
|  | #define NPACKETS 100 | 
|  |  | 
|  | static int root_netns_fd = -1; | 
|  |  | 
|  | static void restore_root_netns(void) | 
|  | { | 
|  | ASSERT_OK(setns(root_netns_fd, CLONE_NEWNET), "restore_root_netns"); | 
|  | } | 
|  |  | 
|  | static int setns_by_name(char *name) | 
|  | { | 
|  | int nsfd, err; | 
|  | char nspath[PATH_MAX]; | 
|  |  | 
|  | snprintf(nspath, sizeof(nspath), "%s/%s", "/var/run/netns", name); | 
|  | nsfd = open(nspath, O_RDONLY | O_CLOEXEC); | 
|  | if (nsfd < 0) | 
|  | return -1; | 
|  |  | 
|  | err = setns(nsfd, CLONE_NEWNET); | 
|  | close(nsfd); | 
|  | return err; | 
|  | } | 
|  |  | 
|  | static int get_rx_packets(const char *iface) | 
|  | { | 
|  | FILE *f; | 
|  | char line[512]; | 
|  | int iface_len = strlen(iface); | 
|  |  | 
|  | f = fopen("/proc/net/dev", "r"); | 
|  | if (!f) | 
|  | return -1; | 
|  |  | 
|  | while (fgets(line, sizeof(line), f)) { | 
|  | char *p = line; | 
|  |  | 
|  | while (*p == ' ') | 
|  | p++; /* skip whitespace */ | 
|  | if (!strncmp(p, iface, iface_len)) { | 
|  | p += iface_len; | 
|  | if (*p++ != ':') | 
|  | continue; | 
|  | while (*p == ' ') | 
|  | p++; /* skip whitespace */ | 
|  | while (*p && *p != ' ') | 
|  | p++; /* skip rx bytes */ | 
|  | while (*p == ' ') | 
|  | p++; /* skip whitespace */ | 
|  | fclose(f); | 
|  | return atoi(p); | 
|  | } | 
|  | } | 
|  | fclose(f); | 
|  | return -1; | 
|  | } | 
|  |  | 
|  | #define MAX_BPF_LINKS 8 | 
|  |  | 
|  | struct skeletons { | 
|  | struct xdp_dummy *xdp_dummy; | 
|  | struct xdp_tx *xdp_tx; | 
|  | struct xdp_redirect_multi_kern *xdp_redirect_multi_kern; | 
|  |  | 
|  | int nlinks; | 
|  | struct bpf_link *links[MAX_BPF_LINKS]; | 
|  | }; | 
|  |  | 
|  | static int xdp_attach(struct skeletons *skeletons, struct bpf_program *prog, char *iface) | 
|  | { | 
|  | struct bpf_link *link; | 
|  | int ifindex; | 
|  |  | 
|  | ifindex = if_nametoindex(iface); | 
|  | if (!ASSERT_GT(ifindex, 0, "get ifindex")) | 
|  | return -1; | 
|  |  | 
|  | if (!ASSERT_LE(skeletons->nlinks+1, MAX_BPF_LINKS, "too many XDP programs attached")) | 
|  | return -1; | 
|  |  | 
|  | link = bpf_program__attach_xdp(prog, ifindex); | 
|  | if (!ASSERT_OK_PTR(link, "attach xdp program")) | 
|  | return -1; | 
|  |  | 
|  | skeletons->links[skeletons->nlinks++] = link; | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | enum { | 
|  | BOND_ONE_NO_ATTACH = 0, | 
|  | BOND_BOTH_AND_ATTACH, | 
|  | }; | 
|  |  | 
|  | static const char * const mode_names[] = { | 
|  | [BOND_MODE_ROUNDROBIN]   = "balance-rr", | 
|  | [BOND_MODE_ACTIVEBACKUP] = "active-backup", | 
|  | [BOND_MODE_XOR]          = "balance-xor", | 
|  | [BOND_MODE_BROADCAST]    = "broadcast", | 
|  | [BOND_MODE_8023AD]       = "802.3ad", | 
|  | [BOND_MODE_TLB]          = "balance-tlb", | 
|  | [BOND_MODE_ALB]          = "balance-alb", | 
|  | }; | 
|  |  | 
|  | static const char * const xmit_policy_names[] = { | 
|  | [BOND_XMIT_POLICY_LAYER2]       = "layer2", | 
|  | [BOND_XMIT_POLICY_LAYER34]      = "layer3+4", | 
|  | [BOND_XMIT_POLICY_LAYER23]      = "layer2+3", | 
|  | [BOND_XMIT_POLICY_ENCAP23]      = "encap2+3", | 
|  | [BOND_XMIT_POLICY_ENCAP34]      = "encap3+4", | 
|  | }; | 
|  |  | 
|  | static int bonding_setup(struct skeletons *skeletons, int mode, int xmit_policy, | 
|  | int bond_both_attach) | 
|  | { | 
|  | #define SYS(fmt, ...)						\ | 
|  | ({							\ | 
|  | char cmd[1024];					\ | 
|  | snprintf(cmd, sizeof(cmd), fmt, ##__VA_ARGS__);	\ | 
|  | if (!ASSERT_OK(system(cmd), cmd))		\ | 
|  | return -1;				\ | 
|  | }) | 
|  |  | 
|  | SYS("ip netns add ns_dst"); | 
|  | SYS("ip link add veth1_1 type veth peer name veth2_1 netns ns_dst"); | 
|  | SYS("ip link add veth1_2 type veth peer name veth2_2 netns ns_dst"); | 
|  |  | 
|  | SYS("ip link add bond1 type bond mode %s xmit_hash_policy %s", | 
|  | mode_names[mode], xmit_policy_names[xmit_policy]); | 
|  | SYS("ip link set bond1 up address " BOND1_MAC_STR " addrgenmode none"); | 
|  | SYS("ip -netns ns_dst link add bond2 type bond mode %s xmit_hash_policy %s", | 
|  | mode_names[mode], xmit_policy_names[xmit_policy]); | 
|  | SYS("ip -netns ns_dst link set bond2 up address " BOND2_MAC_STR " addrgenmode none"); | 
|  |  | 
|  | SYS("ip link set veth1_1 master bond1"); | 
|  | if (bond_both_attach == BOND_BOTH_AND_ATTACH) { | 
|  | SYS("ip link set veth1_2 master bond1"); | 
|  | } else { | 
|  | SYS("ip link set veth1_2 up addrgenmode none"); | 
|  |  | 
|  | if (xdp_attach(skeletons, skeletons->xdp_dummy->progs.xdp_dummy_prog, "veth1_2")) | 
|  | return -1; | 
|  | } | 
|  |  | 
|  | SYS("ip -netns ns_dst link set veth2_1 master bond2"); | 
|  |  | 
|  | if (bond_both_attach == BOND_BOTH_AND_ATTACH) | 
|  | SYS("ip -netns ns_dst link set veth2_2 master bond2"); | 
|  | else | 
|  | SYS("ip -netns ns_dst link set veth2_2 up addrgenmode none"); | 
|  |  | 
|  | /* Load a dummy program on sending side as with veth peer needs to have a | 
|  | * XDP program loaded as well. | 
|  | */ | 
|  | if (xdp_attach(skeletons, skeletons->xdp_dummy->progs.xdp_dummy_prog, "bond1")) | 
|  | return -1; | 
|  |  | 
|  | if (bond_both_attach == BOND_BOTH_AND_ATTACH) { | 
|  | if (!ASSERT_OK(setns_by_name("ns_dst"), "set netns to ns_dst")) | 
|  | return -1; | 
|  |  | 
|  | if (xdp_attach(skeletons, skeletons->xdp_tx->progs.xdp_tx, "bond2")) | 
|  | return -1; | 
|  |  | 
|  | restore_root_netns(); | 
|  | } | 
|  |  | 
|  | return 0; | 
|  |  | 
|  | #undef SYS | 
|  | } | 
|  |  | 
|  | static void bonding_cleanup(struct skeletons *skeletons) | 
|  | { | 
|  | restore_root_netns(); | 
|  | while (skeletons->nlinks) { | 
|  | skeletons->nlinks--; | 
|  | bpf_link__destroy(skeletons->links[skeletons->nlinks]); | 
|  | } | 
|  | ASSERT_OK(system("ip link delete bond1"), "delete bond1"); | 
|  | ASSERT_OK(system("ip link delete veth1_1"), "delete veth1_1"); | 
|  | ASSERT_OK(system("ip link delete veth1_2"), "delete veth1_2"); | 
|  | ASSERT_OK(system("ip netns delete ns_dst"), "delete ns_dst"); | 
|  | } | 
|  |  | 
|  | static int send_udp_packets(int vary_dst_ip) | 
|  | { | 
|  | struct ethhdr eh = { | 
|  | .h_source = BOND1_MAC, | 
|  | .h_dest = BOND2_MAC, | 
|  | .h_proto = htons(ETH_P_IP), | 
|  | }; | 
|  | struct iphdr iph = {}; | 
|  | struct udphdr uh = {}; | 
|  | uint8_t buf[128]; | 
|  | int i, s = -1; | 
|  | int ifindex; | 
|  |  | 
|  | s = socket(AF_PACKET, SOCK_RAW, IPPROTO_RAW); | 
|  | if (!ASSERT_GE(s, 0, "socket")) | 
|  | goto err; | 
|  |  | 
|  | ifindex = if_nametoindex("bond1"); | 
|  | if (!ASSERT_GT(ifindex, 0, "get bond1 ifindex")) | 
|  | goto err; | 
|  |  | 
|  | iph.ihl = 5; | 
|  | iph.version = 4; | 
|  | iph.tos = 16; | 
|  | iph.id = 1; | 
|  | iph.ttl = 64; | 
|  | iph.protocol = IPPROTO_UDP; | 
|  | iph.saddr = 1; | 
|  | iph.daddr = 2; | 
|  | iph.tot_len = htons(sizeof(buf) - ETH_HLEN); | 
|  | iph.check = 0; | 
|  |  | 
|  | for (i = 1; i <= NPACKETS; i++) { | 
|  | int n; | 
|  | struct sockaddr_ll saddr_ll = { | 
|  | .sll_ifindex = ifindex, | 
|  | .sll_halen = ETH_ALEN, | 
|  | .sll_addr = BOND2_MAC, | 
|  | }; | 
|  |  | 
|  | /* vary the UDP destination port for even distribution with roundrobin/xor modes */ | 
|  | uh.dest++; | 
|  |  | 
|  | if (vary_dst_ip) | 
|  | iph.daddr++; | 
|  |  | 
|  | /* construct a packet */ | 
|  | memcpy(buf, &eh, sizeof(eh)); | 
|  | memcpy(buf + sizeof(eh), &iph, sizeof(iph)); | 
|  | memcpy(buf + sizeof(eh) + sizeof(iph), &uh, sizeof(uh)); | 
|  |  | 
|  | n = sendto(s, buf, sizeof(buf), 0, (struct sockaddr *)&saddr_ll, sizeof(saddr_ll)); | 
|  | if (!ASSERT_EQ(n, sizeof(buf), "sendto")) | 
|  | goto err; | 
|  | } | 
|  |  | 
|  | return 0; | 
|  |  | 
|  | err: | 
|  | if (s >= 0) | 
|  | close(s); | 
|  | return -1; | 
|  | } | 
|  |  | 
|  | static void test_xdp_bonding_with_mode(struct skeletons *skeletons, int mode, int xmit_policy) | 
|  | { | 
|  | int bond1_rx; | 
|  |  | 
|  | if (bonding_setup(skeletons, mode, xmit_policy, BOND_BOTH_AND_ATTACH)) | 
|  | goto out; | 
|  |  | 
|  | if (send_udp_packets(xmit_policy != BOND_XMIT_POLICY_LAYER34)) | 
|  | goto out; | 
|  |  | 
|  | bond1_rx = get_rx_packets("bond1"); | 
|  | ASSERT_EQ(bond1_rx, NPACKETS, "expected more received packets"); | 
|  |  | 
|  | switch (mode) { | 
|  | case BOND_MODE_ROUNDROBIN: | 
|  | case BOND_MODE_XOR: { | 
|  | int veth1_rx = get_rx_packets("veth1_1"); | 
|  | int veth2_rx = get_rx_packets("veth1_2"); | 
|  | int diff = abs(veth1_rx - veth2_rx); | 
|  |  | 
|  | ASSERT_GE(veth1_rx + veth2_rx, NPACKETS, "expected more packets"); | 
|  |  | 
|  | switch (xmit_policy) { | 
|  | case BOND_XMIT_POLICY_LAYER2: | 
|  | ASSERT_GE(diff, NPACKETS, | 
|  | "expected packets on only one of the interfaces"); | 
|  | break; | 
|  | case BOND_XMIT_POLICY_LAYER23: | 
|  | case BOND_XMIT_POLICY_LAYER34: | 
|  | ASSERT_LT(diff, NPACKETS/2, | 
|  | "expected even distribution of packets"); | 
|  | break; | 
|  | default: | 
|  | PRINT_FAIL("Unimplemented xmit_policy=%d\n", xmit_policy); | 
|  | break; | 
|  | } | 
|  | break; | 
|  | } | 
|  | case BOND_MODE_ACTIVEBACKUP: { | 
|  | int veth1_rx = get_rx_packets("veth1_1"); | 
|  | int veth2_rx = get_rx_packets("veth1_2"); | 
|  | int diff = abs(veth1_rx - veth2_rx); | 
|  |  | 
|  | ASSERT_GE(diff, NPACKETS, | 
|  | "expected packets on only one of the interfaces"); | 
|  | break; | 
|  | } | 
|  | default: | 
|  | PRINT_FAIL("Unimplemented xmit_policy=%d\n", xmit_policy); | 
|  | break; | 
|  | } | 
|  |  | 
|  | out: | 
|  | bonding_cleanup(skeletons); | 
|  | } | 
|  |  | 
|  | /* Test the broadcast redirection using xdp_redirect_map_multi_prog and adding | 
|  | * all the interfaces to it and checking that broadcasting won't send the packet | 
|  | * to neither the ingress bond device (bond2) or its slave (veth2_1). | 
|  | */ | 
|  | static void test_xdp_bonding_redirect_multi(struct skeletons *skeletons) | 
|  | { | 
|  | static const char * const ifaces[] = {"bond2", "veth2_1", "veth2_2"}; | 
|  | int veth1_1_rx, veth1_2_rx; | 
|  | int err; | 
|  |  | 
|  | if (bonding_setup(skeletons, BOND_MODE_ROUNDROBIN, BOND_XMIT_POLICY_LAYER23, | 
|  | BOND_ONE_NO_ATTACH)) | 
|  | goto out; | 
|  |  | 
|  |  | 
|  | if (!ASSERT_OK(setns_by_name("ns_dst"), "could not set netns to ns_dst")) | 
|  | goto out; | 
|  |  | 
|  | /* populate the devmap with the relevant interfaces */ | 
|  | for (int i = 0; i < ARRAY_SIZE(ifaces); i++) { | 
|  | int ifindex = if_nametoindex(ifaces[i]); | 
|  | int map_fd = bpf_map__fd(skeletons->xdp_redirect_multi_kern->maps.map_all); | 
|  |  | 
|  | if (!ASSERT_GT(ifindex, 0, "could not get interface index")) | 
|  | goto out; | 
|  |  | 
|  | err = bpf_map_update_elem(map_fd, &ifindex, &ifindex, 0); | 
|  | if (!ASSERT_OK(err, "add interface to map_all")) | 
|  | goto out; | 
|  | } | 
|  |  | 
|  | if (xdp_attach(skeletons, | 
|  | skeletons->xdp_redirect_multi_kern->progs.xdp_redirect_map_multi_prog, | 
|  | "bond2")) | 
|  | goto out; | 
|  |  | 
|  | restore_root_netns(); | 
|  |  | 
|  | if (send_udp_packets(BOND_MODE_ROUNDROBIN)) | 
|  | goto out; | 
|  |  | 
|  | veth1_1_rx = get_rx_packets("veth1_1"); | 
|  | veth1_2_rx = get_rx_packets("veth1_2"); | 
|  |  | 
|  | ASSERT_EQ(veth1_1_rx, 0, "expected no packets on veth1_1"); | 
|  | ASSERT_GE(veth1_2_rx, NPACKETS, "expected packets on veth1_2"); | 
|  |  | 
|  | out: | 
|  | restore_root_netns(); | 
|  | bonding_cleanup(skeletons); | 
|  | } | 
|  |  | 
|  | /* Test that XDP programs cannot be attached to both the bond master and slaves simultaneously */ | 
|  | static void test_xdp_bonding_attach(struct skeletons *skeletons) | 
|  | { | 
|  | struct bpf_link *link = NULL; | 
|  | struct bpf_link *link2 = NULL; | 
|  | int veth, bond, err; | 
|  |  | 
|  | if (!ASSERT_OK(system("ip link add veth type veth"), "add veth")) | 
|  | goto out; | 
|  | if (!ASSERT_OK(system("ip link add bond type bond"), "add bond")) | 
|  | goto out; | 
|  |  | 
|  | veth = if_nametoindex("veth"); | 
|  | if (!ASSERT_GE(veth, 0, "if_nametoindex veth")) | 
|  | goto out; | 
|  | bond = if_nametoindex("bond"); | 
|  | if (!ASSERT_GE(bond, 0, "if_nametoindex bond")) | 
|  | goto out; | 
|  |  | 
|  | /* enslaving with a XDP program loaded is allowed */ | 
|  | link = bpf_program__attach_xdp(skeletons->xdp_dummy->progs.xdp_dummy_prog, veth); | 
|  | if (!ASSERT_OK_PTR(link, "attach program to veth")) | 
|  | goto out; | 
|  |  | 
|  | err = system("ip link set veth master bond"); | 
|  | if (!ASSERT_OK(err, "set veth master")) | 
|  | goto out; | 
|  |  | 
|  | bpf_link__destroy(link); | 
|  | link = NULL; | 
|  |  | 
|  | /* attaching to slave when master has no program is allowed */ | 
|  | link = bpf_program__attach_xdp(skeletons->xdp_dummy->progs.xdp_dummy_prog, veth); | 
|  | if (!ASSERT_OK_PTR(link, "attach program to slave when enslaved")) | 
|  | goto out; | 
|  |  | 
|  | /* attaching to master not allowed when slave has program loaded */ | 
|  | link2 = bpf_program__attach_xdp(skeletons->xdp_dummy->progs.xdp_dummy_prog, bond); | 
|  | if (!ASSERT_ERR_PTR(link2, "attach program to master when slave has program")) | 
|  | goto out; | 
|  |  | 
|  | bpf_link__destroy(link); | 
|  | link = NULL; | 
|  |  | 
|  | /* attaching XDP program to master allowed when slave has no program */ | 
|  | link = bpf_program__attach_xdp(skeletons->xdp_dummy->progs.xdp_dummy_prog, bond); | 
|  | if (!ASSERT_OK_PTR(link, "attach program to master")) | 
|  | goto out; | 
|  |  | 
|  | /* attaching to slave not allowed when master has program loaded */ | 
|  | link2 = bpf_program__attach_xdp(skeletons->xdp_dummy->progs.xdp_dummy_prog, veth); | 
|  | if (!ASSERT_ERR_PTR(link2, "attach program to slave when master has program")) | 
|  | goto out; | 
|  |  | 
|  | bpf_link__destroy(link); | 
|  | link = NULL; | 
|  |  | 
|  | /* test program unwinding with a non-XDP slave */ | 
|  | if (!ASSERT_OK(system("ip link add vxlan type vxlan id 1 remote 1.2.3.4 dstport 0 dev lo"), | 
|  | "add vxlan")) | 
|  | goto out; | 
|  |  | 
|  | err = system("ip link set vxlan master bond"); | 
|  | if (!ASSERT_OK(err, "set vxlan master")) | 
|  | goto out; | 
|  |  | 
|  | /* attaching not allowed when one slave does not support XDP */ | 
|  | link = bpf_program__attach_xdp(skeletons->xdp_dummy->progs.xdp_dummy_prog, bond); | 
|  | if (!ASSERT_ERR_PTR(link, "attach program to master when slave does not support XDP")) | 
|  | goto out; | 
|  |  | 
|  | out: | 
|  | bpf_link__destroy(link); | 
|  | bpf_link__destroy(link2); | 
|  |  | 
|  | system("ip link del veth"); | 
|  | system("ip link del bond"); | 
|  | system("ip link del vxlan"); | 
|  | } | 
|  |  | 
|  | /* Test with nested bonding devices to catch issue with negative jump label count */ | 
|  | static void test_xdp_bonding_nested(struct skeletons *skeletons) | 
|  | { | 
|  | struct bpf_link *link = NULL; | 
|  | int bond, err; | 
|  |  | 
|  | if (!ASSERT_OK(system("ip link add bond type bond"), "add bond")) | 
|  | goto out; | 
|  |  | 
|  | bond = if_nametoindex("bond"); | 
|  | if (!ASSERT_GE(bond, 0, "if_nametoindex bond")) | 
|  | goto out; | 
|  |  | 
|  | if (!ASSERT_OK(system("ip link add bond_nest1 type bond"), "add bond_nest1")) | 
|  | goto out; | 
|  |  | 
|  | err = system("ip link set bond_nest1 master bond"); | 
|  | if (!ASSERT_OK(err, "set bond_nest1 master")) | 
|  | goto out; | 
|  |  | 
|  | if (!ASSERT_OK(system("ip link add bond_nest2 type bond"), "add bond_nest1")) | 
|  | goto out; | 
|  |  | 
|  | err = system("ip link set bond_nest2 master bond_nest1"); | 
|  | if (!ASSERT_OK(err, "set bond_nest2 master")) | 
|  | goto out; | 
|  |  | 
|  | link = bpf_program__attach_xdp(skeletons->xdp_dummy->progs.xdp_dummy_prog, bond); | 
|  | ASSERT_OK_PTR(link, "attach program to master"); | 
|  |  | 
|  | out: | 
|  | bpf_link__destroy(link); | 
|  | system("ip link del bond"); | 
|  | system("ip link del bond_nest1"); | 
|  | system("ip link del bond_nest2"); | 
|  | } | 
|  |  | 
|  | static int libbpf_debug_print(enum libbpf_print_level level, | 
|  | const char *format, va_list args) | 
|  | { | 
|  | if (level != LIBBPF_WARN) | 
|  | vprintf(format, args); | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | struct bond_test_case { | 
|  | char *name; | 
|  | int mode; | 
|  | int xmit_policy; | 
|  | }; | 
|  |  | 
|  | static struct bond_test_case bond_test_cases[] = { | 
|  | { "xdp_bonding_roundrobin", BOND_MODE_ROUNDROBIN, BOND_XMIT_POLICY_LAYER23, }, | 
|  | { "xdp_bonding_activebackup", BOND_MODE_ACTIVEBACKUP, BOND_XMIT_POLICY_LAYER23 }, | 
|  |  | 
|  | { "xdp_bonding_xor_layer2", BOND_MODE_XOR, BOND_XMIT_POLICY_LAYER2, }, | 
|  | { "xdp_bonding_xor_layer23", BOND_MODE_XOR, BOND_XMIT_POLICY_LAYER23, }, | 
|  | { "xdp_bonding_xor_layer34", BOND_MODE_XOR, BOND_XMIT_POLICY_LAYER34, }, | 
|  | }; | 
|  |  | 
|  | void serial_test_xdp_bonding(void) | 
|  | { | 
|  | libbpf_print_fn_t old_print_fn; | 
|  | struct skeletons skeletons = {}; | 
|  | int i; | 
|  |  | 
|  | old_print_fn = libbpf_set_print(libbpf_debug_print); | 
|  |  | 
|  | root_netns_fd = open("/proc/self/ns/net", O_RDONLY); | 
|  | if (!ASSERT_GE(root_netns_fd, 0, "open /proc/self/ns/net")) | 
|  | goto out; | 
|  |  | 
|  | skeletons.xdp_dummy = xdp_dummy__open_and_load(); | 
|  | if (!ASSERT_OK_PTR(skeletons.xdp_dummy, "xdp_dummy__open_and_load")) | 
|  | goto out; | 
|  |  | 
|  | skeletons.xdp_tx = xdp_tx__open_and_load(); | 
|  | if (!ASSERT_OK_PTR(skeletons.xdp_tx, "xdp_tx__open_and_load")) | 
|  | goto out; | 
|  |  | 
|  | skeletons.xdp_redirect_multi_kern = xdp_redirect_multi_kern__open_and_load(); | 
|  | if (!ASSERT_OK_PTR(skeletons.xdp_redirect_multi_kern, | 
|  | "xdp_redirect_multi_kern__open_and_load")) | 
|  | goto out; | 
|  |  | 
|  | if (test__start_subtest("xdp_bonding_attach")) | 
|  | test_xdp_bonding_attach(&skeletons); | 
|  |  | 
|  | if (test__start_subtest("xdp_bonding_nested")) | 
|  | test_xdp_bonding_nested(&skeletons); | 
|  |  | 
|  | for (i = 0; i < ARRAY_SIZE(bond_test_cases); i++) { | 
|  | struct bond_test_case *test_case = &bond_test_cases[i]; | 
|  |  | 
|  | if (test__start_subtest(test_case->name)) | 
|  | test_xdp_bonding_with_mode( | 
|  | &skeletons, | 
|  | test_case->mode, | 
|  | test_case->xmit_policy); | 
|  | } | 
|  |  | 
|  | if (test__start_subtest("xdp_bonding_redirect_multi")) | 
|  | test_xdp_bonding_redirect_multi(&skeletons); | 
|  |  | 
|  | out: | 
|  | xdp_dummy__destroy(skeletons.xdp_dummy); | 
|  | xdp_tx__destroy(skeletons.xdp_tx); | 
|  | xdp_redirect_multi_kern__destroy(skeletons.xdp_redirect_multi_kern); | 
|  |  | 
|  | libbpf_set_print(old_print_fn); | 
|  | if (root_netns_fd >= 0) | 
|  | close(root_netns_fd); | 
|  | } |