mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/stable/linux.git
synced 2025-01-01 10:45:49 +00:00
bpf-next-for-netdev
-----BEGIN PGP SIGNATURE----- iIsEABYIADMWIQTFp0I1jqZrAX+hPRXbK58LschIgwUCZyP6TxUcZGFuaWVsQGlv Z2VhcmJveC5uZXQACgkQ2yufC7HISINz7QD/RTuJAzPJXPQmjdzMj7pepjnSQH4K DnOc1soDqjJPSFkBAMlklDCZqSsFoNtNxagbyILrYQBC/MsV9jngimK46DEN =pDzC -----END PGP SIGNATURE----- Merge tag 'for-netdev' of https://git.kernel.org/pub/scm/linux/kernel/git/bpf/bpf-next Daniel Borkmann says: ==================== pull-request: bpf-next 2024-10-31 We've added 13 non-merge commits during the last 16 day(s) which contain a total of 16 files changed, 710 insertions(+), 668 deletions(-). The main changes are: 1) Optimize and homogenize bpf_csum_diff helper for all archs and also add a batch of new BPF selftests for it, from Puranjay Mohan. 2) Rewrite and migrate the test_tcp_check_syncookie.sh BPF selftest into test_progs so that it can be run in BPF CI, from Alexis Lothoré. 3) Two BPF sockmap selftest fixes, from Zijian Zhang. 4) Small XDP synproxy BPF selftest cleanup to remove IP_DF check, from Vincent Li. * tag 'for-netdev' of https://git.kernel.org/pub/scm/linux/kernel/git/bpf/bpf-next: selftests/bpf: Add a selftest for bpf_csum_diff() selftests/bpf: Don't mask result of bpf_csum_diff() in test_verifier bpf: bpf_csum_diff: Optimize and homogenize for all archs net: checksum: Move from32to16() to generic header selftests/bpf: remove xdp_synproxy IP_DF check selftests/bpf: remove test_tcp_check_syncookie selftests/bpf: test MSS value returned with bpf_tcp_gen_syncookie selftests/bpf: add ipv4 and dual ipv4/ipv6 support in btf_skc_cls_ingress selftests/bpf: get rid of global vars in btf_skc_cls_ingress selftests/bpf: add missing ns cleanups in btf_skc_cls_ingress selftests/bpf: factorize conn and syncookies tests in a single runner selftests/bpf: Fix txmsg_redir of test_txmsg_pull in test_sockmap selftests/bpf: Fix msg_verify_data in test_sockmap ==================== Link: https://patch.msgid.link/20241031221543.108853-1-daniel@iogearbox.net Signed-off-by: Jakub Kicinski <kuba@kernel.org>
This commit is contained in:
commit
cbf49bed6a
@ -25,15 +25,6 @@
|
||||
: "=r"(_t) \
|
||||
: "r"(_r), "0"(_t));
|
||||
|
||||
static inline unsigned short from32to16(unsigned int x)
|
||||
{
|
||||
/* 32 bits --> 16 bits + carry */
|
||||
x = (x & 0xffff) + (x >> 16);
|
||||
/* 16 bits + carry --> 16 bits including carry */
|
||||
x = (x & 0xffff) + (x >> 16);
|
||||
return (unsigned short)x;
|
||||
}
|
||||
|
||||
static inline unsigned int do_csum(const unsigned char * buff, int len)
|
||||
{
|
||||
int odd, count;
|
||||
@ -85,7 +76,7 @@ static inline unsigned int do_csum(const unsigned char * buff, int len)
|
||||
}
|
||||
if (len & 1)
|
||||
result += le16_to_cpu(*buff);
|
||||
result = from32to16(result);
|
||||
result = csum_from32to16(result);
|
||||
if (odd)
|
||||
result = swab16(result);
|
||||
out:
|
||||
@ -102,7 +93,7 @@ __wsum csum_partial(const void *buff, int len, __wsum sum)
|
||||
{
|
||||
unsigned int result = do_csum(buff, len);
|
||||
addc(result, sum);
|
||||
return (__force __wsum)from32to16(result);
|
||||
return (__force __wsum)csum_from32to16(result);
|
||||
}
|
||||
|
||||
EXPORT_SYMBOL(csum_partial);
|
||||
|
@ -151,6 +151,12 @@ static inline void csum_replace(__wsum *csum, __wsum old, __wsum new)
|
||||
*csum = csum_add(csum_sub(*csum, old), new);
|
||||
}
|
||||
|
||||
static inline unsigned short csum_from32to16(unsigned int sum)
|
||||
{
|
||||
sum += (sum >> 16) | (sum << 16);
|
||||
return (unsigned short)(sum >> 16);
|
||||
}
|
||||
|
||||
struct sk_buff;
|
||||
void inet_proto_csum_replace4(__sum16 *sum, struct sk_buff *skb,
|
||||
__be32 from, __be32 to, bool pseudohdr);
|
||||
|
@ -34,15 +34,6 @@
|
||||
#include <asm/byteorder.h>
|
||||
|
||||
#ifndef do_csum
|
||||
static inline unsigned short from32to16(unsigned int x)
|
||||
{
|
||||
/* add up 16-bit and 16-bit for 16+c bit */
|
||||
x = (x & 0xffff) + (x >> 16);
|
||||
/* add up carry.. */
|
||||
x = (x & 0xffff) + (x >> 16);
|
||||
return x;
|
||||
}
|
||||
|
||||
static unsigned int do_csum(const unsigned char *buff, int len)
|
||||
{
|
||||
int odd;
|
||||
@ -90,7 +81,7 @@ static unsigned int do_csum(const unsigned char *buff, int len)
|
||||
#else
|
||||
result += (*buff << 8);
|
||||
#endif
|
||||
result = from32to16(result);
|
||||
result = csum_from32to16(result);
|
||||
if (odd)
|
||||
result = ((result >> 8) & 0xff) | ((result & 0xff) << 8);
|
||||
out:
|
||||
|
@ -1654,18 +1654,6 @@ void sk_reuseport_prog_free(struct bpf_prog *prog)
|
||||
bpf_prog_destroy(prog);
|
||||
}
|
||||
|
||||
struct bpf_scratchpad {
|
||||
union {
|
||||
__be32 diff[MAX_BPF_STACK / sizeof(__be32)];
|
||||
u8 buff[MAX_BPF_STACK];
|
||||
};
|
||||
local_lock_t bh_lock;
|
||||
};
|
||||
|
||||
static DEFINE_PER_CPU(struct bpf_scratchpad, bpf_sp) = {
|
||||
.bh_lock = INIT_LOCAL_LOCK(bh_lock),
|
||||
};
|
||||
|
||||
static inline int __bpf_try_make_writable(struct sk_buff *skb,
|
||||
unsigned int write_len)
|
||||
{
|
||||
@ -2022,11 +2010,6 @@ static const struct bpf_func_proto bpf_l4_csum_replace_proto = {
|
||||
BPF_CALL_5(bpf_csum_diff, __be32 *, from, u32, from_size,
|
||||
__be32 *, to, u32, to_size, __wsum, seed)
|
||||
{
|
||||
struct bpf_scratchpad *sp = this_cpu_ptr(&bpf_sp);
|
||||
u32 diff_size = from_size + to_size;
|
||||
int i, j = 0;
|
||||
__wsum ret;
|
||||
|
||||
/* This is quite flexible, some examples:
|
||||
*
|
||||
* from_size == 0, to_size > 0, seed := csum --> pushing data
|
||||
@ -2035,19 +2018,19 @@ BPF_CALL_5(bpf_csum_diff, __be32 *, from, u32, from_size,
|
||||
*
|
||||
* Even for diffing, from_size and to_size don't need to be equal.
|
||||
*/
|
||||
if (unlikely(((from_size | to_size) & (sizeof(__be32) - 1)) ||
|
||||
diff_size > sizeof(sp->diff)))
|
||||
return -EINVAL;
|
||||
|
||||
local_lock_nested_bh(&bpf_sp.bh_lock);
|
||||
for (i = 0; i < from_size / sizeof(__be32); i++, j++)
|
||||
sp->diff[j] = ~from[i];
|
||||
for (i = 0; i < to_size / sizeof(__be32); i++, j++)
|
||||
sp->diff[j] = to[i];
|
||||
__wsum ret = seed;
|
||||
|
||||
ret = csum_partial(sp->diff, diff_size, seed);
|
||||
local_unlock_nested_bh(&bpf_sp.bh_lock);
|
||||
return ret;
|
||||
if (from_size && to_size)
|
||||
ret = csum_sub(csum_partial(to, to_size, ret),
|
||||
csum_partial(from, from_size, 0));
|
||||
else if (to_size)
|
||||
ret = csum_partial(to, to_size, ret);
|
||||
|
||||
else if (from_size)
|
||||
ret = ~csum_partial(from, from_size, ~ret);
|
||||
|
||||
return csum_from32to16((__force unsigned int)ret);
|
||||
}
|
||||
|
||||
static const struct bpf_func_proto bpf_csum_diff_proto = {
|
||||
|
1
tools/testing/selftests/bpf/.gitignore
vendored
1
tools/testing/selftests/bpf/.gitignore
vendored
@ -24,7 +24,6 @@ test_flow_dissector
|
||||
flow_dissector_load
|
||||
test_tcpnotify_user
|
||||
test_libbpf
|
||||
test_tcp_check_syncookie_user
|
||||
test_sysctl
|
||||
xdping
|
||||
test_cpp
|
||||
|
@ -137,7 +137,6 @@ TEST_PROGS := test_kmod.sh \
|
||||
test_xdp_vlan_mode_generic.sh \
|
||||
test_xdp_vlan_mode_native.sh \
|
||||
test_lwt_ip_encap.sh \
|
||||
test_tcp_check_syncookie.sh \
|
||||
test_tc_tunnel.sh \
|
||||
test_tc_edt.sh \
|
||||
test_xdping.sh \
|
||||
@ -154,11 +153,23 @@ TEST_PROGS_EXTENDED := with_addr.sh \
|
||||
|
||||
# Compile but not part of 'make run_tests'
|
||||
TEST_GEN_PROGS_EXTENDED = \
|
||||
flow_dissector_load test_flow_dissector test_tcp_check_syncookie_user \
|
||||
test_lirc_mode2_user xdping test_cpp runqslower bench bpf_testmod.ko \
|
||||
xskxceiver xdp_redirect_multi xdp_synproxy veristat xdp_hw_metadata \
|
||||
xdp_features bpf_test_no_cfi.ko bpf_test_modorder_x.ko \
|
||||
bpf_test_modorder_y.ko
|
||||
bench \
|
||||
bpf_testmod.ko \
|
||||
bpf_test_modorder_x.ko \
|
||||
bpf_test_modorder_y.ko \
|
||||
bpf_test_no_cfi.ko \
|
||||
flow_dissector_load \
|
||||
runqslower \
|
||||
test_cpp \
|
||||
test_flow_dissector \
|
||||
test_lirc_mode2_user \
|
||||
veristat \
|
||||
xdp_features \
|
||||
xdp_hw_metadata \
|
||||
xdp_redirect_multi \
|
||||
xdp_synproxy \
|
||||
xdping \
|
||||
xskxceiver
|
||||
|
||||
TEST_GEN_FILES += liburandom_read.so urandom_read sign-file uprobe_multi
|
||||
|
||||
@ -361,7 +372,6 @@ $(OUTPUT)/flow_dissector_load: $(TESTING_HELPERS)
|
||||
$(OUTPUT)/test_maps: $(TESTING_HELPERS)
|
||||
$(OUTPUT)/test_verifier: $(TESTING_HELPERS) $(CAP_HELPERS) $(UNPRIV_HELPERS)
|
||||
$(OUTPUT)/xsk.o: $(BPFOBJ)
|
||||
$(OUTPUT)/test_tcp_check_syncookie_user: $(NETWORK_HELPERS)
|
||||
|
||||
BPFTOOL ?= $(DEFAULT_BPFTOOL)
|
||||
$(DEFAULT_BPFTOOL): $(wildcard $(BPFTOOLDIR)/*.[ch] $(BPFTOOLDIR)/Makefile) \
|
||||
|
@ -17,32 +17,37 @@
|
||||
#include "test_progs.h"
|
||||
#include "test_btf_skc_cls_ingress.skel.h"
|
||||
|
||||
static struct test_btf_skc_cls_ingress *skel;
|
||||
static struct sockaddr_in6 srv_sa6;
|
||||
static __u32 duration;
|
||||
#define TEST_NS "skc_cls_ingress"
|
||||
|
||||
static int prepare_netns(void)
|
||||
#define BIT(n) (1 << (n))
|
||||
#define TEST_MODE_IPV4 BIT(0)
|
||||
#define TEST_MODE_IPV6 BIT(1)
|
||||
#define TEST_MODE_DUAL (TEST_MODE_IPV4 | TEST_MODE_IPV6)
|
||||
|
||||
#define SERVER_ADDR_IPV4 "127.0.0.1"
|
||||
#define SERVER_ADDR_IPV6 "::1"
|
||||
#define SERVER_ADDR_DUAL "::0"
|
||||
/* RFC791, 576 for minimal IPv4 datagram, minus 40 bytes of TCP header */
|
||||
#define MIN_IPV4_MSS 536
|
||||
|
||||
static struct netns_obj *prepare_netns(struct test_btf_skc_cls_ingress *skel)
|
||||
{
|
||||
LIBBPF_OPTS(bpf_tc_hook, qdisc_lo, .attach_point = BPF_TC_INGRESS);
|
||||
LIBBPF_OPTS(bpf_tc_opts, tc_attach,
|
||||
.prog_fd = bpf_program__fd(skel->progs.cls_ingress));
|
||||
struct netns_obj *ns = NULL;
|
||||
|
||||
if (CHECK(unshare(CLONE_NEWNET), "create netns",
|
||||
"unshare(CLONE_NEWNET): %s (%d)",
|
||||
strerror(errno), errno))
|
||||
return -1;
|
||||
|
||||
if (CHECK(system("ip link set dev lo up"),
|
||||
"ip link set dev lo up", "failed\n"))
|
||||
return -1;
|
||||
ns = netns_new(TEST_NS, true);
|
||||
if (!ASSERT_OK_PTR(ns, "create and join netns"))
|
||||
return ns;
|
||||
|
||||
qdisc_lo.ifindex = if_nametoindex("lo");
|
||||
if (!ASSERT_OK(bpf_tc_hook_create(&qdisc_lo), "qdisc add dev lo clsact"))
|
||||
return -1;
|
||||
goto free_ns;
|
||||
|
||||
if (!ASSERT_OK(bpf_tc_attach(&qdisc_lo, &tc_attach),
|
||||
"filter add dev lo ingress"))
|
||||
return -1;
|
||||
goto free_ns;
|
||||
|
||||
/* Ensure 20 bytes options (i.e. in total 40 bytes tcp header) for the
|
||||
* bpf_tcp_gen_syncookie() helper.
|
||||
@ -50,71 +55,142 @@ static int prepare_netns(void)
|
||||
if (write_sysctl("/proc/sys/net/ipv4/tcp_window_scaling", "1") ||
|
||||
write_sysctl("/proc/sys/net/ipv4/tcp_timestamps", "1") ||
|
||||
write_sysctl("/proc/sys/net/ipv4/tcp_sack", "1"))
|
||||
return -1;
|
||||
goto free_ns;
|
||||
|
||||
return 0;
|
||||
return ns;
|
||||
|
||||
free_ns:
|
||||
netns_free(ns);
|
||||
return NULL;
|
||||
}
|
||||
|
||||
static void reset_test(void)
|
||||
static void reset_test(struct test_btf_skc_cls_ingress *skel)
|
||||
{
|
||||
memset(&skel->bss->srv_sa4, 0, sizeof(skel->bss->srv_sa4));
|
||||
memset(&skel->bss->srv_sa6, 0, sizeof(skel->bss->srv_sa6));
|
||||
skel->bss->listen_tp_sport = 0;
|
||||
skel->bss->req_sk_sport = 0;
|
||||
skel->bss->recv_cookie = 0;
|
||||
skel->bss->gen_cookie = 0;
|
||||
skel->bss->linum = 0;
|
||||
skel->bss->mss = 0;
|
||||
}
|
||||
|
||||
static void print_err_line(void)
|
||||
static void print_err_line(struct test_btf_skc_cls_ingress *skel)
|
||||
{
|
||||
if (skel->bss->linum)
|
||||
printf("bpf prog error at line %u\n", skel->bss->linum);
|
||||
}
|
||||
|
||||
static void test_conn(void)
|
||||
static int v6only_true(int fd, void *opts)
|
||||
{
|
||||
int mode = true;
|
||||
|
||||
return setsockopt(fd, IPPROTO_IPV6, IPV6_V6ONLY, &mode, sizeof(mode));
|
||||
}
|
||||
|
||||
static int v6only_false(int fd, void *opts)
|
||||
{
|
||||
int mode = false;
|
||||
|
||||
return setsockopt(fd, IPPROTO_IPV6, IPV6_V6ONLY, &mode, sizeof(mode));
|
||||
}
|
||||
|
||||
static void run_test(struct test_btf_skc_cls_ingress *skel, bool gen_cookies,
|
||||
int ip_mode)
|
||||
{
|
||||
const char *tcp_syncookies = gen_cookies ? "2" : "1";
|
||||
int listen_fd = -1, cli_fd = -1, srv_fd = -1, err;
|
||||
socklen_t addrlen = sizeof(srv_sa6);
|
||||
struct network_helper_opts opts = { 0 };
|
||||
struct sockaddr_storage *addr;
|
||||
struct sockaddr_in6 srv_sa6;
|
||||
struct sockaddr_in srv_sa4;
|
||||
socklen_t addr_len;
|
||||
int sock_family;
|
||||
char *srv_addr;
|
||||
int srv_port;
|
||||
|
||||
if (write_sysctl("/proc/sys/net/ipv4/tcp_syncookies", "1"))
|
||||
switch (ip_mode) {
|
||||
case TEST_MODE_IPV4:
|
||||
sock_family = AF_INET;
|
||||
srv_addr = SERVER_ADDR_IPV4;
|
||||
addr = (struct sockaddr_storage *)&srv_sa4;
|
||||
addr_len = sizeof(srv_sa4);
|
||||
break;
|
||||
case TEST_MODE_IPV6:
|
||||
opts.post_socket_cb = v6only_true;
|
||||
sock_family = AF_INET6;
|
||||
srv_addr = SERVER_ADDR_IPV6;
|
||||
addr = (struct sockaddr_storage *)&srv_sa6;
|
||||
addr_len = sizeof(srv_sa6);
|
||||
break;
|
||||
case TEST_MODE_DUAL:
|
||||
opts.post_socket_cb = v6only_false;
|
||||
sock_family = AF_INET6;
|
||||
srv_addr = SERVER_ADDR_DUAL;
|
||||
addr = (struct sockaddr_storage *)&srv_sa6;
|
||||
addr_len = sizeof(srv_sa6);
|
||||
break;
|
||||
default:
|
||||
PRINT_FAIL("Unknown IP mode %d", ip_mode);
|
||||
return;
|
||||
}
|
||||
|
||||
if (write_sysctl("/proc/sys/net/ipv4/tcp_syncookies", tcp_syncookies))
|
||||
return;
|
||||
|
||||
listen_fd = start_server(AF_INET6, SOCK_STREAM, "::1", 0, 0);
|
||||
if (CHECK_FAIL(listen_fd == -1))
|
||||
listen_fd = start_server_str(sock_family, SOCK_STREAM, srv_addr, 0,
|
||||
&opts);
|
||||
if (!ASSERT_OK_FD(listen_fd, "start server"))
|
||||
return;
|
||||
|
||||
err = getsockname(listen_fd, (struct sockaddr *)&srv_sa6, &addrlen);
|
||||
if (CHECK(err, "getsockname(listen_fd)", "err:%d errno:%d\n", err,
|
||||
errno))
|
||||
err = getsockname(listen_fd, (struct sockaddr *)addr, &addr_len);
|
||||
if (!ASSERT_OK(err, "getsockname(listen_fd)"))
|
||||
goto done;
|
||||
memcpy(&skel->bss->srv_sa6, &srv_sa6, sizeof(srv_sa6));
|
||||
srv_port = ntohs(srv_sa6.sin6_port);
|
||||
|
||||
switch (ip_mode) {
|
||||
case TEST_MODE_IPV4:
|
||||
memcpy(&skel->bss->srv_sa4, &srv_sa4, sizeof(srv_sa4));
|
||||
srv_port = ntohs(srv_sa4.sin_port);
|
||||
break;
|
||||
case TEST_MODE_IPV6:
|
||||
case TEST_MODE_DUAL:
|
||||
memcpy(&skel->bss->srv_sa6, &srv_sa6, sizeof(srv_sa6));
|
||||
srv_port = ntohs(srv_sa6.sin6_port);
|
||||
break;
|
||||
default:
|
||||
goto done;
|
||||
}
|
||||
|
||||
cli_fd = connect_to_fd(listen_fd, 0);
|
||||
if (CHECK_FAIL(cli_fd == -1))
|
||||
if (!ASSERT_OK_FD(cli_fd, "connect client"))
|
||||
goto done;
|
||||
|
||||
srv_fd = accept(listen_fd, NULL, NULL);
|
||||
if (CHECK_FAIL(srv_fd == -1))
|
||||
if (!ASSERT_OK_FD(srv_fd, "accept connection"))
|
||||
goto done;
|
||||
|
||||
if (CHECK(skel->bss->listen_tp_sport != srv_port ||
|
||||
skel->bss->req_sk_sport != srv_port,
|
||||
"Unexpected sk src port",
|
||||
"listen_tp_sport:%u req_sk_sport:%u expected:%u\n",
|
||||
skel->bss->listen_tp_sport, skel->bss->req_sk_sport,
|
||||
srv_port))
|
||||
goto done;
|
||||
ASSERT_EQ(skel->bss->listen_tp_sport, srv_port, "listen tp src port");
|
||||
|
||||
if (CHECK(skel->bss->gen_cookie || skel->bss->recv_cookie,
|
||||
"Unexpected syncookie states",
|
||||
"gen_cookie:%u recv_cookie:%u\n",
|
||||
skel->bss->gen_cookie, skel->bss->recv_cookie))
|
||||
goto done;
|
||||
|
||||
CHECK(skel->bss->linum, "bpf prog detected error", "at line %u\n",
|
||||
skel->bss->linum);
|
||||
if (!gen_cookies) {
|
||||
ASSERT_EQ(skel->bss->req_sk_sport, srv_port,
|
||||
"request socket source port with syncookies disabled");
|
||||
ASSERT_EQ(skel->bss->gen_cookie, 0,
|
||||
"generated syncookie with syncookies disabled");
|
||||
ASSERT_EQ(skel->bss->recv_cookie, 0,
|
||||
"received syncookie with syncookies disabled");
|
||||
} else {
|
||||
ASSERT_EQ(skel->bss->req_sk_sport, 0,
|
||||
"request socket source port with syncookies enabled");
|
||||
ASSERT_NEQ(skel->bss->gen_cookie, 0,
|
||||
"syncookie properly generated");
|
||||
ASSERT_EQ(skel->bss->gen_cookie, skel->bss->recv_cookie,
|
||||
"matching syncookies on client and server");
|
||||
ASSERT_GT(skel->bss->mss, MIN_IPV4_MSS,
|
||||
"MSS in cookie min value");
|
||||
ASSERT_LT(skel->bss->mss, USHRT_MAX,
|
||||
"MSS in cookie max value");
|
||||
}
|
||||
|
||||
done:
|
||||
if (listen_fd != -1)
|
||||
@ -125,96 +201,74 @@ static void test_conn(void)
|
||||
close(srv_fd);
|
||||
}
|
||||
|
||||
static void test_syncookie(void)
|
||||
static void test_conn_ipv4(struct test_btf_skc_cls_ingress *skel)
|
||||
{
|
||||
int listen_fd = -1, cli_fd = -1, srv_fd = -1, err;
|
||||
socklen_t addrlen = sizeof(srv_sa6);
|
||||
int srv_port;
|
||||
run_test(skel, false, TEST_MODE_IPV4);
|
||||
}
|
||||
|
||||
/* Enforce syncookie mode */
|
||||
if (write_sysctl("/proc/sys/net/ipv4/tcp_syncookies", "2"))
|
||||
return;
|
||||
static void test_conn_ipv6(struct test_btf_skc_cls_ingress *skel)
|
||||
{
|
||||
run_test(skel, false, TEST_MODE_IPV6);
|
||||
}
|
||||
|
||||
listen_fd = start_server(AF_INET6, SOCK_STREAM, "::1", 0, 0);
|
||||
if (CHECK_FAIL(listen_fd == -1))
|
||||
return;
|
||||
static void test_conn_dual(struct test_btf_skc_cls_ingress *skel)
|
||||
{
|
||||
run_test(skel, false, TEST_MODE_DUAL);
|
||||
}
|
||||
|
||||
err = getsockname(listen_fd, (struct sockaddr *)&srv_sa6, &addrlen);
|
||||
if (CHECK(err, "getsockname(listen_fd)", "err:%d errno:%d\n", err,
|
||||
errno))
|
||||
goto done;
|
||||
memcpy(&skel->bss->srv_sa6, &srv_sa6, sizeof(srv_sa6));
|
||||
srv_port = ntohs(srv_sa6.sin6_port);
|
||||
static void test_syncookie_ipv4(struct test_btf_skc_cls_ingress *skel)
|
||||
{
|
||||
run_test(skel, true, TEST_MODE_IPV4);
|
||||
}
|
||||
|
||||
cli_fd = connect_to_fd(listen_fd, 0);
|
||||
if (CHECK_FAIL(cli_fd == -1))
|
||||
goto done;
|
||||
static void test_syncookie_ipv6(struct test_btf_skc_cls_ingress *skel)
|
||||
{
|
||||
run_test(skel, true, TEST_MODE_IPV6);
|
||||
}
|
||||
|
||||
srv_fd = accept(listen_fd, NULL, NULL);
|
||||
if (CHECK_FAIL(srv_fd == -1))
|
||||
goto done;
|
||||
|
||||
if (CHECK(skel->bss->listen_tp_sport != srv_port,
|
||||
"Unexpected tp src port",
|
||||
"listen_tp_sport:%u expected:%u\n",
|
||||
skel->bss->listen_tp_sport, srv_port))
|
||||
goto done;
|
||||
|
||||
if (CHECK(skel->bss->req_sk_sport,
|
||||
"Unexpected req_sk src port",
|
||||
"req_sk_sport:%u expected:0\n",
|
||||
skel->bss->req_sk_sport))
|
||||
goto done;
|
||||
|
||||
if (CHECK(!skel->bss->gen_cookie ||
|
||||
skel->bss->gen_cookie != skel->bss->recv_cookie,
|
||||
"Unexpected syncookie states",
|
||||
"gen_cookie:%u recv_cookie:%u\n",
|
||||
skel->bss->gen_cookie, skel->bss->recv_cookie))
|
||||
goto done;
|
||||
|
||||
CHECK(skel->bss->linum, "bpf prog detected error", "at line %u\n",
|
||||
skel->bss->linum);
|
||||
|
||||
done:
|
||||
if (listen_fd != -1)
|
||||
close(listen_fd);
|
||||
if (cli_fd != -1)
|
||||
close(cli_fd);
|
||||
if (srv_fd != -1)
|
||||
close(srv_fd);
|
||||
static void test_syncookie_dual(struct test_btf_skc_cls_ingress *skel)
|
||||
{
|
||||
run_test(skel, true, TEST_MODE_DUAL);
|
||||
}
|
||||
|
||||
struct test {
|
||||
const char *desc;
|
||||
void (*run)(void);
|
||||
void (*run)(struct test_btf_skc_cls_ingress *skel);
|
||||
};
|
||||
|
||||
#define DEF_TEST(name) { #name, test_##name }
|
||||
static struct test tests[] = {
|
||||
DEF_TEST(conn),
|
||||
DEF_TEST(syncookie),
|
||||
DEF_TEST(conn_ipv4),
|
||||
DEF_TEST(conn_ipv6),
|
||||
DEF_TEST(conn_dual),
|
||||
DEF_TEST(syncookie_ipv4),
|
||||
DEF_TEST(syncookie_ipv6),
|
||||
DEF_TEST(syncookie_dual),
|
||||
};
|
||||
|
||||
void test_btf_skc_cls_ingress(void)
|
||||
{
|
||||
struct test_btf_skc_cls_ingress *skel;
|
||||
struct netns_obj *ns;
|
||||
int i;
|
||||
|
||||
skel = test_btf_skc_cls_ingress__open_and_load();
|
||||
if (CHECK(!skel, "test_btf_skc_cls_ingress__open_and_load", "failed\n"))
|
||||
if (!ASSERT_OK_PTR(skel, "test_btf_skc_cls_ingress__open_and_load"))
|
||||
return;
|
||||
|
||||
for (i = 0; i < ARRAY_SIZE(tests); i++) {
|
||||
if (!test__start_subtest(tests[i].desc))
|
||||
continue;
|
||||
|
||||
if (prepare_netns())
|
||||
ns = prepare_netns(skel);
|
||||
if (!ns)
|
||||
break;
|
||||
|
||||
tests[i].run();
|
||||
tests[i].run(skel);
|
||||
|
||||
print_err_line();
|
||||
reset_test();
|
||||
print_err_line(skel);
|
||||
reset_test(skel);
|
||||
netns_free(ns);
|
||||
}
|
||||
|
||||
test_btf_skc_cls_ingress__destroy(skel);
|
||||
|
408
tools/testing/selftests/bpf/prog_tests/test_csum_diff.c
Normal file
408
tools/testing/selftests/bpf/prog_tests/test_csum_diff.c
Normal file
@ -0,0 +1,408 @@
|
||||
// SPDX-License-Identifier: GPL-2.0
|
||||
/* Copyright Amazon.com Inc. or its affiliates */
|
||||
#include <test_progs.h>
|
||||
#include "csum_diff_test.skel.h"
|
||||
|
||||
#define BUFF_SZ 512
|
||||
|
||||
struct testcase {
|
||||
unsigned long long to_buff[BUFF_SZ / 8];
|
||||
unsigned int to_buff_len;
|
||||
unsigned long long from_buff[BUFF_SZ / 8];
|
||||
unsigned int from_buff_len;
|
||||
unsigned short seed;
|
||||
unsigned short result;
|
||||
};
|
||||
|
||||
#define NUM_PUSH_TESTS 4
|
||||
|
||||
struct testcase push_tests[NUM_PUSH_TESTS] = {
|
||||
{
|
||||
.to_buff = {
|
||||
0xdeadbeefdeadbeef,
|
||||
},
|
||||
.to_buff_len = 8,
|
||||
.from_buff = {},
|
||||
.from_buff_len = 0,
|
||||
.seed = 0,
|
||||
.result = 0x3b3b
|
||||
},
|
||||
{
|
||||
.to_buff = {
|
||||
0xdeadbeefdeadbeef,
|
||||
0xbeefdeadbeefdead,
|
||||
},
|
||||
.to_buff_len = 16,
|
||||
.from_buff = {},
|
||||
.from_buff_len = 0,
|
||||
.seed = 0x1234,
|
||||
.result = 0x88aa
|
||||
},
|
||||
{
|
||||
.to_buff = {
|
||||
0xdeadbeefdeadbeef,
|
||||
0xbeefdeadbeefdead,
|
||||
},
|
||||
.to_buff_len = 15,
|
||||
.from_buff = {},
|
||||
.from_buff_len = 0,
|
||||
.seed = 0x1234,
|
||||
#if __BYTE_ORDER__ == __ORDER_LITTLE_ENDIAN__
|
||||
.result = 0xcaa9
|
||||
#else
|
||||
.result = 0x87fd
|
||||
#endif
|
||||
},
|
||||
{
|
||||
.to_buff = {
|
||||
0x327b23c66b8b4567,
|
||||
0x66334873643c9869,
|
||||
0x19495cff74b0dc51,
|
||||
0x625558ec2ae8944a,
|
||||
0x46e87ccd238e1f29,
|
||||
0x507ed7ab3d1b58ba,
|
||||
0x41b71efb2eb141f2,
|
||||
0x7545e14679e2a9e3,
|
||||
0x5bd062c2515f007c,
|
||||
0x4db127f812200854,
|
||||
0x1f16e9e80216231b,
|
||||
0x66ef438d1190cde7,
|
||||
0x3352255a140e0f76,
|
||||
0x0ded7263109cf92e,
|
||||
0x1befd79f7fdcc233,
|
||||
0x6b68079a41a7c4c9,
|
||||
0x25e45d324e6afb66,
|
||||
0x431bd7b7519b500d,
|
||||
0x7c83e4583f2dba31,
|
||||
0x62bbd95a257130a3,
|
||||
0x628c895d436c6125,
|
||||
0x721da317333ab105,
|
||||
0x2d1d5ae92443a858,
|
||||
0x75a2a8d46763845e,
|
||||
0x79838cb208edbdab,
|
||||
0x0b03e0c64353d0cd,
|
||||
0x54e49eb4189a769b,
|
||||
0x2ca8861171f32454,
|
||||
0x02901d820836c40e,
|
||||
0x081386413a95f874,
|
||||
0x7c3dbd3d1e7ff521,
|
||||
0x6ceaf087737b8ddc,
|
||||
0x4516dde922221a70,
|
||||
0x614fd4a13006c83e,
|
||||
0x5577f8e1419ac241,
|
||||
0x05072367440badfc,
|
||||
0x77465f013804823e,
|
||||
0x5c482a977724c67e,
|
||||
0x5e884adc2463b9ea,
|
||||
0x2d51779651ead36b,
|
||||
0x153ea438580bd78f,
|
||||
0x70a64e2a3855585c,
|
||||
0x2a487cb06a2342ec,
|
||||
0x725a06fb1d4ed43b,
|
||||
0x57e4ccaf2cd89a32,
|
||||
0x4b588f547a6d8d3c,
|
||||
0x6de91b18542289ec,
|
||||
0x7644a45c38437fdb,
|
||||
0x684a481a32fff902,
|
||||
0x749abb43579478fe,
|
||||
0x1ba026fa3dc240fb,
|
||||
0x75c6c33a79a1deaa,
|
||||
0x70c6a52912e685fb,
|
||||
0x374a3fe6520eedd1,
|
||||
0x23f9c13c4f4ef005,
|
||||
0x275ac794649bb77c,
|
||||
0x1cf10fd839386575,
|
||||
0x235ba861180115be,
|
||||
0x354fe9f947398c89,
|
||||
0x741226bb15b5af5c,
|
||||
0x10233c990d34b6a8,
|
||||
0x615740953f6ab60f,
|
||||
0x77ae35eb7e0c57b1,
|
||||
0x310c50b3579be4f1,
|
||||
},
|
||||
.to_buff_len = 512,
|
||||
.from_buff = {},
|
||||
.from_buff_len = 0,
|
||||
.seed = 0xffff,
|
||||
.result = 0xca45
|
||||
},
|
||||
};
|
||||
|
||||
#define NUM_PULL_TESTS 4
|
||||
|
||||
struct testcase pull_tests[NUM_PULL_TESTS] = {
|
||||
{
|
||||
.from_buff = {
|
||||
0xdeadbeefdeadbeef,
|
||||
},
|
||||
.from_buff_len = 8,
|
||||
.to_buff = {},
|
||||
.to_buff_len = 0,
|
||||
.seed = 0,
|
||||
.result = 0xc4c4
|
||||
},
|
||||
{
|
||||
.from_buff = {
|
||||
0xdeadbeefdeadbeef,
|
||||
0xbeefdeadbeefdead,
|
||||
},
|
||||
.from_buff_len = 16,
|
||||
.to_buff = {},
|
||||
.to_buff_len = 0,
|
||||
.seed = 0x1234,
|
||||
.result = 0x9bbd
|
||||
},
|
||||
{
|
||||
.from_buff = {
|
||||
0xdeadbeefdeadbeef,
|
||||
0xbeefdeadbeefdead,
|
||||
},
|
||||
.from_buff_len = 15,
|
||||
.to_buff = {},
|
||||
.to_buff_len = 0,
|
||||
.seed = 0x1234,
|
||||
#if __BYTE_ORDER__ == __ORDER_LITTLE_ENDIAN__
|
||||
.result = 0x59be
|
||||
#else
|
||||
.result = 0x9c6a
|
||||
#endif
|
||||
},
|
||||
{
|
||||
.from_buff = {
|
||||
0x327b23c66b8b4567,
|
||||
0x66334873643c9869,
|
||||
0x19495cff74b0dc51,
|
||||
0x625558ec2ae8944a,
|
||||
0x46e87ccd238e1f29,
|
||||
0x507ed7ab3d1b58ba,
|
||||
0x41b71efb2eb141f2,
|
||||
0x7545e14679e2a9e3,
|
||||
0x5bd062c2515f007c,
|
||||
0x4db127f812200854,
|
||||
0x1f16e9e80216231b,
|
||||
0x66ef438d1190cde7,
|
||||
0x3352255a140e0f76,
|
||||
0x0ded7263109cf92e,
|
||||
0x1befd79f7fdcc233,
|
||||
0x6b68079a41a7c4c9,
|
||||
0x25e45d324e6afb66,
|
||||
0x431bd7b7519b500d,
|
||||
0x7c83e4583f2dba31,
|
||||
0x62bbd95a257130a3,
|
||||
0x628c895d436c6125,
|
||||
0x721da317333ab105,
|
||||
0x2d1d5ae92443a858,
|
||||
0x75a2a8d46763845e,
|
||||
0x79838cb208edbdab,
|
||||
0x0b03e0c64353d0cd,
|
||||
0x54e49eb4189a769b,
|
||||
0x2ca8861171f32454,
|
||||
0x02901d820836c40e,
|
||||
0x081386413a95f874,
|
||||
0x7c3dbd3d1e7ff521,
|
||||
0x6ceaf087737b8ddc,
|
||||
0x4516dde922221a70,
|
||||
0x614fd4a13006c83e,
|
||||
0x5577f8e1419ac241,
|
||||
0x05072367440badfc,
|
||||
0x77465f013804823e,
|
||||
0x5c482a977724c67e,
|
||||
0x5e884adc2463b9ea,
|
||||
0x2d51779651ead36b,
|
||||
0x153ea438580bd78f,
|
||||
0x70a64e2a3855585c,
|
||||
0x2a487cb06a2342ec,
|
||||
0x725a06fb1d4ed43b,
|
||||
0x57e4ccaf2cd89a32,
|
||||
0x4b588f547a6d8d3c,
|
||||
0x6de91b18542289ec,
|
||||
0x7644a45c38437fdb,
|
||||
0x684a481a32fff902,
|
||||
0x749abb43579478fe,
|
||||
0x1ba026fa3dc240fb,
|
||||
0x75c6c33a79a1deaa,
|
||||
0x70c6a52912e685fb,
|
||||
0x374a3fe6520eedd1,
|
||||
0x23f9c13c4f4ef005,
|
||||
0x275ac794649bb77c,
|
||||
0x1cf10fd839386575,
|
||||
0x235ba861180115be,
|
||||
0x354fe9f947398c89,
|
||||
0x741226bb15b5af5c,
|
||||
0x10233c990d34b6a8,
|
||||
0x615740953f6ab60f,
|
||||
0x77ae35eb7e0c57b1,
|
||||
0x310c50b3579be4f1,
|
||||
},
|
||||
.from_buff_len = 512,
|
||||
.to_buff = {},
|
||||
.to_buff_len = 0,
|
||||
.seed = 0xffff,
|
||||
.result = 0x35ba
|
||||
},
|
||||
};
|
||||
|
||||
#define NUM_DIFF_TESTS 4
|
||||
|
||||
struct testcase diff_tests[NUM_DIFF_TESTS] = {
|
||||
{
|
||||
.from_buff = {
|
||||
0xdeadbeefdeadbeef,
|
||||
},
|
||||
.from_buff_len = 8,
|
||||
.to_buff = {
|
||||
0xabababababababab,
|
||||
},
|
||||
.to_buff_len = 8,
|
||||
.seed = 0,
|
||||
.result = 0x7373
|
||||
},
|
||||
{
|
||||
.from_buff = {
|
||||
0xdeadbeefdeadbeef,
|
||||
},
|
||||
.from_buff_len = 7,
|
||||
.to_buff = {
|
||||
0xabababababababab,
|
||||
},
|
||||
.to_buff_len = 7,
|
||||
.seed = 0,
|
||||
#if __BYTE_ORDER__ == __ORDER_LITTLE_ENDIAN__
|
||||
.result = 0xa673
|
||||
#else
|
||||
.result = 0x73b7
|
||||
#endif
|
||||
},
|
||||
{
|
||||
.from_buff = {
|
||||
0,
|
||||
},
|
||||
.from_buff_len = 8,
|
||||
.to_buff = {
|
||||
0xabababababababab,
|
||||
},
|
||||
.to_buff_len = 8,
|
||||
.seed = 0,
|
||||
.result = 0xaeae
|
||||
},
|
||||
{
|
||||
.from_buff = {
|
||||
0xdeadbeefdeadbeef
|
||||
},
|
||||
.from_buff_len = 8,
|
||||
.to_buff = {
|
||||
0,
|
||||
},
|
||||
.to_buff_len = 8,
|
||||
.seed = 0xffff,
|
||||
.result = 0xc4c4
|
||||
},
|
||||
};
|
||||
|
||||
#define NUM_EDGE_TESTS 4
|
||||
|
||||
struct testcase edge_tests[NUM_EDGE_TESTS] = {
|
||||
{
|
||||
.from_buff = {},
|
||||
.from_buff_len = 0,
|
||||
.to_buff = {},
|
||||
.to_buff_len = 0,
|
||||
.seed = 0,
|
||||
.result = 0
|
||||
},
|
||||
{
|
||||
.from_buff = {
|
||||
0x1234
|
||||
},
|
||||
.from_buff_len = 0,
|
||||
.to_buff = {
|
||||
0x1234
|
||||
},
|
||||
.to_buff_len = 0,
|
||||
.seed = 0,
|
||||
.result = 0
|
||||
},
|
||||
{
|
||||
.from_buff = {},
|
||||
.from_buff_len = 0,
|
||||
.to_buff = {},
|
||||
.to_buff_len = 0,
|
||||
.seed = 0x1234,
|
||||
.result = 0x1234
|
||||
},
|
||||
{
|
||||
.from_buff = {},
|
||||
.from_buff_len = 512,
|
||||
.to_buff = {},
|
||||
.to_buff_len = 0,
|
||||
.seed = 0xffff,
|
||||
.result = 0xffff
|
||||
},
|
||||
};
|
||||
|
||||
static unsigned short trigger_csum_diff(const struct csum_diff_test *skel)
|
||||
{
|
||||
u8 tmp_out[64 << 2] = {};
|
||||
u8 tmp_in[64] = {};
|
||||
int err;
|
||||
int pfd;
|
||||
|
||||
LIBBPF_OPTS(bpf_test_run_opts, topts,
|
||||
.data_in = tmp_in,
|
||||
.data_size_in = sizeof(tmp_in),
|
||||
.data_out = tmp_out,
|
||||
.data_size_out = sizeof(tmp_out),
|
||||
.repeat = 1,
|
||||
);
|
||||
pfd = bpf_program__fd(skel->progs.compute_checksum);
|
||||
err = bpf_prog_test_run_opts(pfd, &topts);
|
||||
if (err)
|
||||
return -1;
|
||||
|
||||
return skel->bss->result;
|
||||
}
|
||||
|
||||
static void test_csum_diff(struct testcase *tests, int num_tests)
|
||||
{
|
||||
struct csum_diff_test *skel;
|
||||
unsigned short got;
|
||||
int err;
|
||||
|
||||
for (int i = 0; i < num_tests; i++) {
|
||||
skel = csum_diff_test__open();
|
||||
if (!ASSERT_OK_PTR(skel, "csum_diff_test open"))
|
||||
return;
|
||||
|
||||
skel->rodata->to_buff_len = tests[i].to_buff_len;
|
||||
skel->rodata->from_buff_len = tests[i].from_buff_len;
|
||||
|
||||
err = csum_diff_test__load(skel);
|
||||
if (!ASSERT_EQ(err, 0, "csum_diff_test load"))
|
||||
goto out;
|
||||
|
||||
memcpy(skel->bss->to_buff, tests[i].to_buff, tests[i].to_buff_len);
|
||||
memcpy(skel->bss->from_buff, tests[i].from_buff, tests[i].from_buff_len);
|
||||
skel->bss->seed = tests[i].seed;
|
||||
|
||||
got = trigger_csum_diff(skel);
|
||||
ASSERT_EQ(got, tests[i].result, "csum_diff result");
|
||||
|
||||
csum_diff_test__destroy(skel);
|
||||
}
|
||||
|
||||
return;
|
||||
out:
|
||||
csum_diff_test__destroy(skel);
|
||||
}
|
||||
|
||||
void test_test_csum_diff(void)
|
||||
{
|
||||
if (test__start_subtest("csum_diff_push"))
|
||||
test_csum_diff(push_tests, NUM_PUSH_TESTS);
|
||||
if (test__start_subtest("csum_diff_pull"))
|
||||
test_csum_diff(pull_tests, NUM_PULL_TESTS);
|
||||
if (test__start_subtest("csum_diff_diff"))
|
||||
test_csum_diff(diff_tests, NUM_DIFF_TESTS);
|
||||
if (test__start_subtest("csum_diff_edge"))
|
||||
test_csum_diff(edge_tests, NUM_EDGE_TESTS);
|
||||
}
|
42
tools/testing/selftests/bpf/progs/csum_diff_test.c
Normal file
42
tools/testing/selftests/bpf/progs/csum_diff_test.c
Normal file
@ -0,0 +1,42 @@
|
||||
// SPDX-License-Identifier: GPL-2.0
|
||||
/* Copyright Amazon.com Inc. or its affiliates */
|
||||
#include <linux/types.h>
|
||||
#include <linux/bpf.h>
|
||||
#include <bpf/bpf_helpers.h>
|
||||
#include <bpf/bpf_tracing.h>
|
||||
|
||||
#define BUFF_SZ 512
|
||||
|
||||
/* Will be updated by benchmark before program loading */
|
||||
char to_buff[BUFF_SZ];
|
||||
const volatile unsigned int to_buff_len = 0;
|
||||
char from_buff[BUFF_SZ];
|
||||
const volatile unsigned int from_buff_len = 0;
|
||||
unsigned short seed = 0;
|
||||
|
||||
short result;
|
||||
|
||||
char _license[] SEC("license") = "GPL";
|
||||
|
||||
SEC("tc")
|
||||
int compute_checksum(void *ctx)
|
||||
{
|
||||
int to_len_half = to_buff_len / 2;
|
||||
int from_len_half = from_buff_len / 2;
|
||||
short result2;
|
||||
|
||||
/* Calculate checksum in one go */
|
||||
result2 = bpf_csum_diff((void *)from_buff, from_buff_len,
|
||||
(void *)to_buff, to_buff_len, seed);
|
||||
|
||||
/* Calculate checksum by concatenating bpf_csum_diff()*/
|
||||
result = bpf_csum_diff((void *)from_buff, from_buff_len - from_len_half,
|
||||
(void *)to_buff, to_buff_len - to_len_half, seed);
|
||||
|
||||
result = bpf_csum_diff((void *)from_buff + (from_buff_len - from_len_half), from_len_half,
|
||||
(void *)to_buff + (to_buff_len - to_len_half), to_len_half, result);
|
||||
|
||||
result = (result == result2) ? result : 0;
|
||||
|
||||
return 0;
|
||||
}
|
@ -10,16 +10,18 @@
|
||||
#endif
|
||||
|
||||
struct sockaddr_in6 srv_sa6 = {};
|
||||
struct sockaddr_in srv_sa4 = {};
|
||||
__u16 listen_tp_sport = 0;
|
||||
__u16 req_sk_sport = 0;
|
||||
__u32 recv_cookie = 0;
|
||||
__u32 gen_cookie = 0;
|
||||
__u32 mss = 0;
|
||||
__u32 linum = 0;
|
||||
|
||||
#define LOG() ({ if (!linum) linum = __LINE__; })
|
||||
|
||||
static void test_syncookie_helper(struct ipv6hdr *ip6h, struct tcphdr *th,
|
||||
struct tcp_sock *tp,
|
||||
static void test_syncookie_helper(void *iphdr, int iphdr_size,
|
||||
struct tcphdr *th, struct tcp_sock *tp,
|
||||
struct __sk_buff *skb)
|
||||
{
|
||||
if (th->syn) {
|
||||
@ -38,17 +40,18 @@ static void test_syncookie_helper(struct ipv6hdr *ip6h, struct tcphdr *th,
|
||||
return;
|
||||
}
|
||||
|
||||
mss_cookie = bpf_tcp_gen_syncookie(tp, ip6h, sizeof(*ip6h),
|
||||
mss_cookie = bpf_tcp_gen_syncookie(tp, iphdr, iphdr_size,
|
||||
th, 40);
|
||||
if (mss_cookie < 0) {
|
||||
if (mss_cookie != -ENOENT)
|
||||
LOG();
|
||||
} else {
|
||||
gen_cookie = (__u32)mss_cookie;
|
||||
mss = mss_cookie >> 32;
|
||||
}
|
||||
} else if (gen_cookie) {
|
||||
/* It was in cookie mode */
|
||||
int ret = bpf_tcp_check_syncookie(tp, ip6h, sizeof(*ip6h),
|
||||
int ret = bpf_tcp_check_syncookie(tp, iphdr, iphdr_size,
|
||||
th, sizeof(*th));
|
||||
|
||||
if (ret < 0) {
|
||||
@ -60,26 +63,58 @@ static void test_syncookie_helper(struct ipv6hdr *ip6h, struct tcphdr *th,
|
||||
}
|
||||
}
|
||||
|
||||
static int handle_ip6_tcp(struct ipv6hdr *ip6h, struct __sk_buff *skb)
|
||||
static int handle_ip_tcp(struct ethhdr *eth, struct __sk_buff *skb)
|
||||
{
|
||||
struct bpf_sock_tuple *tuple;
|
||||
struct bpf_sock_tuple *tuple = NULL;
|
||||
unsigned int tuple_len = 0;
|
||||
struct bpf_sock *bpf_skc;
|
||||
unsigned int tuple_len;
|
||||
void *data_end, *iphdr;
|
||||
struct ipv6hdr *ip6h;
|
||||
struct iphdr *ip4h;
|
||||
struct tcphdr *th;
|
||||
void *data_end;
|
||||
int iphdr_size;
|
||||
|
||||
data_end = (void *)(long)(skb->data_end);
|
||||
|
||||
th = (struct tcphdr *)(ip6h + 1);
|
||||
if (th + 1 > data_end)
|
||||
switch (eth->h_proto) {
|
||||
case bpf_htons(ETH_P_IP):
|
||||
ip4h = (struct iphdr *)(eth + 1);
|
||||
if (ip4h + 1 > data_end)
|
||||
return TC_ACT_OK;
|
||||
if (ip4h->protocol != IPPROTO_TCP)
|
||||
return TC_ACT_OK;
|
||||
th = (struct tcphdr *)(ip4h + 1);
|
||||
if (th + 1 > data_end)
|
||||
return TC_ACT_OK;
|
||||
/* Is it the testing traffic? */
|
||||
if (th->dest != srv_sa4.sin_port)
|
||||
return TC_ACT_OK;
|
||||
tuple_len = sizeof(tuple->ipv4);
|
||||
tuple = (struct bpf_sock_tuple *)&ip4h->saddr;
|
||||
iphdr = ip4h;
|
||||
iphdr_size = sizeof(*ip4h);
|
||||
break;
|
||||
case bpf_htons(ETH_P_IPV6):
|
||||
ip6h = (struct ipv6hdr *)(eth + 1);
|
||||
if (ip6h + 1 > data_end)
|
||||
return TC_ACT_OK;
|
||||
if (ip6h->nexthdr != IPPROTO_TCP)
|
||||
return TC_ACT_OK;
|
||||
th = (struct tcphdr *)(ip6h + 1);
|
||||
if (th + 1 > data_end)
|
||||
return TC_ACT_OK;
|
||||
/* Is it the testing traffic? */
|
||||
if (th->dest != srv_sa6.sin6_port)
|
||||
return TC_ACT_OK;
|
||||
tuple_len = sizeof(tuple->ipv6);
|
||||
tuple = (struct bpf_sock_tuple *)&ip6h->saddr;
|
||||
iphdr = ip6h;
|
||||
iphdr_size = sizeof(*ip6h);
|
||||
break;
|
||||
default:
|
||||
return TC_ACT_OK;
|
||||
}
|
||||
|
||||
/* Is it the testing traffic? */
|
||||
if (th->dest != srv_sa6.sin6_port)
|
||||
return TC_ACT_OK;
|
||||
|
||||
tuple_len = sizeof(tuple->ipv6);
|
||||
tuple = (struct bpf_sock_tuple *)&ip6h->saddr;
|
||||
if ((void *)tuple + tuple_len > data_end) {
|
||||
LOG();
|
||||
return TC_ACT_OK;
|
||||
@ -126,7 +161,7 @@ static int handle_ip6_tcp(struct ipv6hdr *ip6h, struct __sk_buff *skb)
|
||||
|
||||
listen_tp_sport = tp->inet_conn.icsk_inet.sk.__sk_common.skc_num;
|
||||
|
||||
test_syncookie_helper(ip6h, th, tp, skb);
|
||||
test_syncookie_helper(iphdr, iphdr_size, th, tp, skb);
|
||||
bpf_sk_release(tp);
|
||||
return TC_ACT_OK;
|
||||
}
|
||||
@ -142,7 +177,6 @@ static int handle_ip6_tcp(struct ipv6hdr *ip6h, struct __sk_buff *skb)
|
||||
SEC("tc")
|
||||
int cls_ingress(struct __sk_buff *skb)
|
||||
{
|
||||
struct ipv6hdr *ip6h;
|
||||
struct ethhdr *eth;
|
||||
void *data_end;
|
||||
|
||||
@ -152,17 +186,11 @@ int cls_ingress(struct __sk_buff *skb)
|
||||
if (eth + 1 > data_end)
|
||||
return TC_ACT_OK;
|
||||
|
||||
if (eth->h_proto != bpf_htons(ETH_P_IPV6))
|
||||
if (eth->h_proto != bpf_htons(ETH_P_IP) &&
|
||||
eth->h_proto != bpf_htons(ETH_P_IPV6))
|
||||
return TC_ACT_OK;
|
||||
|
||||
ip6h = (struct ipv6hdr *)(eth + 1);
|
||||
if (ip6h + 1 > data_end)
|
||||
return TC_ACT_OK;
|
||||
|
||||
if (ip6h->nexthdr == IPPROTO_TCP)
|
||||
return handle_ip6_tcp(ip6h, skb);
|
||||
|
||||
return TC_ACT_OK;
|
||||
return handle_ip_tcp(eth, skb);
|
||||
}
|
||||
|
||||
char _license[] SEC("license") = "GPL";
|
||||
|
@ -1,167 +0,0 @@
|
||||
// SPDX-License-Identifier: GPL-2.0
|
||||
// Copyright (c) 2018 Facebook
|
||||
// Copyright (c) 2019 Cloudflare
|
||||
|
||||
#include <string.h>
|
||||
|
||||
#include <linux/bpf.h>
|
||||
#include <linux/pkt_cls.h>
|
||||
#include <linux/if_ether.h>
|
||||
#include <linux/in.h>
|
||||
#include <linux/ip.h>
|
||||
#include <linux/ipv6.h>
|
||||
#include <sys/socket.h>
|
||||
#include <linux/tcp.h>
|
||||
|
||||
#include <bpf/bpf_helpers.h>
|
||||
#include <bpf/bpf_endian.h>
|
||||
|
||||
struct {
|
||||
__uint(type, BPF_MAP_TYPE_ARRAY);
|
||||
__type(key, __u32);
|
||||
__type(value, __u32);
|
||||
__uint(max_entries, 3);
|
||||
} results SEC(".maps");
|
||||
|
||||
static __always_inline __s64 gen_syncookie(void *data_end, struct bpf_sock *sk,
|
||||
void *iph, __u32 ip_size,
|
||||
struct tcphdr *tcph)
|
||||
{
|
||||
__u32 thlen = tcph->doff * 4;
|
||||
|
||||
if (tcph->syn && !tcph->ack) {
|
||||
// packet should only have an MSS option
|
||||
if (thlen != 24)
|
||||
return 0;
|
||||
|
||||
if ((void *)tcph + thlen > data_end)
|
||||
return 0;
|
||||
|
||||
return bpf_tcp_gen_syncookie(sk, iph, ip_size, tcph, thlen);
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
static __always_inline void check_syncookie(void *ctx, void *data,
|
||||
void *data_end)
|
||||
{
|
||||
struct bpf_sock_tuple tup;
|
||||
struct bpf_sock *sk;
|
||||
struct ethhdr *ethh;
|
||||
struct iphdr *ipv4h;
|
||||
struct ipv6hdr *ipv6h;
|
||||
struct tcphdr *tcph;
|
||||
int ret;
|
||||
__u32 key_mss = 2;
|
||||
__u32 key_gen = 1;
|
||||
__u32 key = 0;
|
||||
__s64 seq_mss;
|
||||
|
||||
ethh = data;
|
||||
if (ethh + 1 > data_end)
|
||||
return;
|
||||
|
||||
switch (bpf_ntohs(ethh->h_proto)) {
|
||||
case ETH_P_IP:
|
||||
ipv4h = data + sizeof(struct ethhdr);
|
||||
if (ipv4h + 1 > data_end)
|
||||
return;
|
||||
|
||||
if (ipv4h->ihl != 5)
|
||||
return;
|
||||
|
||||
tcph = data + sizeof(struct ethhdr) + sizeof(struct iphdr);
|
||||
if (tcph + 1 > data_end)
|
||||
return;
|
||||
|
||||
tup.ipv4.saddr = ipv4h->saddr;
|
||||
tup.ipv4.daddr = ipv4h->daddr;
|
||||
tup.ipv4.sport = tcph->source;
|
||||
tup.ipv4.dport = tcph->dest;
|
||||
|
||||
sk = bpf_skc_lookup_tcp(ctx, &tup, sizeof(tup.ipv4),
|
||||
BPF_F_CURRENT_NETNS, 0);
|
||||
if (!sk)
|
||||
return;
|
||||
|
||||
if (sk->state != BPF_TCP_LISTEN)
|
||||
goto release;
|
||||
|
||||
seq_mss = gen_syncookie(data_end, sk, ipv4h, sizeof(*ipv4h),
|
||||
tcph);
|
||||
|
||||
ret = bpf_tcp_check_syncookie(sk, ipv4h, sizeof(*ipv4h),
|
||||
tcph, sizeof(*tcph));
|
||||
break;
|
||||
|
||||
case ETH_P_IPV6:
|
||||
ipv6h = data + sizeof(struct ethhdr);
|
||||
if (ipv6h + 1 > data_end)
|
||||
return;
|
||||
|
||||
if (ipv6h->nexthdr != IPPROTO_TCP)
|
||||
return;
|
||||
|
||||
tcph = data + sizeof(struct ethhdr) + sizeof(struct ipv6hdr);
|
||||
if (tcph + 1 > data_end)
|
||||
return;
|
||||
|
||||
memcpy(tup.ipv6.saddr, &ipv6h->saddr, sizeof(tup.ipv6.saddr));
|
||||
memcpy(tup.ipv6.daddr, &ipv6h->daddr, sizeof(tup.ipv6.daddr));
|
||||
tup.ipv6.sport = tcph->source;
|
||||
tup.ipv6.dport = tcph->dest;
|
||||
|
||||
sk = bpf_skc_lookup_tcp(ctx, &tup, sizeof(tup.ipv6),
|
||||
BPF_F_CURRENT_NETNS, 0);
|
||||
if (!sk)
|
||||
return;
|
||||
|
||||
if (sk->state != BPF_TCP_LISTEN)
|
||||
goto release;
|
||||
|
||||
seq_mss = gen_syncookie(data_end, sk, ipv6h, sizeof(*ipv6h),
|
||||
tcph);
|
||||
|
||||
ret = bpf_tcp_check_syncookie(sk, ipv6h, sizeof(*ipv6h),
|
||||
tcph, sizeof(*tcph));
|
||||
break;
|
||||
|
||||
default:
|
||||
return;
|
||||
}
|
||||
|
||||
if (seq_mss > 0) {
|
||||
__u32 cookie = (__u32)seq_mss;
|
||||
__u32 mss = seq_mss >> 32;
|
||||
|
||||
bpf_map_update_elem(&results, &key_gen, &cookie, 0);
|
||||
bpf_map_update_elem(&results, &key_mss, &mss, 0);
|
||||
}
|
||||
|
||||
if (ret == 0) {
|
||||
__u32 cookie = bpf_ntohl(tcph->ack_seq) - 1;
|
||||
|
||||
bpf_map_update_elem(&results, &key, &cookie, 0);
|
||||
}
|
||||
|
||||
release:
|
||||
bpf_sk_release(sk);
|
||||
}
|
||||
|
||||
SEC("tc")
|
||||
int check_syncookie_clsact(struct __sk_buff *skb)
|
||||
{
|
||||
check_syncookie(skb, (void *)(long)skb->data,
|
||||
(void *)(long)skb->data_end);
|
||||
return TC_ACT_OK;
|
||||
}
|
||||
|
||||
SEC("xdp")
|
||||
int check_syncookie_xdp(struct xdp_md *ctx)
|
||||
{
|
||||
check_syncookie(ctx, (void *)(long)ctx->data,
|
||||
(void *)(long)ctx->data_end);
|
||||
return XDP_PASS;
|
||||
}
|
||||
|
||||
char _license[] SEC("license") = "GPL";
|
@ -368,8 +368,7 @@ __naked void a_read_only_array_2_1(void)
|
||||
r4 = 0; \
|
||||
r5 = 0; \
|
||||
call %[bpf_csum_diff]; \
|
||||
l0_%=: r0 &= 0xffff; \
|
||||
exit; \
|
||||
l0_%=: exit; \
|
||||
" :
|
||||
: __imm(bpf_csum_diff),
|
||||
__imm(bpf_map_lookup_elem),
|
||||
|
@ -21,7 +21,6 @@
|
||||
|
||||
#define tcp_flag_word(tp) (((union tcp_word_hdr *)(tp))->words[3])
|
||||
|
||||
#define IP_DF 0x4000
|
||||
#define IP_MF 0x2000
|
||||
#define IP_OFFSET 0x1fff
|
||||
|
||||
@ -442,7 +441,7 @@ static __always_inline int tcp_lookup(void *ctx, struct header_pointers *hdr, bo
|
||||
/* TCP doesn't normally use fragments, and XDP can't reassemble
|
||||
* them.
|
||||
*/
|
||||
if ((hdr->ipv4->frag_off & bpf_htons(IP_DF | IP_MF | IP_OFFSET)) != bpf_htons(IP_DF))
|
||||
if ((hdr->ipv4->frag_off & bpf_htons(IP_MF | IP_OFFSET)) != 0)
|
||||
return XDP_DROP;
|
||||
|
||||
tup.ipv4.saddr = hdr->ipv4->saddr;
|
||||
|
@ -56,6 +56,8 @@ static void running_handler(int a);
|
||||
#define BPF_SOCKHASH_FILENAME "test_sockhash_kern.bpf.o"
|
||||
#define CG_PATH "/sockmap"
|
||||
|
||||
#define EDATAINTEGRITY 2001
|
||||
|
||||
/* global sockets */
|
||||
int s1, s2, c1, c2, p1, p2;
|
||||
int test_cnt;
|
||||
@ -510,23 +512,25 @@ static int msg_alloc_iov(struct msghdr *msg,
|
||||
return -ENOMEM;
|
||||
}
|
||||
|
||||
static int msg_verify_data(struct msghdr *msg, int size, int chunk_sz)
|
||||
/* TODO: Add verification logic for push, pull and pop data */
|
||||
static int msg_verify_data(struct msghdr *msg, int size, int chunk_sz,
|
||||
unsigned char *k_p, int *bytes_cnt_p)
|
||||
{
|
||||
int i, j = 0, bytes_cnt = 0;
|
||||
unsigned char k = 0;
|
||||
int i, j, bytes_cnt = *bytes_cnt_p;
|
||||
unsigned char k = *k_p;
|
||||
|
||||
for (i = 0; i < msg->msg_iovlen; i++) {
|
||||
for (i = 0, j = 0; i < msg->msg_iovlen && size; i++, j = 0) {
|
||||
unsigned char *d = msg->msg_iov[i].iov_base;
|
||||
|
||||
/* Special case test for skb ingress + ktls */
|
||||
if (i == 0 && txmsg_ktls_skb) {
|
||||
if (msg->msg_iov[i].iov_len < 4)
|
||||
return -EIO;
|
||||
return -EDATAINTEGRITY;
|
||||
if (memcmp(d, "PASS", 4) != 0) {
|
||||
fprintf(stderr,
|
||||
"detected skb data error with skb ingress update @iov[%i]:%i \"%02x %02x %02x %02x\" != \"PASS\"\n",
|
||||
i, 0, d[0], d[1], d[2], d[3]);
|
||||
return -EIO;
|
||||
return -EDATAINTEGRITY;
|
||||
}
|
||||
j = 4; /* advance index past PASS header */
|
||||
}
|
||||
@ -536,7 +540,7 @@ static int msg_verify_data(struct msghdr *msg, int size, int chunk_sz)
|
||||
fprintf(stderr,
|
||||
"detected data corruption @iov[%i]:%i %02x != %02x, %02x ?= %02x\n",
|
||||
i, j, d[j], k - 1, d[j+1], k);
|
||||
return -EIO;
|
||||
return -EDATAINTEGRITY;
|
||||
}
|
||||
bytes_cnt++;
|
||||
if (bytes_cnt == chunk_sz) {
|
||||
@ -546,6 +550,8 @@ static int msg_verify_data(struct msghdr *msg, int size, int chunk_sz)
|
||||
size--;
|
||||
}
|
||||
}
|
||||
*k_p = k;
|
||||
*bytes_cnt_p = bytes_cnt;
|
||||
return 0;
|
||||
}
|
||||
|
||||
@ -602,6 +608,8 @@ static int msg_loop(int fd, int iov_count, int iov_length, int cnt,
|
||||
float total_bytes, txmsg_pop_total;
|
||||
int fd_flags = O_NONBLOCK;
|
||||
struct timeval timeout;
|
||||
unsigned char k = 0;
|
||||
int bytes_cnt = 0;
|
||||
fd_set w;
|
||||
|
||||
fcntl(fd, fd_flags);
|
||||
@ -696,7 +704,7 @@ static int msg_loop(int fd, int iov_count, int iov_length, int cnt,
|
||||
iov_length * cnt :
|
||||
iov_length * iov_count;
|
||||
|
||||
errno = msg_verify_data(&msg, recv, chunk_sz);
|
||||
errno = msg_verify_data(&msg, recv, chunk_sz, &k, &bytes_cnt);
|
||||
if (errno) {
|
||||
perror("data verify msg failed");
|
||||
goto out_errno;
|
||||
@ -704,7 +712,9 @@ static int msg_loop(int fd, int iov_count, int iov_length, int cnt,
|
||||
if (recvp) {
|
||||
errno = msg_verify_data(&msg_peek,
|
||||
recvp,
|
||||
chunk_sz);
|
||||
chunk_sz,
|
||||
&k,
|
||||
&bytes_cnt);
|
||||
if (errno) {
|
||||
perror("data verify msg_peek failed");
|
||||
goto out_errno;
|
||||
@ -812,7 +822,7 @@ static int sendmsg_test(struct sockmap_options *opt)
|
||||
s.bytes_sent, sent_Bps, sent_Bps/giga,
|
||||
s.bytes_recvd, recvd_Bps, recvd_Bps/giga,
|
||||
peek_flag ? "(peek_msg)" : "");
|
||||
if (err && txmsg_cork)
|
||||
if (err && err != -EDATAINTEGRITY && txmsg_cork)
|
||||
err = 0;
|
||||
exit(err ? 1 : 0);
|
||||
} else if (rxpid == -1) {
|
||||
@ -1596,7 +1606,7 @@ static void test_txmsg_pull(int cgrp, struct sockmap_options *opt)
|
||||
test_send_large(opt, cgrp);
|
||||
|
||||
/* Test pull + redirect */
|
||||
txmsg_redir = 0;
|
||||
txmsg_redir = 1;
|
||||
txmsg_start = 1;
|
||||
txmsg_end = 2;
|
||||
test_send(opt, cgrp);
|
||||
|
@ -1,85 +0,0 @@
|
||||
#!/bin/sh
|
||||
# SPDX-License-Identifier: GPL-2.0
|
||||
# Copyright (c) 2018 Facebook
|
||||
# Copyright (c) 2019 Cloudflare
|
||||
|
||||
set -eu
|
||||
readonly NS1="ns1-$(mktemp -u XXXXXX)"
|
||||
|
||||
wait_for_ip()
|
||||
{
|
||||
local _i
|
||||
printf "Wait for IP %s to become available " "$1"
|
||||
for _i in $(seq ${MAX_PING_TRIES}); do
|
||||
printf "."
|
||||
if ns1_exec ping -c 1 -W 1 "$1" >/dev/null 2>&1; then
|
||||
echo " OK"
|
||||
return
|
||||
fi
|
||||
sleep 1
|
||||
done
|
||||
echo 1>&2 "ERROR: Timeout waiting for test IP to become available."
|
||||
exit 1
|
||||
}
|
||||
|
||||
get_prog_id()
|
||||
{
|
||||
awk '/ id / {sub(/.* id /, "", $0); print($1)}'
|
||||
}
|
||||
|
||||
ns1_exec()
|
||||
{
|
||||
ip netns exec ${NS1} "$@"
|
||||
}
|
||||
|
||||
setup()
|
||||
{
|
||||
ip netns add ${NS1}
|
||||
ns1_exec ip link set lo up
|
||||
|
||||
ns1_exec sysctl -w net.ipv4.tcp_syncookies=2
|
||||
ns1_exec sysctl -w net.ipv4.tcp_window_scaling=0
|
||||
ns1_exec sysctl -w net.ipv4.tcp_timestamps=0
|
||||
ns1_exec sysctl -w net.ipv4.tcp_sack=0
|
||||
|
||||
wait_for_ip 127.0.0.1
|
||||
wait_for_ip ::1
|
||||
}
|
||||
|
||||
cleanup()
|
||||
{
|
||||
ip netns del ns1 2>/dev/null || :
|
||||
}
|
||||
|
||||
main()
|
||||
{
|
||||
trap cleanup EXIT 2 3 6 15
|
||||
setup
|
||||
|
||||
printf "Testing clsact..."
|
||||
ns1_exec tc qdisc add dev "${TEST_IF}" clsact
|
||||
ns1_exec tc filter add dev "${TEST_IF}" ingress \
|
||||
bpf obj "${BPF_PROG_OBJ}" sec "${CLSACT_SECTION}" da
|
||||
|
||||
BPF_PROG_ID=$(ns1_exec tc filter show dev "${TEST_IF}" ingress | \
|
||||
get_prog_id)
|
||||
ns1_exec "${PROG}" "${BPF_PROG_ID}"
|
||||
ns1_exec tc qdisc del dev "${TEST_IF}" clsact
|
||||
|
||||
printf "Testing XDP..."
|
||||
ns1_exec ip link set "${TEST_IF}" xdp \
|
||||
object "${BPF_PROG_OBJ}" section "${XDP_SECTION}"
|
||||
BPF_PROG_ID=$(ns1_exec ip link show "${TEST_IF}" | get_prog_id)
|
||||
ns1_exec "${PROG}" "${BPF_PROG_ID}"
|
||||
}
|
||||
|
||||
DIR=$(dirname $0)
|
||||
TEST_IF=lo
|
||||
MAX_PING_TRIES=5
|
||||
BPF_PROG_OBJ="${DIR}/test_tcp_check_syncookie_kern.bpf.o"
|
||||
CLSACT_SECTION="tc"
|
||||
XDP_SECTION="xdp"
|
||||
BPF_PROG_ID=0
|
||||
PROG="${DIR}/test_tcp_check_syncookie_user"
|
||||
|
||||
main
|
@ -1,213 +0,0 @@
|
||||
// SPDX-License-Identifier: GPL-2.0
|
||||
// Copyright (c) 2018 Facebook
|
||||
// Copyright (c) 2019 Cloudflare
|
||||
|
||||
#include <limits.h>
|
||||
#include <string.h>
|
||||
#include <stdlib.h>
|
||||
#include <unistd.h>
|
||||
|
||||
#include <arpa/inet.h>
|
||||
#include <netinet/in.h>
|
||||
#include <sys/types.h>
|
||||
#include <sys/socket.h>
|
||||
|
||||
#include <bpf/bpf.h>
|
||||
#include <bpf/libbpf.h>
|
||||
|
||||
#include "cgroup_helpers.h"
|
||||
#include "network_helpers.h"
|
||||
|
||||
static int get_map_fd_by_prog_id(int prog_id, bool *xdp)
|
||||
{
|
||||
struct bpf_prog_info info = {};
|
||||
__u32 info_len = sizeof(info);
|
||||
__u32 map_ids[1];
|
||||
int prog_fd = -1;
|
||||
int map_fd = -1;
|
||||
|
||||
prog_fd = bpf_prog_get_fd_by_id(prog_id);
|
||||
if (prog_fd < 0) {
|
||||
log_err("Failed to get fd by prog id %d", prog_id);
|
||||
goto err;
|
||||
}
|
||||
|
||||
info.nr_map_ids = 1;
|
||||
info.map_ids = (__u64)(unsigned long)map_ids;
|
||||
|
||||
if (bpf_prog_get_info_by_fd(prog_fd, &info, &info_len)) {
|
||||
log_err("Failed to get info by prog fd %d", prog_fd);
|
||||
goto err;
|
||||
}
|
||||
|
||||
if (!info.nr_map_ids) {
|
||||
log_err("No maps found for prog fd %d", prog_fd);
|
||||
goto err;
|
||||
}
|
||||
|
||||
*xdp = info.type == BPF_PROG_TYPE_XDP;
|
||||
|
||||
map_fd = bpf_map_get_fd_by_id(map_ids[0]);
|
||||
if (map_fd < 0)
|
||||
log_err("Failed to get fd by map id %d", map_ids[0]);
|
||||
err:
|
||||
if (prog_fd >= 0)
|
||||
close(prog_fd);
|
||||
return map_fd;
|
||||
}
|
||||
|
||||
static int run_test(int server_fd, int results_fd, bool xdp)
|
||||
{
|
||||
int client = -1, srv_client = -1;
|
||||
int ret = 0;
|
||||
__u32 key = 0;
|
||||
__u32 key_gen = 1;
|
||||
__u32 key_mss = 2;
|
||||
__u32 value = 0;
|
||||
__u32 value_gen = 0;
|
||||
__u32 value_mss = 0;
|
||||
|
||||
if (bpf_map_update_elem(results_fd, &key, &value, 0) < 0) {
|
||||
log_err("Can't clear results");
|
||||
goto err;
|
||||
}
|
||||
|
||||
if (bpf_map_update_elem(results_fd, &key_gen, &value_gen, 0) < 0) {
|
||||
log_err("Can't clear results");
|
||||
goto err;
|
||||
}
|
||||
|
||||
if (bpf_map_update_elem(results_fd, &key_mss, &value_mss, 0) < 0) {
|
||||
log_err("Can't clear results");
|
||||
goto err;
|
||||
}
|
||||
|
||||
client = connect_to_fd(server_fd, 0);
|
||||
if (client == -1)
|
||||
goto err;
|
||||
|
||||
srv_client = accept(server_fd, NULL, 0);
|
||||
if (srv_client == -1) {
|
||||
log_err("Can't accept connection");
|
||||
goto err;
|
||||
}
|
||||
|
||||
if (bpf_map_lookup_elem(results_fd, &key, &value) < 0) {
|
||||
log_err("Can't lookup result");
|
||||
goto err;
|
||||
}
|
||||
|
||||
if (value == 0) {
|
||||
log_err("Didn't match syncookie: %u", value);
|
||||
goto err;
|
||||
}
|
||||
|
||||
if (bpf_map_lookup_elem(results_fd, &key_gen, &value_gen) < 0) {
|
||||
log_err("Can't lookup result");
|
||||
goto err;
|
||||
}
|
||||
|
||||
if (xdp && value_gen == 0) {
|
||||
// SYN packets do not get passed through generic XDP, skip the
|
||||
// rest of the test.
|
||||
printf("Skipping XDP cookie check\n");
|
||||
goto out;
|
||||
}
|
||||
|
||||
if (bpf_map_lookup_elem(results_fd, &key_mss, &value_mss) < 0) {
|
||||
log_err("Can't lookup result");
|
||||
goto err;
|
||||
}
|
||||
|
||||
if (value != value_gen) {
|
||||
log_err("BPF generated cookie does not match kernel one");
|
||||
goto err;
|
||||
}
|
||||
|
||||
if (value_mss < 536 || value_mss > USHRT_MAX) {
|
||||
log_err("Unexpected MSS retrieved");
|
||||
goto err;
|
||||
}
|
||||
|
||||
goto out;
|
||||
|
||||
err:
|
||||
ret = 1;
|
||||
out:
|
||||
close(client);
|
||||
close(srv_client);
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int v6only_true(int fd, void *opts)
|
||||
{
|
||||
int mode = true;
|
||||
|
||||
return setsockopt(fd, IPPROTO_IPV6, IPV6_V6ONLY, &mode, sizeof(mode));
|
||||
}
|
||||
|
||||
static int v6only_false(int fd, void *opts)
|
||||
{
|
||||
int mode = false;
|
||||
|
||||
return setsockopt(fd, IPPROTO_IPV6, IPV6_V6ONLY, &mode, sizeof(mode));
|
||||
}
|
||||
|
||||
int main(int argc, char **argv)
|
||||
{
|
||||
struct network_helper_opts opts = { 0 };
|
||||
int server = -1;
|
||||
int server_v6 = -1;
|
||||
int server_dual = -1;
|
||||
int results = -1;
|
||||
int err = 0;
|
||||
bool xdp;
|
||||
|
||||
if (argc < 2) {
|
||||
fprintf(stderr, "Usage: %s prog_id\n", argv[0]);
|
||||
exit(1);
|
||||
}
|
||||
|
||||
/* Use libbpf 1.0 API mode */
|
||||
libbpf_set_strict_mode(LIBBPF_STRICT_ALL);
|
||||
|
||||
results = get_map_fd_by_prog_id(atoi(argv[1]), &xdp);
|
||||
if (results < 0) {
|
||||
log_err("Can't get map");
|
||||
goto err;
|
||||
}
|
||||
|
||||
server = start_server_str(AF_INET, SOCK_STREAM, "127.0.0.1", 0, NULL);
|
||||
if (server == -1)
|
||||
goto err;
|
||||
|
||||
opts.post_socket_cb = v6only_true;
|
||||
server_v6 = start_server_str(AF_INET6, SOCK_STREAM, "::1", 0, &opts);
|
||||
if (server_v6 == -1)
|
||||
goto err;
|
||||
|
||||
opts.post_socket_cb = v6only_false;
|
||||
server_dual = start_server_str(AF_INET6, SOCK_STREAM, "::0", 0, &opts);
|
||||
if (server_dual == -1)
|
||||
goto err;
|
||||
|
||||
if (run_test(server, results, xdp))
|
||||
goto err;
|
||||
|
||||
if (run_test(server_v6, results, xdp))
|
||||
goto err;
|
||||
|
||||
if (run_test(server_dual, results, xdp))
|
||||
goto err;
|
||||
|
||||
printf("ok\n");
|
||||
goto out;
|
||||
err:
|
||||
err = 1;
|
||||
out:
|
||||
close(server);
|
||||
close(server_v6);
|
||||
close(server_dual);
|
||||
close(results);
|
||||
return err;
|
||||
}
|
Loading…
Reference in New Issue
Block a user