Parametrize the SK_REUSEPORT tests so that the map type for storing sockets can be selected at run-time. Also allow choosing which L4 protocols get tested. Run the extended reuseport program test two times, once for REUSEPORT_ARRAY, and once for SOCKMAP but just with TCP to cover the newly enabled map type. Signed-off-by: Jakub Sitnicki <jakub@xxxxxxxxxxxxxx> --- tools/testing/selftests/bpf/Makefile | 7 +- .../selftests/bpf/test_select_reuseport.c | 141 ++++++++++++++---- .../selftests/bpf/test_select_reuseport.sh | 14 ++ 3 files changed, 131 insertions(+), 31 deletions(-) create mode 100755 tools/testing/selftests/bpf/test_select_reuseport.sh diff --git a/tools/testing/selftests/bpf/Makefile b/tools/testing/selftests/bpf/Makefile index 4ff5f4aada08..a32646f2e80a 100644 --- a/tools/testing/selftests/bpf/Makefile +++ b/tools/testing/selftests/bpf/Makefile @@ -28,7 +28,7 @@ LDLIBS += -lcap -lelf -lrt -lpthread TEST_GEN_PROGS = test_verifier test_tag test_maps test_lru_map test_lpm_map test_progs \ test_align test_verifier_log test_dev_cgroup test_tcpbpf_user \ test_sock test_btf test_sockmap get_cgroup_id_user test_socket_cookie \ - test_cgroup_storage test_select_reuseport test_section_names \ + test_cgroup_storage test_section_names \ test_netcnt test_tcpnotify_user test_sock_fields test_sysctl test_hashmap \ test_cgroup_attach xdping @@ -69,7 +69,8 @@ TEST_PROGS := test_kmod.sh \ test_tc_tunnel.sh \ test_tc_edt.sh \ test_xdping.sh \ - test_bpftool_build.sh + test_bpftool_build.sh \ + test_select_reuseport.sh TEST_PROGS_EXTENDED := with_addr.sh \ with_tunnels.sh \ @@ -80,7 +81,7 @@ TEST_PROGS_EXTENDED := with_addr.sh \ # Compile but not part of 'make run_tests' TEST_GEN_PROGS_EXTENDED = test_sock_addr test_skb_cgroup_id_user \ flow_dissector_load test_flow_dissector test_tcp_check_syncookie_user \ - test_lirc_mode2_user + test_lirc_mode2_user test_select_reuseport TEST_CUSTOM_PROGS = urandom_read diff --git a/tools/testing/selftests/bpf/test_select_reuseport.c b/tools/testing/selftests/bpf/test_select_reuseport.c index 7566c13eb51a..732cfeee189f 100644 --- a/tools/testing/selftests/bpf/test_select_reuseport.c +++ b/tools/testing/selftests/bpf/test_select_reuseport.c @@ -1,6 +1,7 @@ // SPDX-License-Identifier: GPL-2.0 /* Copyright (c) 2018 Facebook */ +#define _GNU_SOURCE #include <stdlib.h> #include <unistd.h> #include <stdbool.h> @@ -29,6 +30,12 @@ #define TCP_FO_SYSCTL "/proc/sys/net/ipv4/tcp_fastopen" #define REUSEPORT_ARRAY_SIZE 32 +#define BIND_TO_INANY true +#define BIND_TO_LOOPBACK (!BIND_TO_INANY) + +static enum bpf_map_type cfg_map_type = BPF_MAP_TYPE_REUSEPORT_SOCKARRAY; +static unsigned int cfg_sock_types = (1 << SOCK_STREAM) | (1 << SOCK_DGRAM); + static int result_map, tmp_index_ovr_map, linum_map, data_check_map; static enum result expected_results[NR_RESULTS]; static int sk_fds[REUSEPORT_ARRAY_SIZE]; @@ -61,7 +68,7 @@ static void create_maps(void) /* Creating reuseport_array */ attr.name = "reuseport_array"; - attr.map_type = BPF_MAP_TYPE_REUSEPORT_SOCKARRAY; + attr.map_type = cfg_map_type; attr.key_size = sizeof(__u32); attr.value_size = sizeof(__u32); attr.max_entries = REUSEPORT_ARRAY_SIZE; @@ -680,53 +687,131 @@ static void cleanup(void) bpf_object__close(obj); } +static const char *family_to_str(int family) +{ + switch (family) { + case AF_INET: + return "IPv4"; + case AF_INET6: + return "IPv6"; + default: + return "unknown"; + } +} + +static const char *type_to_str(int type) +{ + switch (type) { + case SOCK_STREAM: + return "TCP"; + case SOCK_DGRAM: + return "UDP"; + default: + return "unknown"; + } +} + +static void test_one(int family, int type, bool inany) +{ + int err; + + printf("######## %s/%s %-8s ########\n", + family_to_str(family), type_to_str(type), + inany ? "INANY" : "LOOPBACK"); + + setup_per_test(type, family, inany); + + test_err_inner_map(type, family); + + /* Install reuseport_array to the outer_map */ + err = bpf_map_update_elem(outer_map, &index_zero, &reuseport_array, + BPF_ANY); + CHECK(err == -1, "update_elem(outer_map)", + "err:%d errno:%d\n", err, errno); + + test_err_skb_data(type, family); + test_err_sk_select_port(type, family); + test_pass(type, family); + test_syncookie(type, family); + test_pass_on_err(type, family); + /* Must be the last test */ + test_detach_bpf(type, family); + + cleanup_per_test(); + printf("\n"); +} + static void test_all(void) { - /* Extra SOCK_STREAM to test bind_inany==true */ - const int types[] = { SOCK_STREAM, SOCK_DGRAM, SOCK_STREAM }; - const char * const type_strings[] = { "TCP", "UDP", "TCP" }; - const char * const family_strings[] = { "IPv6", "IPv4" }; + const int types[] = { SOCK_STREAM, SOCK_DGRAM }; const unsigned short families[] = { AF_INET6, AF_INET }; - const bool bind_inany[] = { false, false, true }; - int t, f, err; + int t, f; for (f = 0; f < ARRAY_SIZE(families); f++) { unsigned short family = families[f]; for (t = 0; t < ARRAY_SIZE(types); t++) { - bool inany = bind_inany[t]; int type = types[t]; - printf("######## %s/%s %s ########\n", - family_strings[f], type_strings[t], - inany ? " INANY " : "LOOPBACK"); + /* Socket type excluded from tests? */ + if (~cfg_sock_types & (1 << type)) + continue; - setup_per_test(type, family, inany); + test_one(family, type, BIND_TO_LOOPBACK); + test_one(family, type, BIND_TO_INANY); + } + } +} - test_err_inner_map(type, family); +static void __attribute__((noreturn)) usage(void) +{ + fprintf(stderr, + "Usage: %s [-m reuseport_sockarray|sockmap] [-t] [-u]\n", + program_invocation_short_name); + exit(1); +} - /* Install reuseport_array to the outer_map */ - err = bpf_map_update_elem(outer_map, &index_zero, - &reuseport_array, BPF_ANY); - CHECK(err == -1, "update_elem(outer_map)", - "err:%d errno:%d\n", err, errno); +static enum bpf_map_type parse_map_type(const char *optarg) +{ + if (!strcmp(optarg, "reuseport_sockarray")) + return BPF_MAP_TYPE_REUSEPORT_SOCKARRAY; + if (!strcmp(optarg, "sockmap")) + return BPF_MAP_TYPE_SOCKMAP; - test_err_skb_data(type, family); - test_err_sk_select_port(type, family); - test_pass(type, family); - test_syncookie(type, family); - test_pass_on_err(type, family); - /* Must be the last test */ - test_detach_bpf(type, family); + return BPF_MAP_TYPE_UNSPEC; +} - cleanup_per_test(); - printf("\n"); +static void parse_opts(int argc, char **argv) +{ + unsigned int sock_types = 0; + int c; + + while ((c = getopt(argc, argv, "hm:tu")) != -1) { + switch (c) { + case 'h': + usage(); + break; + case 'm': + cfg_map_type = parse_map_type(optarg); + break; + case 't': + sock_types |= 1 << SOCK_STREAM; + break; + case 'u': + sock_types |= 1 << SOCK_DGRAM; + break; } } + + if (cfg_map_type == BPF_MAP_TYPE_UNSPEC) + usage(); + if (sock_types != 0) + cfg_sock_types = sock_types; } -int main(int argc, const char **argv) +int main(int argc, char **argv) { + parse_opts(argc, argv); create_maps(); prepare_bpf_obj(); saved_tcp_fo = read_int_sysctl(TCP_FO_SYSCTL); diff --git a/tools/testing/selftests/bpf/test_select_reuseport.sh b/tools/testing/selftests/bpf/test_select_reuseport.sh new file mode 100755 index 000000000000..1951b4886021 --- /dev/null +++ b/tools/testing/selftests/bpf/test_select_reuseport.sh @@ -0,0 +1,14 @@ +#!/bin/bash +# SPDX-License-Identifier: GPL-2.0 + +set -eu + +DIR=$(dirname $0) + +echo "Testing reuseport with REUSEPORT_SOCKARRAY..." +$DIR/test_select_reuseport -m reuseport_sockarray + +echo "Testing reuseport with SOCKMAP (TCP only)..." +$DIR/test_select_reuseport -m sockmap -t + +exit 0 -- 2.20.1