mirror of
				https://github.com/torvalds/linux.git
				synced 2025-11-04 10:40:15 +02:00 
			
		
		
		
	Remove duplicated include. Signed-off-by: YueHaibing <yuehaibing@huawei.com> Acked-by: Song Liu <songliubraving@fb.com> Signed-off-by: Daniel Borkmann <daniel@iogearbox.net>
		
			
				
	
	
		
			1849 lines
		
	
	
	
		
			38 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			1849 lines
		
	
	
	
		
			38 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
// SPDX-License-Identifier: GPL-2.0
 | 
						|
// Copyright (c) 2017-2018 Covalent IO, Inc. http://covalent.io
 | 
						|
#include <stdio.h>
 | 
						|
#include <stdlib.h>
 | 
						|
#include <sys/socket.h>
 | 
						|
#include <sys/ioctl.h>
 | 
						|
#include <sys/select.h>
 | 
						|
#include <netinet/in.h>
 | 
						|
#include <arpa/inet.h>
 | 
						|
#include <unistd.h>
 | 
						|
#include <string.h>
 | 
						|
#include <errno.h>
 | 
						|
#include <stdbool.h>
 | 
						|
#include <signal.h>
 | 
						|
#include <fcntl.h>
 | 
						|
#include <sys/wait.h>
 | 
						|
#include <time.h>
 | 
						|
#include <sched.h>
 | 
						|
 | 
						|
#include <sys/time.h>
 | 
						|
#include <sys/resource.h>
 | 
						|
#include <sys/types.h>
 | 
						|
#include <sys/sendfile.h>
 | 
						|
 | 
						|
#include <linux/netlink.h>
 | 
						|
#include <linux/socket.h>
 | 
						|
#include <linux/sock_diag.h>
 | 
						|
#include <linux/bpf.h>
 | 
						|
#include <linux/if_link.h>
 | 
						|
#include <linux/tls.h>
 | 
						|
#include <assert.h>
 | 
						|
#include <libgen.h>
 | 
						|
 | 
						|
#include <getopt.h>
 | 
						|
 | 
						|
#include <bpf/bpf.h>
 | 
						|
#include <bpf/libbpf.h>
 | 
						|
 | 
						|
#include "bpf_util.h"
 | 
						|
#include "bpf_rlimit.h"
 | 
						|
#include "cgroup_helpers.h"
 | 
						|
 | 
						|
int running;
 | 
						|
static void running_handler(int a);
 | 
						|
 | 
						|
#ifndef TCP_ULP
 | 
						|
# define TCP_ULP 31
 | 
						|
#endif
 | 
						|
#ifndef SOL_TLS
 | 
						|
# define SOL_TLS 282
 | 
						|
#endif
 | 
						|
 | 
						|
/* randomly selected ports for testing on lo */
 | 
						|
#define S1_PORT 10000
 | 
						|
#define S2_PORT 10001
 | 
						|
 | 
						|
#define BPF_SOCKMAP_FILENAME "test_sockmap_kern.o"
 | 
						|
#define BPF_SOCKHASH_FILENAME "test_sockhash_kern.o"
 | 
						|
#define CG_PATH "/sockmap"
 | 
						|
 | 
						|
/* global sockets */
 | 
						|
int s1, s2, c1, c2, p1, p2;
 | 
						|
int test_cnt;
 | 
						|
int passed;
 | 
						|
int failed;
 | 
						|
int map_fd[8];
 | 
						|
struct bpf_map *maps[8];
 | 
						|
int prog_fd[11];
 | 
						|
 | 
						|
int txmsg_pass;
 | 
						|
int txmsg_noisy;
 | 
						|
int txmsg_redir;
 | 
						|
int txmsg_redir_noisy;
 | 
						|
int txmsg_drop;
 | 
						|
int txmsg_apply;
 | 
						|
int txmsg_cork;
 | 
						|
int txmsg_start;
 | 
						|
int txmsg_end;
 | 
						|
int txmsg_start_push;
 | 
						|
int txmsg_end_push;
 | 
						|
int txmsg_start_pop;
 | 
						|
int txmsg_pop;
 | 
						|
int txmsg_ingress;
 | 
						|
int txmsg_skb;
 | 
						|
int ktls;
 | 
						|
int peek_flag;
 | 
						|
 | 
						|
static const struct option long_options[] = {
 | 
						|
	{"help",	no_argument,		NULL, 'h' },
 | 
						|
	{"cgroup",	required_argument,	NULL, 'c' },
 | 
						|
	{"rate",	required_argument,	NULL, 'r' },
 | 
						|
	{"verbose",	no_argument,		NULL, 'v' },
 | 
						|
	{"iov_count",	required_argument,	NULL, 'i' },
 | 
						|
	{"length",	required_argument,	NULL, 'l' },
 | 
						|
	{"test",	required_argument,	NULL, 't' },
 | 
						|
	{"data_test",   no_argument,		NULL, 'd' },
 | 
						|
	{"txmsg",		no_argument,	&txmsg_pass,  1  },
 | 
						|
	{"txmsg_noisy",		no_argument,	&txmsg_noisy, 1  },
 | 
						|
	{"txmsg_redir",		no_argument,	&txmsg_redir, 1  },
 | 
						|
	{"txmsg_redir_noisy",	no_argument,	&txmsg_redir_noisy, 1},
 | 
						|
	{"txmsg_drop",		no_argument,	&txmsg_drop, 1 },
 | 
						|
	{"txmsg_apply",	required_argument,	NULL, 'a'},
 | 
						|
	{"txmsg_cork",	required_argument,	NULL, 'k'},
 | 
						|
	{"txmsg_start", required_argument,	NULL, 's'},
 | 
						|
	{"txmsg_end",	required_argument,	NULL, 'e'},
 | 
						|
	{"txmsg_start_push", required_argument,	NULL, 'p'},
 | 
						|
	{"txmsg_end_push",   required_argument,	NULL, 'q'},
 | 
						|
	{"txmsg_start_pop",  required_argument,	NULL, 'w'},
 | 
						|
	{"txmsg_pop",	     required_argument,	NULL, 'x'},
 | 
						|
	{"txmsg_ingress", no_argument,		&txmsg_ingress, 1 },
 | 
						|
	{"txmsg_skb", no_argument,		&txmsg_skb, 1 },
 | 
						|
	{"ktls", no_argument,			&ktls, 1 },
 | 
						|
	{"peek", no_argument,			&peek_flag, 1 },
 | 
						|
	{0, 0, NULL, 0 }
 | 
						|
};
 | 
						|
 | 
						|
static void usage(char *argv[])
 | 
						|
{
 | 
						|
	int i;
 | 
						|
 | 
						|
	printf(" Usage: %s --cgroup <cgroup_path>\n", argv[0]);
 | 
						|
	printf(" options:\n");
 | 
						|
	for (i = 0; long_options[i].name != 0; i++) {
 | 
						|
		printf(" --%-12s", long_options[i].name);
 | 
						|
		if (long_options[i].flag != NULL)
 | 
						|
			printf(" flag (internal value:%d)\n",
 | 
						|
				*long_options[i].flag);
 | 
						|
		else
 | 
						|
			printf(" -%c\n", long_options[i].val);
 | 
						|
	}
 | 
						|
	printf("\n");
 | 
						|
}
 | 
						|
 | 
						|
char *sock_to_string(int s)
 | 
						|
{
 | 
						|
	if (s == c1)
 | 
						|
		return "client1";
 | 
						|
	else if (s == c2)
 | 
						|
		return "client2";
 | 
						|
	else if (s == s1)
 | 
						|
		return "server1";
 | 
						|
	else if (s == s2)
 | 
						|
		return "server2";
 | 
						|
	else if (s == p1)
 | 
						|
		return "peer1";
 | 
						|
	else if (s == p2)
 | 
						|
		return "peer2";
 | 
						|
	else
 | 
						|
		return "unknown";
 | 
						|
}
 | 
						|
 | 
						|
