[PATCH liburing] Add test for multiple concurrent accepts

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



Add tests for accept that queues multiple accepts and then ensures the
correct things happen.

Check that when connections arrive one at a time that only one CQE is
posted (in blocking and nonblocking sockets), as well as make sure that
closing the accept socket & cancellation all work as expected.

This relies on a kernel with [1] for the tests to pass.

[1]: https://lore.kernel.org/io-uring/20220325093755.4123343-1-dylany@xxxxxx/

Signed-off-by: Dylan Yudaken <dylany@xxxxxx>
---
 test/accept.c | 203 +++++++++++++++++++++++++++++++++++---------------
 1 file changed, 144 insertions(+), 59 deletions(-)

diff --git a/test/accept.c b/test/accept.c
index e2c6b51..c591e76 100644
--- a/test/accept.c
+++ b/test/accept.c
@@ -59,19 +59,24 @@ static void queue_recv(struct io_uring *ring, int fd, bool fixed)
 		sqe->flags |= IOSQE_FIXED_FILE;
 }
 
-static void queue_accept_conn(struct io_uring *ring, int fd, int fixed_idx)
+static void queue_accept_conn(struct io_uring *ring,
+			      int fd, int fixed_idx,
+			      int count)
 {
 	struct io_uring_sqe *sqe;
 	int ret;
 
-	sqe = io_uring_get_sqe(ring);
-	if (fixed_idx < 0)
-		io_uring_prep_accept(sqe, fd, NULL, NULL, 0);
-	else
-		io_uring_prep_accept_direct(sqe, fd, NULL, NULL, 0, fixed_idx);
+	while (count--) {
+		sqe = io_uring_get_sqe(ring);
+		if (fixed_idx < 0)
+			io_uring_prep_accept(sqe, fd, NULL, NULL, 0);
+		else
+			io_uring_prep_accept_direct(sqe, fd, NULL, NULL, 0,
+						    fixed_idx);
 
-	ret = io_uring_submit(ring);
-	assert(ret != -1);
+		ret = io_uring_submit(ring);
+		assert(ret != -1);
+	}
 }
 
 static int accept_conn(struct io_uring *ring, int fixed_idx)
@@ -131,18 +136,19 @@ struct accept_test_args {
 	bool fixed;
 	bool nonblock;
 	bool queue_accept_before_connect;
+	int extra_loops;
 };
 
