mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/next/linux-next.git
synced 2025-01-10 15:58:47 +00:00
xsk: Change the default frame size to 4096 and allow controlling it
The typical XDP memory scheme is one packet per page. Change the AF_XDP frame size in libbpf to 4096, which is the page size on x86, to allow libbpf to be used with the drivers with the packet-per-page scheme. Add a command line option -f to xdpsock to allow to specify a custom frame size. Signed-off-by: Maxim Mikityanskiy <maximmi@mellanox.com> Signed-off-by: Tariq Toukan <tariqt@mellanox.com> Acked-by: Saeed Mahameed <saeedm@mellanox.com> Signed-off-by: Daniel Borkmann <daniel@iogearbox.net>
This commit is contained in:
parent
2761ed4b6e
commit
123e8da1d3
@ -68,6 +68,7 @@ static int opt_queue;
|
|||||||
static int opt_poll;
|
static int opt_poll;
|
||||||
static int opt_interval = 1;
|
static int opt_interval = 1;
|
||||||
static u32 opt_xdp_bind_flags;
|
static u32 opt_xdp_bind_flags;
|
||||||
|
static int opt_xsk_frame_size = XSK_UMEM__DEFAULT_FRAME_SIZE;
|
||||||
static __u32 prog_id;
|
static __u32 prog_id;
|
||||||
|
|
||||||
struct xsk_umem_info {
|
struct xsk_umem_info {
|
||||||
@ -276,6 +277,12 @@ static size_t gen_eth_frame(struct xsk_umem_info *umem, u64 addr)
|
|||||||
static struct xsk_umem_info *xsk_configure_umem(void *buffer, u64 size)
|
static struct xsk_umem_info *xsk_configure_umem(void *buffer, u64 size)
|
||||||
{
|
{
|
||||||
struct xsk_umem_info *umem;
|
struct xsk_umem_info *umem;
|
||||||
|
struct xsk_umem_config cfg = {
|
||||||
|
.fill_size = XSK_RING_PROD__DEFAULT_NUM_DESCS,
|
||||||
|
.comp_size = XSK_RING_CONS__DEFAULT_NUM_DESCS,
|
||||||
|
.frame_size = opt_xsk_frame_size,
|
||||||
|
.frame_headroom = XSK_UMEM__DEFAULT_FRAME_HEADROOM,
|
||||||
|
};
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
umem = calloc(1, sizeof(*umem));
|
umem = calloc(1, sizeof(*umem));
|
||||||
@ -283,7 +290,7 @@ static struct xsk_umem_info *xsk_configure_umem(void *buffer, u64 size)
|
|||||||
exit_with_error(errno);
|
exit_with_error(errno);
|
||||||
|
|
||||||
ret = xsk_umem__create(&umem->umem, buffer, size, &umem->fq, &umem->cq,
|
ret = xsk_umem__create(&umem->umem, buffer, size, &umem->fq, &umem->cq,
|
||||||
NULL);
|
&cfg);
|
||||||
if (ret)
|
if (ret)
|
||||||
exit_with_error(-ret);
|
exit_with_error(-ret);
|
||||||
|
|
||||||
@ -323,11 +330,9 @@ static struct xsk_socket_info *xsk_configure_socket(struct xsk_umem_info *umem)
|
|||||||
&idx);
|
&idx);
|
||||||
if (ret != XSK_RING_PROD__DEFAULT_NUM_DESCS)
|
if (ret != XSK_RING_PROD__DEFAULT_NUM_DESCS)
|
||||||
exit_with_error(-ret);
|
exit_with_error(-ret);
|
||||||
for (i = 0;
|
for (i = 0; i < XSK_RING_PROD__DEFAULT_NUM_DESCS; i++)
|
||||||
i < XSK_RING_PROD__DEFAULT_NUM_DESCS *
|
*xsk_ring_prod__fill_addr(&xsk->umem->fq, idx++) =
|
||||||
XSK_UMEM__DEFAULT_FRAME_SIZE;
|
i * opt_xsk_frame_size;
|
||||||
i += XSK_UMEM__DEFAULT_FRAME_SIZE)
|
|
||||||
*xsk_ring_prod__fill_addr(&xsk->umem->fq, idx++) = i;
|
|
||||||
xsk_ring_prod__submit(&xsk->umem->fq,
|
xsk_ring_prod__submit(&xsk->umem->fq,
|
||||||
XSK_RING_PROD__DEFAULT_NUM_DESCS);
|
XSK_RING_PROD__DEFAULT_NUM_DESCS);
|
||||||
|
|
||||||
@ -346,6 +351,7 @@ static struct option long_options[] = {
|
|||||||
{"interval", required_argument, 0, 'n'},
|
{"interval", required_argument, 0, 'n'},
|
||||||
{"zero-copy", no_argument, 0, 'z'},
|
{"zero-copy", no_argument, 0, 'z'},
|
||||||
{"copy", no_argument, 0, 'c'},
|
{"copy", no_argument, 0, 'c'},
|
||||||
|
{"frame-size", required_argument, 0, 'f'},
|
||||||
{0, 0, 0, 0}
|
{0, 0, 0, 0}
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -365,8 +371,9 @@ static void usage(const char *prog)
|
|||||||
" -n, --interval=n Specify statistics update interval (default 1 sec).\n"
|
" -n, --interval=n Specify statistics update interval (default 1 sec).\n"
|
||||||
" -z, --zero-copy Force zero-copy mode.\n"
|
" -z, --zero-copy Force zero-copy mode.\n"
|
||||||
" -c, --copy Force copy mode.\n"
|
" -c, --copy Force copy mode.\n"
|
||||||
|
" -f, --frame-size=n Set the frame size (must be a power of two, default is %d).\n"
|
||||||
"\n";
|
"\n";
|
||||||
fprintf(stderr, str, prog);
|
fprintf(stderr, str, prog, XSK_UMEM__DEFAULT_FRAME_SIZE);
|
||||||
exit(EXIT_FAILURE);
|
exit(EXIT_FAILURE);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -377,7 +384,7 @@ static void parse_command_line(int argc, char **argv)
|
|||||||
opterr = 0;
|
opterr = 0;
|
||||||
|
|
||||||
for (;;) {
|
for (;;) {
|
||||||
c = getopt_long(argc, argv, "Frtli:q:psSNn:cz", long_options,
|
c = getopt_long(argc, argv, "Frtli:q:psSNn:czf:", long_options,
|
||||||
&option_index);
|
&option_index);
|
||||||
if (c == -1)
|
if (c == -1)
|
||||||
break;
|
break;
|
||||||
@ -420,6 +427,9 @@ static void parse_command_line(int argc, char **argv)
|
|||||||
case 'F':
|
case 'F':
|
||||||
opt_xdp_flags &= ~XDP_FLAGS_UPDATE_IF_NOEXIST;
|
opt_xdp_flags &= ~XDP_FLAGS_UPDATE_IF_NOEXIST;
|
||||||
break;
|
break;
|
||||||
|
case 'f':
|
||||||
|
opt_xsk_frame_size = atoi(optarg);
|
||||||
|
break;
|
||||||
default:
|
default:
|
||||||
usage(basename(argv[0]));
|
usage(basename(argv[0]));
|
||||||
}
|
}
|
||||||
@ -432,6 +442,11 @@ static void parse_command_line(int argc, char **argv)
|
|||||||
usage(basename(argv[0]));
|
usage(basename(argv[0]));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (opt_xsk_frame_size & (opt_xsk_frame_size - 1)) {
|
||||||
|
fprintf(stderr, "--frame-size=%d is not a power of two\n",
|
||||||
|
opt_xsk_frame_size);
|
||||||
|
usage(basename(argv[0]));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static void kick_tx(struct xsk_socket_info *xsk)
|
static void kick_tx(struct xsk_socket_info *xsk)
|
||||||
@ -583,8 +598,7 @@ static void tx_only(struct xsk_socket_info *xsk)
|
|||||||
|
|
||||||
for (i = 0; i < BATCH_SIZE; i++) {
|
for (i = 0; i < BATCH_SIZE; i++) {
|
||||||
xsk_ring_prod__tx_desc(&xsk->tx, idx + i)->addr
|
xsk_ring_prod__tx_desc(&xsk->tx, idx + i)->addr
|
||||||
= (frame_nb + i) <<
|
= (frame_nb + i) * opt_xsk_frame_size;
|
||||||
XSK_UMEM__DEFAULT_FRAME_SHIFT;
|
|
||||||
xsk_ring_prod__tx_desc(&xsk->tx, idx + i)->len =
|
xsk_ring_prod__tx_desc(&xsk->tx, idx + i)->len =
|
||||||
sizeof(pkt_data) - 1;
|
sizeof(pkt_data) - 1;
|
||||||
}
|
}
|
||||||
@ -661,21 +675,19 @@ int main(int argc, char **argv)
|
|||||||
}
|
}
|
||||||
|
|
||||||
ret = posix_memalign(&bufs, getpagesize(), /* PAGE_SIZE aligned */
|
ret = posix_memalign(&bufs, getpagesize(), /* PAGE_SIZE aligned */
|
||||||
NUM_FRAMES * XSK_UMEM__DEFAULT_FRAME_SIZE);
|
NUM_FRAMES * opt_xsk_frame_size);
|
||||||
if (ret)
|
if (ret)
|
||||||
exit_with_error(ret);
|
exit_with_error(ret);
|
||||||
|
|
||||||
/* Create sockets... */
|
/* Create sockets... */
|
||||||
umem = xsk_configure_umem(bufs,
|
umem = xsk_configure_umem(bufs, NUM_FRAMES * opt_xsk_frame_size);
|
||||||
NUM_FRAMES * XSK_UMEM__DEFAULT_FRAME_SIZE);
|
|
||||||
xsks[num_socks++] = xsk_configure_socket(umem);
|
xsks[num_socks++] = xsk_configure_socket(umem);
|
||||||
|
|
||||||
if (opt_bench == BENCH_TXONLY) {
|
if (opt_bench == BENCH_TXONLY) {
|
||||||
int i;
|
int i;
|
||||||
|
|
||||||
for (i = 0; i < NUM_FRAMES * XSK_UMEM__DEFAULT_FRAME_SIZE;
|
for (i = 0; i < NUM_FRAMES; i++)
|
||||||
i += XSK_UMEM__DEFAULT_FRAME_SIZE)
|
(void)gen_eth_frame(umem, i * opt_xsk_frame_size);
|
||||||
(void)gen_eth_frame(umem, i);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
signal(SIGINT, int_exit);
|
signal(SIGINT, int_exit);
|
||||||
|
@ -167,7 +167,7 @@ LIBBPF_API int xsk_socket__fd(const struct xsk_socket *xsk);
|
|||||||
|
|
||||||
#define XSK_RING_CONS__DEFAULT_NUM_DESCS 2048
|
#define XSK_RING_CONS__DEFAULT_NUM_DESCS 2048
|
||||||
#define XSK_RING_PROD__DEFAULT_NUM_DESCS 2048
|
#define XSK_RING_PROD__DEFAULT_NUM_DESCS 2048
|
||||||
#define XSK_UMEM__DEFAULT_FRAME_SHIFT 11 /* 2048 bytes */
|
#define XSK_UMEM__DEFAULT_FRAME_SHIFT 12 /* 4096 bytes */
|
||||||
#define XSK_UMEM__DEFAULT_FRAME_SIZE (1 << XSK_UMEM__DEFAULT_FRAME_SHIFT)
|
#define XSK_UMEM__DEFAULT_FRAME_SIZE (1 << XSK_UMEM__DEFAULT_FRAME_SHIFT)
|
||||||
#define XSK_UMEM__DEFAULT_FRAME_HEADROOM 0
|
#define XSK_UMEM__DEFAULT_FRAME_HEADROOM 0
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user