static int sockmap_init_ktls(int verbose, int s)
 | 
						|
{
 | 
						|
	struct tls12_crypto_info_aes_gcm_128 tls_tx = {
 | 
						|
		.info = {
 | 
						|
			.version     = TLS_1_2_VERSION,
 | 
						|
			.cipher_type = TLS_CIPHER_AES_GCM_128,
 | 
						|
		},
 | 
						|
	};
 | 
						|
	struct tls12_crypto_info_aes_gcm_128 tls_rx = {
 | 
						|
		.info = {
 | 
						|
			.version     = TLS_1_2_VERSION,
 | 
						|
			.cipher_type = TLS_CIPHER_AES_GCM_128,
 | 
						|
		},
 | 
						|
	};
 | 
						|
	int so_buf = 6553500;
 | 
						|
	int err;
 | 
						|
 | 
						|
	err = setsockopt(s, 6, TCP_ULP, "tls", sizeof("tls"));
 | 
						|
	if (err) {
 | 
						|
		fprintf(stderr, "setsockopt: TCP_ULP(%s) failed with error %i\n", sock_to_string(s), err);
 | 
						|
		return -EINVAL;
 | 
						|
	}
 | 
						|
	err = setsockopt(s, SOL_TLS, TLS_TX, (void *)&tls_tx, sizeof(tls_tx));
 | 
						|
	if (err) {
 | 
						|
		fprintf(stderr, "setsockopt: TLS_TX(%s) failed with error %i\n", sock_to_string(s), err);
 | 
						|
		return -EINVAL;
 | 
						|
	}
 | 
						|
	err = setsockopt(s, SOL_TLS, TLS_RX, (void *)&tls_rx, sizeof(tls_rx));
 | 
						|
	if (err) {
 | 
						|
		fprintf(stderr, "setsockopt: TLS_RX(%s) failed with error %i\n", sock_to_string(s), err);
 | 
						|
		return -EINVAL;
 | 
						|
	}
 | 
						|
	err = setsockopt(s, SOL_SOCKET, SO_SNDBUF, &so_buf, sizeof(so_buf));
 | 
						|
	if (err) {
 | 
						|
		fprintf(stderr, "setsockopt: (%s) failed sndbuf with error %i\n", sock_to_string(s), err);
 | 
						|
		return -EINVAL;
 | 
						|
	}
 | 
						|
	err = setsockopt(s, SOL_SOCKET, SO_RCVBUF, &so_buf, sizeof(so_buf));
 | 
						|
	if (err) {
 | 
						|
		fprintf(stderr, "setsockopt: (%s) failed rcvbuf with error %i\n", sock_to_string(s), err);
 | 
						|
		return -EINVAL;
 | 
						|
	}
 | 
						|
 | 
						|
	if (verbose)
 | 
						|
		fprintf(stdout, "socket(%s) kTLS enabled\n", sock_to_string(s));
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
static int sockmap_init_sockets(int verbose)
 | 
						|
{
 | 
						|
	int i, err, one = 1;
 | 
						|
	struct sockaddr_in addr;
 | 
						|
	int *fds[4] = {&s1, &s2, &c1, &c2};
 | 
						|
 | 
						|
	s1 = s2 = p1 = p2 = c1 = c2 = 0;
 | 
						|
 | 
						|
	/* Init sockets */
 | 
						|
	for (i = 0; i < 4; i++) {
 | 
						|
		*fds[i] = socket(AF_INET, SOCK_STREAM, 0);
 | 
						|
		if (*fds[i] < 0) {
 | 
						|
			perror("socket s1 failed()");
 | 
						|
			return errno;
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	/* Allow reuse */
 | 
						|
	for (i = 0; i < 2; i++) {
 | 
						|
		err = setsockopt(*fds[i], SOL_SOCKET, SO_REUSEADDR,
 | 
						|
				 (char *)&one, sizeof(one));
 | 
						|
		if (err) {
 | 
						|
			perror("setsockopt failed()");
 | 
						|
			return errno;
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	/* Non-blocking sockets */
 | 
						|
	for (i = 0; i < 2; i++) {
 | 
						|
		err = ioctl(*fds[i], FIONBIO, (char *)&one);
 | 
						|
		if (err < 0) {
 | 
						|
			perror("ioctl s1 failed()");
 | 
						|
			return errno;
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	/* Bind server sockets */
 | 
						|
	memset(&addr, 0, sizeof(struct sockaddr_in));
 | 
						|
	addr.sin_family = AF_INET;
 | 
						|
	addr.sin_addr.s_addr = inet_addr("127.0.0.1");
 | 
						|
 | 
						|
	addr.sin_port = htons(S1_PORT);
 | 
						|
	err = bind(s1, (struct sockaddr *)&addr, sizeof(addr));
 | 
						|
	if (err < 0) {
 | 
						|
		perror("bind s1 failed()\n");
 | 
						|
		return errno;
 | 
						|
	}
 | 
						|
 | 
						|
	addr.sin_port = htons(S2_PORT);
 | 
						|
	err = bind(s2, (struct sockaddr *)&addr, sizeof(addr));
 | 
						|
	if (err < 0) {
 | 
						|
		perror("bind s2 failed()\n");
 | 
						|
		return errno;
 | 
						|
	}
 | 
						|
 | 
						|
	/* Listen server sockets */
 | 
						|
	addr.sin_port = htons(S1_PORT);
 | 
						|
	err = listen(s1, 32);
 | 
						|
	if (err < 0) {
 | 
						|
		perror("listen s1 failed()\n");
 | 
						|
		return errno;
 | 
						|
	}
 | 
						|
 | 
						|
	addr.sin_port = htons(S2_PORT);
 | 
						|
	err = listen(s2, 32);
 | 
						|
	if (err < 0) {
 | 
						|
		perror("listen s1 failed()\n");
 | 
						|
		return errno;
 | 
						|
	}
 | 
						|
 | 
						|
	/* Initiate Connect */
 | 
						|
	addr.sin_port = htons(S1_PORT);
 | 
						|
	err = connect(c1, (struct sockaddr *)&addr, sizeof(addr));
 | 
						|
	if (err < 0 && errno != EINPROGRESS) {
 | 
						|
		perror("connect c1 failed()\n");
 | 
						|
		return errno;
 | 
						|
	}
 | 
						|
 | 
						|
	addr.sin_port = htons(S2_PORT);
 | 
						|
	err = connect(c2, (struct sockaddr *)&addr, sizeof(addr));
 | 
						|
	if (err < 0 && errno != EINPROGRESS) {
 | 
						|
		perror("connect c2 failed()\n");
 | 
						|
		return errno;
 | 
						|
	} else if (err < 0) {
 | 
						|
		err = 0;
 | 
						|
	}
 | 
						|
 | 
						|
	/* Accept Connecrtions */
 | 
						|
	p1 = accept(s1, NULL, NULL);
 | 
						|
	if (p1 < 0) {
 | 
						|
		perror("accept s1 failed()\n");
 | 
						|
		return errno;
 | 
						|
	}
 | 
						|
 | 
						|
	p2 = accept(s2, NULL, NULL);
 | 
						|
	if (p2 < 0) {
 | 
						|
		perror("accept s1 failed()\n");
 | 
						|
		return errno;
 | 
						|
	}
 | 
						|
 | 
						|
	if (verbose) {
 | 
						|
		printf("connected sockets: c1 <-> p1, c2 <-> p2\n");
 | 
						|
		printf("cgroups binding: c1(%i) <-> s1(%i) - - - c2(%i) <-> s2(%i)\n",
 | 
						|
			c1, s1, c2, s2);
 | 
						|
	}
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
struct msg_stats {
 | 
						|
	size_t bytes_sent;
 | 
						|
	size_t bytes_recvd;
 | 
						|
	struct timespec start;
 | 
						|
	struct timespec end;
 | 
						|
};
 | 
						|
 | 
						|
struct sockmap_options {
 | 
						|
	int verbose;
 | 
						|
	bool base;
 | 
						|
	bool sendpage;
 | 
						|
	bool data_test;
 | 
						|
	bool drop_expected;
 | 
						|
	int iov_count;
 | 
						|
	int iov_length;
 | 
						|
	int rate;
 | 
						|
};
 | 
						|
 | 
						|
static int msg_loop_sendpage(int fd, int iov_length, int cnt,
 | 
						|
			     struct msg_stats *s,
 | 
						|
			     struct sockmap_options *opt)
 | 
						|
{
 | 
						|
	bool drop = opt->drop_expected;
 | 
						|
	unsigned char k = 0;
 | 
						|
	FILE *file;
 | 
						|
	int i, fp;
 | 
						|
 | 
						|
	file = fopen(".sendpage_tst.tmp", "w+");
 | 
						|
	for (i = 0; i < iov_length * cnt; i++, k++)
 | 
						|
		fwrite(&k, sizeof(char), 1, file);
 | 
						|
	fflush(file);
 | 
						|
	fseek(file, 0, SEEK_SET);
 | 
						|
	fclose(file);
 | 
						|
 | 
						|
	fp = open(".sendpage_tst.tmp", O_RDONLY);
 | 
						|
	clock_gettime(CLOCK_MONOTONIC, &s->start);
 | 
						|
	for (i = 0; i < cnt; i++) {
 | 
						|
		int sent = sendfile(fd, fp, NULL, iov_length);
 | 
						|
 | 
						|
		if (!drop && sent < 0) {
 | 
						|
			perror("send loop error:");
 | 
						|
			close(fp);
 | 
						|
			return sent;
 | 
						|
		} else if (drop && sent >= 0) {
 | 
						|
			printf("sendpage loop error expected: %i\n", sent);
 | 
						|
			close(fp);
 | 
						|
			return -EIO;
 | 
						|
		}
 | 
						|
 | 
						|
		if (sent > 0)
 | 
						|
			s->bytes_sent += sent;
 | 
						|
	}
 | 
						|
	clock_gettime(CLOCK_MONOTONIC, &s->end);
 | 
						|
	close(fp);
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static void msg_free_iov(struct msghdr *msg)
 | 
						|
{
 | 
						|
	int i;
 | 
						|
 | 
						|
	for (i = 0; i < msg->msg_iovlen; i++)
 | 
						|
		free(msg->msg_iov[i].iov_base);
 | 
						|
	free(msg->msg_iov);
 | 
						|
	msg->msg_iov = NULL;
 | 
						|
	msg->msg_iovlen = 0;
 | 
						|
}
 | 
						|
 | 
						|
static int msg_alloc_iov(struct msghdr *msg,
 | 
						|
			 int iov_count, int iov_length,
 | 
						|
			 bool data, bool xmit)
 | 
						|
{
 | 
						|
	unsigned char k = 0;
 | 
						|
	struct iovec *iov;
 | 
						|
	int i;
 | 
						|
 | 
						|
	iov = calloc(iov_count, sizeof(struct iovec));
 | 
						|
	if (!iov)
 | 
						|
		return errno;
 | 
						|
 | 
						|
	for (i = 0; i < iov_count; i++) {
 | 
						|
		unsigned char *d = calloc(iov_length, sizeof(char));
 | 
						|
 | 
						|
		if (!d) {
 | 
						|
			fprintf(stderr, "iov_count %i/%i OOM\n", i, iov_count);
 | 
						|
			goto unwind_iov;
 | 
						|
		}
 | 
						|
		iov[i].iov_base = d;
 | 
						|
		iov[i].iov_len = iov_length;
 | 
						|
 | 
						|
		if (data && xmit) {
 | 
						|
			int j;
 | 
						|
 | 
						|
			for (j = 0; j < iov_length; j++)
 | 
						|
				d[j] = k++;
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	msg->msg_iov = iov;
 | 
						|
	msg->msg_iovlen = iov_count;
 | 
						|
 | 
						|
	return 0;
 | 
						|
unwind_iov:
 | 
						|
	for (i--; i >= 0 ; i--)
 | 
						|
		free(msg->msg_iov[i].iov_base);
 | 
						|
	return -ENOMEM;
 | 
						|
}
 | 
						|
 | 
						|
static int msg_verify_data(struct msghdr *msg, int size, int chunk_sz)
 | 
						|
{
 | 
						|
	int i, j, bytes_cnt = 0;
 | 
						|
	unsigned char k = 0;
 | 
						|
 | 
						|
	for (i = 0; i < msg->msg_iovlen; i++) {
 | 
						|
		unsigned char *d = msg->msg_iov[i].iov_base;
 | 
						|
 | 
						|
		for (j = 0;
 | 
						|
		     j < msg->msg_iov[i].iov_len && size; j++) {
 | 
						|
			if (d[j] != k++) {
 | 
						|
				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;
 | 
						|
			}
 | 
						|
			bytes_cnt++;
 | 
						|
			if (bytes_cnt == chunk_sz) {
 | 
						|
				k = 0;
 | 
						|
				bytes_cnt = 0;
 | 
						|
			}
 | 
						|
			size--;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static int msg_loop(int fd, int iov_count, int iov_length, int cnt,
 | 
						|
		    struct msg_stats *s, bool tx,
 | 
						|
		    struct sockmap_options *opt)
 | 
						|
{
 | 
						|
	struct msghdr msg = {0}, msg_peek = {0};
 | 
						|
	int err, i, flags = MSG_NOSIGNAL;
 | 
						|
	bool drop = opt->drop_expected;
 | 
						|
	bool data = opt->data_test;
 | 
						|
 | 
						|
	err = msg_alloc_iov(&msg, iov_count, iov_length, data, tx);
 | 
						|
	if (err)
 | 
						|
		goto out_errno;
 | 
						|
	if (peek_flag) {
 | 
						|
		err = msg_alloc_iov(&msg_peek, iov_count, iov_length, data, tx);
 | 
						|
		if (err)
 | 
						|
			goto out_errno;
 | 
						|
	}
 | 
						|
 | 
						|
	if (tx) {
 | 
						|
		clock_gettime(CLOCK_MONOTONIC, &s->start);
 | 
						|
		for (i = 0; i < cnt; i++) {
 | 
						|
			int sent = sendmsg(fd, &msg, flags);
 | 
						|
 | 
						|
			if (!drop && sent < 0) {
 | 
						|
				perror("send loop error:");
 | 
						|
				goto out_errno;
 | 
						|
			} else if (drop && sent >= 0) {
 | 
						|
				printf("send loop error expected: %i\n", sent);
 | 
						|
				errno = -EIO;
 | 
						|
				goto out_errno;
 | 
						|
			}
 | 
						|
			if (sent > 0)
 | 
						|
				s->bytes_sent += sent;
 | 
						|
		}
 | 
						|
		clock_gettime(CLOCK_MONOTONIC, &s->end);
 | 
						|
	} else {
 | 
						|
		int slct, recvp = 0, recv, max_fd = fd;
 | 
						|
		float total_bytes, txmsg_pop_total;
 | 
						|
		int fd_flags = O_NONBLOCK;
 | 
						|
		struct timeval timeout;
 | 
						|
		fd_set w;
 | 
						|
 | 
						|
		fcntl(fd, fd_flags);
 | 
						|
		/* Account for pop bytes noting each iteration of apply will
 | 
						|
		 * call msg_pop_data helper so we need to account for this
 | 
						|
		 * by calculating the number of apply iterations. Note user
 | 
						|
		 * of the tool can create cases where no data is sent by
 | 
						|
		 * manipulating pop/push/pull/etc. For example txmsg_apply 1
 | 
						|
		 * with txmsg_pop 1 will try to apply 1B at a time but each
 | 
						|
		 * iteration will then pop 1B so no data will ever be sent.
 | 
						|
		 * This is really only useful for testing edge cases in code
 | 
						|
		 * paths.
 | 
						|
		 */
 | 
						|
		total_bytes = (float)iov_count * (float)iov_length * (float)cnt;
 | 
						|
		txmsg_pop_total = txmsg_pop;
 | 
						|
		if (txmsg_apply)
 | 
						|
			txmsg_pop_total *= (total_bytes / txmsg_apply);
 | 
						|
		total_bytes -= txmsg_pop_total;
 | 
						|
		err = clock_gettime(CLOCK_MONOTONIC, &s->start);
 | 
						|
		if (err < 0)
 | 
						|
			perror("recv start time: ");
 | 
						|
		while (s->bytes_recvd < total_bytes) {
 | 
						|
			if (txmsg_cork) {
 | 
						|
				timeout.tv_sec = 0;
 | 
						|
				timeout.tv_usec = 300000;
 | 
						|
			} else {
 | 
						|
				timeout.tv_sec = 3;
 | 
						|
				timeout.tv_usec = 0;
 | 
						|
			}
 | 
						|
 | 
						|
			/* FD sets */
 | 
						|
			FD_ZERO(&w);
 | 
						|
			FD_SET(fd, &w);
 | 
						|
 | 
						|
			slct = select(max_fd + 1, &w, NULL, NULL, &timeout);
 | 
						|
			if (slct == -1) {
 | 
						|
				perror("select()");
 | 
						|
				clock_gettime(CLOCK_MONOTONIC, &s->end);
 | 
						|
				goto out_errno;
 | 
						|
			} else if (!slct) {
 | 
						|
				if (opt->verbose)
 | 
						|
					fprintf(stderr, "unexpected timeout: recved %zu/%f pop_total %f\n", s->bytes_recvd, total_bytes, txmsg_pop_total);
 | 
						|
				errno = -EIO;
 | 
						|
				clock_gettime(CLOCK_MONOTONIC, &s->end);
 | 
						|
				goto out_errno;
 | 
						|
			}
 | 
						|
 | 
						|
			errno = 0;
 | 
						|
			if (peek_flag) {
 | 
						|
				flags |= MSG_PEEK;
 | 
						|
				recvp = recvmsg(fd, &msg_peek, flags);
 | 
						|
				if (recvp < 0) {
 | 
						|
					if (errno != EWOULDBLOCK) {
 | 
						|
						clock_gettime(CLOCK_MONOTONIC, &s->end);
 | 
						|
						goto out_errno;
 | 
						|
					}
 | 
						|
				}
 | 
						|
				flags = 0;
 | 
						|
			}
 | 
						|
 | 
						|
			recv = recvmsg(fd, &msg, flags);
 | 
						|
			if (recv < 0) {
 | 
						|
				if (errno != EWOULDBLOCK) {
 | 
						|
					clock_gettime(CLOCK_MONOTONIC, &s->end);
 | 
						|
					perror("recv failed()\n");
 | 
						|
					goto out_errno;
 | 
						|
				}
 | 
						|
			}
 | 
						|
 | 
						|
			s->bytes_recvd += recv;
 | 
						|
 | 
						|
			if (data) {
 | 
						|
				int chunk_sz = opt->sendpage ?
 | 
						|
						iov_length * cnt :
 | 
						|
						iov_length * iov_count;
 | 
						|
 | 
						|
				errno = msg_verify_data(&msg, recv, chunk_sz);
 | 
						|
				if (errno) {
 | 
						|
					perror("data verify msg failed\n");
 | 
						|
					goto out_errno;
 | 
						|
				}
 | 
						|
				if (recvp) {
 | 
						|
					errno = msg_verify_data(&msg_peek,
 | 
						|
								recvp,
 | 
						|
								chunk_sz);
 | 
						|
					if (errno) {
 | 
						|
						perror("data verify msg_peek failed\n");
 | 
						|
						goto out_errno;
 | 
						|
					}
 | 
						|
				}
 | 
						|
			}
 | 
						|
		}
 | 
						|
		clock_gettime(CLOCK_MONOTONIC, &s->end);
 | 
						|
	}
 | 
						|
 | 
						|
	msg_free_iov(&msg);
 | 
						|
	msg_free_iov(&msg_peek);
 | 
						|
	return err;
 | 
						|
out_errno:
 | 
						|
	msg_free_iov(&msg);
 | 
						|
	msg_free_iov(&msg_peek);
 | 
						|
	return errno;
 | 
						|
}
 | 
						|
 | 
						|
static float giga = 1000000000;
 | 
						|
 | 
						|
static inline float sentBps(struct msg_stats s)
 | 
						|
{
 | 
						|
	return s.bytes_sent / (s.end.tv_sec - s.start.tv_sec);
 | 
						|
}
 | 
						|
 | 
						|
static inline float recvdBps(struct msg_stats s)
 | 
						|
{
 | 
						|
	return s.bytes_recvd / (s.end.tv_sec - s.start.tv_sec);
 | 
						|
}
 | 
						|
 | 
						|
static int sendmsg_test(struct sockmap_options *opt)
 | 
						|
{
 | 
						|
	float sent_Bps = 0, recvd_Bps = 0;
 | 
						|
	int rx_fd, txpid, rxpid, err = 0;
 | 
						|
	struct msg_stats s = {0};
 | 
						|
	int iov_count = opt->iov_count;
 | 
						|
	int iov_buf = opt->iov_length;
 | 
						|
	int rx_status, tx_status;
 | 
						|
	int cnt = opt->rate;
 | 
						|
 | 
						|
	errno = 0;
 | 
						|
 | 
						|
	if (opt->base)
 | 
						|
		rx_fd = p1;
 | 
						|
	else
 | 
						|
		rx_fd = p2;
 | 
						|
 | 
						|
	if (ktls) {
 | 
						|
		/* Redirecting into non-TLS socket which sends into a TLS
 | 
						|
		 * socket is not a valid test. So in this case lets not
 | 
						|
		 * enable kTLS but still run the test.
 | 
						|
		 */
 | 
						|
		if (!txmsg_redir || (txmsg_redir && txmsg_ingress)) {
 | 
						|
			err = sockmap_init_ktls(opt->verbose, rx_fd);
 | 
						|
			if (err)
 | 
						|
				return err;
 | 
						|
		}
 | 
						|
		err = sockmap_init_ktls(opt->verbose, c1);
 | 
						|
		if (err)
 | 
						|
			return err;
 | 
						|
	}
 | 
						|
 | 
						|
	rxpid = fork();
 | 
						|
	if (rxpid == 0) {
 | 
						|
		if (opt->drop_expected)
 | 
						|
			exit(0);
 | 
						|
 | 
						|
		if (opt->sendpage)
 | 
						|
			iov_count = 1;
 | 
						|
		err = msg_loop(rx_fd, iov_count, iov_buf,
 | 
						|
			       cnt, &s, false, opt);
 | 
						|
		if (opt->verbose)
 | 
						|
			fprintf(stderr,
 | 
						|
				"msg_loop_rx: iov_count %i iov_buf %i cnt %i err %i\n",
 | 
						|
				iov_count, iov_buf, cnt, err);
 | 
						|
		if (s.end.tv_sec - s.start.tv_sec) {
 | 
						|
			sent_Bps = sentBps(s);
 | 
						|
			recvd_Bps = recvdBps(s);
 | 
						|
		}
 | 
						|
		if (opt->verbose)
 | 
						|
			fprintf(stdout,
 | 
						|
				"rx_sendmsg: TX: %zuB %fB/s %fGB/s RX: %zuB %fB/s %fGB/s %s\n",
 | 
						|
				s.bytes_sent, sent_Bps, sent_Bps/giga,
 | 
						|
				s.bytes_recvd, recvd_Bps, recvd_Bps/giga,
 | 
						|
				peek_flag ? "(peek_msg)" : "");
 | 
						|
		if (err && txmsg_cork)
 | 
						|
			err = 0;
 | 
						|
		exit(err ? 1 : 0);
 | 
						|
	} else if (rxpid == -1) {
 | 
						|
		perror("msg_loop_rx: ");
 | 
						|
		return errno;
 | 
						|
	}
 | 
						|
 | 
						|
	txpid = fork();
 | 
						|
	if (txpid == 0) {
 | 
						|
		if (opt->sendpage)
 | 
						|
			err = msg_loop_sendpage(c1, iov_buf, cnt, &s, opt);
 | 
						|
		else
 | 
						|
			err = msg_loop(c1, iov_count, iov_buf,
 | 
						|
				       cnt, &s, true, opt);
 | 
						|
 | 
						|
		if (err)
 | 
						|
			fprintf(stderr,
 | 
						|
				"msg_loop_tx: iov_count %i iov_buf %i cnt %i err %i\n",
 | 
						|
				iov_count, iov_buf, cnt, err);
 | 
						|
		if (s.end.tv_sec - s.start.tv_sec) {
 | 
						|
			sent_Bps = sentBps(s);
 | 
						|
			recvd_Bps = recvdBps(s);
 | 
						|
		}
 | 
						|
		if (opt->verbose)
 | 
						|
			fprintf(stdout,
 | 
						|
				"tx_sendmsg: TX: %zuB %fB/s %f GB/s RX: %zuB %fB/s %fGB/s\n",
 | 
						|
				s.bytes_sent, sent_Bps, sent_Bps/giga,
 | 
						|
				s.bytes_recvd, recvd_Bps, recvd_Bps/giga);
 | 
						|
		exit(err ? 1 : 0);
 | 
						|
	} else if (txpid == -1) {
 | 
						|
		perror("msg_loop_tx: ");
 | 
						|
		return errno;
 | 
						|
	}
 | 
						|
 | 
						|
	assert(waitpid(rxpid, &rx_status, 0) == rxpid);
 | 
						|
	assert(waitpid(txpid, &tx_status, 0) == txpid);
 | 
						|
	if (WIFEXITED(rx_status)) {
 | 
						|
		err = WEXITSTATUS(rx_status);
 | 
						|
		if (err) {
 | 
						|
			fprintf(stderr, "rx thread exited with err %d. ", err);
 | 
						|
			goto out;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	if (WIFEXITED(tx_status)) {
 | 
						|
		err = WEXITSTATUS(tx_status);
 | 
						|
		if (err)
 | 
						|
			fprintf(stderr, "tx thread exited with err %d. ", err);
 | 
						|
	}
 | 
						|
out:
 | 
						|
	return err;
 | 
						|
}
 | 
						|
 | 
						|
static int forever_ping_pong(int rate, struct sockmap_options *opt)
 | 
						|
{
 | 
						|
	struct timeval timeout;
 | 
						|
	char buf[1024] = {0};
 | 
						|
	int sc;
 | 
						|
 | 
						|
	timeout.tv_sec = 10;
 | 
						|
	timeout.tv_usec = 0;
 | 
						|
 | 
						|
	/* Ping/Pong data from client to server */
 | 
						|
	sc = send(c1, buf, sizeof(buf), 0);
 | 
						|
	if (sc < 0) {
 | 
						|
		perror("send failed()\n");
 | 
						|
		return sc;
 | 
						|
	}
 | 
						|
 | 
						|
	do {
 | 
						|
		int s, rc, i, max_fd = p2;
 | 
						|
		fd_set w;
 | 
						|
 | 
						|
		/* FD sets */
 | 
						|
		FD_ZERO(&w);
 | 
						|
		FD_SET(c1, &w);
 | 
						|
		FD_SET(c2, &w);
 | 
						|
		FD_SET(p1, &w);
 | 
						|
		FD_SET(p2, &w);
 | 
						|
 | 
						|
		s = select(max_fd + 1, &w, NULL, NULL, &timeout);
 | 
						|
		if (s == -1) {
 | 
						|
			perror("select()");
 | 
						|
			break;
 | 
						|
		} else if (!s) {
 | 
						|
			fprintf(stderr, "unexpected timeout\n");
 | 
						|
			break;
 | 
						|
		}
 | 
						|
 | 
						|
		for (i = 0; i <= max_fd && s > 0; ++i) {
 | 
						|
			if (!FD_ISSET(i, &w))
 | 
						|
				continue;
 | 
						|
 | 
						|
			s--;
 | 
						|
 | 
						|
			rc = recv(i, buf, sizeof(buf), 0);
 | 
						|
			if (rc < 0) {
 | 
						|
				if (errno != EWOULDBLOCK) {
 | 
						|
					perror("recv failed()\n");
 | 
						|
					return rc;
 | 
						|
				}
 | 
						|
			}
 | 
						|
 | 
						|
			if (rc == 0) {
 | 
						|
				close(i);
 | 
						|
				break;
 | 
						|
			}
 | 
						|
 | 
						|
			sc = send(i, buf, rc, 0);
 | 
						|
			if (sc < 0) {
 | 
						|
				perror("send failed()\n");
 | 
						|
				return sc;
 | 
						|
			}
 | 
						|
		}
 | 
						|
 | 
						|
		if (rate)
 | 
						|
			sleep(rate);
 | 
						|
 | 
						|
		if (opt->verbose) {
 | 
						|
			printf(".");
 | 
						|
			fflush(stdout);
 | 
						|
 | 
						|
		}
 | 
						|
	} while (running);
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
enum {
 | 
						|
	PING_PONG,
 | 
						|
	SENDMSG,
 | 
						|
	BASE,
 | 
						|
	BASE_SENDPAGE,
 | 
						|
	SENDPAGE,
 | 
						|
};
 | 
						|
 | 
						|
static int run_options(struct sockmap_options *options, int cg_fd,  int test)
 | 
						|
{
 | 
						|
	int i, key, next_key, err, tx_prog_fd = -1, zero = 0;
 | 
						|
 | 
						|
	/* If base test skip BPF setup */
 | 
						|
	if (test == BASE || test == BASE_SENDPAGE)
 | 
						|
		goto run;
 | 
						|
 | 
						|
	/* Attach programs to sockmap */
 | 
						|
	err = bpf_prog_attach(prog_fd[0], map_fd[0],
 | 
						|
				BPF_SK_SKB_STREAM_PARSER, 0);
 | 
						|
	if (err) {
 | 
						|
		fprintf(stderr,
 | 
						|
			"ERROR: bpf_prog_attach (sockmap %i->%i): %d (%s)\n",
 | 
						|
			prog_fd[0], map_fd[0], err, strerror(errno));
 | 
						|
		return err;
 | 
						|
	}
 | 
						|
 | 
						|
	err = bpf_prog_attach(prog_fd[1], map_fd[0],
 | 
						|
				BPF_SK_SKB_STREAM_VERDICT, 0);
 | 
						|
	if (err) {
 | 
						|
		fprintf(stderr, "ERROR: bpf_prog_attach (sockmap): %d (%s)\n",
 | 
						|
			err, strerror(errno));
 | 
						|
		return err;
 | 
						|
	}
 | 
						|
 | 
						|
	/* Attach to cgroups */
 | 
						|
	err = bpf_prog_attach(prog_fd[2], cg_fd, BPF_CGROUP_SOCK_OPS, 0);
 | 
						|
	if (err) {
 | 
						|
		fprintf(stderr, "ERROR: bpf_prog_attach (groups): %d (%s)\n",
 | 
						|
			err, strerror(errno));
 | 
						|
		return err;
 | 
						|
	}
 | 
						|
 | 
						|
run:
 | 
						|
	err = sockmap_init_sockets(options->verbose);
 | 
						|
	if (err) {
 | 
						|
		fprintf(stderr, "ERROR: test socket failed: %d\n", err);
 | 
						|
		goto out;
 | 
						|
	}
 | 
						|
 | 
						|
	/* Attach txmsg program to sockmap */
 | 
						|
	if (txmsg_pass)
 | 
						|
		tx_prog_fd = prog_fd[3];
 | 
						|
	else if (txmsg_noisy)
 | 
						|
		tx_prog_fd = prog_fd[4];
 | 
						|
	else if (txmsg_redir)
 | 
						|
		tx_prog_fd = prog_fd[5];
 | 
						|
	else if (txmsg_redir_noisy)
 | 
						|
		tx_prog_fd = prog_fd[6];
 | 
						|
	else if (txmsg_drop)
 | 
						|
		tx_prog_fd = prog_fd[9];
 | 
						|
	/* apply and cork must be last */
 | 
						|
	else if (txmsg_apply)
 | 
						|
		tx_prog_fd = prog_fd[7];
 | 
						|
	else if (txmsg_cork)
 | 
						|
		tx_prog_fd = prog_fd[8];
 | 
						|
	else
 | 
						|
		tx_prog_fd = 0;
 | 
						|
 | 
						|
	if (tx_prog_fd) {
 | 
						|
		int redir_fd, i = 0;
 | 
						|
 | 
						|
		err = bpf_prog_attach(tx_prog_fd,
 | 
						|
				      map_fd[1], BPF_SK_MSG_VERDICT, 0);
 | 
						|
		if (err) {
 | 
						|
			fprintf(stderr,
 | 
						|
				"ERROR: bpf_prog_attach (txmsg): %d (%s)\n",
 | 
						|
				err, strerror(errno));
 | 
						|
			goto out;
 | 
						|
		}
 | 
						|
 | 
						|
		err = bpf_map_update_elem(map_fd[1], &i, &c1, BPF_ANY);
 | 
						|
		if (err) {
 | 
						|
			fprintf(stderr,
 | 
						|
				"ERROR: bpf_map_update_elem (txmsg):  %d (%s\n",
 | 
						|
				err, strerror(errno));
 | 
						|
			goto out;
 | 
						|
		}
 | 
						|
 | 
						|
		if (txmsg_redir || txmsg_redir_noisy)
 | 
						|
			redir_fd = c2;
 | 
						|
		else
 | 
						|
			redir_fd = c1;
 | 
						|
 | 
						|
		err = bpf_map_update_elem(map_fd[2], &i, &redir_fd, BPF_ANY);
 | 
						|
		if (err) {
 | 
						|
			fprintf(stderr,
 | 
						|
				"ERROR: bpf_map_update_elem (txmsg):  %d (%s\n",
 | 
						|
				err, strerror(errno));
 | 
						|
			goto out;
 | 
						|
		}
 | 
						|
 | 
						|
		if (txmsg_apply) {
 | 
						|
			err = bpf_map_update_elem(map_fd[3],
 | 
						|
						  &i, &txmsg_apply, BPF_ANY);
 | 
						|
			if (err) {
 | 
						|
				fprintf(stderr,
 | 
						|
					"ERROR: bpf_map_update_elem (apply_bytes):  %d (%s\n",
 | 
						|
					err, strerror(errno));
 | 
						|
				goto out;
 | 
						|
			}
 | 
						|
		}
 | 
						|
 | 
						|
		if (txmsg_cork) {
 | 
						|
			err = bpf_map_update_elem(map_fd[4],
 | 
						|
						  &i, &txmsg_cork, BPF_ANY);
 | 
						|
			if (err) {
 | 
						|
				fprintf(stderr,
 | 
						|
					"ERROR: bpf_map_update_elem (cork_bytes):  %d (%s\n",
 | 
						|
					err, strerror(errno));
 | 
						|
				goto out;
 | 
						|
			}
 | 
						|
		}
 | 
						|
 | 
						|
		if (txmsg_start) {
 | 
						|
			err = bpf_map_update_elem(map_fd[5],
 | 
						|
						  &i, &txmsg_start, BPF_ANY);
 | 
						|
			if (err) {
 | 
						|
				fprintf(stderr,
 | 
						|
					"ERROR: bpf_map_update_elem (txmsg_start):  %d (%s)\n",
 | 
						|
					err, strerror(errno));
 | 
						|
				goto out;
 | 
						|
			}
 | 
						|
		}
 | 
						|
 | 
						|
		if (txmsg_end) {
 | 
						|
			i = 1;
 | 
						|
			err = bpf_map_update_elem(map_fd[5],
 | 
						|
						  &i, &txmsg_end, BPF_ANY);
 | 
						|
			if (err) {
 | 
						|
				fprintf(stderr,
 | 
						|
					"ERROR: bpf_map_update_elem (txmsg_end):  %d (%s)\n",
 | 
						|
					err, strerror(errno));
 | 
						|
				goto out;
 | 
						|
			}
 | 
						|
		}
 | 
						|
 | 
						|
		if (txmsg_start_push) {
 | 
						|
			i = 2;
 | 
						|
			err = bpf_map_update_elem(map_fd[5],
 | 
						|
						  &i, &txmsg_start_push, BPF_ANY);
 | 
						|
			if (err) {
 | 
						|
				fprintf(stderr,
 | 
						|
					"ERROR: bpf_map_update_elem (txmsg_start_push):  %d (%s)\n",
 | 
						|
					err, strerror(errno));
 | 
						|
				goto out;
 | 
						|
			}
 | 
						|
		}
 | 
						|
 | 
						|
		if (txmsg_end_push) {
 | 
						|
			i = 3;
 | 
						|
			err = bpf_map_update_elem(map_fd[5],
 | 
						|
						  &i, &txmsg_end_push, BPF_ANY);
 | 
						|
			if (err) {
 | 
						|
				fprintf(stderr,
 | 
						|
					"ERROR: bpf_map_update_elem %i@%i (txmsg_end_push):  %d (%s)\n",
 | 
						|
					txmsg_end_push, i, err, strerror(errno));
 | 
						|
				goto out;
 | 
						|
			}
 | 
						|
		}
 | 
						|
 | 
						|
		if (txmsg_start_pop) {
 | 
						|
			i = 4;
 | 
						|
			err = bpf_map_update_elem(map_fd[5],
 | 
						|
						  &i, &txmsg_start_pop, BPF_ANY);
 | 
						|
			if (err) {
 | 
						|
				fprintf(stderr,
 | 
						|
					"ERROR: bpf_map_update_elem %i@%i (txmsg_start_pop):  %d (%s)\n",
 | 
						|
					txmsg_start_pop, i, err, strerror(errno));
 | 
						|
				goto out;
 | 
						|
			}
 | 
						|
		} else {
 | 
						|
			i = 4;
 | 
						|
			bpf_map_update_elem(map_fd[5],
 | 
						|
						  &i, &txmsg_start_pop, BPF_ANY);
 | 
						|
		}
 | 
						|
 | 
						|
		if (txmsg_pop) {
 | 
						|
			i = 5;
 | 
						|
			err = bpf_map_update_elem(map_fd[5],
 | 
						|
						  &i, &txmsg_pop, BPF_ANY);
 | 
						|
			if (err) {
 | 
						|
				fprintf(stderr,
 | 
						|
					"ERROR: bpf_map_update_elem %i@%i (txmsg_pop):  %d (%s)\n",
 | 
						|
					txmsg_pop, i, err, strerror(errno));
 | 
						|
				goto out;
 | 
						|
			}
 | 
						|
		} else {
 | 
						|
			i = 5;
 | 
						|
			bpf_map_update_elem(map_fd[5],
 | 
						|
					    &i, &txmsg_pop, BPF_ANY);
 | 
						|
 | 
						|
		}
 | 
						|
 | 
						|
		if (txmsg_ingress) {
 | 
						|
			int in = BPF_F_INGRESS;
 | 
						|
 | 
						|
			i = 0;
 | 
						|
			err = bpf_map_update_elem(map_fd[6], &i, &in, BPF_ANY);
 | 
						|
			if (err) {
 | 
						|
				fprintf(stderr,
 | 
						|
					"ERROR: bpf_map_update_elem (txmsg_ingress): %d (%s)\n",
 | 
						|
					err, strerror(errno));
 | 
						|
			}
 | 
						|
			i = 1;
 | 
						|
			err = bpf_map_update_elem(map_fd[1], &i, &p1, BPF_ANY);
 | 
						|
			if (err) {
 | 
						|
				fprintf(stderr,
 | 
						|
					"ERROR: bpf_map_update_elem (p1 txmsg): %d (%s)\n",
 | 
						|
					err, strerror(errno));
 | 
						|
			}
 | 
						|
			err = bpf_map_update_elem(map_fd[2], &i, &p1, BPF_ANY);
 | 
						|
			if (err) {
 | 
						|
				fprintf(stderr,
 | 
						|
					"ERROR: bpf_map_update_elem (p1 redir): %d (%s)\n",
 | 
						|
					err, strerror(errno));
 | 
						|
			}
 | 
						|
 | 
						|
			i = 2;
 | 
						|
			err = bpf_map_update_elem(map_fd[2], &i, &p2, BPF_ANY);
 | 
						|
			if (err) {
 | 
						|
				fprintf(stderr,
 | 
						|
					"ERROR: bpf_map_update_elem (p2 txmsg): %d (%s)\n",
 | 
						|
					err, strerror(errno));
 | 
						|
			}
 | 
						|
		}
 | 
						|
 | 
						|
		if (txmsg_skb) {
 | 
						|
			int skb_fd = (test == SENDMSG || test == SENDPAGE) ?
 | 
						|
					p2 : p1;
 | 
						|
			int ingress = BPF_F_INGRESS;
 | 
						|
 | 
						|
			i = 0;
 | 
						|
			err = bpf_map_update_elem(map_fd[7],
 | 
						|
						  &i, &ingress, BPF_ANY);
 | 
						|
			if (err) {
 | 
						|
				fprintf(stderr,
 | 
						|
					"ERROR: bpf_map_update_elem (txmsg_ingress): %d (%s)\n",
 | 
						|
					err, strerror(errno));
 | 
						|
			}
 | 
						|
 | 
						|
			i = 3;
 | 
						|
			err = bpf_map_update_elem(map_fd[0],
 | 
						|
						  &i, &skb_fd, BPF_ANY);
 | 
						|
			if (err) {
 | 
						|
				fprintf(stderr,
 | 
						|
					"ERROR: bpf_map_update_elem (c1 sockmap): %d (%s)\n",
 | 
						|
					err, strerror(errno));
 | 
						|
			}
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	if (txmsg_drop)
 | 
						|
		options->drop_expected = true;
 | 
						|
 | 
						|
	if (test == PING_PONG)
 | 
						|
		err = forever_ping_pong(options->rate, options);
 | 
						|
	else if (test == SENDMSG) {
 | 
						|
		options->base = false;
 | 
						|
		options->sendpage = false;
 | 
						|
		err = sendmsg_test(options);
 | 
						|
	} else if (test == SENDPAGE) {
 | 
						|
		options->base = false;
 | 
						|
		options->sendpage = true;
 | 
						|
		err = sendmsg_test(options);
 | 
						|
	} else if (test == BASE) {
 | 
						|
		options->base = true;
 | 
						|
		options->sendpage = false;
 | 
						|
		err = sendmsg_test(options);
 | 
						|
	} else if (test == BASE_SENDPAGE) {
 | 
						|
		options->base = true;
 | 
						|
		options->sendpage = true;
 | 
						|
		err = sendmsg_test(options);
 | 
						|
	} else
 | 
						|
		fprintf(stderr, "unknown test\n");
 | 
						|
out:
 | 
						|
	/* Detatch and zero all the maps */
 | 
						|
	bpf_prog_detach2(prog_fd[2], cg_fd, BPF_CGROUP_SOCK_OPS);
 | 
						|
	bpf_prog_detach2(prog_fd[0], map_fd[0], BPF_SK_SKB_STREAM_PARSER);
 | 
						|
	bpf_prog_detach2(prog_fd[1], map_fd[0], BPF_SK_SKB_STREAM_VERDICT);
 | 
						|
	if (tx_prog_fd >= 0)
 | 
						|
		bpf_prog_detach2(tx_prog_fd, map_fd[1], BPF_SK_MSG_VERDICT);
 | 
						|
 | 
						|
	for (i = 0; i < 8; i++) {
 | 
						|
		key = next_key = 0;
 | 
						|
		bpf_map_update_elem(map_fd[i], &key, &zero, BPF_ANY);
 | 
						|
		while (bpf_map_get_next_key(map_fd[i], &key, &next_key) == 0) {
 | 
						|
			bpf_map_update_elem(map_fd[i], &key, &zero, BPF_ANY);
 | 
						|
			key = next_key;
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	close(s1);
 | 
						|
	close(s2);
 | 
						|
	close(p1);
 | 
						|
	close(p2);
 | 
						|
	close(c1);
 | 
						|
	close(c2);
 | 
						|
	return err;
 | 
						|
}
 | 
						|
 | 
						|
static char *test_to_str(int test)
 | 
						|
{
 | 
						|
	switch (test) {
 | 
						|
	case SENDMSG:
 | 
						|
		return "sendmsg";
 | 
						|
	case SENDPAGE:
 | 
						|
		return "sendpage";
 | 
						|
	}
 | 
						|
	return "unknown";
 | 
						|
}
 | 
						|
 | 
						|
#define OPTSTRING 60
 | 
						|
static void test_options(char *options)
 | 
						|
{
 | 
						|
	char tstr[OPTSTRING];
 | 
						|
 | 
						|
	memset(options, 0, OPTSTRING);
 | 
						|
 | 
						|
	if (txmsg_pass)
 | 
						|
		strncat(options, "pass,", OPTSTRING);
 | 
						|
	if (txmsg_noisy)
 | 
						|
		strncat(options, "pass_noisy,", OPTSTRING);
 | 
						|
	if (txmsg_redir)
 | 
						|
		strncat(options, "redir,", OPTSTRING);
 | 
						|
	if (txmsg_redir_noisy)
 | 
						|
		strncat(options, "redir_noisy,", OPTSTRING);
 | 
						|
	if (txmsg_drop)
 | 
						|
		strncat(options, "drop,", OPTSTRING);
 | 
						|
	if (txmsg_apply) {
 | 
						|
		snprintf(tstr, OPTSTRING, "apply %d,", txmsg_apply);
 | 
						|
		strncat(options, tstr, OPTSTRING);
 | 
						|
	}
 | 
						|
	if (txmsg_cork) {
 | 
						|
		snprintf(tstr, OPTSTRING, "cork %d,", txmsg_cork);
 | 
						|
		strncat(options, tstr, OPTSTRING);
 | 
						|
	}
 | 
						|
	if (txmsg_start) {
 | 
						|
		snprintf(tstr, OPTSTRING, "start %d,", txmsg_start);
 | 
						|
		strncat(options, tstr, OPTSTRING);
 | 
						|
	}
 | 
						|
	if (txmsg_end) {
 | 
						|
		snprintf(tstr, OPTSTRING, "end %d,", txmsg_end);
 | 
						|
		strncat(options, tstr, OPTSTRING);
 | 
						|
	}
 | 
						|
	if (txmsg_start_pop) {
 | 
						|
		snprintf(tstr, OPTSTRING, "pop (%d,%d),",
 | 
						|
			 txmsg_start_pop, txmsg_start_pop + txmsg_pop);
 | 
						|
		strncat(options, tstr, OPTSTRING);
 | 
						|
	}
 | 
						|
	if (txmsg_ingress)
 | 
						|
		strncat(options, "ingress,", OPTSTRING);
 | 
						|
	if (txmsg_skb)
 | 
						|
		strncat(options, "skb,", OPTSTRING);
 | 
						|
	if (ktls)
 | 
						|
		strncat(options, "ktls,", OPTSTRING);
 | 
						|
	if (peek_flag)
 | 
						|
		strncat(options, "peek,", OPTSTRING);
 | 
						|
}
 | 
						|
 | 
						|
static int __test_exec(int cgrp, int test, struct sockmap_options *opt)
 | 
						|
{
 | 
						|
	char *options = calloc(OPTSTRING, sizeof(char));
 | 
						|
	int err;
 | 
						|
 | 
						|
	if (test == SENDPAGE)
 | 
						|
		opt->sendpage = true;
 | 
						|
	else
 | 
						|
		opt->sendpage = false;
 | 
						|
 | 
						|
	if (txmsg_drop)
 | 
						|
		opt->drop_expected = true;
 | 
						|
	else
 | 
						|
		opt->drop_expected = false;
 | 
						|
 | 
						|
	test_options(options);
 | 
						|
 | 
						|
	fprintf(stdout,
 | 
						|
		"[TEST %i]: (%i, %i, %i, %s, %s): ",
 | 
						|
		test_cnt, opt->rate, opt->iov_count, opt->iov_length,
 | 
						|
		test_to_str(test), options);
 | 
						|
	fflush(stdout);
 | 
						|
	err = run_options(opt, cgrp, test);
 | 
						|
	fprintf(stdout, "%s\n", !err ? "PASS" : "FAILED");
 | 
						|
	test_cnt++;
 | 
						|
	!err ? passed++ : failed++;
 | 
						|
	free(options);
 | 
						|
	return err;
 | 
						|
}
 | 
						|
 | 
						|
static int test_exec(int cgrp, struct sockmap_options *opt)
 | 
						|
{
 | 
						|
	int err = __test_exec(cgrp, SENDMSG, opt);
 | 
						|
 | 
						|
	if (err)
 | 
						|
		goto out;
 | 
						|
 | 
						|
	err = __test_exec(cgrp, SENDPAGE, opt);
 | 
						|
out:
 | 
						|
	return err;
 | 
						|
}
 | 
						|
 | 
						|
static int test_loop(int cgrp)
 | 
						|
{
 | 
						|
	struct sockmap_options opt;
 | 
						|
 | 
						|
	int err, i, l, r;
 | 
						|
 | 
						|
	opt.verbose = 0;
 | 
						|
	opt.base = false;
 | 
						|
	opt.sendpage = false;
 | 
						|
	opt.data_test = false;
 | 
						|
	opt.drop_expected = false;
 | 
						|
	opt.iov_count = 0;
 | 
						|
	opt.iov_length = 0;
 | 
						|
	opt.rate = 0;
 | 
						|
 | 
						|
	r = 1;
 | 
						|
	for (i = 1; i < 100; i += 33) {
 | 
						|
		for (l = 1; l < 100; l += 33) {
 | 
						|
			opt.rate = r;
 | 
						|
			opt.iov_count = i;
 | 
						|
			opt.iov_length = l;
 | 
						|
			err = test_exec(cgrp, &opt);
 | 
						|
			if (err)
 | 
						|
				goto out;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	sched_yield();
 | 
						|
out:
 | 
						|
	return err;
 | 
						|
}
 | 
						|
 | 
						|
static int test_txmsg(int cgrp)
 | 
						|
{
 | 
						|
	int err;
 | 
						|
 | 
						|
	txmsg_pass = txmsg_noisy = txmsg_redir_noisy = txmsg_drop = 0;
 | 
						|
	txmsg_apply = txmsg_cork = 0;
 | 
						|
	txmsg_ingress = txmsg_skb = 0;
 | 
						|
 | 
						|
	txmsg_pass = 1;
 | 
						|
	err = test_loop(cgrp);
 | 
						|
	txmsg_pass = 0;
 | 
						|
	if (err)
 | 
						|
		goto out;
 | 
						|
 | 
						|
	txmsg_redir = 1;
 | 
						|
	err = test_loop(cgrp);
 | 
						|
	txmsg_redir = 0;
 | 
						|
	if (err)
 | 
						|
		goto out;
 | 
						|
 | 
						|
	txmsg_drop = 1;
 | 
						|
	err = test_loop(cgrp);
 | 
						|
	txmsg_drop = 0;
 | 
						|
	if (err)
 | 
						|
		goto out;
 | 
						|
 | 
						|
	txmsg_redir = 1;
 | 
						|
	txmsg_ingress = 1;
 | 
						|
	err = test_loop(cgrp);
 | 
						|
	txmsg_redir = 0;
 | 
						|
	txmsg_ingress = 0;
 | 
						|
	if (err)
 | 
						|
		goto out;
 | 
						|
out:
 | 
						|
	txmsg_pass = 0;
 | 
						|
	txmsg_redir = 0;
 | 
						|
	txmsg_drop = 0;
 | 
						|
	return err;
 | 
						|
}
 | 
						|
 | 
						|
static int test_send(struct sockmap_options *opt, int cgrp)
 | 
						|
{
 | 
						|
	int err;
 | 
						|
 | 
						|
	opt->iov_length = 1;
 | 
						|
	opt->iov_count = 1;
 | 
						|
	opt->rate = 1;
 | 
						|
	err = test_exec(cgrp, opt);
 | 
						|
	if (err)
 | 
						|
		goto out;
 | 
						|
 | 
						|
	opt->iov_length = 1;
 | 
						|
	opt->iov_count = 1024;
 | 
						|
	opt->rate = 1;
 | 
						|
	err = test_exec(cgrp, opt);
 | 
						|
	if (err)
 | 
						|
		goto out;
 | 
						|
 | 
						|
	opt->iov_length = 1024;
 | 
						|
	opt->iov_count = 1;
 | 
						|
	opt->rate = 1;
 | 
						|
	err = test_exec(cgrp, opt);
 | 
						|
	if (err)
 | 
						|
		goto out;
 | 
						|
 | 
						|
	opt->iov_length = 1;
 | 
						|
	opt->iov_count = 1;
 | 
						|
	opt->rate = 512;
 | 
						|
	err = test_exec(cgrp, opt);
 | 
						|
	if (err)
 | 
						|
		goto out;
 | 
						|
 | 
						|
	opt->iov_length = 256;
 | 
						|
	opt->iov_count = 1024;
 | 
						|
	opt->rate = 2;
 | 
						|
	err = test_exec(cgrp, opt);
 | 
						|
	if (err)
 | 
						|
		goto out;
 | 
						|
 | 
						|
	opt->rate = 100;
 | 
						|
	opt->iov_count = 1;
 | 
						|
	opt->iov_length = 5;
 | 
						|
	err = test_exec(cgrp, opt);
 | 
						|
	if (err)
 | 
						|
		goto out;
 | 
						|
out:
 | 
						|
	sched_yield();
 | 
						|
	return err;
 | 
						|
}
 | 
						|
 | 
						|
static int test_mixed(int cgrp)
 | 
						|
{
 | 
						|
	struct sockmap_options opt = {0};
 | 
						|
	int err;
 | 
						|
 | 
						|
	txmsg_pass = txmsg_noisy = txmsg_redir_noisy = txmsg_drop = 0;
 | 
						|
	txmsg_apply = txmsg_cork = 0;
 | 
						|
	txmsg_start = txmsg_end = 0;
 | 
						|
	txmsg_start_push = txmsg_end_push = 0;
 | 
						|
	txmsg_start_pop = txmsg_pop = 0;
 | 
						|
 | 
						|
	/* Test small and large iov_count values with pass/redir/apply/cork */
 | 
						|
	txmsg_pass = 1;
 | 
						|
	txmsg_redir = 0;
 | 
						|
	txmsg_apply = 1;
 | 
						|
	txmsg_cork = 0;
 | 
						|
	err = test_send(&opt, cgrp);
 | 
						|
	if (err)
 | 
						|
		goto out;
 | 
						|
 | 
						|
	txmsg_pass = 1;
 | 
						|
	txmsg_redir = 0;
 | 
						|
	txmsg_apply = 0;
 | 
						|
	txmsg_cork = 1;
 | 
						|
	err = test_send(&opt, cgrp);
 | 
						|
	if (err)
 | 
						|
		goto out;
 | 
						|
 | 
						|
	txmsg_pass = 1;
 | 
						|
	txmsg_redir = 0;
 | 
						|
	txmsg_apply = 1;
 | 
						|
	txmsg_cork = 1;
 | 
						|
	err = test_send(&opt, cgrp);
 | 
						|
	if (err)
 | 
						|
		goto out;
 | 
						|
 | 
						|
	txmsg_pass = 1;
 | 
						|
	txmsg_redir = 0;
 | 
						|
	txmsg_apply = 1024;
 | 
						|
	txmsg_cork = 0;
 | 
						|
	err = test_send(&opt, cgrp);
 | 
						|
	if (err)
 | 
						|
		goto out;
 | 
						|
 | 
						|
	txmsg_pass = 1;
 | 
						|
	txmsg_redir = 0;
 | 
						|
	txmsg_apply = 0;
 | 
						|
	txmsg_cork = 1024;
 | 
						|
	err = test_send(&opt, cgrp);
 | 
						|
	if (err)
 | 
						|
		goto out;
 | 
						|
 | 
						|
	txmsg_pass = 1;
 | 
						|
	txmsg_redir = 0;
 | 
						|
	txmsg_apply = 1024;
 | 
						|
	txmsg_cork = 1024;
 | 
						|
	err = test_send(&opt, cgrp);
 | 
						|
	if (err)
 | 
						|
		goto out;
 | 
						|
 | 
						|
	txmsg_pass = 1;
 | 
						|
	txmsg_redir = 0;
 | 
						|
	txmsg_cork = 4096;
 | 
						|
	txmsg_apply = 4096;
 | 
						|
	err = test_send(&opt, cgrp);
 | 
						|
	if (err)
 | 
						|
		goto out;
 | 
						|
 | 
						|
	txmsg_pass = 0;
 | 
						|
	txmsg_redir = 1;
 | 
						|
	txmsg_apply = 1;
 | 
						|
	txmsg_cork = 0;
 | 
						|
	err = test_send(&opt, cgrp);
 | 
						|
	if (err)
 | 
						|
		goto out;
 | 
						|
 | 
						|
	txmsg_pass = 0;
 | 
						|
	txmsg_redir = 1;
 | 
						|
	txmsg_apply = 0;
 | 
						|
	txmsg_cork = 1;
 | 
						|
	err = test_send(&opt, cgrp);
 | 
						|
	if (err)
 | 
						|
		goto out;
 | 
						|
 | 
						|
	txmsg_pass = 0;
 | 
						|
	txmsg_redir = 1;
 | 
						|
	txmsg_apply = 1024;
 | 
						|
	txmsg_cork = 0;
 | 
						|
	err = test_send(&opt, cgrp);
 | 
						|
	if (err)
 | 
						|
		goto out;
 | 
						|
 | 
						|
	txmsg_pass = 0;
 | 
						|
	txmsg_redir = 1;
 | 
						|
	txmsg_apply = 0;
 | 
						|
	txmsg_cork = 1024;
 | 
						|
	err = test_send(&opt, cgrp);
 | 
						|
	if (err)
 | 
						|
		goto out;
 | 
						|
 | 
						|
	txmsg_pass = 0;
 | 
						|
	txmsg_redir = 1;
 | 
						|
	txmsg_apply = 1024;
 | 
						|
	txmsg_cork = 1024;
 | 
						|
	err = test_send(&opt, cgrp);
 | 
						|
	if (err)
 | 
						|
		goto out;
 | 
						|
 | 
						|
	txmsg_pass = 0;
 | 
						|
	txmsg_redir = 1;
 | 
						|
	txmsg_cork = 4096;
 | 
						|
	txmsg_apply = 4096;
 | 
						|
	err = test_send(&opt, cgrp);
 | 
						|
	if (err)
 | 
						|
		goto out;
 | 
						|
out:
 | 
						|
	return err;
 | 
						|
}
 | 
						|
 | 
						|
static int test_start_end(int cgrp)
 | 
						|
{
 | 
						|
	struct sockmap_options opt = {0};
 | 
						|
	int err, i;
 | 
						|
 | 
						|
	/* Test basic start/end with lots of iov_count and iov_lengths */
 | 
						|
	txmsg_start = 1;
 | 
						|
	txmsg_end = 2;
 | 
						|
	txmsg_start_push = 1;
 | 
						|
	txmsg_end_push = 2;
 | 
						|
	txmsg_start_pop = 1;
 | 
						|
	txmsg_pop = 1;
 | 
						|
	err = test_txmsg(cgrp);
 | 
						|
	if (err)
 | 
						|
		goto out;
 | 
						|
 | 
						|
	/* Cut a byte of pushed data but leave reamining in place */
 | 
						|
	txmsg_start = 1;
 | 
						|
	txmsg_end = 2;
 | 
						|
	txmsg_start_push = 1;
 | 
						|
	txmsg_end_push = 3;
 | 
						|
	txmsg_start_pop = 1;
 | 
						|
	txmsg_pop = 1;
 | 
						|
	err = test_txmsg(cgrp);
 | 
						|
	if (err)
 | 
						|
		goto out;
 | 
						|
 | 
						|
	/* Test start/end with cork */
 | 
						|
	opt.rate = 16;
 | 
						|
	opt.iov_count = 1;
 | 
						|
	opt.iov_length = 100;
 | 
						|
	txmsg_cork = 1600;
 | 
						|
 | 
						|
	txmsg_start_pop = 0;
 | 
						|
	txmsg_pop = 0;
 | 
						|
 | 
						|
	for (i = 99; i <= 1600; i += 500) {
 | 
						|
		txmsg_start = 0;
 | 
						|
		txmsg_end = i;
 | 
						|
		txmsg_start_push = 0;
 | 
						|
		txmsg_end_push = i;
 | 
						|
		err = test_exec(cgrp, &opt);
 | 
						|
		if (err)
 | 
						|
			goto out;
 | 
						|
	}
 | 
						|
 | 
						|
	/* Test pop data in middle of cork */
 | 
						|
	for (i = 99; i <= 1600; i += 500) {
 | 
						|
		txmsg_start_pop = 10;
 | 
						|
		txmsg_pop = i;
 | 
						|
		err = test_exec(cgrp, &opt);
 | 
						|
		if (err)
 | 
						|
			goto out;
 | 
						|
	}
 | 
						|
	txmsg_start_pop = 0;
 | 
						|
	txmsg_pop = 0;
 | 
						|
 | 
						|
	/* Test start/end with cork but pull data in middle */
 | 
						|
	for (i = 199; i <= 1600; i += 500) {
 | 
						|
		txmsg_start = 100;
 | 
						|
		txmsg_end = i;
 | 
						|
		txmsg_start_push = 100;
 | 
						|
		txmsg_end_push = i;
 | 
						|
		err = test_exec(cgrp, &opt);
 | 
						|
		if (err)
 | 
						|
			goto out;
 | 
						|
	}
 | 
						|
 | 
						|
	/* Test start/end with cork pulling last sg entry */
 | 
						|
	txmsg_start = 1500;
 | 
						|
	txmsg_end = 1600;
 | 
						|
	txmsg_start_push = 1500;
 | 
						|
	txmsg_end_push = 1600;
 | 
						|
	err = test_exec(cgrp, &opt);
 | 
						|
	if (err)
 | 
						|
		goto out;
 | 
						|
 | 
						|
	/* Test pop with cork pulling last sg entry */
 | 
						|
	txmsg_start_pop = 1500;
 | 
						|
	txmsg_pop = 1600;
 | 
						|
	err = test_exec(cgrp, &opt);
 | 
						|
	if (err)
 | 
						|
		goto out;
 | 
						|
	txmsg_start_pop = 0;
 | 
						|
	txmsg_pop = 0;
 | 
						|
 | 
						|
	/* Test start/end pull of single byte in last page */
 | 
						|
	txmsg_start = 1111;
 | 
						|
	txmsg_end = 1112;
 | 
						|
	txmsg_start_push = 1111;
 | 
						|
	txmsg_end_push = 1112;
 | 
						|
	err = test_exec(cgrp, &opt);
 | 
						|
	if (err)
 | 
						|
		goto out;
 | 
						|
 | 
						|
	/* Test pop of single byte in last page */
 | 
						|
	txmsg_start_pop = 1111;
 | 
						|
	txmsg_pop = 1112;
 | 
						|
	err = test_exec(cgrp, &opt);
 | 
						|
	if (err)
 | 
						|
		goto out;
 | 
						|
 | 
						|
	/* Test start/end with end < start */
 | 
						|
	txmsg_start = 1111;
 | 
						|
	txmsg_end = 0;
 | 
						|
	txmsg_start_push = 1111;
 | 
						|
	txmsg_end_push = 0;
 | 
						|
	err = test_exec(cgrp, &opt);
 | 
						|
	if (err)
 | 
						|
		goto out;
 | 
						|
 | 
						|
	/* Test start/end with end > data */
 | 
						|
	txmsg_start = 0;
 | 
						|
	txmsg_end = 1601;
 | 
						|
	txmsg_start_push = 0;
 | 
						|
	txmsg_end_push = 1601;
 | 
						|
	err = test_exec(cgrp, &opt);
 | 
						|
	if (err)
 | 
						|
		goto out;
 | 
						|
 | 
						|
	/* Test start/end with start > data */
 | 
						|
	txmsg_start = 1601;
 | 
						|
	txmsg_end = 1600;
 | 
						|
	txmsg_start_push = 1601;
 | 
						|
	txmsg_end_push = 1600;
 | 
						|
	err = test_exec(cgrp, &opt);
 | 
						|
	if (err)
 | 
						|
		goto out;
 | 
						|
 | 
						|
	/* Test pop with start > data */
 | 
						|
	txmsg_start_pop = 1601;
 | 
						|
	txmsg_pop = 1;
 | 
						|
	err = test_exec(cgrp, &opt);
 | 
						|
	if (err)
 | 
						|
		goto out;
 | 
						|
 | 
						|
	/* Test pop with pop range > data */
 | 
						|
	txmsg_start_pop = 1599;
 | 
						|
	txmsg_pop = 10;
 | 
						|
	err = test_exec(cgrp, &opt);
 | 
						|
out:
 | 
						|
	txmsg_start = 0;
 | 
						|
	txmsg_end = 0;
 | 
						|
	sched_yield();
 | 
						|
	return err;
 | 
						|
}
 | 
						|
 | 
						|
char *map_names[] = {
 | 
						|
	"sock_map",
 | 
						|
	"sock_map_txmsg",
 | 
						|
	"sock_map_redir",
 | 
						|
	"sock_apply_bytes",
 | 
						|
	"sock_cork_bytes",
 | 
						|
	"sock_bytes",
 | 
						|
	"sock_redir_flags",
 | 
						|
	"sock_skb_opts",
 | 
						|
};
 | 
						|
 | 
						|
int prog_attach_type[] = {
 | 
						|
	BPF_SK_SKB_STREAM_PARSER,
 | 
						|
	BPF_SK_SKB_STREAM_VERDICT,
 | 
						|
	BPF_CGROUP_SOCK_OPS,
 | 
						|
	BPF_SK_MSG_VERDICT,
 | 
						|
	BPF_SK_MSG_VERDICT,
 | 
						|
	BPF_SK_MSG_VERDICT,
 | 
						|
	BPF_SK_MSG_VERDICT,
 | 
						|
	BPF_SK_MSG_VERDICT,
 | 
						|
	BPF_SK_MSG_VERDICT,
 | 
						|
	BPF_SK_MSG_VERDICT,
 | 
						|
};
 | 
						|
 | 
						|
int prog_type[] = {
 | 
						|
	BPF_PROG_TYPE_SK_SKB,
 | 
						|
	BPF_PROG_TYPE_SK_SKB,
 | 
						|
	BPF_PROG_TYPE_SOCK_OPS,
 | 
						|
	BPF_PROG_TYPE_SK_MSG,
 | 
						|
	BPF_PROG_TYPE_SK_MSG,
 | 
						|
	BPF_PROG_TYPE_SK_MSG,
 | 
						|
	BPF_PROG_TYPE_SK_MSG,
 | 
						|
	BPF_PROG_TYPE_SK_MSG,
 | 
						|
	BPF_PROG_TYPE_SK_MSG,
 | 
						|
	BPF_PROG_TYPE_SK_MSG,
 | 
						|
};
 | 
						|
 | 
						|
static int populate_progs(char *bpf_file)
 | 
						|
{
 | 
						|
	struct bpf_program *prog;
 | 
						|
	struct bpf_object *obj;
 | 
						|
	int i = 0;
 | 
						|
	long err;
 | 
						|
 | 
						|
	obj = bpf_object__open(bpf_file);
 | 
						|
	err = libbpf_get_error(obj);
 | 
						|
	if (err) {
 | 
						|
		char err_buf[256];
 | 
						|
 | 
						|
		libbpf_strerror(err, err_buf, sizeof(err_buf));
 | 
						|
		printf("Unable to load eBPF objects in file '%s' : %s\n",
 | 
						|
		       bpf_file, err_buf);
 | 
						|
		return -1;
 | 
						|
	}
 | 
						|
 | 
						|
	bpf_object__for_each_program(prog, obj) {
 | 
						|
		bpf_program__set_type(prog, prog_type[i]);
 | 
						|
		bpf_program__set_expected_attach_type(prog,
 | 
						|
						      prog_attach_type[i]);
 | 
						|
		i++;
 | 
						|
	}
 | 
						|
 | 
						|
	i = bpf_object__load(obj);
 | 
						|
	i = 0;
 | 
						|
	bpf_object__for_each_program(prog, obj) {
 | 
						|
		prog_fd[i] = bpf_program__fd(prog);
 | 
						|
		i++;
 | 
						|
	}
 | 
						|
 | 
						|
	for (i = 0; i < sizeof(map_fd)/sizeof(int); i++) {
 | 
						|
		maps[i] = bpf_object__find_map_by_name(obj, map_names[i]);
 | 
						|
		map_fd[i] = bpf_map__fd(maps[i]);
 | 
						|
		if (map_fd[i] < 0) {
 | 
						|
			fprintf(stderr, "load_bpf_file: (%i) %s\n",
 | 
						|
				map_fd[i], strerror(errno));
 | 
						|
			return -1;
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static int __test_suite(int cg_fd, char *bpf_file)
 | 
						|
{
 | 
						|
	int err, cleanup = cg_fd;
 | 
						|
 | 
						|
	err = populate_progs(bpf_file);
 | 
						|
	if (err < 0) {
 | 
						|
		fprintf(stderr, "ERROR: (%i) load bpf failed\n", err);
 | 
						|
		return err;
 | 
						|
	}
 | 
						|
 | 
						|
	if (cg_fd < 0) {
 | 
						|
		if (setup_cgroup_environment()) {
 | 
						|
			fprintf(stderr, "ERROR: cgroup env failed\n");
 | 
						|
			return -EINVAL;
 | 
						|
		}
 | 
						|
 | 
						|
		cg_fd = create_and_get_cgroup(CG_PATH);
 | 
						|
		if (cg_fd < 0) {
 | 
						|
			fprintf(stderr,
 | 
						|
				"ERROR: (%i) open cg path failed: %s\n",
 | 
						|
				cg_fd, optarg);
 | 
						|
			return cg_fd;
 | 
						|
		}
 | 
						|
 | 
						|
		if (join_cgroup(CG_PATH)) {
 | 
						|
			fprintf(stderr, "ERROR: failed to join cgroup\n");
 | 
						|
			return -EINVAL;
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	/* Tests basic commands and APIs with range of iov values */
 | 
						|
	txmsg_start = txmsg_end = txmsg_start_push = txmsg_end_push = 0;
 | 
						|
	err = test_txmsg(cg_fd);
 | 
						|
	if (err)
 | 
						|
		goto out;
 | 
						|
 | 
						|
	/* Tests interesting combinations of APIs used together */
 | 
						|
	err = test_mixed(cg_fd);
 | 
						|
	if (err)
 | 
						|
		goto out;
 | 
						|
 | 
						|
	/* Tests pull_data API using start/end API */
 | 
						|
	err = test_start_end(cg_fd);
 | 
						|
	if (err)
 | 
						|
		goto out;
 | 
						|
 | 
						|
out:
 | 
						|
	printf("Summary: %i PASSED %i FAILED\n", passed, failed);
 | 
						|
	if (cleanup < 0) {
 | 
						|
		cleanup_cgroup_environment();
 | 
						|
		close(cg_fd);
 | 
						|
	}
 | 
						|
	return err;
 | 
						|
}
 | 
						|
 | 
						|
static int test_suite(int cg_fd)
 | 
						|
{
 | 
						|
	int err;
 | 
						|
 | 
						|
	err = __test_suite(cg_fd, BPF_SOCKMAP_FILENAME);
 | 
						|
	if (err)
 | 
						|
		goto out;
 | 
						|
	err = __test_suite(cg_fd, BPF_SOCKHASH_FILENAME);
 | 
						|
out:
 | 
						|
	if (cg_fd > -1)
 | 
						|
		close(cg_fd);
 | 
						|
	return err;
 | 
						|
}
 | 
						|
 | 
						|
int main(int argc, char **argv)
 | 
						|
{
 | 
						|
	int iov_count = 1, length = 1024, rate = 1;
 | 
						|
	struct sockmap_options options = {0};
 | 
						|
	int opt, longindex, err, cg_fd = 0;
 | 
						|
	char *bpf_file = BPF_SOCKMAP_FILENAME;
 | 
						|
	int test = PING_PONG;
 | 
						|
 | 
						|
	if (argc < 2)
 | 
						|
		return test_suite(-1);
 | 
						|
 | 
						|
	while ((opt = getopt_long(argc, argv, ":dhvc:r:i:l:t:p:q:",
 | 
						|
				  long_options, &longindex)) != -1) {
 | 
						|
		switch (opt) {
 | 
						|
		case 's':
 | 
						|
			txmsg_start = atoi(optarg);
 | 
						|
			break;
 | 
						|
		case 'e':
 | 
						|
			txmsg_end = atoi(optarg);
 | 
						|
			break;
 | 
						|
		case 'p':
 | 
						|
			txmsg_start_push = atoi(optarg);
 | 
						|
			break;
 | 
						|
		case 'q':
 | 
						|
			txmsg_end_push = atoi(optarg);
 | 
						|
			break;
 | 
						|
		case 'w':
 | 
						|
			txmsg_start_pop = atoi(optarg);
 | 
						|
			break;
 | 
						|
		case 'x':
 | 
						|
			txmsg_pop = atoi(optarg);
 | 
						|
			break;
 | 
						|
		case 'a':
 | 
						|
			txmsg_apply = atoi(optarg);
 | 
						|
			break;
 | 
						|
		case 'k':
 | 
						|
			txmsg_cork = atoi(optarg);
 | 
						|
			break;
 | 
						|
		case 'c':
 | 
						|
			cg_fd = open(optarg, O_DIRECTORY, O_RDONLY);
 | 
						|
			if (cg_fd < 0) {
 | 
						|
				fprintf(stderr,
 | 
						|
					"ERROR: (%i) open cg path failed: %s\n",
 | 
						|
					cg_fd, optarg);
 | 
						|
				return cg_fd;
 | 
						|
			}
 | 
						|
			break;
 | 
						|
		case 'r':
 | 
						|
			rate = atoi(optarg);
 | 
						|
			break;
 | 
						|
		case 'v':
 | 
						|
			options.verbose = 1;
 | 
						|
			break;
 | 
						|
		case 'i':
 | 
						|
			iov_count = atoi(optarg);
 | 
						|
			break;
 | 
						|
		case 'l':
 | 
						|
			length = atoi(optarg);
 | 
						|
			break;
 | 
						|
		case 'd':
 | 
						|
			options.data_test = true;
 | 
						|
			break;
 | 
						|
		case 't':
 | 
						|
			if (strcmp(optarg, "ping") == 0) {
 | 
						|
				test = PING_PONG;
 | 
						|
			} else if (strcmp(optarg, "sendmsg") == 0) {
 | 
						|
				test = SENDMSG;
 | 
						|
			} else if (strcmp(optarg, "base") == 0) {
 | 
						|
				test = BASE;
 | 
						|
			} else if (strcmp(optarg, "base_sendpage") == 0) {
 | 
						|
				test = BASE_SENDPAGE;
 | 
						|
			} else if (strcmp(optarg, "sendpage") == 0) {
 | 
						|
				test = SENDPAGE;
 | 
						|
			} else {
 | 
						|
				usage(argv);
 | 
						|
				return -1;
 | 
						|
			}
 | 
						|
			break;
 | 
						|
		case 0:
 | 
						|
			break;
 | 
						|
		case 'h':
 | 
						|
		default:
 | 
						|
			usage(argv);
 | 
						|
			return -1;
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	if (argc <= 3 && cg_fd)
 | 
						|
		return test_suite(cg_fd);
 | 
						|
 | 
						|
	if (!cg_fd) {
 | 
						|
		fprintf(stderr, "%s requires cgroup option: --cgroup <path>\n",
 | 
						|
			argv[0]);
 | 
						|
		return -1;
 | 
						|
	}
 | 
						|
 | 
						|
	err = populate_progs(bpf_file);
 | 
						|
	if (err) {
 | 
						|
		fprintf(stderr, "populate program: (%s) %s\n",
 | 
						|
			bpf_file, strerror(errno));
 | 
						|
		return 1;
 | 
						|
	}
 | 
						|
	running = 1;
 | 
						|
 | 
						|
	/* catch SIGINT */
 | 
						|
	signal(SIGINT, running_handler);
 | 
						|
 | 
						|
	options.iov_count = iov_count;
 | 
						|
	options.iov_length = length;
 | 
						|
	options.rate = rate;
 | 
						|
 | 
						|
	err = run_options(&options, cg_fd, test);
 | 
						|
	close(cg_fd);
 | 
						|
	return err;
 | 
						|
}
 | 
						|
 | 
						|
void running_handler(int a)
 | 
						|
{
 | 
						|
	running = 0;
 | 
						|
}
 |