-static int test(struct io_uring *ring, struct accept_test_args args)
+
+static int test_loop(struct io_uring *ring,
+		     struct accept_test_args args,
+		     int recv_s0,
+		     struct sockaddr_in *addr)
 {
 	struct io_uring_cqe *cqe;
-	struct sockaddr_in addr;
 	uint32_t head, count = 0;
 	int ret, p_fd[2], done = 0;
-
 	int32_t val;
-	int32_t recv_s0 = start_accept_listen(&addr, 0,
-					      args.nonblock ? O_NONBLOCK : 0);
 
 	p_fd[1] = socket(AF_INET, SOCK_STREAM | SOCK_CLOEXEC, IPPROTO_TCP);
 
@@ -157,10 +163,7 @@ static int test(struct io_uring *ring, struct accept_test_args args)
 	ret = fcntl(p_fd[1], F_SETFL, flags);
 	assert(ret != -1);
 
-	if (args.queue_accept_before_connect)
-		queue_accept_conn(ring, recv_s0, args.fixed ? 0 : -1);
-
-	ret = connect(p_fd[1], (struct sockaddr*)&addr, sizeof(addr));
+	ret = connect(p_fd[1], (struct sockaddr *)addr, sizeof(*addr));
 	assert(ret == -1);
 
 	flags = fcntl(p_fd[1], F_GETFL, 0);
@@ -171,7 +174,7 @@ static int test(struct io_uring *ring, struct accept_test_args args)
 	assert(ret != -1);
 
 	if (!args.queue_accept_before_connect)
-		queue_accept_conn(ring, recv_s0, args.fixed ? 0 : -1);
+		queue_accept_conn(ring, recv_s0, args.fixed ? 0 : -1, 1);
 
 	p_fd[0] = accept_conn(ring, args.fixed ? 0 : -1);
 	if (p_fd[0] == -EINVAL) {
@@ -219,16 +222,34 @@ out:
 	if (!args.fixed)
 		close(p_fd[0]);
 	close(p_fd[1]);
-	close(recv_s0);
 	return 0;
 err:
 	if (!args.fixed)
 		close(p_fd[0]);
 	close(p_fd[1]);
-	close(recv_s0);
 	return 1;
 }
 
+static int test(struct io_uring *ring, struct accept_test_args args)
+{
+	struct sockaddr_in addr;
+	int ret = 0;
+	int loop;
+	int32_t recv_s0 = start_accept_listen(&addr, 0,
+					      args.nonblock ? O_NONBLOCK : 0);
+	if (args.queue_accept_before_connect)
+		queue_accept_conn(ring, recv_s0, args.fixed ? 0 : -1,
+				  1 + args.extra_loops);
+	for (loop = 0; loop < 1 + args.extra_loops; loop++) {
+		ret = test_loop(ring, args, recv_s0, &addr);
+		if (ret)
+			break;
+	}
+
+	close(recv_s0);
+	return ret;
+}
+
 static void sig_alrm(int sig)
 {
 	exit(0);
@@ -261,10 +282,17 @@ static int test_accept_pending_on_exit(void)
 	return 0;
 }
 
+struct test_accept_many_args {
+	unsigned int usecs;
+	bool nonblock;
+	bool single_sock;
+	bool close_fds;
+};
+
 /*
  * Test issue many accepts and see if we handle cancellation on exit
  */
-static int test_accept_many(unsigned nr, unsigned usecs, bool nonblock)
+static int test_accept_many(struct test_accept_many_args args)
 {
 	struct io_uring m_io_uring;
 	struct io_uring_cqe *cqe;
@@ -272,6 +300,8 @@ static int test_accept_many(unsigned nr, unsigned usecs, bool nonblock)
 	unsigned long cur_lim;
 	struct rlimit rlim;
 	int *fds, i, ret;
+	unsigned int nr = 128;
+	int nr_socks = args.single_sock ? 1 : nr;
 
 	if (getrlimit(RLIMIT_NPROC, &rlim) < 0) {
 		perror("getrlimit");
@@ -289,28 +319,32 @@ static int test_accept_many(unsigned nr, unsigned usecs, bool nonblock)
 	ret = io_uring_queue_init(2 * nr, &m_io_uring, 0);
 	assert(ret >= 0);
 
-	fds = t_calloc(nr, sizeof(int));
+	fds = t_calloc(nr_socks, sizeof(int));
 
-	for (i = 0; i < nr; i++)
+	for (i = 0; i < nr_socks; i++)
 		fds[i] = start_accept_listen(NULL, i,
-					     nonblock ? O_NONBLOCK : 0);
+					     args.nonblock ? O_NONBLOCK : 0);
 
 	for (i = 0; i < nr; i++) {
+		int sock_idx = args.single_sock ? 0 : i;
 		sqe = io_uring_get_sqe(&m_io_uring);
-		io_uring_prep_accept(sqe, fds[i], NULL, NULL, 0);
+		io_uring_prep_accept(sqe, fds[sock_idx], NULL, NULL, 0);
 		sqe->user_data = 1 + i;
 		ret = io_uring_submit(&m_io_uring);
 		assert(ret == 1);
 	}
 
-	if (usecs)
-		usleep(usecs);
+	if (args.usecs)
+		usleep(args.usecs);
+
+	if (args.close_fds)
+		for (i = 0; i < nr_socks; i++)
+			close(fds[i]);
 
 	for (i = 0; i < nr; i++) {
 		if (io_uring_peek_cqe(&m_io_uring, &cqe))
 			break;
-		if (cqe->res != -ECANCELED &&
-		    !(cqe->res == -EAGAIN && nonblock)) {
+		if (cqe->res != -ECANCELED) {
 			fprintf(stderr, "Expected cqe to be cancelled %d\n", cqe->res);
 			ret = 1;
 			goto out;
@@ -330,7 +364,7 @@ out:
 	return ret;
 }
 
-static int test_accept_cancel(unsigned usecs)
+static int test_accept_cancel(unsigned usecs, unsigned int nr)
 {
 	struct io_uring m_io_uring;
 	struct io_uring_cqe *cqe;
@@ -342,22 +376,25 @@ static int test_accept_cancel(unsigned usecs)
 
 	fd = start_accept_listen(NULL, 0, 0);
 
-	sqe = io_uring_get_sqe(&m_io_uring);
-	io_uring_prep_accept(sqe, fd, NULL, NULL, 0);
-	sqe->user_data = 1;
-	ret = io_uring_submit(&m_io_uring);
-	assert(ret == 1);
+	for (i = 1; i <= nr; i++) {
+		sqe = io_uring_get_sqe(&m_io_uring);
+		io_uring_prep_accept(sqe, fd, NULL, NULL, 0);
+		sqe->user_data = i;
+		ret = io_uring_submit(&m_io_uring);
+		assert(ret == 1);
+	}
 
 	if (usecs)
 		usleep(usecs);
 
-	sqe = io_uring_get_sqe(&m_io_uring);
-	io_uring_prep_cancel(sqe, 1, 0);
-	sqe->user_data = 2;
-	ret = io_uring_submit(&m_io_uring);
-	assert(ret == 1);
-
-	for (i = 0; i < 2; i++) {
+	for (i = 1; i <= nr; i++) {
+		sqe = io_uring_get_sqe(&m_io_uring);
+		io_uring_prep_cancel(sqe, i, 0);
+		sqe->user_data = nr + i;
+		ret = io_uring_submit(&m_io_uring);
+		assert(ret == 1);
+	}
+	for (i = 0; i < nr * 2; i++) {
 		ret = io_uring_wait_cqe(&m_io_uring, &cqe);
 		assert(!ret);
 		/*
@@ -370,12 +407,15 @@ static int test_accept_cancel(unsigned usecs)
 		 *    should get '-EALREADY' for the cancel request and
 		 *    '-EINTR' for the accept request.
 		 */
-		if (cqe->user_data == 1) {
+		if (cqe->user_data == 0) {
+			fprintf(stderr, "unexpected 0 user data\n");
+			goto err;
+		} else if (cqe->user_data <= nr) {
 			if (cqe->res != -EINTR && cqe->res != -ECANCELED) {
 				fprintf(stderr, "Cancelled accept got %d\n", cqe->res);
 				goto err;
 			}
-		} else if (cqe->user_data == 2) {
+		} else if (cqe->user_data <= nr * 2) {
 			if (cqe->res != -EALREADY && cqe->res != 0) {
 				fprintf(stderr, "Cancel got %d\n", cqe->res);
 				goto err;
@@ -391,11 +431,14 @@ err:
 	return 1;
 }
 
-static int test_accept(void)
+static int test_accept(int count, bool before)
 {
 	struct io_uring m_io_uring;
 	int ret;
-	struct accept_test_args args = { };
+	struct accept_test_args args = {
+		.queue_accept_before_connect = before,
+		.extra_loops = count - 1
+	};
 
 	ret = io_uring_queue_init(32, &m_io_uring, 0);
 	assert(ret >= 0);
@@ -404,13 +447,14 @@ static int test_accept(void)
 	return ret;
 }
 
-static int test_accept_nonblock(bool queue_before_connect)
+static int test_accept_nonblock(bool queue_before_connect, int count)
 {
 	struct io_uring m_io_uring;
 	int ret;
 	struct accept_test_args args = {
 		.nonblock = true,
-		.queue_accept_before_connect = queue_before_connect
+		.queue_accept_before_connect = queue_before_connect,
+		.extra_loops = count - 1
 	};
 
 	ret = io_uring_queue_init(32, &m_io_uring, 0);
@@ -467,7 +511,7 @@ int main(int argc, char *argv[])
 	if (argc > 1)
 		return 0;
 
-	ret = test_accept();
+	ret = test_accept(1, false);
 	if (ret) {
 		fprintf(stderr, "test_accept failed\n");
 		return ret;
@@ -475,15 +519,33 @@ int main(int argc, char *argv[])
 	if (no_accept)
 		return 0;
 
-	ret = test_accept_nonblock(false);
+	ret = test_accept(2, false);
+	if (ret) {
+		fprintf(stderr, "test_accept(2) failed\n");
+		return ret;
+	}
+
+	ret = test_accept(2, true);
+	if (ret) {
+		fprintf(stderr, "test_accept(2, true) failed\n");
+		return ret;
+	}
+
+	ret = test_accept_nonblock(false, 1);
 	if (ret) {
 		fprintf(stderr, "test_accept_nonblock failed\n");
 		return ret;
 	}
 
-	ret = test_accept_nonblock(true);
+	ret = test_accept_nonblock(true, 1);
+	if (ret) {
+		fprintf(stderr, "test_accept_nonblock(before, 1) failed\n");
+		return ret;
+	}
+
+	ret = test_accept_nonblock(true, 3);
 	if (ret) {
-		fprintf(stderr, "test_accept_nonblock(queue_before) failed\n");
+		fprintf(stderr, "test_accept_nonblock(before,3) failed\n");
 		return ret;
 	}
 
@@ -499,33 +561,56 @@ int main(int argc, char *argv[])
 		return ret;
 	}
 
-	ret = test_accept_cancel(0);
+	ret = test_accept_cancel(0, 1);
 	if (ret) {
 		fprintf(stderr, "test_accept_cancel nodelay failed\n");
 		return ret;
 	}
 
-	ret = test_accept_cancel(10000);
+	ret = test_accept_cancel(10000, 1);
 	if (ret) {
 		fprintf(stderr, "test_accept_cancel delay failed\n");
 		return ret;
 	}
 
-	ret = test_accept_many(128, 0, false);
+	ret = test_accept_cancel(0, 4);
 	if (ret) {
-		fprintf(stderr, "test_accept_many failed\n");
+		fprintf(stderr, "test_accept_cancel nodelay failed\n");
+		return ret;
+	}
+
+	ret = test_accept_cancel(10000, 4);
+	if (ret) {
+		fprintf(stderr, "test_accept_cancel delay failed\n");
 		return ret;
 	}
 
-	ret = test_accept_many(128, 100000, false);
+	ret = test_accept_many((struct test_accept_many_args) {});
 	if (ret) {
 		fprintf(stderr, "test_accept_many failed\n");
 		return ret;
 	}
 
-	ret = test_accept_many(128, 0, true);
+	ret = test_accept_many((struct test_accept_many_args) {
+				.usecs = 100000 });
+	if (ret) {
+		fprintf(stderr, "test_accept_many(sleep) failed\n");
+		return ret;
+	}
+
+	ret = test_accept_many((struct test_accept_many_args) {
+				.nonblock = true });
+	if (ret) {
+		fprintf(stderr, "test_accept_many(nonblock) failed\n");
+		return ret;
+	}
+
+	ret = test_accept_many((struct test_accept_many_args) {
+				.nonblock = true,
+				.single_sock = true,
+				.close_fds = true });
 	if (ret) {
-		fprintf(stderr, "test_accept_many nonblocking failed\n");
+		fprintf(stderr, "test_accept_many(nonblock,close) failed\n");
 		return ret;
 	}
 

base-commit: 7a3a27b6a384f51b67f7e7086f47cf552fa70dc4
-- 
2.30.2





[Index of Archives]     [Linux Samsung SoC]     [Linux Rockchip SoC]     [Linux Actions SoC]     [Linux for Synopsys ARC Processors]     [Linux NFS]     [Linux NILFS]     [Linux USB Devel]     [Video for Linux]     [Linux Audio Users]     [Yosemite News]     [Linux Kernel]     [Linux SCSI]


  Powered by Linux