7cd6df4f5e
Modify the packet pacing algorithm so that it works with multi-buffer packets. This algorithm makes sure we do not send too many buffers to the receiving thread so that packets have to be dropped. The previous algorithm made the assumption that each packet only consumes one buffer, but that is not true anymore when multi-buffer support gets added. Instead, we find out what the largest packet size is in the packet stream and assume that each packet will consume this many buffers. This is conservative and overly cautious as there might be smaller packets in the stream that need fewer buffers per packet. But it keeps the algorithm simple. Also simplify it by removing the pthread conditional and just test if there is enough space in the Rx thread before trying to send one more batch. Also makes the tests run faster. Signed-off-by: Magnus Karlsson <magnus.karlsson@intel.com> Link: https://lore.kernel.org/r/20230516103109.3066-11-magnus.karlsson@gmail.com Signed-off-by: Alexei Starovoitov <ast@kernel.org>
2026 lines
53 KiB
C
2026 lines
53 KiB
C
// SPDX-License-Identifier: GPL-2.0
|
|
/* Copyright(c) 2020 Intel Corporation. */
|
|
|
|
/*
|
|
* Some functions in this program are taken from
|
|
* Linux kernel samples/bpf/xdpsock* and modified
|
|
* for use.
|
|
*
|
|
* See test_xsk.sh for detailed information on test topology
|
|
* and prerequisite network setup.
|
|
*
|
|
* This test program contains two threads, each thread is single socket with
|
|
* a unique UMEM. It validates in-order packet delivery and packet content
|
|
* by sending packets to each other.
|
|
*
|
|
* Tests Information:
|
|
* ------------------
|
|
* These selftests test AF_XDP SKB and Native/DRV modes using veth
|
|
* Virtual Ethernet interfaces.
|
|
*
|
|
* For each mode, the following tests are run:
|
|
* a. nopoll - soft-irq processing in run-to-completion mode
|
|
* b. poll - using poll() syscall
|
|
* c. Socket Teardown
|
|
* Create a Tx and a Rx socket, Tx from one socket, Rx on another. Destroy
|
|
* both sockets, then repeat multiple times. Only nopoll mode is used
|
|
* d. Bi-directional sockets
|
|
* Configure sockets as bi-directional tx/rx sockets, sets up fill and
|
|
* completion rings on each socket, tx/rx in both directions. Only nopoll
|
|
* mode is used
|
|
* e. Statistics
|
|
* Trigger some error conditions and ensure that the appropriate statistics
|
|
* are incremented. Within this test, the following statistics are tested:
|
|
* i. rx dropped
|
|
* Increase the UMEM frame headroom to a value which results in
|
|
* insufficient space in the rx buffer for both the packet and the headroom.
|
|
* ii. tx invalid
|
|
* Set the 'len' field of tx descriptors to an invalid value (umem frame
|
|
* size + 1).
|
|
* iii. rx ring full
|
|
* Reduce the size of the RX ring to a fraction of the fill ring size.
|
|
* iv. fill queue empty
|
|
* Do not populate the fill queue and then try to receive pkts.
|
|
* f. bpf_link resource persistence
|
|
* Configure sockets at indexes 0 and 1, run a traffic on queue ids 0,
|
|
* then remove xsk sockets from queue 0 on both veth interfaces and
|
|
* finally run a traffic on queues ids 1
|
|
* g. unaligned mode
|
|
* h. tests for invalid and corner case Tx descriptors so that the correct ones
|
|
* are discarded and let through, respectively.
|
|
* i. 2K frame size tests
|
|
*
|
|
* Total tests: 12
|
|
*
|
|
* Flow:
|
|
* -----
|
|
* - Single process spawns two threads: Tx and Rx
|
|
* - Each of these two threads attach to a veth interface
|
|
* - Each thread creates one AF_XDP socket connected to a unique umem for each
|
|
* veth interface
|
|
* - Tx thread Transmits a number of packets from veth<xxxx> to veth<yyyy>
|
|
* - Rx thread verifies if all packets were received and delivered in-order,
|
|
* and have the right content
|
|
*
|
|
* Enable/disable packet dump mode:
|
|
* --------------------------
|
|
* To enable L2 - L4 headers and payload dump of each packet on STDOUT, add
|
|
* parameter -D to params array in test_xsk.sh, i.e. params=("-S" "-D")
|
|
*/
|
|
|
|
#define _GNU_SOURCE
|
|
#include <assert.h>
|
|
#include <fcntl.h>
|
|
#include <errno.h>
|
|
#include <getopt.h>
|
|
#include <asm/barrier.h>
|
|
#include <linux/if_link.h>
|
|
#include <linux/if_ether.h>
|
|
#include <linux/mman.h>
|
|
#include <arpa/inet.h>
|
|
#include <net/if.h>
|
|
#include <locale.h>
|
|
#include <poll.h>
|
|
#include <pthread.h>
|
|
#include <signal.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <stddef.h>
|
|
#include <sys/mman.h>
|
|
#include <sys/socket.h>
|
|
#include <sys/time.h>
|
|
#include <sys/types.h>
|
|
#include <time.h>
|
|
#include <unistd.h>
|
|
|
|
#include "xsk_xdp_progs.skel.h"
|
|
#include "xsk.h"
|
|
#include "xskxceiver.h"
|
|
#include <bpf/bpf.h>
|
|
#include <linux/filter.h>
|
|
#include "../kselftest.h"
|
|
#include "xsk_xdp_metadata.h"
|
|
|
|
static const char *MAC1 = "\x00\x0A\x56\x9E\xEE\x62";
|
|
static const char *MAC2 = "\x00\x0A\x56\x9E\xEE\x61";
|
|
|
|
static void __exit_with_error(int error, const char *file, const char *func, int line)
|
|
{
|
|
ksft_test_result_fail("[%s:%s:%i]: ERROR: %d/\"%s\"\n", file, func, line, error,
|
|
strerror(error));
|
|
ksft_exit_xfail();
|
|
}
|
|
|
|
#define exit_with_error(error) __exit_with_error(error, __FILE__, __func__, __LINE__)
|
|
#define busy_poll_string(test) (test)->ifobj_tx->busy_poll ? "BUSY-POLL " : ""
|
|
static char *mode_string(struct test_spec *test)
|
|
{
|
|
switch (test->mode) {
|
|
case TEST_MODE_SKB:
|
|
return "SKB";
|
|
case TEST_MODE_DRV:
|
|
return "DRV";
|
|
case TEST_MODE_ZC:
|
|
return "ZC";
|
|
default:
|
|
return "BOGUS";
|
|
}
|
|
}
|
|
|
|
static void report_failure(struct test_spec *test)
|
|
{
|
|
if (test->fail)
|
|
return;
|
|
|
|
ksft_test_result_fail("FAIL: %s %s%s\n", mode_string(test), busy_poll_string(test),
|
|
test->name);
|
|
test->fail = true;
|
|
}
|
|
|
|
/* The payload is a word consisting of a packet sequence number in the upper
|
|
* 16-bits and a intra packet data sequence number in the lower 16 bits. So the 3rd packet's
|
|
* 5th word of data will contain the number (2<<16) | 4 as they are numbered from 0.
|
|
*/
|
|
static void write_payload(void *dest, u32 pkt_nb, u32 start, u32 size)
|
|
{
|
|
u32 *ptr = (u32 *)dest, i;
|
|
|
|
start /= sizeof(*ptr);
|
|
size /= sizeof(*ptr);
|
|
for (i = 0; i < size; i++)
|
|
ptr[i] = htonl(pkt_nb << 16 | (i + start));
|
|
}
|
|
|
|
static void gen_eth_hdr(struct ifobject *ifobject, struct ethhdr *eth_hdr)
|
|
{
|
|
memcpy(eth_hdr->h_dest, ifobject->dst_mac, ETH_ALEN);
|
|
memcpy(eth_hdr->h_source, ifobject->src_mac, ETH_ALEN);
|
|
eth_hdr->h_proto = htons(ETH_P_LOOPBACK);
|
|
}
|
|
|
|
static bool is_umem_valid(struct ifobject *ifobj)
|
|
{
|
|
return !!ifobj->umem->umem;
|
|
}
|
|
|
|
static u32 mode_to_xdp_flags(enum test_mode mode)
|
|
{
|
|
return (mode == TEST_MODE_SKB) ? XDP_FLAGS_SKB_MODE : XDP_FLAGS_DRV_MODE;
|
|
}
|
|
|
|
static u64 umem_size(struct xsk_umem_info *umem)
|
|
{
|
|
return umem->num_frames * umem->frame_size;
|
|
}
|
|
|
|
static int xsk_configure_umem(struct ifobject *ifobj, struct xsk_umem_info *umem, void *buffer,
|
|
u64 size)
|
|
{
|
|
struct xsk_umem_config cfg = {
|
|
.fill_size = XSK_RING_PROD__DEFAULT_NUM_DESCS,
|
|
.comp_size = XSK_RING_CONS__DEFAULT_NUM_DESCS,
|
|
.frame_size = umem->frame_size,
|
|
.frame_headroom = umem->frame_headroom,
|
|
.flags = XSK_UMEM__DEFAULT_FLAGS
|
|
};
|
|
int ret;
|
|
|
|
if (umem->unaligned_mode)
|
|
cfg.flags |= XDP_UMEM_UNALIGNED_CHUNK_FLAG;
|
|
|
|
ret = xsk_umem__create(&umem->umem, buffer, size,
|
|
&umem->fq, &umem->cq, &cfg);
|
|
if (ret)
|
|
return ret;
|
|
|
|
umem->buffer = buffer;
|
|
if (ifobj->shared_umem && ifobj->rx_on) {
|
|
umem->base_addr = umem_size(umem);
|
|
umem->next_buffer = umem_size(umem);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static u64 umem_alloc_buffer(struct xsk_umem_info *umem)
|
|
{
|
|
u64 addr;
|
|
|
|
addr = umem->next_buffer;
|
|
umem->next_buffer += umem->frame_size;
|
|
if (umem->next_buffer >= umem->base_addr + umem_size(umem))
|
|
umem->next_buffer = umem->base_addr;
|
|
|
|
return addr;
|
|
}
|
|
|
|
static void umem_reset_alloc(struct xsk_umem_info *umem)
|
|
{
|
|
umem->next_buffer = 0;
|
|
}
|
|
|
|
static void enable_busy_poll(struct xsk_socket_info *xsk)
|
|
{
|
|
int sock_opt;
|
|
|
|
sock_opt = 1;
|
|
if (setsockopt(xsk_socket__fd(xsk->xsk), SOL_SOCKET, SO_PREFER_BUSY_POLL,
|
|
(void *)&sock_opt, sizeof(sock_opt)) < 0)
|
|
exit_with_error(errno);
|
|
|
|
sock_opt = 20;
|
|
if (setsockopt(xsk_socket__fd(xsk->xsk), SOL_SOCKET, SO_BUSY_POLL,
|
|
(void *)&sock_opt, sizeof(sock_opt)) < 0)
|
|
exit_with_error(errno);
|
|
|
|
sock_opt = BATCH_SIZE;
|
|
if (setsockopt(xsk_socket__fd(xsk->xsk), SOL_SOCKET, SO_BUSY_POLL_BUDGET,
|
|
(void *)&sock_opt, sizeof(sock_opt)) < 0)
|
|
exit_with_error(errno);
|
|
}
|
|
|
|
static int __xsk_configure_socket(struct xsk_socket_info *xsk, struct xsk_umem_info *umem,
|
|
struct ifobject *ifobject, bool shared)
|
|
{
|
|
struct xsk_socket_config cfg = {};
|
|
struct xsk_ring_cons *rxr;
|
|
struct xsk_ring_prod *txr;
|
|
|
|
xsk->umem = umem;
|
|
cfg.rx_size = xsk->rxqsize;
|
|
cfg.tx_size = XSK_RING_PROD__DEFAULT_NUM_DESCS;
|
|
cfg.bind_flags = ifobject->bind_flags;
|
|
if (shared)
|
|
cfg.bind_flags |= XDP_SHARED_UMEM;
|
|
|
|
txr = ifobject->tx_on ? &xsk->tx : NULL;
|
|
rxr = ifobject->rx_on ? &xsk->rx : NULL;
|
|
return xsk_socket__create(&xsk->xsk, ifobject->ifindex, 0, umem->umem, rxr, txr, &cfg);
|
|
}
|
|
|
|
static bool ifobj_zc_avail(struct ifobject *ifobject)
|
|
{
|
|
size_t umem_sz = DEFAULT_UMEM_BUFFERS * XSK_UMEM__DEFAULT_FRAME_SIZE;
|
|
int mmap_flags = MAP_PRIVATE | MAP_ANONYMOUS | MAP_NORESERVE;
|
|
struct xsk_socket_info *xsk;
|
|
struct xsk_umem_info *umem;
|
|
bool zc_avail = false;
|
|
void *bufs;
|
|
int ret;
|
|
|
|
bufs = mmap(NULL, umem_sz, PROT_READ | PROT_WRITE, mmap_flags, -1, 0);
|
|
if (bufs == MAP_FAILED)
|
|
exit_with_error(errno);
|
|
|
|
umem = calloc(1, sizeof(struct xsk_umem_info));
|
|
if (!umem) {
|
|
munmap(bufs, umem_sz);
|
|
exit_with_error(ENOMEM);
|
|
}
|
|
umem->frame_size = XSK_UMEM__DEFAULT_FRAME_SIZE;
|
|
ret = xsk_configure_umem(ifobject, umem, bufs, umem_sz);
|
|
if (ret)
|
|
exit_with_error(-ret);
|
|
|
|
xsk = calloc(1, sizeof(struct xsk_socket_info));
|
|
if (!xsk)
|
|
goto out;
|
|
ifobject->bind_flags = XDP_USE_NEED_WAKEUP | XDP_ZEROCOPY;
|
|
ifobject->rx_on = true;
|
|
xsk->rxqsize = XSK_RING_CONS__DEFAULT_NUM_DESCS;
|
|
ret = __xsk_configure_socket(xsk, umem, ifobject, false);
|
|
if (!ret)
|
|
zc_avail = true;
|
|
|
|
xsk_socket__delete(xsk->xsk);
|
|
free(xsk);
|
|
out:
|
|
munmap(umem->buffer, umem_sz);
|
|
xsk_umem__delete(umem->umem);
|
|
free(umem);
|
|
return zc_avail;
|
|
}
|
|
|
|
static struct option long_options[] = {
|
|
{"interface", required_argument, 0, 'i'},
|
|
{"busy-poll", no_argument, 0, 'b'},
|
|
{"verbose", no_argument, 0, 'v'},
|
|
{0, 0, 0, 0}
|
|
};
|
|
|
|
static void usage(const char *prog)
|
|
{
|
|
const char *str =
|
|
" Usage: %s [OPTIONS]\n"
|
|
" Options:\n"
|
|
" -i, --interface Use interface\n"
|
|
" -v, --verbose Verbose output\n"
|
|
" -b, --busy-poll Enable busy poll\n";
|
|
|
|
ksft_print_msg(str, prog);
|
|
}
|
|
|
|
static bool validate_interface(struct ifobject *ifobj)
|
|
{
|
|
if (!strcmp(ifobj->ifname, ""))
|
|
return false;
|
|
return true;
|
|
}
|
|
|
|
static void parse_command_line(struct ifobject *ifobj_tx, struct ifobject *ifobj_rx, int argc,
|
|
char **argv)
|
|
{
|
|
struct ifobject *ifobj;
|
|
u32 interface_nb = 0;
|
|
int option_index, c;
|
|
|
|
opterr = 0;
|
|
|
|
for (;;) {
|
|
c = getopt_long(argc, argv, "i:vb", long_options, &option_index);
|
|
if (c == -1)
|
|
break;
|
|
|
|
switch (c) {
|
|
case 'i':
|
|
if (interface_nb == 0)
|
|
ifobj = ifobj_tx;
|
|
else if (interface_nb == 1)
|
|
ifobj = ifobj_rx;
|
|
else
|
|
break;
|
|
|
|
memcpy(ifobj->ifname, optarg,
|
|
min_t(size_t, MAX_INTERFACE_NAME_CHARS, strlen(optarg)));
|
|
|
|
ifobj->ifindex = if_nametoindex(ifobj->ifname);
|
|
if (!ifobj->ifindex)
|
|
exit_with_error(errno);
|
|
|
|
interface_nb++;
|
|
break;
|
|
case 'v':
|
|
opt_verbose = true;
|
|
break;
|
|
case 'b':
|
|
ifobj_tx->busy_poll = true;
|
|
ifobj_rx->busy_poll = true;
|
|
break;
|
|
default:
|
|
usage(basename(argv[0]));
|
|
ksft_exit_xfail();
|
|
}
|
|
}
|
|
}
|
|
|
|
static void __test_spec_init(struct test_spec *test, struct ifobject *ifobj_tx,
|
|
struct ifobject *ifobj_rx)
|
|
{
|
|
u32 i, j;
|
|
|
|
for (i = 0; i < MAX_INTERFACES; i++) {
|
|
struct ifobject *ifobj = i ? ifobj_rx : ifobj_tx;
|
|
|
|
ifobj->xsk = &ifobj->xsk_arr[0];
|
|
ifobj->use_poll = false;
|
|
ifobj->use_fill_ring = true;
|
|
ifobj->release_rx = true;
|
|
ifobj->validation_func = NULL;
|
|
ifobj->use_metadata = false;
|
|
|
|
if (i == 0) {
|
|
ifobj->rx_on = false;
|
|
ifobj->tx_on = true;
|
|
ifobj->pkt_stream = test->tx_pkt_stream_default;
|
|
} else {
|
|
ifobj->rx_on = true;
|
|
ifobj->tx_on = false;
|
|
ifobj->pkt_stream = test->rx_pkt_stream_default;
|
|
}
|
|
|
|
memset(ifobj->umem, 0, sizeof(*ifobj->umem));
|
|
ifobj->umem->num_frames = DEFAULT_UMEM_BUFFERS;
|
|
ifobj->umem->frame_size = XSK_UMEM__DEFAULT_FRAME_SIZE;
|
|
|
|
for (j = 0; j < MAX_SOCKETS; j++) {
|
|
memset(&ifobj->xsk_arr[j], 0, sizeof(ifobj->xsk_arr[j]));
|
|
ifobj->xsk_arr[j].rxqsize = XSK_RING_CONS__DEFAULT_NUM_DESCS;
|
|
}
|
|
}
|
|
|
|
test->ifobj_tx = ifobj_tx;
|
|
test->ifobj_rx = ifobj_rx;
|
|
test->current_step = 0;
|
|
test->total_steps = 1;
|
|
test->nb_sockets = 1;
|
|
test->fail = false;
|
|
test->xdp_prog_rx = ifobj_rx->xdp_progs->progs.xsk_def_prog;
|
|
test->xskmap_rx = ifobj_rx->xdp_progs->maps.xsk;
|
|
test->xdp_prog_tx = ifobj_tx->xdp_progs->progs.xsk_def_prog;
|
|
test->xskmap_tx = ifobj_tx->xdp_progs->maps.xsk;
|
|
}
|
|
|
|
static void test_spec_init(struct test_spec *test, struct ifobject *ifobj_tx,
|
|
struct ifobject *ifobj_rx, enum test_mode mode)
|
|
{
|
|
struct pkt_stream *tx_pkt_stream;
|
|
struct pkt_stream *rx_pkt_stream;
|
|
u32 i;
|
|
|
|
tx_pkt_stream = test->tx_pkt_stream_default;
|
|
rx_pkt_stream = test->rx_pkt_stream_default;
|
|
memset(test, 0, sizeof(*test));
|
|
test->tx_pkt_stream_default = tx_pkt_stream;
|
|
test->rx_pkt_stream_default = rx_pkt_stream;
|
|
|
|
for (i = 0; i < MAX_INTERFACES; i++) {
|
|
struct ifobject *ifobj = i ? ifobj_rx : ifobj_tx;
|
|
|
|
ifobj->bind_flags = XDP_USE_NEED_WAKEUP;
|
|
if (mode == TEST_MODE_ZC)
|
|
ifobj->bind_flags |= XDP_ZEROCOPY;
|
|
else
|
|
ifobj->bind_flags |= XDP_COPY;
|
|
}
|
|
|
|
test->mode = mode;
|
|
__test_spec_init(test, ifobj_tx, ifobj_rx);
|
|
}
|
|
|
|
static void test_spec_reset(struct test_spec *test)
|
|
{
|
|
__test_spec_init(test, test->ifobj_tx, test->ifobj_rx);
|
|
}
|
|
|
|
static void test_spec_set_name(struct test_spec *test, const char *name)
|
|
{
|
|
strncpy(test->name, name, MAX_TEST_NAME_SIZE);
|
|
}
|
|
|
|
static void test_spec_set_xdp_prog(struct test_spec *test, struct bpf_program *xdp_prog_rx,
|
|
struct bpf_program *xdp_prog_tx, struct bpf_map *xskmap_rx,
|
|
struct bpf_map *xskmap_tx)
|
|
{
|
|
test->xdp_prog_rx = xdp_prog_rx;
|
|
test->xdp_prog_tx = xdp_prog_tx;
|
|
test->xskmap_rx = xskmap_rx;
|
|
test->xskmap_tx = xskmap_tx;
|
|
}
|
|
|
|
static void pkt_stream_reset(struct pkt_stream *pkt_stream)
|
|
{
|
|
if (pkt_stream)
|
|
pkt_stream->current_pkt_nb = 0;
|
|
}
|
|
|
|
static struct pkt *pkt_stream_get_next_tx_pkt(struct pkt_stream *pkt_stream)
|
|
{
|
|
if (pkt_stream->current_pkt_nb >= pkt_stream->nb_pkts)
|
|
return NULL;
|
|
|
|
return &pkt_stream->pkts[pkt_stream->current_pkt_nb++];
|
|
}
|
|
|
|
static struct pkt *pkt_stream_get_next_rx_pkt(struct pkt_stream *pkt_stream, u32 *pkts_sent)
|
|
{
|
|
while (pkt_stream->current_pkt_nb < pkt_stream->nb_pkts) {
|
|
(*pkts_sent)++;
|
|
if (pkt_stream->pkts[pkt_stream->current_pkt_nb].valid)
|
|
return &pkt_stream->pkts[pkt_stream->current_pkt_nb++];
|
|
pkt_stream->current_pkt_nb++;
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
static void pkt_stream_delete(struct pkt_stream *pkt_stream)
|
|
{
|
|
free(pkt_stream->pkts);
|
|
free(pkt_stream);
|
|
}
|
|
|
|
static void pkt_stream_restore_default(struct test_spec *test)
|
|
{
|
|
struct pkt_stream *tx_pkt_stream = test->ifobj_tx->pkt_stream;
|
|
struct pkt_stream *rx_pkt_stream = test->ifobj_rx->pkt_stream;
|
|
|
|
if (tx_pkt_stream != test->tx_pkt_stream_default) {
|
|
pkt_stream_delete(test->ifobj_tx->pkt_stream);
|
|
test->ifobj_tx->pkt_stream = test->tx_pkt_stream_default;
|
|
}
|
|
|
|
if (rx_pkt_stream != test->rx_pkt_stream_default) {
|
|
pkt_stream_delete(test->ifobj_rx->pkt_stream);
|
|
test->ifobj_rx->pkt_stream = test->rx_pkt_stream_default;
|
|
}
|
|
}
|
|
|
|
static struct pkt_stream *__pkt_stream_alloc(u32 nb_pkts)
|
|
{
|
|
struct pkt_stream *pkt_stream;
|
|
|
|
pkt_stream = calloc(1, sizeof(*pkt_stream));
|
|
if (!pkt_stream)
|
|
return NULL;
|
|
|
|
pkt_stream->pkts = calloc(nb_pkts, sizeof(*pkt_stream->pkts));
|
|
if (!pkt_stream->pkts) {
|
|
free(pkt_stream);
|
|
return NULL;
|
|
}
|
|
|
|
pkt_stream->nb_pkts = nb_pkts;
|
|
return pkt_stream;
|
|
}
|
|
|
|
static u32 ceil_u32(u32 a, u32 b)
|
|
{
|
|
return (a + b - 1) / b;
|
|
}
|
|
|
|
static u32 pkt_nb_frags(u32 frame_size, struct pkt *pkt)
|
|
{
|
|
if (!pkt || !pkt->valid)
|
|
return 1;
|
|
return ceil_u32(pkt->len, frame_size);
|
|
}
|
|
|
|
static void pkt_set(struct xsk_umem_info *umem, struct pkt *pkt, int offset, u32 len)
|
|
{
|
|
pkt->offset = offset;
|
|
pkt->len = len;
|
|
if (len > umem->frame_size - XDP_PACKET_HEADROOM - MIN_PKT_SIZE * 2 - umem->frame_headroom)
|
|
pkt->valid = false;
|
|
else
|
|
pkt->valid = true;
|
|
}
|
|
|
|
static u32 pkt_get_buffer_len(struct xsk_umem_info *umem, u32 len)
|
|
{
|
|
return ceil_u32(len, umem->frame_size) * umem->frame_size;
|
|
}
|
|
|
|
static struct pkt_stream *pkt_stream_generate(struct xsk_umem_info *umem, u32 nb_pkts, u32 pkt_len)
|
|
{
|
|
struct pkt_stream *pkt_stream;
|
|
u32 i;
|
|
|
|
pkt_stream = __pkt_stream_alloc(nb_pkts);
|
|
if (!pkt_stream)
|
|
exit_with_error(ENOMEM);
|
|
|
|
pkt_stream->nb_pkts = nb_pkts;
|
|
pkt_stream->max_pkt_len = pkt_len;
|
|
for (i = 0; i < nb_pkts; i++) {
|
|
struct pkt *pkt = &pkt_stream->pkts[i];
|
|
|
|
pkt_set(umem, pkt, 0, pkt_len);
|
|
pkt->pkt_nb = i;
|
|
}
|
|
|
|
return pkt_stream;
|
|
}
|
|
|
|
static struct pkt_stream *pkt_stream_clone(struct xsk_umem_info *umem,
|
|
struct pkt_stream *pkt_stream)
|
|
{
|
|
return pkt_stream_generate(umem, pkt_stream->nb_pkts, pkt_stream->pkts[0].len);
|
|
}
|
|
|
|
static void pkt_stream_replace(struct test_spec *test, u32 nb_pkts, u32 pkt_len)
|
|
{
|
|
struct pkt_stream *pkt_stream;
|
|
|
|
pkt_stream = pkt_stream_generate(test->ifobj_tx->umem, nb_pkts, pkt_len);
|
|
test->ifobj_tx->pkt_stream = pkt_stream;
|
|
pkt_stream = pkt_stream_generate(test->ifobj_rx->umem, nb_pkts, pkt_len);
|
|
test->ifobj_rx->pkt_stream = pkt_stream;
|
|
}
|
|
|
|
static void __pkt_stream_replace_half(struct ifobject *ifobj, u32 pkt_len,
|
|
int offset)
|
|
{
|
|
struct xsk_umem_info *umem = ifobj->umem;
|
|
struct pkt_stream *pkt_stream;
|
|
u32 i;
|
|
|
|
pkt_stream = pkt_stream_clone(umem, ifobj->pkt_stream);
|
|
for (i = 1; i < ifobj->pkt_stream->nb_pkts; i += 2)
|
|
pkt_set(umem, &pkt_stream->pkts[i], offset, pkt_len);
|
|
|
|
ifobj->pkt_stream = pkt_stream;
|
|
}
|
|
|
|
static void pkt_stream_replace_half(struct test_spec *test, u32 pkt_len, int offset)
|
|
{
|
|
__pkt_stream_replace_half(test->ifobj_tx, pkt_len, offset);
|
|
__pkt_stream_replace_half(test->ifobj_rx, pkt_len, offset);
|
|
}
|
|
|
|
static void pkt_stream_receive_half(struct test_spec *test)
|
|
{
|
|
struct xsk_umem_info *umem = test->ifobj_rx->umem;
|
|
struct pkt_stream *pkt_stream = test->ifobj_tx->pkt_stream;
|
|
u32 i;
|
|
|
|
test->ifobj_rx->pkt_stream = pkt_stream_generate(umem, pkt_stream->nb_pkts,
|
|
pkt_stream->pkts[0].len);
|
|
pkt_stream = test->ifobj_rx->pkt_stream;
|
|
for (i = 1; i < pkt_stream->nb_pkts; i += 2)
|
|
pkt_stream->pkts[i].valid = false;
|
|
}
|
|
|
|
static u64 pkt_get_addr(struct pkt *pkt, struct xsk_umem_info *umem)
|
|
{
|
|
if (!pkt->valid)
|
|
return pkt->offset;
|
|
return pkt->offset + umem_alloc_buffer(umem);
|
|
}
|
|
|
|
static void pkt_generate(struct ifobject *ifobject, u64 addr, u32 len, u32 pkt_nb,
|
|
u32 bytes_written)
|
|
{
|
|
void *data = xsk_umem__get_data(ifobject->umem->buffer, addr);
|
|
|
|
if (len < MIN_PKT_SIZE)
|
|
return;
|
|
|
|
if (!bytes_written) {
|
|
gen_eth_hdr(ifobject, data);
|
|
|
|
len -= PKT_HDR_SIZE;
|
|
data += PKT_HDR_SIZE;
|
|
} else {
|
|
bytes_written -= PKT_HDR_SIZE;
|
|
}
|
|
|
|
write_payload(data, pkt_nb, bytes_written, len);
|
|
}
|
|
|
|
static void __pkt_stream_generate_custom(struct ifobject *ifobj,
|
|
struct pkt *pkts, u32 nb_pkts)
|
|
{
|
|
struct pkt_stream *pkt_stream;
|
|
u32 i;
|
|
|
|
pkt_stream = __pkt_stream_alloc(nb_pkts);
|
|
if (!pkt_stream)
|
|
exit_with_error(ENOMEM);
|
|
|
|
for (i = 0; i < nb_pkts; i++) {
|
|
struct pkt *pkt = &pkt_stream->pkts[i];
|
|
|
|
pkt->offset = pkts[i].offset;
|
|
pkt->len = pkts[i].len;
|
|
pkt->pkt_nb = i;
|
|
pkt->valid = pkts[i].valid;
|
|
if (pkt->len > pkt_stream->max_pkt_len)
|
|
pkt_stream->max_pkt_len = pkt->len;
|
|
}
|
|
|
|
ifobj->pkt_stream = pkt_stream;
|
|
}
|
|
|
|
static void pkt_stream_generate_custom(struct test_spec *test, struct pkt *pkts, u32 nb_pkts)
|
|
{
|
|
__pkt_stream_generate_custom(test->ifobj_tx, pkts, nb_pkts);
|
|
__pkt_stream_generate_custom(test->ifobj_rx, pkts, nb_pkts);
|
|
}
|
|
|
|
static void pkt_print_data(u32 *data, u32 cnt)
|
|
{
|
|
u32 i;
|
|
|
|
for (i = 0; i < cnt; i++) {
|
|
u32 seqnum, pkt_nb;
|
|
|
|
seqnum = ntohl(*data) & 0xffff;
|
|
pkt_nb = ntohl(*data) >> 16;
|
|
fprintf(stdout, "%u:%u ", pkt_nb, seqnum);
|
|
data++;
|
|
}
|
|
}
|
|
|
|
static void pkt_dump(void *pkt, u32 len, bool eth_header)
|
|
{
|
|
struct ethhdr *ethhdr = pkt;
|
|
u32 i, *data;
|
|
|
|
if (eth_header) {
|
|
/*extract L2 frame */
|
|
fprintf(stdout, "DEBUG>> L2: dst mac: ");
|
|
for (i = 0; i < ETH_ALEN; i++)
|
|
fprintf(stdout, "%02X", ethhdr->h_dest[i]);
|
|
|
|
fprintf(stdout, "\nDEBUG>> L2: src mac: ");
|
|
for (i = 0; i < ETH_ALEN; i++)
|
|
fprintf(stdout, "%02X", ethhdr->h_source[i]);
|
|
|
|
data = pkt + PKT_HDR_SIZE;
|
|
} else {
|
|
data = pkt;
|
|
}
|
|
|
|
/*extract L5 frame */
|
|
fprintf(stdout, "\nDEBUG>> L5: seqnum: ");
|
|
pkt_print_data(data, PKT_DUMP_NB_TO_PRINT);
|
|
fprintf(stdout, "....");
|
|
if (len > PKT_DUMP_NB_TO_PRINT * sizeof(u32)) {
|
|
fprintf(stdout, "\n.... ");
|
|
pkt_print_data(data + len / sizeof(u32) - PKT_DUMP_NB_TO_PRINT,
|
|
PKT_DUMP_NB_TO_PRINT);
|
|
}
|
|
fprintf(stdout, "\n---------------------------------------\n");
|
|
}
|
|
|
|
static bool is_offset_correct(struct xsk_umem_info *umem, struct pkt *pkt, u64 addr)
|
|
{
|
|
u32 headroom = umem->unaligned_mode ? 0 : umem->frame_headroom;
|
|
u32 offset = addr % umem->frame_size, expected_offset;
|
|
int pkt_offset = pkt->valid ? pkt->offset : 0;
|
|
|
|
if (!umem->unaligned_mode)
|
|
pkt_offset = 0;
|
|
|
|
expected_offset = (pkt_offset + headroom + XDP_PACKET_HEADROOM) % umem->frame_size;
|
|
|
|
if (offset == expected_offset)
|
|
return true;
|
|
|
|
ksft_print_msg("[%s] expected [%u], got [%u]\n", __func__, expected_offset, offset);
|
|
return false;
|
|
}
|
|
|
|
static bool is_metadata_correct(struct pkt *pkt, void *buffer, u64 addr)
|
|
{
|
|
void *data = xsk_umem__get_data(buffer, addr);
|
|
struct xdp_info *meta = data - sizeof(struct xdp_info);
|
|
|
|
if (meta->count != pkt->pkt_nb) {
|
|
ksft_print_msg("[%s] expected meta_count [%d], got meta_count [%d]\n",
|
|
__func__, pkt->pkt_nb, meta->count);
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
static bool is_pkt_valid(struct pkt *pkt, void *buffer, u64 addr, u32 len)
|
|
{
|
|
void *data = xsk_umem__get_data(buffer, addr);
|
|
u32 seqnum, pkt_data;
|
|
|
|
if (!pkt) {
|
|
ksft_print_msg("[%s] too many packets received\n", __func__);
|
|
goto error;
|
|
}
|
|
|
|
if (len < MIN_PKT_SIZE || pkt->len < MIN_PKT_SIZE) {
|
|
/* Do not try to verify packets that are smaller than minimum size. */
|
|
return true;
|
|
}
|
|
|
|
if (pkt->len != len) {
|
|
ksft_print_msg("[%s] expected length [%d], got length [%d]\n",
|
|
__func__, pkt->len, len);
|
|
goto error;
|
|
}
|
|
|
|
pkt_data = ntohl(*((u32 *)(data + PKT_HDR_SIZE)));
|
|
seqnum = pkt_data >> 16;
|
|
|
|
if (pkt->pkt_nb != seqnum) {
|
|
ksft_print_msg("[%s] expected seqnum [%d], got seqnum [%d]\n",
|
|
__func__, pkt->pkt_nb, seqnum);
|
|
goto error;
|
|
}
|
|
|
|
return true;
|
|
|
|
error:
|
|
pkt_dump(data, len, true);
|
|
return false;
|
|
}
|
|
|
|
static void kick_tx(struct xsk_socket_info *xsk)
|
|
{
|
|
int ret;
|
|
|
|
ret = sendto(xsk_socket__fd(xsk->xsk), NULL, 0, MSG_DONTWAIT, NULL, 0);
|
|
if (ret >= 0)
|
|
return;
|
|
if (errno == ENOBUFS || errno == EAGAIN || errno == EBUSY || errno == ENETDOWN) {
|
|
usleep(100);
|
|
return;
|
|
}
|
|
exit_with_error(errno);
|
|
}
|
|
|
|
static void kick_rx(struct xsk_socket_info *xsk)
|
|
{
|
|
int ret;
|
|
|
|
ret = recvfrom(xsk_socket__fd(xsk->xsk), NULL, 0, MSG_DONTWAIT, NULL, NULL);
|
|
if (ret < 0)
|
|
exit_with_error(errno);
|
|
}
|
|
|
|
static int complete_pkts(struct xsk_socket_info *xsk, int batch_size)
|
|
{
|
|
unsigned int rcvd;
|
|
u32 idx;
|
|
|
|
if (xsk_ring_prod__needs_wakeup(&xsk->tx))
|
|
kick_tx(xsk);
|
|
|
|
rcvd = xsk_ring_cons__peek(&xsk->umem->cq, batch_size, &idx);
|
|
if (rcvd) {
|
|
if (rcvd > xsk->outstanding_tx) {
|
|
u64 addr = *xsk_ring_cons__comp_addr(&xsk->umem->cq, idx + rcvd - 1);
|
|
|
|
ksft_print_msg("[%s] Too many packets completed\n", __func__);
|
|
ksft_print_msg("Last completion address: %llx\n", addr);
|
|
return TEST_FAILURE;
|
|
}
|
|
|
|
xsk_ring_cons__release(&xsk->umem->cq, rcvd);
|
|
xsk->outstanding_tx -= rcvd;
|
|
}
|
|
|
|
return TEST_PASS;
|
|
}
|
|
|
|
static int receive_pkts(struct test_spec *test, struct pollfd *fds)
|
|
{
|
|
struct timeval tv_end, tv_now, tv_timeout = {THREAD_TMOUT, 0};
|
|
struct pkt_stream *pkt_stream = test->ifobj_rx->pkt_stream;
|
|
u32 idx_rx = 0, idx_fq = 0, rcvd, i, pkts_sent = 0;
|
|
struct xsk_socket_info *xsk = test->ifobj_rx->xsk;
|
|
struct ifobject *ifobj = test->ifobj_rx;
|
|
struct xsk_umem_info *umem = xsk->umem;
|
|
struct pkt *pkt;
|
|
int ret;
|
|
|
|
ret = gettimeofday(&tv_now, NULL);
|
|
if (ret)
|
|
exit_with_error(errno);
|
|
timeradd(&tv_now, &tv_timeout, &tv_end);
|
|
|
|
pkt = pkt_stream_get_next_rx_pkt(pkt_stream, &pkts_sent);
|
|
while (pkt) {
|
|
ret = gettimeofday(&tv_now, NULL);
|
|
if (ret)
|
|
exit_with_error(errno);
|
|
if (timercmp(&tv_now, &tv_end, >)) {
|
|
ksft_print_msg("ERROR: [%s] Receive loop timed out\n", __func__);
|
|
return TEST_FAILURE;
|
|
}
|
|
|
|
kick_rx(xsk);
|
|
if (ifobj->use_poll) {
|
|
ret = poll(fds, 1, POLL_TMOUT);
|
|
if (ret < 0)
|
|
exit_with_error(errno);
|
|
|
|
if (!ret) {
|
|
if (!is_umem_valid(test->ifobj_tx))
|
|
return TEST_PASS;
|
|
|
|
ksft_print_msg("ERROR: [%s] Poll timed out\n", __func__);
|
|
return TEST_FAILURE;
|
|
|
|
}
|
|
|
|
if (!(fds->revents & POLLIN))
|
|
continue;
|
|
}
|
|
|
|
rcvd = xsk_ring_cons__peek(&xsk->rx, BATCH_SIZE, &idx_rx);
|
|
if (!rcvd)
|
|
continue;
|
|
|
|
if (ifobj->use_fill_ring) {
|
|
ret = xsk_ring_prod__reserve(&umem->fq, rcvd, &idx_fq);
|
|
while (ret != rcvd) {
|
|
if (ret < 0)
|
|
exit_with_error(-ret);
|
|
if (xsk_ring_prod__needs_wakeup(&umem->fq)) {
|
|
ret = poll(fds, 1, POLL_TMOUT);
|
|
if (ret < 0)
|
|
exit_with_error(errno);
|
|
}
|
|
ret = xsk_ring_prod__reserve(&umem->fq, rcvd, &idx_fq);
|
|
}
|
|
}
|
|
|
|
for (i = 0; i < rcvd; i++) {
|
|
const struct xdp_desc *desc = xsk_ring_cons__rx_desc(&xsk->rx, idx_rx++);
|
|
u64 addr = desc->addr, orig;
|
|
|
|
orig = xsk_umem__extract_addr(addr);
|
|
addr = xsk_umem__add_offset_to_addr(addr);
|
|
|
|
if (!is_pkt_valid(pkt, umem->buffer, addr, desc->len) ||
|
|
!is_offset_correct(umem, pkt, addr) ||
|
|
(ifobj->use_metadata && !is_metadata_correct(pkt, umem->buffer, addr)))
|
|
return TEST_FAILURE;
|
|
|
|
if (ifobj->use_fill_ring)
|
|
*xsk_ring_prod__fill_addr(&umem->fq, idx_fq++) = orig;
|
|
pkt = pkt_stream_get_next_rx_pkt(pkt_stream, &pkts_sent);
|
|
}
|
|
|
|
if (ifobj->use_fill_ring)
|
|
xsk_ring_prod__submit(&umem->fq, rcvd);
|
|
if (ifobj->release_rx)
|
|
xsk_ring_cons__release(&xsk->rx, rcvd);
|
|
|
|
pthread_mutex_lock(&pacing_mutex);
|
|
pkts_in_flight -= pkts_sent;
|
|
pthread_mutex_unlock(&pacing_mutex);
|
|
pkts_sent = 0;
|
|
}
|
|
|
|
return TEST_PASS;
|
|
}
|
|
|
|
static int __send_pkts(struct ifobject *ifobject, struct pollfd *fds, bool timeout)
|
|
{
|
|
struct xsk_socket_info *xsk = ifobject->xsk;
|
|
struct xsk_umem_info *umem = ifobject->umem;
|
|
u32 i, idx = 0, valid_pkts = 0, buffer_len;
|
|
bool use_poll = ifobject->use_poll;
|
|
int ret;
|
|
|
|
buffer_len = pkt_get_buffer_len(umem, ifobject->pkt_stream->max_pkt_len);
|
|
/* pkts_in_flight might be negative if many invalid packets are sent */
|
|
if (pkts_in_flight >= (int)((umem_size(umem) - BATCH_SIZE * buffer_len) / buffer_len)) {
|
|
kick_tx(xsk);
|
|
return TEST_CONTINUE;
|
|
}
|
|
|
|
while (xsk_ring_prod__reserve(&xsk->tx, BATCH_SIZE, &idx) < BATCH_SIZE) {
|
|
if (use_poll) {
|
|
ret = poll(fds, 1, POLL_TMOUT);
|
|
if (timeout) {
|
|
if (ret < 0) {
|
|
ksft_print_msg("ERROR: [%s] Poll error %d\n",
|
|
__func__, errno);
|
|
return TEST_FAILURE;
|
|
}
|
|
if (ret == 0)
|
|
return TEST_PASS;
|
|
break;
|
|
}
|
|
if (ret <= 0) {
|
|
ksft_print_msg("ERROR: [%s] Poll error %d\n",
|
|
__func__, errno);
|
|
return TEST_FAILURE;
|
|
}
|
|
}
|
|
|
|
complete_pkts(xsk, BATCH_SIZE);
|
|
}
|
|
|
|
for (i = 0; i < BATCH_SIZE; i++) {
|
|
struct xdp_desc *tx_desc = xsk_ring_prod__tx_desc(&xsk->tx, idx + i);
|
|
struct pkt *pkt = pkt_stream_get_next_tx_pkt(ifobject->pkt_stream);
|
|
|
|
if (!pkt)
|
|
break;
|
|
|
|
tx_desc->addr = pkt_get_addr(pkt, umem);
|
|
tx_desc->len = pkt->len;
|
|
if (pkt->valid) {
|
|
valid_pkts++;
|
|
pkt_generate(ifobject, tx_desc->addr, tx_desc->len, pkt->pkt_nb, 0);
|
|
}
|
|
}
|
|
|
|
pthread_mutex_lock(&pacing_mutex);
|
|
pkts_in_flight += valid_pkts;
|
|
pthread_mutex_unlock(&pacing_mutex);
|
|
|
|
xsk_ring_prod__submit(&xsk->tx, i);
|
|
xsk->outstanding_tx += valid_pkts;
|
|
|
|
if (use_poll) {
|
|
ret = poll(fds, 1, POLL_TMOUT);
|
|
if (ret <= 0) {
|
|
if (ret == 0 && timeout)
|
|
return TEST_PASS;
|
|
|
|
ksft_print_msg("ERROR: [%s] Poll error %d\n", __func__, ret);
|
|
return TEST_FAILURE;
|
|
}
|
|
}
|
|
|
|
if (!timeout) {
|
|
if (complete_pkts(xsk, i))
|
|
return TEST_FAILURE;
|
|
|
|
usleep(10);
|
|
return TEST_PASS;
|
|
}
|
|
|
|
return TEST_CONTINUE;
|
|
}
|
|
|
|
static void wait_for_tx_completion(struct xsk_socket_info *xsk)
|
|
{
|
|
while (xsk->outstanding_tx)
|
|
complete_pkts(xsk, BATCH_SIZE);
|
|
}
|
|
|
|
static int send_pkts(struct test_spec *test, struct ifobject *ifobject)
|
|
{
|
|
struct pkt_stream *pkt_stream = ifobject->pkt_stream;
|
|
bool timeout = !is_umem_valid(test->ifobj_rx);
|
|
struct pollfd fds = { };
|
|
u32 ret;
|
|
|
|
fds.fd = xsk_socket__fd(ifobject->xsk->xsk);
|
|
fds.events = POLLOUT;
|
|
|
|
while (pkt_stream->current_pkt_nb < pkt_stream->nb_pkts) {
|
|
ret = __send_pkts(ifobject, &fds, timeout);
|
|
if (ret == TEST_CONTINUE && !test->fail)
|
|
continue;
|
|
if ((ret || test->fail) && !timeout)
|
|
return TEST_FAILURE;
|
|
if (ret == TEST_PASS && timeout)
|
|
return ret;
|
|
}
|
|
|
|
wait_for_tx_completion(ifobject->xsk);
|
|
return TEST_PASS;
|
|
}
|
|
|
|
static int get_xsk_stats(struct xsk_socket *xsk, struct xdp_statistics *stats)
|
|
{
|
|
int fd = xsk_socket__fd(xsk), err;
|
|
socklen_t optlen, expected_len;
|
|
|
|
optlen = sizeof(*stats);
|
|
err = getsockopt(fd, SOL_XDP, XDP_STATISTICS, stats, &optlen);
|
|
if (err) {
|
|
ksft_print_msg("[%s] getsockopt(XDP_STATISTICS) error %u %s\n",
|
|
__func__, -err, strerror(-err));
|
|
return TEST_FAILURE;
|
|
}
|
|
|
|
expected_len = sizeof(struct xdp_statistics);
|
|
if (optlen != expected_len) {
|
|
ksft_print_msg("[%s] getsockopt optlen error. Expected: %u got: %u\n",
|
|
__func__, expected_len, optlen);
|
|
return TEST_FAILURE;
|
|
}
|
|
|
|
return TEST_PASS;
|
|
}
|
|
|
|
static int validate_rx_dropped(struct ifobject *ifobject)
|
|
{
|
|
struct xsk_socket *xsk = ifobject->xsk->xsk;
|
|
struct xdp_statistics stats;
|
|
int err;
|
|
|
|
kick_rx(ifobject->xsk);
|
|
|
|
err = get_xsk_stats(xsk, &stats);
|
|
if (err)
|
|
return TEST_FAILURE;
|
|
|
|
/* The receiver calls getsockopt after receiving the last (valid)
|
|
* packet which is not the final packet sent in this test (valid and
|
|
* invalid packets are sent in alternating fashion with the final
|
|
* packet being invalid). Since the last packet may or may not have
|
|
* been dropped already, both outcomes must be allowed.
|
|
*/
|
|
if (stats.rx_dropped == ifobject->pkt_stream->nb_pkts / 2 ||
|
|
stats.rx_dropped == ifobject->pkt_stream->nb_pkts / 2 - 1)
|
|
return TEST_PASS;
|
|
|
|
return TEST_FAILURE;
|
|
}
|
|
|
|
static int validate_rx_full(struct ifobject *ifobject)
|
|
{
|
|
struct xsk_socket *xsk = ifobject->xsk->xsk;
|
|
struct xdp_statistics stats;
|
|
int err;
|
|
|
|
usleep(1000);
|
|
kick_rx(ifobject->xsk);
|
|
|
|
err = get_xsk_stats(xsk, &stats);
|
|
if (err)
|
|
return TEST_FAILURE;
|
|
|
|
if (stats.rx_ring_full)
|
|
return TEST_PASS;
|
|
|
|
return TEST_FAILURE;
|
|
}
|
|
|
|
static int validate_fill_empty(struct ifobject *ifobject)
|
|
{
|
|
struct xsk_socket *xsk = ifobject->xsk->xsk;
|
|
struct xdp_statistics stats;
|
|
int err;
|
|
|
|
usleep(1000);
|
|
kick_rx(ifobject->xsk);
|
|
|
|
err = get_xsk_stats(xsk, &stats);
|
|
if (err)
|
|
return TEST_FAILURE;
|
|
|
|
if (stats.rx_fill_ring_empty_descs)
|
|
return TEST_PASS;
|
|
|
|
return TEST_FAILURE;
|
|
}
|
|
|
|
static int validate_tx_invalid_descs(struct ifobject *ifobject)
|
|
{
|
|
struct xsk_socket *xsk = ifobject->xsk->xsk;
|
|
int fd = xsk_socket__fd(xsk);
|
|
struct xdp_statistics stats;
|
|
socklen_t optlen;
|
|
int err;
|
|
|
|
optlen = sizeof(stats);
|
|
err = getsockopt(fd, SOL_XDP, XDP_STATISTICS, &stats, &optlen);
|
|
if (err) {
|
|
ksft_print_msg("[%s] getsockopt(XDP_STATISTICS) error %u %s\n",
|
|
__func__, -err, strerror(-err));
|
|
return TEST_FAILURE;
|
|
}
|
|
|
|
if (stats.tx_invalid_descs != ifobject->pkt_stream->nb_pkts / 2) {
|
|
ksft_print_msg("[%s] tx_invalid_descs incorrect. Got [%u] expected [%u]\n",
|
|
__func__, stats.tx_invalid_descs, ifobject->pkt_stream->nb_pkts);
|
|
return TEST_FAILURE;
|
|
}
|
|
|
|
return TEST_PASS;
|
|
}
|
|
|
|
static void xsk_configure_socket(struct test_spec *test, struct ifobject *ifobject,
|
|
struct xsk_umem_info *umem, bool tx)
|
|
{
|
|
int i, ret;
|
|
|
|
for (i = 0; i < test->nb_sockets; i++) {
|
|
bool shared = (ifobject->shared_umem && tx) ? true : !!i;
|
|
u32 ctr = 0;
|
|
|
|
while (ctr++ < SOCK_RECONF_CTR) {
|
|
ret = __xsk_configure_socket(&ifobject->xsk_arr[i], umem,
|
|
ifobject, shared);
|
|
if (!ret)
|
|
break;
|
|
|
|
/* Retry if it fails as xsk_socket__create() is asynchronous */
|
|
if (ctr >= SOCK_RECONF_CTR)
|
|
exit_with_error(-ret);
|
|
usleep(USLEEP_MAX);
|
|
}
|
|
if (ifobject->busy_poll)
|
|
enable_busy_poll(&ifobject->xsk_arr[i]);
|
|
}
|
|
}
|
|
|
|
static void thread_common_ops_tx(struct test_spec *test, struct ifobject *ifobject)
|
|
{
|
|
xsk_configure_socket(test, ifobject, test->ifobj_rx->umem, true);
|
|
ifobject->xsk = &ifobject->xsk_arr[0];
|
|
ifobject->xskmap = test->ifobj_rx->xskmap;
|
|
memcpy(ifobject->umem, test->ifobj_rx->umem, sizeof(struct xsk_umem_info));
|
|
ifobject->umem->base_addr = 0;
|
|
}
|
|
|
|
static void xsk_populate_fill_ring(struct xsk_umem_info *umem, struct pkt_stream *pkt_stream,
|
|
bool fill_up)
|
|
{
|
|
u32 rx_frame_size = umem->frame_size - XDP_PACKET_HEADROOM;
|
|
u32 idx = 0, filled = 0, buffers_to_fill, nb_pkts;
|
|
int ret;
|
|
|
|
if (umem->num_frames < XSK_RING_PROD__DEFAULT_NUM_DESCS)
|
|
buffers_to_fill = umem->num_frames;
|
|
else
|
|
buffers_to_fill = XSK_RING_PROD__DEFAULT_NUM_DESCS;
|
|
|
|
ret = xsk_ring_prod__reserve(&umem->fq, buffers_to_fill, &idx);
|
|
if (ret != buffers_to_fill)
|
|
exit_with_error(ENOSPC);
|
|
|
|
while (filled < buffers_to_fill) {
|
|
struct pkt *pkt = pkt_stream_get_next_rx_pkt(pkt_stream, &nb_pkts);
|
|
u64 addr;
|
|
u32 i;
|
|
|
|
for (i = 0; i < pkt_nb_frags(rx_frame_size, pkt); i++) {
|
|
if (!pkt) {
|
|
if (!fill_up)
|
|
break;
|
|
addr = filled * umem->frame_size + umem->base_addr;
|
|
} else if (pkt->offset >= 0) {
|
|
addr = pkt->offset % umem->frame_size + umem_alloc_buffer(umem);
|
|
} else {
|
|
addr = pkt->offset + umem_alloc_buffer(umem);
|
|
}
|
|
|
|
*xsk_ring_prod__fill_addr(&umem->fq, idx++) = addr;
|
|
if (++filled >= buffers_to_fill)
|
|
break;
|
|
}
|
|
}
|
|
xsk_ring_prod__submit(&umem->fq, filled);
|
|
xsk_ring_prod__cancel(&umem->fq, buffers_to_fill - filled);
|
|
|
|
pkt_stream_reset(pkt_stream);
|
|
umem_reset_alloc(umem);
|
|
}
|
|
|
|
static void thread_common_ops(struct test_spec *test, struct ifobject *ifobject)
|
|
{
|
|
u64 umem_sz = ifobject->umem->num_frames * ifobject->umem->frame_size;
|
|
int mmap_flags = MAP_PRIVATE | MAP_ANONYMOUS | MAP_NORESERVE;
|
|
LIBBPF_OPTS(bpf_xdp_query_opts, opts);
|
|
void *bufs;
|
|
int ret;
|
|
|
|
if (ifobject->umem->unaligned_mode)
|
|
mmap_flags |= MAP_HUGETLB | MAP_HUGE_2MB;
|
|
|
|
if (ifobject->shared_umem)
|
|
umem_sz *= 2;
|
|
|
|
bufs = mmap(NULL, umem_sz, PROT_READ | PROT_WRITE, mmap_flags, -1, 0);
|
|
if (bufs == MAP_FAILED)
|
|
exit_with_error(errno);
|
|
|
|
ret = xsk_configure_umem(ifobject, ifobject->umem, bufs, umem_sz);
|
|
if (ret)
|
|
exit_with_error(-ret);
|
|
|
|
xsk_configure_socket(test, ifobject, ifobject->umem, false);
|
|
|
|
ifobject->xsk = &ifobject->xsk_arr[0];
|
|
|
|
if (!ifobject->rx_on)
|
|
return;
|
|
|
|
xsk_populate_fill_ring(ifobject->umem, ifobject->pkt_stream, ifobject->use_fill_ring);
|
|
|
|
ret = xsk_update_xskmap(ifobject->xskmap, ifobject->xsk->xsk);
|
|
if (ret)
|
|
exit_with_error(errno);
|
|
}
|
|
|
|
static void *worker_testapp_validate_tx(void *arg)
|
|
{
|
|
struct test_spec *test = (struct test_spec *)arg;
|
|
struct ifobject *ifobject = test->ifobj_tx;
|
|
int err;
|
|
|
|
if (test->current_step == 1) {
|
|
if (!ifobject->shared_umem)
|
|
thread_common_ops(test, ifobject);
|
|
else
|
|
thread_common_ops_tx(test, ifobject);
|
|
}
|
|
|
|
print_verbose("Sending %d packets on interface %s\n", ifobject->pkt_stream->nb_pkts,
|
|
ifobject->ifname);
|
|
err = send_pkts(test, ifobject);
|
|
|
|
if (!err && ifobject->validation_func)
|
|
err = ifobject->validation_func(ifobject);
|
|
if (err)
|
|
report_failure(test);
|
|
|
|
pthread_exit(NULL);
|
|
}
|
|
|
|
static void *worker_testapp_validate_rx(void *arg)
|
|
{
|
|
struct test_spec *test = (struct test_spec *)arg;
|
|
struct ifobject *ifobject = test->ifobj_rx;
|
|
struct pollfd fds = { };
|
|
int err;
|
|
|
|
if (test->current_step == 1) {
|
|
thread_common_ops(test, ifobject);
|
|
} else {
|
|
xsk_clear_xskmap(ifobject->xskmap);
|
|
err = xsk_update_xskmap(ifobject->xskmap, ifobject->xsk->xsk);
|
|
if (err) {
|
|
printf("Error: Failed to update xskmap, error %s\n", strerror(-err));
|
|
exit_with_error(-err);
|
|
}
|
|
}
|
|
|
|
fds.fd = xsk_socket__fd(ifobject->xsk->xsk);
|
|
fds.events = POLLIN;
|
|
|
|
pthread_barrier_wait(&barr);
|
|
|
|
err = receive_pkts(test, &fds);
|
|
|
|
if (!err && ifobject->validation_func)
|
|
err = ifobject->validation_func(ifobject);
|
|
if (err)
|
|
report_failure(test);
|
|
|
|
pthread_exit(NULL);
|
|
}
|
|
|
|
static u64 ceil_u64(u64 a, u64 b)
|
|
{
|
|
return (a + b - 1) / b;
|
|
}
|
|
|
|
static void testapp_clean_xsk_umem(struct ifobject *ifobj)
|
|
{
|
|
u64 umem_sz = ifobj->umem->num_frames * ifobj->umem->frame_size;
|
|
|
|
if (ifobj->shared_umem)
|
|
umem_sz *= 2;
|
|
|
|
umem_sz = ceil_u64(umem_sz, HUGEPAGE_SIZE) * HUGEPAGE_SIZE;
|
|
xsk_umem__delete(ifobj->umem->umem);
|
|
munmap(ifobj->umem->buffer, umem_sz);
|
|
}
|
|
|
|
static void handler(int signum)
|
|
{
|
|
pthread_exit(NULL);
|
|
}
|
|
|
|
static bool xdp_prog_changed_rx(struct test_spec *test)
|
|
{
|
|
struct ifobject *ifobj = test->ifobj_rx;
|
|
|
|
return ifobj->xdp_prog != test->xdp_prog_rx || ifobj->mode != test->mode;
|
|
}
|
|
|
|
static bool xdp_prog_changed_tx(struct test_spec *test)
|
|
{
|
|
struct ifobject *ifobj = test->ifobj_tx;
|
|
|
|
return ifobj->xdp_prog != test->xdp_prog_tx || ifobj->mode != test->mode;
|
|
}
|
|
|
|
static void xsk_reattach_xdp(struct ifobject *ifobj, struct bpf_program *xdp_prog,
|
|
struct bpf_map *xskmap, enum test_mode mode)
|
|
{
|
|
int err;
|
|
|
|
xsk_detach_xdp_program(ifobj->ifindex, mode_to_xdp_flags(ifobj->mode));
|
|
err = xsk_attach_xdp_program(xdp_prog, ifobj->ifindex, mode_to_xdp_flags(mode));
|
|
if (err) {
|
|
printf("Error attaching XDP program\n");
|
|
exit_with_error(-err);
|
|
}
|
|
|
|
if (ifobj->mode != mode && (mode == TEST_MODE_DRV || mode == TEST_MODE_ZC))
|
|
if (!xsk_is_in_mode(ifobj->ifindex, XDP_FLAGS_DRV_MODE)) {
|
|
ksft_print_msg("ERROR: XDP prog not in DRV mode\n");
|
|
exit_with_error(EINVAL);
|
|
}
|
|
|
|
ifobj->xdp_prog = xdp_prog;
|
|
ifobj->xskmap = xskmap;
|
|
ifobj->mode = mode;
|
|
}
|
|
|
|
static void xsk_attach_xdp_progs(struct test_spec *test, struct ifobject *ifobj_rx,
|
|
struct ifobject *ifobj_tx)
|
|
{
|
|
if (xdp_prog_changed_rx(test))
|
|
xsk_reattach_xdp(ifobj_rx, test->xdp_prog_rx, test->xskmap_rx, test->mode);
|
|
|
|
if (!ifobj_tx || ifobj_tx->shared_umem)
|
|
return;
|
|
|
|
if (xdp_prog_changed_tx(test))
|
|
xsk_reattach_xdp(ifobj_tx, test->xdp_prog_tx, test->xskmap_tx, test->mode);
|
|
}
|
|
|
|
static int __testapp_validate_traffic(struct test_spec *test, struct ifobject *ifobj1,
|
|
struct ifobject *ifobj2)
|
|
{
|
|
pthread_t t0, t1;
|
|
|
|
if (ifobj2) {
|
|
if (pthread_barrier_init(&barr, NULL, 2))
|
|
exit_with_error(errno);
|
|
pkt_stream_reset(ifobj2->pkt_stream);
|
|
}
|
|
|
|
test->current_step++;
|
|
pkt_stream_reset(ifobj1->pkt_stream);
|
|
pkts_in_flight = 0;
|
|
|
|
signal(SIGUSR1, handler);
|
|
/*Spawn RX thread */
|
|
pthread_create(&t0, NULL, ifobj1->func_ptr, test);
|
|
|
|
if (ifobj2) {
|
|
pthread_barrier_wait(&barr);
|
|
if (pthread_barrier_destroy(&barr))
|
|
exit_with_error(errno);
|
|
|
|
/*Spawn TX thread */
|
|
pthread_create(&t1, NULL, ifobj2->func_ptr, test);
|
|
|
|
pthread_join(t1, NULL);
|
|
}
|
|
|
|
if (!ifobj2)
|
|
pthread_kill(t0, SIGUSR1);
|
|
else
|
|
pthread_join(t0, NULL);
|
|
|
|
if (test->total_steps == test->current_step || test->fail) {
|
|
if (ifobj2)
|
|
xsk_socket__delete(ifobj2->xsk->xsk);
|
|
xsk_socket__delete(ifobj1->xsk->xsk);
|
|
testapp_clean_xsk_umem(ifobj1);
|
|
if (ifobj2 && !ifobj2->shared_umem)
|
|
testapp_clean_xsk_umem(ifobj2);
|
|
}
|
|
|
|
return !!test->fail;
|
|
}
|
|
|
|
static int testapp_validate_traffic(struct test_spec *test)
|
|
{
|
|
struct ifobject *ifobj_rx = test->ifobj_rx;
|
|
struct ifobject *ifobj_tx = test->ifobj_tx;
|
|
|
|
if ((ifobj_rx->umem->unaligned_mode && !ifobj_rx->unaligned_supp) ||
|
|
(ifobj_tx->umem->unaligned_mode && !ifobj_tx->unaligned_supp)) {
|
|
ksft_test_result_skip("No huge pages present.\n");
|
|
return TEST_SKIP;
|
|
}
|
|
|
|
xsk_attach_xdp_progs(test, ifobj_rx, ifobj_tx);
|
|
return __testapp_validate_traffic(test, ifobj_rx, ifobj_tx);
|
|
}
|
|
|
|
static int testapp_validate_traffic_single_thread(struct test_spec *test, struct ifobject *ifobj)
|
|
{
|
|
return __testapp_validate_traffic(test, ifobj, NULL);
|
|
}
|
|
|
|
static int testapp_teardown(struct test_spec *test)
|
|
{
|
|
int i;
|
|
|
|
test_spec_set_name(test, "TEARDOWN");
|
|
for (i = 0; i < MAX_TEARDOWN_ITER; i++) {
|
|
if (testapp_validate_traffic(test))
|
|
return TEST_FAILURE;
|
|
test_spec_reset(test);
|
|
}
|
|
|
|
return TEST_PASS;
|
|
}
|
|
|
|
static void swap_directions(struct ifobject **ifobj1, struct ifobject **ifobj2)
|
|
{
|
|
thread_func_t tmp_func_ptr = (*ifobj1)->func_ptr;
|
|
struct ifobject *tmp_ifobj = (*ifobj1);
|
|
|
|
(*ifobj1)->func_ptr = (*ifobj2)->func_ptr;
|
|
(*ifobj2)->func_ptr = tmp_func_ptr;
|
|
|
|
*ifobj1 = *ifobj2;
|
|
*ifobj2 = tmp_ifobj;
|
|
}
|
|
|
|
static int testapp_bidi(struct test_spec *test)
|
|
{
|
|
int res;
|
|
|
|
test_spec_set_name(test, "BIDIRECTIONAL");
|
|
test->ifobj_tx->rx_on = true;
|
|
test->ifobj_rx->tx_on = true;
|
|
test->total_steps = 2;
|
|
if (testapp_validate_traffic(test))
|
|
return TEST_FAILURE;
|
|
|
|
print_verbose("Switching Tx/Rx vectors\n");
|
|
swap_directions(&test->ifobj_rx, &test->ifobj_tx);
|
|
res = __testapp_validate_traffic(test, test->ifobj_rx, test->ifobj_tx);
|
|
|
|
swap_directions(&test->ifobj_rx, &test->ifobj_tx);
|
|
return res;
|
|
}
|
|
|
|
static void swap_xsk_resources(struct ifobject *ifobj_tx, struct ifobject *ifobj_rx)
|
|
{
|
|
int ret;
|
|
|
|
xsk_socket__delete(ifobj_tx->xsk->xsk);
|
|
xsk_socket__delete(ifobj_rx->xsk->xsk);
|
|
ifobj_tx->xsk = &ifobj_tx->xsk_arr[1];
|
|
ifobj_rx->xsk = &ifobj_rx->xsk_arr[1];
|
|
|
|
ret = xsk_update_xskmap(ifobj_rx->xskmap, ifobj_rx->xsk->xsk);
|
|
if (ret)
|
|
exit_with_error(errno);
|
|
}
|
|
|
|
static int testapp_bpf_res(struct test_spec *test)
|
|
{
|
|
test_spec_set_name(test, "BPF_RES");
|
|
test->total_steps = 2;
|
|
test->nb_sockets = 2;
|
|
if (testapp_validate_traffic(test))
|
|
return TEST_FAILURE;
|
|
|
|
swap_xsk_resources(test->ifobj_tx, test->ifobj_rx);
|
|
return testapp_validate_traffic(test);
|
|
}
|
|
|
|
static int testapp_headroom(struct test_spec *test)
|
|
{
|
|
test_spec_set_name(test, "UMEM_HEADROOM");
|
|
test->ifobj_rx->umem->frame_headroom = UMEM_HEADROOM_TEST_SIZE;
|
|
return testapp_validate_traffic(test);
|
|
}
|
|
|
|
static int testapp_stats_rx_dropped(struct test_spec *test)
|
|
{
|
|
test_spec_set_name(test, "STAT_RX_DROPPED");
|
|
if (test->mode == TEST_MODE_ZC) {
|
|
ksft_test_result_skip("Can not run RX_DROPPED test for ZC mode\n");
|
|
return TEST_SKIP;
|
|
}
|
|
|
|
pkt_stream_replace_half(test, MIN_PKT_SIZE * 4, 0);
|
|
test->ifobj_rx->umem->frame_headroom = test->ifobj_rx->umem->frame_size -
|
|
XDP_PACKET_HEADROOM - MIN_PKT_SIZE * 3;
|
|
pkt_stream_receive_half(test);
|
|
test->ifobj_rx->validation_func = validate_rx_dropped;
|
|
return testapp_validate_traffic(test);
|
|
}
|
|
|
|
static int testapp_stats_tx_invalid_descs(struct test_spec *test)
|
|
{
|
|
test_spec_set_name(test, "STAT_TX_INVALID");
|
|
pkt_stream_replace_half(test, XSK_UMEM__INVALID_FRAME_SIZE, 0);
|
|
test->ifobj_tx->validation_func = validate_tx_invalid_descs;
|
|
return testapp_validate_traffic(test);
|
|
}
|
|
|
|
static int testapp_stats_rx_full(struct test_spec *test)
|
|
{
|
|
test_spec_set_name(test, "STAT_RX_FULL");
|
|
pkt_stream_replace(test, DEFAULT_UMEM_BUFFERS + DEFAULT_UMEM_BUFFERS / 2, MIN_PKT_SIZE);
|
|
test->ifobj_rx->pkt_stream = pkt_stream_generate(test->ifobj_rx->umem,
|
|
DEFAULT_UMEM_BUFFERS, MIN_PKT_SIZE);
|
|
|
|
test->ifobj_rx->xsk->rxqsize = DEFAULT_UMEM_BUFFERS;
|
|
test->ifobj_rx->release_rx = false;
|
|
test->ifobj_rx->validation_func = validate_rx_full;
|
|
return testapp_validate_traffic(test);
|
|
}
|
|
|
|
static int testapp_stats_fill_empty(struct test_spec *test)
|
|
{
|
|
test_spec_set_name(test, "STAT_RX_FILL_EMPTY");
|
|
pkt_stream_replace(test, DEFAULT_UMEM_BUFFERS + DEFAULT_UMEM_BUFFERS / 2, MIN_PKT_SIZE);
|
|
test->ifobj_rx->pkt_stream = pkt_stream_generate(test->ifobj_rx->umem,
|
|
DEFAULT_UMEM_BUFFERS, MIN_PKT_SIZE);
|
|
|
|
test->ifobj_rx->use_fill_ring = false;
|
|
test->ifobj_rx->validation_func = validate_fill_empty;
|
|
return testapp_validate_traffic(test);
|
|
}
|
|
|
|
static int testapp_unaligned(struct test_spec *test)
|
|
{
|
|
test_spec_set_name(test, "UNALIGNED_MODE");
|
|
test->ifobj_tx->umem->unaligned_mode = true;
|
|
test->ifobj_rx->umem->unaligned_mode = true;
|
|
/* Let half of the packets straddle a 4K buffer boundary */
|
|
pkt_stream_replace_half(test, MIN_PKT_SIZE, -MIN_PKT_SIZE / 2);
|
|
|
|
return testapp_validate_traffic(test);
|
|
}
|
|
|
|
static int testapp_single_pkt(struct test_spec *test)
|
|
{
|
|
struct pkt pkts[] = {{0, MIN_PKT_SIZE, 0, true}};
|
|
|
|
pkt_stream_generate_custom(test, pkts, ARRAY_SIZE(pkts));
|
|
return testapp_validate_traffic(test);
|
|
}
|
|
|
|
static int testapp_invalid_desc(struct test_spec *test)
|
|
{
|
|
struct xsk_umem_info *umem = test->ifobj_tx->umem;
|
|
u64 umem_size = umem->num_frames * umem->frame_size;
|
|
struct pkt pkts[] = {
|
|
/* Zero packet address allowed */
|
|
{0, MIN_PKT_SIZE, 0, true},
|
|
/* Allowed packet */
|
|
{0, MIN_PKT_SIZE, 0, true},
|
|
/* Straddling the start of umem */
|
|
{-2, MIN_PKT_SIZE, 0, false},
|
|
/* Packet too large */
|
|
{0, XSK_UMEM__INVALID_FRAME_SIZE, 0, false},
|
|
/* Up to end of umem allowed */
|
|
{umem_size - MIN_PKT_SIZE - 2 * umem->frame_size, MIN_PKT_SIZE, 0, true},
|
|
/* After umem ends */
|
|
{umem_size, MIN_PKT_SIZE, 0, false},
|
|
/* Straddle the end of umem */
|
|
{umem_size - MIN_PKT_SIZE / 2, MIN_PKT_SIZE, 0, false},
|
|
/* Straddle a 4K boundary */
|
|
{0x1000 - MIN_PKT_SIZE / 2, MIN_PKT_SIZE, 0, false},
|
|
/* Straddle a 2K boundary */
|
|
{0x800 - MIN_PKT_SIZE / 2, MIN_PKT_SIZE, 0, true},
|
|
/* Valid packet for synch so that something is received */
|
|
{0, MIN_PKT_SIZE, 0, true}};
|
|
|
|
if (umem->unaligned_mode) {
|
|
/* Crossing a page boundary allowed */
|
|
pkts[7].valid = true;
|
|
}
|
|
if (umem->frame_size == XSK_UMEM__DEFAULT_FRAME_SIZE / 2) {
|
|
/* Crossing a 2K frame size boundary not allowed */
|
|
pkts[8].valid = false;
|
|
}
|
|
|
|
if (test->ifobj_tx->shared_umem) {
|
|
pkts[4].offset += umem_size;
|
|
pkts[5].offset += umem_size;
|
|
pkts[6].offset += umem_size;
|
|
}
|
|
|
|
pkt_stream_generate_custom(test, pkts, ARRAY_SIZE(pkts));
|
|
return testapp_validate_traffic(test);
|
|
}
|
|
|
|
static int testapp_xdp_drop(struct test_spec *test)
|
|
{
|
|
struct xsk_xdp_progs *skel_rx = test->ifobj_rx->xdp_progs;
|
|
struct xsk_xdp_progs *skel_tx = test->ifobj_tx->xdp_progs;
|
|
|
|
test_spec_set_name(test, "XDP_DROP_HALF");
|
|
test_spec_set_xdp_prog(test, skel_rx->progs.xsk_xdp_drop, skel_tx->progs.xsk_xdp_drop,
|
|
skel_rx->maps.xsk, skel_tx->maps.xsk);
|
|
|
|
pkt_stream_receive_half(test);
|
|
return testapp_validate_traffic(test);
|
|
}
|
|
|
|
static int testapp_xdp_metadata_count(struct test_spec *test)
|
|
{
|
|
struct xsk_xdp_progs *skel_rx = test->ifobj_rx->xdp_progs;
|
|
struct xsk_xdp_progs *skel_tx = test->ifobj_tx->xdp_progs;
|
|
struct bpf_map *data_map;
|
|
int count = 0;
|
|
int key = 0;
|
|
|
|
test_spec_set_name(test, "XDP_METADATA_COUNT");
|
|
test_spec_set_xdp_prog(test, skel_rx->progs.xsk_xdp_populate_metadata,
|
|
skel_tx->progs.xsk_xdp_populate_metadata,
|
|
skel_rx->maps.xsk, skel_tx->maps.xsk);
|
|
test->ifobj_rx->use_metadata = true;
|
|
|
|
data_map = bpf_object__find_map_by_name(skel_rx->obj, "xsk_xdp_.bss");
|
|
if (!data_map || !bpf_map__is_internal(data_map))
|
|
exit_with_error(ENOMEM);
|
|
|
|
if (bpf_map_update_elem(bpf_map__fd(data_map), &key, &count, BPF_ANY))
|
|
exit_with_error(errno);
|
|
|
|
return testapp_validate_traffic(test);
|
|
}
|
|
|
|
static int testapp_poll_txq_tmout(struct test_spec *test)
|
|
{
|
|
test_spec_set_name(test, "POLL_TXQ_FULL");
|
|
|
|
test->ifobj_tx->use_poll = true;
|
|
/* create invalid frame by set umem frame_size and pkt length equal to 2048 */
|
|
test->ifobj_tx->umem->frame_size = 2048;
|
|
pkt_stream_replace(test, 2 * DEFAULT_PKT_CNT, 2048);
|
|
return testapp_validate_traffic_single_thread(test, test->ifobj_tx);
|
|
}
|
|
|
|
static int testapp_poll_rxq_tmout(struct test_spec *test)
|
|
{
|
|
test_spec_set_name(test, "POLL_RXQ_EMPTY");
|
|
test->ifobj_rx->use_poll = true;
|
|
return testapp_validate_traffic_single_thread(test, test->ifobj_rx);
|
|
}
|
|
|
|
static int xsk_load_xdp_programs(struct ifobject *ifobj)
|
|
{
|
|
ifobj->xdp_progs = xsk_xdp_progs__open_and_load();
|
|
if (libbpf_get_error(ifobj->xdp_progs))
|
|
return libbpf_get_error(ifobj->xdp_progs);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void xsk_unload_xdp_programs(struct ifobject *ifobj)
|
|
{
|
|
xsk_xdp_progs__destroy(ifobj->xdp_progs);
|
|
}
|
|
|
|
/* Simple test */
|
|
static bool hugepages_present(void)
|
|
{
|
|
size_t mmap_sz = 2 * DEFAULT_UMEM_BUFFERS * XSK_UMEM__DEFAULT_FRAME_SIZE;
|
|
void *bufs;
|
|
|
|
bufs = mmap(NULL, mmap_sz, PROT_READ | PROT_WRITE,
|
|
MAP_PRIVATE | MAP_ANONYMOUS | MAP_HUGETLB, -1, MAP_HUGE_2MB);
|
|
if (bufs == MAP_FAILED)
|
|
return false;
|
|
|
|
mmap_sz = ceil_u64(mmap_sz, HUGEPAGE_SIZE) * HUGEPAGE_SIZE;
|
|
munmap(bufs, mmap_sz);
|
|
return true;
|
|
}
|
|
|
|
static void init_iface(struct ifobject *ifobj, const char *dst_mac, const char *src_mac,
|
|
thread_func_t func_ptr)
|
|
{
|
|
int err;
|
|
|
|
memcpy(ifobj->dst_mac, dst_mac, ETH_ALEN);
|
|
memcpy(ifobj->src_mac, src_mac, ETH_ALEN);
|
|
|
|
ifobj->func_ptr = func_ptr;
|
|
|
|
err = xsk_load_xdp_programs(ifobj);
|
|
if (err) {
|
|
printf("Error loading XDP program\n");
|
|
exit_with_error(err);
|
|
}
|
|
|
|
if (hugepages_present())
|
|
ifobj->unaligned_supp = true;
|
|
}
|
|
|
|
static void run_pkt_test(struct test_spec *test, enum test_mode mode, enum test_type type)
|
|
{
|
|
int ret = TEST_SKIP;
|
|
|
|
switch (type) {
|
|
case TEST_TYPE_STATS_RX_DROPPED:
|
|
ret = testapp_stats_rx_dropped(test);
|
|
break;
|
|
case TEST_TYPE_STATS_TX_INVALID_DESCS:
|
|
ret = testapp_stats_tx_invalid_descs(test);
|
|
break;
|
|
case TEST_TYPE_STATS_RX_FULL:
|
|
ret = testapp_stats_rx_full(test);
|
|
break;
|
|
case TEST_TYPE_STATS_FILL_EMPTY:
|
|
ret = testapp_stats_fill_empty(test);
|
|
break;
|
|
case TEST_TYPE_TEARDOWN:
|
|
ret = testapp_teardown(test);
|
|
break;
|
|
case TEST_TYPE_BIDI:
|
|
ret = testapp_bidi(test);
|
|
break;
|
|
case TEST_TYPE_BPF_RES:
|
|
ret = testapp_bpf_res(test);
|
|
break;
|
|
case TEST_TYPE_RUN_TO_COMPLETION:
|
|
test_spec_set_name(test, "RUN_TO_COMPLETION");
|
|
ret = testapp_validate_traffic(test);
|
|
break;
|
|
case TEST_TYPE_RUN_TO_COMPLETION_SINGLE_PKT:
|
|
test_spec_set_name(test, "RUN_TO_COMPLETION_SINGLE_PKT");
|
|
ret = testapp_single_pkt(test);
|
|
break;
|
|
case TEST_TYPE_RUN_TO_COMPLETION_2K_FRAME:
|
|
test_spec_set_name(test, "RUN_TO_COMPLETION_2K_FRAME_SIZE");
|
|
test->ifobj_tx->umem->frame_size = 2048;
|
|
test->ifobj_rx->umem->frame_size = 2048;
|
|
pkt_stream_replace(test, DEFAULT_PKT_CNT, MIN_PKT_SIZE);
|
|
ret = testapp_validate_traffic(test);
|
|
break;
|
|
case TEST_TYPE_RX_POLL:
|
|
test->ifobj_rx->use_poll = true;
|
|
test_spec_set_name(test, "POLL_RX");
|
|
ret = testapp_validate_traffic(test);
|
|
break;
|
|
case TEST_TYPE_TX_POLL:
|
|
test->ifobj_tx->use_poll = true;
|
|
test_spec_set_name(test, "POLL_TX");
|
|
ret = testapp_validate_traffic(test);
|
|
break;
|
|
case TEST_TYPE_POLL_TXQ_TMOUT:
|
|
ret = testapp_poll_txq_tmout(test);
|
|
break;
|
|
case TEST_TYPE_POLL_RXQ_TMOUT:
|
|
ret = testapp_poll_rxq_tmout(test);
|
|
break;
|
|
case TEST_TYPE_ALIGNED_INV_DESC:
|
|
test_spec_set_name(test, "ALIGNED_INV_DESC");
|
|
ret = testapp_invalid_desc(test);
|
|
break;
|
|
case TEST_TYPE_ALIGNED_INV_DESC_2K_FRAME:
|
|
test_spec_set_name(test, "ALIGNED_INV_DESC_2K_FRAME_SIZE");
|
|
test->ifobj_tx->umem->frame_size = 2048;
|
|
test->ifobj_rx->umem->frame_size = 2048;
|
|
ret = testapp_invalid_desc(test);
|
|
break;
|
|
case TEST_TYPE_UNALIGNED_INV_DESC:
|
|
test_spec_set_name(test, "UNALIGNED_INV_DESC");
|
|
test->ifobj_tx->umem->unaligned_mode = true;
|
|
test->ifobj_rx->umem->unaligned_mode = true;
|
|
ret = testapp_invalid_desc(test);
|
|
break;
|
|
case TEST_TYPE_UNALIGNED_INV_DESC_4K1_FRAME: {
|
|
u64 page_size, umem_size;
|
|
|
|
test_spec_set_name(test, "UNALIGNED_INV_DESC_4K1_FRAME_SIZE");
|
|
/* Odd frame size so the UMEM doesn't end near a page boundary. */
|
|
test->ifobj_tx->umem->frame_size = 4001;
|
|
test->ifobj_rx->umem->frame_size = 4001;
|
|
test->ifobj_tx->umem->unaligned_mode = true;
|
|
test->ifobj_rx->umem->unaligned_mode = true;
|
|
/* This test exists to test descriptors that staddle the end of
|
|
* the UMEM but not a page.
|
|
*/
|
|
page_size = sysconf(_SC_PAGESIZE);
|
|
umem_size = test->ifobj_tx->umem->num_frames * test->ifobj_tx->umem->frame_size;
|
|
assert(umem_size % page_size > MIN_PKT_SIZE);
|
|
assert(umem_size % page_size < page_size - MIN_PKT_SIZE);
|
|
ret = testapp_invalid_desc(test);
|
|
break;
|
|
}
|
|
case TEST_TYPE_UNALIGNED:
|
|
ret = testapp_unaligned(test);
|
|
break;
|
|
case TEST_TYPE_HEADROOM:
|
|
ret = testapp_headroom(test);
|
|
break;
|
|
case TEST_TYPE_XDP_DROP_HALF:
|
|
ret = testapp_xdp_drop(test);
|
|
break;
|
|
case TEST_TYPE_XDP_METADATA_COUNT:
|
|
ret = testapp_xdp_metadata_count(test);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
if (ret == TEST_PASS)
|
|
ksft_test_result_pass("PASS: %s %s%s\n", mode_string(test), busy_poll_string(test),
|
|
test->name);
|
|
pkt_stream_restore_default(test);
|
|
}
|
|
|
|
static struct ifobject *ifobject_create(void)
|
|
{
|
|
struct ifobject *ifobj;
|
|
|
|
ifobj = calloc(1, sizeof(struct ifobject));
|
|
if (!ifobj)
|
|
return NULL;
|
|
|
|
ifobj->xsk_arr = calloc(MAX_SOCKETS, sizeof(*ifobj->xsk_arr));
|
|
if (!ifobj->xsk_arr)
|
|
goto out_xsk_arr;
|
|
|
|
ifobj->umem = calloc(1, sizeof(*ifobj->umem));
|
|
if (!ifobj->umem)
|
|
goto out_umem;
|
|
|
|
return ifobj;
|
|
|
|
out_umem:
|
|
free(ifobj->xsk_arr);
|
|
out_xsk_arr:
|
|
free(ifobj);
|
|
return NULL;
|
|
}
|
|
|
|
static void ifobject_delete(struct ifobject *ifobj)
|
|
{
|
|
free(ifobj->umem);
|
|
free(ifobj->xsk_arr);
|
|
free(ifobj);
|
|
}
|
|
|
|
static bool is_xdp_supported(int ifindex)
|
|
{
|
|
int flags = XDP_FLAGS_DRV_MODE;
|
|
|
|
LIBBPF_OPTS(bpf_link_create_opts, opts, .flags = flags);
|
|
struct bpf_insn insns[2] = {
|
|
BPF_MOV64_IMM(BPF_REG_0, XDP_PASS),
|
|
BPF_EXIT_INSN()
|
|
};
|
|
int prog_fd, insn_cnt = ARRAY_SIZE(insns);
|
|
int err;
|
|
|
|
prog_fd = bpf_prog_load(BPF_PROG_TYPE_XDP, NULL, "GPL", insns, insn_cnt, NULL);
|
|
if (prog_fd < 0)
|
|
return false;
|
|
|
|
err = bpf_xdp_attach(ifindex, prog_fd, flags, NULL);
|
|
if (err) {
|
|
close(prog_fd);
|
|
return false;
|
|
}
|
|
|
|
bpf_xdp_detach(ifindex, flags, NULL);
|
|
close(prog_fd);
|
|
|
|
return true;
|
|
}
|
|
|
|
int main(int argc, char **argv)
|
|
{
|
|
struct pkt_stream *rx_pkt_stream_default;
|
|
struct pkt_stream *tx_pkt_stream_default;
|
|
struct ifobject *ifobj_tx, *ifobj_rx;
|
|
int modes = TEST_MODE_SKB + 1;
|
|
u32 i, j, failed_tests = 0;
|
|
struct test_spec test;
|
|
bool shared_netdev;
|
|
|
|
/* Use libbpf 1.0 API mode */
|
|
libbpf_set_strict_mode(LIBBPF_STRICT_ALL);
|
|
|
|
ifobj_tx = ifobject_create();
|
|
if (!ifobj_tx)
|
|
exit_with_error(ENOMEM);
|
|
ifobj_rx = ifobject_create();
|
|
if (!ifobj_rx)
|
|
exit_with_error(ENOMEM);
|
|
|
|
setlocale(LC_ALL, "");
|
|
|
|
parse_command_line(ifobj_tx, ifobj_rx, argc, argv);
|
|
|
|
shared_netdev = (ifobj_tx->ifindex == ifobj_rx->ifindex);
|
|
ifobj_tx->shared_umem = shared_netdev;
|
|
ifobj_rx->shared_umem = shared_netdev;
|
|
|
|
if (!validate_interface(ifobj_tx) || !validate_interface(ifobj_rx)) {
|
|
usage(basename(argv[0]));
|
|
ksft_exit_xfail();
|
|
}
|
|
|
|
if (is_xdp_supported(ifobj_tx->ifindex)) {
|
|
modes++;
|
|
if (ifobj_zc_avail(ifobj_tx))
|
|
modes++;
|
|
}
|
|
|
|
init_iface(ifobj_rx, MAC1, MAC2, worker_testapp_validate_rx);
|
|
init_iface(ifobj_tx, MAC2, MAC1, worker_testapp_validate_tx);
|
|
|
|
test_spec_init(&test, ifobj_tx, ifobj_rx, 0);
|
|
tx_pkt_stream_default = pkt_stream_generate(ifobj_tx->umem, DEFAULT_PKT_CNT, MIN_PKT_SIZE);
|
|
rx_pkt_stream_default = pkt_stream_generate(ifobj_rx->umem, DEFAULT_PKT_CNT, MIN_PKT_SIZE);
|
|
if (!tx_pkt_stream_default || !rx_pkt_stream_default)
|
|
exit_with_error(ENOMEM);
|
|
test.tx_pkt_stream_default = tx_pkt_stream_default;
|
|
test.rx_pkt_stream_default = rx_pkt_stream_default;
|
|
|
|
ksft_set_plan(modes * TEST_TYPE_MAX);
|
|
|
|
for (i = 0; i < modes; i++) {
|
|
for (j = 0; j < TEST_TYPE_MAX; j++) {
|
|
test_spec_init(&test, ifobj_tx, ifobj_rx, i);
|
|
run_pkt_test(&test, i, j);
|
|
usleep(USLEEP_MAX);
|
|
|
|
if (test.fail)
|
|
failed_tests++;
|
|
}
|
|
}
|
|
|
|
pkt_stream_delete(tx_pkt_stream_default);
|
|
pkt_stream_delete(rx_pkt_stream_default);
|
|
xsk_unload_xdp_programs(ifobj_tx);
|
|
xsk_unload_xdp_programs(ifobj_rx);
|
|
ifobject_delete(ifobj_tx);
|
|
ifobject_delete(ifobj_rx);
|
|
|
|
if (failed_tests)
|
|
ksft_exit_fail();
|
|
else
|
|
ksft_exit_pass();
|
|
}
|