Blame test/read-write.c

Packit Service 63ea89
/* SPDX-License-Identifier: MIT */
Packit Service 63ea89
/*
Packit Service 63ea89
 * Description: basic read/write tests with buffered, O_DIRECT, and SQPOLL
Packit Service 63ea89
 */
Packit Service 63ea89
#include <errno.h>
Packit Service 63ea89
#include <stdio.h>
Packit Service 63ea89
#include <unistd.h>
Packit Service 63ea89
#include <stdlib.h>
Packit Service 63ea89
#include <string.h>
Packit Service 63ea89
#include <fcntl.h>
Packit Service 63ea89
#include <sys/types.h>
Packit Service 63ea89
#include <sys/poll.h>
Packit Service 63ea89
#include <sys/eventfd.h>
Packit Service 63ea89
#include <sys/resource.h>
Packit Service 63ea89
#include "liburing.h"
Packit Service 63ea89
Packit Service 63ea89
#define FILE_SIZE	(128 * 1024)
Packit Service 63ea89
#define BS		4096
Packit Service 63ea89
#define BUFFERS		(FILE_SIZE / BS)
Packit Service 63ea89
Packit Service 63ea89
static struct iovec *vecs;
Packit Service 63ea89
static int no_read;
Packit Service 63ea89
static int no_buf_select;
Packit Service 63ea89
static int warned;
Packit Service 63ea89
Packit Service 63ea89
static int create_buffers(void)
Packit Service 63ea89
{
Packit Service 63ea89
	int i;
Packit Service 63ea89
Packit Service 63ea89
	vecs = malloc(BUFFERS * sizeof(struct iovec));
Packit Service 63ea89
	for (i = 0; i < BUFFERS; i++) {
Packit Service 63ea89
		if (posix_memalign(&vecs[i].iov_base, BS, BS))
Packit Service 63ea89
			return 1;
Packit Service 63ea89
		vecs[i].iov_len = BS;
Packit Service 63ea89
	}
Packit Service 63ea89
Packit Service 63ea89
	return 0;
Packit Service 63ea89
}
Packit Service 63ea89
Packit Service 63ea89
static int create_file(const char *file)
Packit Service 63ea89
{
Packit Service 63ea89
	ssize_t ret;
Packit Service 63ea89
	char *buf;
Packit Service 63ea89
	int fd;
Packit Service 63ea89
Packit Service 63ea89
	buf = malloc(FILE_SIZE);
Packit Service 63ea89
	memset(buf, 0xaa, FILE_SIZE);
Packit Service 63ea89
Packit Service 63ea89
	fd = open(file, O_WRONLY | O_CREAT, 0644);
Packit Service 63ea89
	if (fd < 0) {
Packit Service 63ea89
		perror("open file");
Packit Service 63ea89
		return 1;
Packit Service 63ea89
	}
Packit Service 63ea89
	ret = write(fd, buf, FILE_SIZE);
Packit Service 63ea89
	close(fd);
Packit Service 63ea89
	return ret != FILE_SIZE;
Packit Service 63ea89
}
Packit Service 63ea89
Packit Service 63ea89
static int __test_io(const char *file, struct io_uring *ring, int write, int buffered,
Packit Service 63ea89
		     int sqthread, int fixed, int mixed_fixed, int nonvec,
Packit Service 63ea89
		     int buf_select, int seq, int exp_len)
Packit Service 63ea89
{
Packit Service 63ea89
	struct io_uring_sqe *sqe;
Packit Service 63ea89
	struct io_uring_cqe *cqe;
Packit Service 63ea89
	int open_flags;
Packit Service 63ea89
	int i, fd, ret;
Packit Service 63ea89
	off_t offset;
Packit Service 63ea89
Packit Service 63ea89
#ifdef VERBOSE
Packit Service 63ea89
	fprintf(stdout, "%s: start %d/%d/%d/%d/%d/%d: ", __FUNCTION__, write,
Packit Service 63ea89
							buffered, sqthread,
Packit Service 63ea89
							fixed, mixed_fixed,
Packit Service 63ea89
							nonvec);
Packit Service 63ea89
#endif
Packit Service 63ea89
	if (sqthread && geteuid()) {
Packit Service 63ea89
#ifdef VERBOSE
Packit Service 63ea89
		fprintf(stdout, "SKIPPED (not root)\n");
Packit Service 63ea89
#endif
Packit Service 63ea89
		return 0;
Packit Service 63ea89
	}
Packit Service 63ea89
Packit Service 63ea89
	if (write)
Packit Service 63ea89
		open_flags = O_WRONLY;
Packit Service 63ea89
	else
Packit Service 63ea89
		open_flags = O_RDONLY;
Packit Service 63ea89
	if (!buffered)
Packit Service 63ea89
		open_flags |= O_DIRECT;
Packit Service 63ea89
Packit Service 63ea89
	fd = open(file, open_flags);
Packit Service 63ea89
	if (fd < 0) {
Packit Service 63ea89
		perror("file open");
Packit Service 63ea89
		goto err;
Packit Service 63ea89
	}
Packit Service 63ea89
Packit Service 63ea89
	if (fixed) {
Packit Service 63ea89
		ret = io_uring_register_buffers(ring, vecs, BUFFERS);
Packit Service 63ea89
		if (ret) {
Packit Service 63ea89
			fprintf(stderr, "buffer reg failed: %d\n", ret);
Packit Service 63ea89
			goto err;
Packit Service 63ea89
		}
Packit Service 63ea89
	}
Packit Service 63ea89
	if (sqthread) {
Packit Service 63ea89
		ret = io_uring_register_files(ring, &fd, 1);
Packit Service 63ea89
		if (ret) {
Packit Service 63ea89
			fprintf(stderr, "file reg failed: %d\n", ret);
Packit Service 63ea89
			goto err;
Packit Service 63ea89
		}
Packit Service 63ea89
	}
Packit Service 63ea89
Packit Service 63ea89
	offset = 0;
Packit Service 63ea89
	for (i = 0; i < BUFFERS; i++) {
Packit Service 63ea89
		sqe = io_uring_get_sqe(ring);
Packit Service 63ea89
		if (!sqe) {
Packit Service 63ea89
			fprintf(stderr, "sqe get failed\n");
Packit Service 63ea89
			goto err;
Packit Service 63ea89
		}
Packit Service 63ea89
		if (!seq)
Packit Service 63ea89
			offset = BS * (rand() % BUFFERS);
Packit Service 63ea89
		if (write) {
Packit Service 63ea89
			int do_fixed = fixed;
Packit Service 63ea89
			int use_fd = fd;
Packit Service 63ea89
Packit Service 63ea89
			if (sqthread)
Packit Service 63ea89
				use_fd = 0;
Packit Service 63ea89
			if (fixed && (i & 1))
Packit Service 63ea89
				do_fixed = 0;
Packit Service 63ea89
			if (do_fixed) {
Packit Service 63ea89
				io_uring_prep_write_fixed(sqe, use_fd, vecs[i].iov_base,
Packit Service 63ea89
								vecs[i].iov_len,
Packit Service 63ea89
								offset, i);
Packit Service 63ea89
			} else if (nonvec) {
Packit Service 63ea89
				io_uring_prep_write(sqe, use_fd, vecs[i].iov_base,
Packit Service 63ea89
							vecs[i].iov_len, offset);
Packit Service 63ea89
			} else {
Packit Service 63ea89
				io_uring_prep_writev(sqe, use_fd, &vecs[i], 1,
Packit Service 63ea89
								offset);
Packit Service 63ea89
			}
Packit Service 63ea89
		} else {
Packit Service 63ea89
			int do_fixed = fixed;
Packit Service 63ea89
			int use_fd = fd;
Packit Service 63ea89
Packit Service 63ea89
			if (sqthread)
Packit Service 63ea89
				use_fd = 0;
Packit Service 63ea89
			if (fixed && (i & 1))
Packit Service 63ea89
				do_fixed = 0;
Packit Service 63ea89
			if (do_fixed) {
Packit Service 63ea89
				io_uring_prep_read_fixed(sqe, use_fd, vecs[i].iov_base,
Packit Service 63ea89
								vecs[i].iov_len,
Packit Service 63ea89
								offset, i);
Packit Service 63ea89
			} else if (nonvec) {
Packit Service 63ea89
				io_uring_prep_read(sqe, use_fd, vecs[i].iov_base,
Packit Service 63ea89
							vecs[i].iov_len, offset);
Packit Service 63ea89
			} else {
Packit Service 63ea89
				io_uring_prep_readv(sqe, use_fd, &vecs[i], 1,
Packit Service 63ea89
								offset);
Packit Service 63ea89
			}
Packit Service 63ea89
Packit Service 63ea89
		}
Packit Service 63ea89
		if (sqthread)
Packit Service 63ea89
			sqe->flags |= IOSQE_FIXED_FILE;
Packit Service 63ea89
		if (buf_select) {
Packit Service 63ea89
			if (nonvec)
Packit Service 63ea89
				sqe->addr = 0;
Packit Service 63ea89
			sqe->flags |= IOSQE_BUFFER_SELECT;
Packit Service 63ea89
			sqe->buf_group = buf_select;
Packit Service 63ea89
			sqe->user_data = i;
Packit Service 63ea89
		}
Packit Service 63ea89
		if (seq)
Packit Service 63ea89
			offset += BS;
Packit Service 63ea89
	}
Packit Service 63ea89
Packit Service 63ea89
	ret = io_uring_submit(ring);
Packit Service 63ea89
	if (ret != BUFFERS) {
Packit Service 63ea89
		fprintf(stderr, "submit got %d, wanted %d\n", ret, BUFFERS);
Packit Service 63ea89
		goto err;
Packit Service 63ea89
	}
Packit Service 63ea89
Packit Service 63ea89
	for (i = 0; i < BUFFERS; i++) {
Packit Service 63ea89
		ret = io_uring_wait_cqe(ring, &cqe);
Packit Service 63ea89
		if (ret) {
Packit Service 63ea89
			fprintf(stderr, "wait_cqe=%d\n", ret);
Packit Service 63ea89
			goto err;
Packit Service 63ea89
		}
Packit Service 63ea89
		if (cqe->res == -EINVAL && nonvec) {
Packit Service 63ea89
			if (!warned) {
Packit Service 63ea89
				fprintf(stdout, "Non-vectored IO not "
Packit Service 63ea89
					"supported, skipping\n");
Packit Service 63ea89
				warned = 1;
Packit Service 63ea89
				no_read = 1;
Packit Service 63ea89
			}
Packit Service 63ea89
		} else if (cqe->res != exp_len) {
Packit Service 63ea89
			fprintf(stderr, "cqe res %d, wanted %d\n", cqe->res, exp_len);
Packit Service 63ea89
			goto err;
Packit Service 63ea89
		}
Packit Service 63ea89
		if (buf_select && exp_len == BS) {
Packit Service 63ea89
			int bid = cqe->flags >> 16;
Packit Service 63ea89
			unsigned char *ptr = vecs[bid].iov_base;
Packit Service 63ea89
			int j;
Packit Service 63ea89
Packit Service 63ea89
			for (j = 0; j < BS; j++) {
Packit Service 63ea89
				if (ptr[j] == cqe->user_data)
Packit Service 63ea89
					continue;
Packit Service 63ea89
Packit Service 63ea89
				fprintf(stderr, "Data mismatch! bid=%d, "
Packit Service 63ea89
						"wanted=%d, got=%d\n", bid,
Packit Service 63ea89
						(int)cqe->user_data, ptr[j]);
Packit Service 63ea89
				return 1;
Packit Service 63ea89
			}
Packit Service 63ea89
		}
Packit Service 63ea89
		io_uring_cqe_seen(ring, cqe);
Packit Service 63ea89
	}
Packit Service 63ea89
Packit Service 63ea89
	if (fixed) {
Packit Service 63ea89
		ret = io_uring_unregister_buffers(ring);
Packit Service 63ea89
		if (ret) {
Packit Service 63ea89
			fprintf(stderr, "buffer unreg failed: %d\n", ret);
Packit Service 63ea89
			goto err;
Packit Service 63ea89
		}
Packit Service 63ea89
	}
Packit Service 63ea89
	if (sqthread) {
Packit Service 63ea89
		ret = io_uring_unregister_files(ring);
Packit Service 63ea89
		if (ret) {
Packit Service 63ea89
			fprintf(stderr, "file unreg failed: %d\n", ret);
Packit Service 63ea89
			goto err;
Packit Service 63ea89
		}
Packit Service 63ea89
	}
Packit Service 63ea89
Packit Service 63ea89
	close(fd);
Packit Service 63ea89
#ifdef VERBOSE
Packit Service 63ea89
	fprintf(stdout, "PASS\n");
Packit Service 63ea89
#endif
Packit Service 63ea89
	return 0;
Packit Service 63ea89
err:
Packit Service 63ea89
#ifdef VERBOSE
Packit Service 63ea89
	fprintf(stderr, "FAILED\n");
Packit Service 63ea89
#endif
Packit Service 63ea89
	if (fd != -1)
Packit Service 63ea89
		close(fd);
Packit Service 63ea89
	return 1;
Packit Service 63ea89
}
Packit Service 63ea89
static int test_io(const char *file, int write, int buffered, int sqthread,
Packit Service 63ea89
		   int fixed, int mixed_fixed, int nonvec)
Packit Service 63ea89
{
Packit Service 63ea89
	struct io_uring ring;
Packit Service 63ea89
	int ret, ring_flags;
Packit Service 63ea89
Packit Service 63ea89
	if (sqthread) {
Packit Service 63ea89
		if (geteuid()) {
Packit Service 63ea89
			if (!warned) {
Packit Service 63ea89
				fprintf(stderr, "SQPOLL requires root, skipping\n");
Packit Service 63ea89
				warned = 1;
Packit Service 63ea89
			}
Packit Service 63ea89
			return 0;
Packit Service 63ea89
		}
Packit Service 63ea89
		ring_flags = IORING_SETUP_SQPOLL;
Packit Service 63ea89
	} else {
Packit Service 63ea89
		ring_flags = 0;
Packit Service 63ea89
	}
Packit Service 63ea89
Packit Service 63ea89
	ret = io_uring_queue_init(64, &ring, ring_flags);
Packit Service 63ea89
	if (ret) {
Packit Service 63ea89
		fprintf(stderr, "ring create failed: %d\n", ret);
Packit Service 63ea89
		return 1;
Packit Service 63ea89
	}
Packit Service 63ea89
Packit Service 63ea89
	ret = __test_io(file, &ring, write, buffered, sqthread, fixed,
Packit Service 63ea89
			mixed_fixed, nonvec, 0, 0, BS);
Packit Service 63ea89
Packit Service 63ea89
	io_uring_queue_exit(&ring);
Packit Service 63ea89
	return ret;
Packit Service 63ea89
}
Packit Service 63ea89
Packit Service 63ea89
static int read_poll_link(const char *file)
Packit Service 63ea89
{
Packit Service 63ea89
	struct __kernel_timespec ts;
Packit Service 63ea89
	struct io_uring_sqe *sqe;
Packit Service 63ea89
	struct io_uring_cqe *cqe;
Packit Service 63ea89
	struct io_uring ring;
Packit Service 63ea89
	int i, fd, ret, fds[2];
Packit Service 63ea89
Packit Service 63ea89
	ret = io_uring_queue_init(8, &ring, 0);
Packit Service 63ea89
	if (ret)
Packit Service 63ea89
		return ret;
Packit Service 63ea89
Packit Service 63ea89
	fd = open(file, O_WRONLY);
Packit Service 63ea89
	if (fd < 0) {
Packit Service 63ea89
		perror("open");
Packit Service 63ea89
		return 1;
Packit Service 63ea89
	}
Packit Service 63ea89
Packit Service 63ea89
	if (pipe(fds)) {
Packit Service 63ea89
		perror("pipe");
Packit Service 63ea89
		return 1;
Packit Service 63ea89
	}
Packit Service 63ea89
Packit Service 63ea89
	sqe = io_uring_get_sqe(&ring);
Packit Service 63ea89
	io_uring_prep_writev(sqe, fd, &vecs[0], 1, 0);
Packit Service 63ea89
	sqe->flags |= IOSQE_IO_LINK;
Packit Service 63ea89
	sqe->user_data = 1;
Packit Service 63ea89
Packit Service 63ea89
	sqe = io_uring_get_sqe(&ring);
Packit Service 63ea89
	io_uring_prep_poll_add(sqe, fds[0], POLLIN);
Packit Service 63ea89
	sqe->flags |= IOSQE_IO_LINK;
Packit Service 63ea89
	sqe->user_data = 2;
Packit Service 63ea89
Packit Service 63ea89
	ts.tv_sec = 1;
Packit Service 63ea89
	ts.tv_nsec = 0;
Packit Service 63ea89
	sqe = io_uring_get_sqe(&ring);
Packit Service 63ea89
	io_uring_prep_link_timeout(sqe, &ts, 0);
Packit Service 63ea89
	sqe->user_data = 3;
Packit Service 63ea89
Packit Service 63ea89
	ret = io_uring_submit(&ring);
Packit Service 63ea89
	if (ret != 3) {
Packit Service 63ea89
		fprintf(stderr, "submitted %d\n", ret);
Packit Service 63ea89
		return 1;
Packit Service 63ea89
	}
Packit Service 63ea89
Packit Service 63ea89
	for (i = 0; i < 3; i++) {
Packit Service 63ea89
		ret = io_uring_wait_cqe(&ring, &cqe);
Packit Service 63ea89
		if (ret) {
Packit Service 63ea89
			fprintf(stderr, "wait_cqe=%d\n", ret);
Packit Service 63ea89
			return 1;
Packit Service 63ea89
		}
Packit Service 63ea89
		io_uring_cqe_seen(&ring, cqe);
Packit Service 63ea89
	}
Packit Service 63ea89
Packit Service 63ea89
	return 0;
Packit Service 63ea89
}
Packit Service 63ea89
Packit Service 63ea89
static int has_nonvec_read(void)
Packit Service 63ea89
{
Packit Service 63ea89
	struct io_uring_probe *p;
Packit Service 63ea89
	struct io_uring ring;
Packit Service 63ea89
	int ret;
Packit Service 63ea89
Packit Service 63ea89
	ret = io_uring_queue_init(1, &ring, 0);
Packit Service 63ea89
	if (ret) {
Packit Service 63ea89
		fprintf(stderr, "queue init failed: %d\n", ret);
Packit Service 63ea89
		exit(ret);
Packit Service 63ea89
	}
Packit Service 63ea89
Packit Service 63ea89
	p = calloc(1, sizeof(*p) + 256 * sizeof(struct io_uring_probe_op));
Packit Service 63ea89
	ret = io_uring_register_probe(&ring, p, 256);
Packit Service 63ea89
	/* if we don't have PROBE_REGISTER, we don't have OP_READ/WRITE */
Packit Service 63ea89
	if (ret == -EINVAL) {
Packit Service 63ea89
out:
Packit Service 63ea89
		io_uring_queue_exit(&ring);
Packit Service 63ea89
		return 0;
Packit Service 63ea89
	} else if (ret) {
Packit Service 63ea89
		fprintf(stderr, "register_probe: %d\n", ret);
Packit Service 63ea89
		goto out;
Packit Service 63ea89
	}
Packit Service 63ea89
Packit Service 63ea89
	if (p->ops_len <= IORING_OP_READ)
Packit Service 63ea89
		goto out;
Packit Service 63ea89
	if (!(p->ops[IORING_OP_READ].flags & IO_URING_OP_SUPPORTED))
Packit Service 63ea89
		goto out;
Packit Service 63ea89
	io_uring_queue_exit(&ring);
Packit Service 63ea89
	return 1;
Packit Service 63ea89
}
Packit Service 63ea89
Packit Service 63ea89
static int test_eventfd_read(void)
Packit Service 63ea89
{
Packit Service 63ea89
	struct io_uring ring;
Packit Service 63ea89
	int fd, ret;
Packit Service 63ea89
	eventfd_t event;
Packit Service 63ea89
	struct io_uring_sqe *sqe;
Packit Service 63ea89
	struct io_uring_cqe *cqe;
Packit Service 63ea89
Packit Service 63ea89
	if (no_read)
Packit Service 63ea89
		return 0;
Packit Service 63ea89
	ret = io_uring_queue_init(8, &ring, 0);
Packit Service 63ea89
	if (ret)
Packit Service 63ea89
		return ret;
Packit Service 63ea89
Packit Service 63ea89
	fd = eventfd(1, 0);
Packit Service 63ea89
	if (fd < 0) {
Packit Service 63ea89
		perror("eventfd");
Packit Service 63ea89
		return 1;
Packit Service 63ea89
	}
Packit Service 63ea89
	sqe = io_uring_get_sqe(&ring);
Packit Service 63ea89
	io_uring_prep_read(sqe, fd, &event, sizeof(eventfd_t), 0);
Packit Service 63ea89
	ret = io_uring_submit(&ring);
Packit Service 63ea89
	if (ret != 1) {
Packit Service 63ea89
		fprintf(stderr, "submitted %d\n", ret);
Packit Service 63ea89
		return 1;
Packit Service 63ea89
	}
Packit Service 63ea89
	eventfd_write(fd, 1);
Packit Service 63ea89
	ret = io_uring_wait_cqe(&ring, &cqe);
Packit Service 63ea89
	if (ret) {
Packit Service 63ea89
		fprintf(stderr, "wait_cqe=%d\n", ret);
Packit Service 63ea89
		return 1;
Packit Service 63ea89
	}
Packit Service 63ea89
	if (cqe->res == -EINVAL) {
Packit Service 63ea89
		fprintf(stdout, "eventfd IO not supported, skipping\n");
Packit Service 63ea89
	} else if (cqe->res != sizeof(eventfd_t)) {
Packit Service 63ea89
		fprintf(stderr, "cqe res %d, wanted %d\n", cqe->res,
Packit Service 63ea89
						(int) sizeof(eventfd_t));
Packit Service 63ea89
		return 1;
Packit Service 63ea89
	}
Packit Service 63ea89
	io_uring_cqe_seen(&ring, cqe);
Packit Service 63ea89
	return 0;
Packit Service 63ea89
}
Packit Service 63ea89
Packit Service 63ea89
static int test_buf_select_short(const char *filename, int nonvec)
Packit Service 63ea89
{
Packit Service 63ea89
	struct io_uring_sqe *sqe;
Packit Service 63ea89
	struct io_uring_cqe *cqe;
Packit Service 63ea89
	struct io_uring ring;
Packit Service 63ea89
	int ret, i, exp_len;
Packit Service 63ea89
Packit Service 63ea89
	if (no_buf_select)
Packit Service 63ea89
		return 0;
Packit Service 63ea89
Packit Service 63ea89
	ret = io_uring_queue_init(64, &ring, 0);
Packit Service 63ea89
	if (ret) {
Packit Service 63ea89
		fprintf(stderr, "ring create failed: %d\n", ret);
Packit Service 63ea89
		return 1;
Packit Service 63ea89
	}
Packit Service 63ea89
Packit Service 63ea89
	exp_len = 0;
Packit Service 63ea89
	for (i = 0; i < BUFFERS; i++) {
Packit Service 63ea89
		sqe = io_uring_get_sqe(&ring);
Packit Service 63ea89
		io_uring_prep_provide_buffers(sqe, vecs[i].iov_base,
Packit Service 63ea89
						vecs[i].iov_len / 2, 1, 1, i);
Packit Service 63ea89
		if (!exp_len)
Packit Service 63ea89
			exp_len = vecs[i].iov_len / 2;
Packit Service 63ea89
	}
Packit Service 63ea89
Packit Service 63ea89
	ret = io_uring_submit(&ring);
Packit Service 63ea89
	if (ret != BUFFERS) {
Packit Service 63ea89
		fprintf(stderr, "submit: %d\n", ret);
Packit Service 63ea89
		return -1;
Packit Service 63ea89
	}
Packit Service 63ea89
Packit Service 63ea89
	for (i = 0; i < BUFFERS; i++) {
Packit Service 63ea89
		ret = io_uring_wait_cqe(&ring, &cqe);
Packit Service 63ea89
		if (cqe->res < 0) {
Packit Service 63ea89
			fprintf(stderr, "cqe->res=%d\n", cqe->res);
Packit Service 63ea89
			return 1;
Packit Service 63ea89
		}
Packit Service 63ea89
		io_uring_cqe_seen(&ring, cqe);
Packit Service 63ea89
	}
Packit Service 63ea89
Packit Service 63ea89
	ret = __test_io(filename, &ring, 0, 0, 0, 0, 0, nonvec, 1, 1, exp_len);
Packit Service 63ea89
Packit Service 63ea89
	io_uring_queue_exit(&ring);
Packit Service 63ea89
	return ret;
Packit Service 63ea89
}
Packit Service 63ea89
Packit Service 63ea89
static int test_buf_select(const char *filename, int nonvec)
Packit Service 63ea89
{
Packit Service 63ea89
	struct io_uring_sqe *sqe;
Packit Service 63ea89
	struct io_uring_cqe *cqe;
Packit Service 63ea89
	struct io_uring_probe *p;
Packit Service 63ea89
	struct io_uring ring;
Packit Service 63ea89
	int ret, i;
Packit Service 63ea89
Packit Service 63ea89
	ret = io_uring_queue_init(64, &ring, 0);
Packit Service 63ea89
	if (ret) {
Packit Service 63ea89
		fprintf(stderr, "ring create failed: %d\n", ret);
Packit Service 63ea89
		return 1;
Packit Service 63ea89
	}
Packit Service 63ea89
Packit Service 63ea89
	p = io_uring_get_probe_ring(&ring);
Packit Service 63ea89
	if (!p || !io_uring_opcode_supported(p, IORING_OP_PROVIDE_BUFFERS)) {
Packit Service 63ea89
		no_buf_select = 1;
Packit Service 63ea89
		fprintf(stdout, "Buffer select not supported, skipping\n");
Packit Service 63ea89
		return 0;
Packit Service 63ea89
	}
Packit Service 63ea89
	free(p);
Packit Service 63ea89
Packit Service 63ea89
	/*
Packit Service 63ea89
	 * Write out data with known pattern
Packit Service 63ea89
	 */
Packit Service 63ea89
	for (i = 0; i < BUFFERS; i++)
Packit Service 63ea89
		memset(vecs[i].iov_base, i, vecs[i].iov_len);
Packit Service 63ea89
Packit Service 63ea89
	ret = __test_io(filename, &ring, 1, 0, 0, 0, 0, 0, 0, 1, BS);
Packit Service 63ea89
	if (ret) {
Packit Service 63ea89
		fprintf(stderr, "failed writing data\n");
Packit Service 63ea89
		return 1;
Packit Service 63ea89
	}
Packit Service 63ea89
Packit Service 63ea89
	for (i = 0; i < BUFFERS; i++)
Packit Service 63ea89
		memset(vecs[i].iov_base, 0x55, vecs[i].iov_len);
Packit Service 63ea89
Packit Service 63ea89
	for (i = 0; i < BUFFERS; i++) {
Packit Service 63ea89
		sqe = io_uring_get_sqe(&ring);
Packit Service 63ea89
		io_uring_prep_provide_buffers(sqe, vecs[i].iov_base,
Packit Service 63ea89
						vecs[i].iov_len, 1, 1, i);
Packit Service 63ea89
	}
Packit Service 63ea89
Packit Service 63ea89
	ret = io_uring_submit(&ring);
Packit Service 63ea89
	if (ret != BUFFERS) {
Packit Service 63ea89
		fprintf(stderr, "submit: %d\n", ret);
Packit Service 63ea89
		return -1;
Packit Service 63ea89
	}
Packit Service 63ea89
Packit Service 63ea89
	for (i = 0; i < BUFFERS; i++) {
Packit Service 63ea89
		ret = io_uring_wait_cqe(&ring, &cqe);
Packit Service 63ea89
		if (cqe->res < 0) {
Packit Service 63ea89
			fprintf(stderr, "cqe->res=%d\n", cqe->res);
Packit Service 63ea89
			return 1;
Packit Service 63ea89
		}
Packit Service 63ea89
		io_uring_cqe_seen(&ring, cqe);
Packit Service 63ea89
	}
Packit Service 63ea89
Packit Service 63ea89
	ret = __test_io(filename, &ring, 0, 0, 0, 0, 0, nonvec, 1, 1, BS);
Packit Service 63ea89
Packit Service 63ea89
	io_uring_queue_exit(&ring);
Packit Service 63ea89
	return ret;
Packit Service 63ea89
}
Packit Service 63ea89
Packit Service 63ea89
static int test_io_link(const char *file)
Packit Service 63ea89
{
Packit Service 63ea89
	const int nr_links = 100;
Packit Service 63ea89
	const int link_len = 100;
Packit Service 63ea89
	const int nr_sqes = nr_links * link_len;
Packit Service 63ea89
	struct io_uring_sqe *sqe;
Packit Service 63ea89
	struct io_uring_cqe *cqe;
Packit Service 63ea89
	struct io_uring ring;
Packit Service 63ea89
	int i, j, fd, ret;
Packit Service 63ea89
Packit Service 63ea89
	fd = open(file, O_WRONLY);
Packit Service 63ea89
	if (fd < 0) {
Packit Service 63ea89
		perror("file open");
Packit Service 63ea89
		goto err;
Packit Service 63ea89
	}
Packit Service 63ea89
Packit Service 63ea89
	ret = io_uring_queue_init(nr_sqes, &ring, 0);
Packit Service 63ea89
	if (ret) {
Packit Service 63ea89
		fprintf(stderr, "ring create failed: %d\n", ret);
Packit Service 63ea89
		goto err;
Packit Service 63ea89
	}
Packit Service 63ea89
Packit Service 63ea89
	for (i = 0; i < nr_links; ++i) {
Packit Service 63ea89
		for (j = 0; j < link_len; ++j) {
Packit Service 63ea89
			sqe = io_uring_get_sqe(&ring);
Packit Service 63ea89
			if (!sqe) {
Packit Service 63ea89
				fprintf(stderr, "sqe get failed\n");
Packit Service 63ea89
				goto err;
Packit Service 63ea89
			}
Packit Service 63ea89
			io_uring_prep_writev(sqe, fd, &vecs[0], 1, 0);
Packit Service 63ea89
			sqe->flags |= IOSQE_ASYNC;
Packit Service 63ea89
			if (j != link_len - 1)
Packit Service 63ea89
				sqe->flags |= IOSQE_IO_LINK;
Packit Service 63ea89
		}
Packit Service 63ea89
	}
Packit Service 63ea89
Packit Service 63ea89
	ret = io_uring_submit(&ring);
Packit Service 63ea89
	if (ret != nr_sqes) {
Packit Service 63ea89
		ret = io_uring_peek_cqe(&ring, &cqe);
Packit Service 63ea89
		if (!ret && cqe->res == -EINVAL) {
Packit Service 63ea89
			fprintf(stdout, "IOSQE_ASYNC not supported, skipped\n");
Packit Service 63ea89
			goto out;
Packit Service 63ea89
		}
Packit Service 63ea89
		fprintf(stderr, "submit got %d, wanted %d\n", ret, nr_sqes);
Packit Service 63ea89
		goto err;
Packit Service 63ea89
	}
Packit Service 63ea89
Packit Service 63ea89
	for (i = 0; i < nr_sqes; i++) {
Packit Service 63ea89
		ret = io_uring_wait_cqe(&ring, &cqe);
Packit Service 63ea89
		if (ret) {
Packit Service 63ea89
			fprintf(stderr, "wait_cqe=%d\n", ret);
Packit Service 63ea89
			goto err;
Packit Service 63ea89
		}
Packit Service 63ea89
		if (cqe->res == -EINVAL) {
Packit Service 63ea89
			if (!warned) {
Packit Service 63ea89
				fprintf(stdout, "Non-vectored IO not "
Packit Service 63ea89
					"supported, skipping\n");
Packit Service 63ea89
				warned = 1;
Packit Service 63ea89
				no_read = 1;
Packit Service 63ea89
			}
Packit Service 63ea89
		} else if (cqe->res != BS) {
Packit Service 63ea89
			fprintf(stderr, "cqe res %d, wanted %d\n", cqe->res, BS);
Packit Service 63ea89
			goto err;
Packit Service 63ea89
		}
Packit Service 63ea89
		io_uring_cqe_seen(&ring, cqe);
Packit Service 63ea89
	}
Packit Service 63ea89
Packit Service 63ea89
out:
Packit Service 63ea89
	io_uring_queue_exit(&ring);
Packit Service 63ea89
	close(fd);
Packit Service 63ea89
	return 0;
Packit Service 63ea89
err:
Packit Service 63ea89
	if (fd != -1)
Packit Service 63ea89
		close(fd);
Packit Service 63ea89
	return 1;
Packit Service 63ea89
}
Packit Service 63ea89
Packit Service 63ea89
static int test_write_efbig(void)
Packit Service 63ea89
{
Packit Service 63ea89
	struct io_uring_sqe *sqe;
Packit Service 63ea89
	struct io_uring_cqe *cqe;
Packit Service 63ea89
	struct io_uring ring;
Packit Service 63ea89
	struct rlimit rlim;
Packit Service 63ea89
	int i, fd, ret;
Packit Service 63ea89
	loff_t off;
Packit Service 63ea89
Packit Service 63ea89
	if (getrlimit(RLIMIT_FSIZE, &rlim) < 0) {
Packit Service 63ea89
		perror("getrlimit");
Packit Service 63ea89
		return 1;
Packit Service 63ea89
	}
Packit Service 63ea89
	rlim.rlim_cur = 64 * 1024;
Packit Service 63ea89
	rlim.rlim_max = 64 * 1024;
Packit Service 63ea89
	if (setrlimit(RLIMIT_FSIZE, &rlim) < 0) {
Packit Service 63ea89
		perror("setrlimit");
Packit Service 63ea89
		return 1;
Packit Service 63ea89
	}
Packit Service 63ea89
Packit Service 63ea89
	fd = open(".efbig", O_WRONLY | O_CREAT, 0644);
Packit Service 63ea89
	if (fd < 0) {
Packit Service 63ea89
		perror("file open");
Packit Service 63ea89
		goto err;
Packit Service 63ea89
	}
Packit Service 63ea89
Packit Service 63ea89
	ret = io_uring_queue_init(32, &ring, 0);
Packit Service 63ea89
	if (ret) {
Packit Service 63ea89
		fprintf(stderr, "ring create failed: %d\n", ret);
Packit Service 63ea89
		goto err;
Packit Service 63ea89
	}
Packit Service 63ea89
Packit Service 63ea89
	off = 0;
Packit Service 63ea89
	for (i = 0; i < 32; i++) {
Packit Service 63ea89
		sqe = io_uring_get_sqe(&ring);
Packit Service 63ea89
		if (!sqe) {
Packit Service 63ea89
			fprintf(stderr, "sqe get failed\n");
Packit Service 63ea89
			goto err;
Packit Service 63ea89
		}
Packit Service 63ea89
		io_uring_prep_writev(sqe, fd, &vecs[i], 1, off);
Packit Service 63ea89
		off += BS;
Packit Service 63ea89
	}
Packit Service 63ea89
Packit Service 63ea89
	ret = io_uring_submit(&ring);
Packit Service 63ea89
	if (ret != 32) {
Packit Service 63ea89
		fprintf(stderr, "submit got %d, wanted %d\n", ret, 32);
Packit Service 63ea89
		goto err;
Packit Service 63ea89
	}
Packit Service 63ea89
Packit Service 63ea89
	for (i = 0; i < 32; i++) {
Packit Service 63ea89
		ret = io_uring_wait_cqe(&ring, &cqe);
Packit Service 63ea89
		if (ret) {
Packit Service 63ea89
			fprintf(stderr, "wait_cqe=%d\n", ret);
Packit Service 63ea89
			goto err;
Packit Service 63ea89
		}
Packit Service 63ea89
		if (i < 16) {
Packit Service 63ea89
			if (cqe->res != BS) {
Packit Service 63ea89
				fprintf(stderr, "bad write: %d\n", cqe->res);
Packit Service 63ea89
				goto err;
Packit Service 63ea89
			}
Packit Service 63ea89
		} else {
Packit Service 63ea89
			if (cqe->res != -EFBIG) {
Packit Service 63ea89
				fprintf(stderr, "Expected -EFBIG: %d\n", cqe->res);
Packit Service 63ea89
				goto err;
Packit Service 63ea89
			}
Packit Service 63ea89
		}
Packit Service 63ea89
		io_uring_cqe_seen(&ring, cqe);
Packit Service 63ea89
	}
Packit Service 63ea89
Packit Service 63ea89
	io_uring_queue_exit(&ring);
Packit Service 63ea89
	close(fd);
Packit Service 63ea89
	unlink(".efbig");
Packit Service 63ea89
	return 0;
Packit Service 63ea89
err:
Packit Service 63ea89
	if (fd != -1)
Packit Service 63ea89
		close(fd);
Packit Service 63ea89
	unlink(".efbig");
Packit Service 63ea89
	return 1;
Packit Service 63ea89
}
Packit Service 63ea89
Packit Service 63ea89
int main(int argc, char *argv[])
Packit Service 63ea89
{
Packit Service 63ea89
	int i, ret, nr;
Packit Service 63ea89
	char *fname;
Packit Service 63ea89
Packit Service 63ea89
	if (argc > 1) {
Packit Service 63ea89
		fname = argv[1];
Packit Service 63ea89
	} else {
Packit Service 63ea89
		fname = ".basic-rw";
Packit Service 63ea89
		if (create_file(fname)) {
Packit Service 63ea89
			fprintf(stderr, "file creation failed\n");
Packit Service 63ea89
			goto err;
Packit Service 63ea89
		}
Packit Service 63ea89
	}
Packit Service 63ea89
Packit Service 63ea89
	if (create_buffers()) {
Packit Service 63ea89
		fprintf(stderr, "file creation failed\n");
Packit Service 63ea89
		goto err;
Packit Service 63ea89
	}
Packit Service 63ea89
Packit Service 63ea89
	/* if we don't have nonvec read, skip testing that */
Packit Service 63ea89
	if (has_nonvec_read())
Packit Service 63ea89
		nr = 64;
Packit Service 63ea89
	else
Packit Service 63ea89
		nr = 32;
Packit Service 63ea89
Packit Service 63ea89
	for (i = 0; i < nr; i++) {
Packit Service 63ea89
		int v1, v2, v3, v4, v5, v6;
Packit Service 63ea89
Packit Service 63ea89
		v1 = (i & 1) != 0;
Packit Service 63ea89
		v2 = (i & 2) != 0;
Packit Service 63ea89
		v3 = (i & 4) != 0;
Packit Service 63ea89
		v4 = (i & 8) != 0;
Packit Service 63ea89
		v5 = (i & 16) != 0;
Packit Service 63ea89
		v6 = (i & 32) != 0;
Packit Service 63ea89
		ret = test_io(fname, v1, v2, v3, v4, v5, v6);
Packit Service 63ea89
		if (ret) {
Packit Service 63ea89
			fprintf(stderr, "test_io failed %d/%d/%d/%d/%d/%d\n",
Packit Service 63ea89
					v1, v2, v3, v4, v5, v6);
Packit Service 63ea89
			goto err;
Packit Service 63ea89
		}
Packit Service 63ea89
	}
Packit Service 63ea89
Packit Service 63ea89
	ret = test_buf_select(fname, 1);
Packit Service 63ea89
	if (ret) {
Packit Service 63ea89
		fprintf(stderr, "test_buf_select nonvec failed\n");
Packit Service 63ea89
		goto err;
Packit Service 63ea89
	}
Packit Service 63ea89
Packit Service 63ea89
	ret = test_buf_select(fname, 0);
Packit Service 63ea89
	if (ret) {
Packit Service 63ea89
		fprintf(stderr, "test_buf_select vec failed\n");
Packit Service 63ea89
		goto err;
Packit Service 63ea89
	}
Packit Service 63ea89
Packit Service 63ea89
	ret = test_buf_select_short(fname, 1);
Packit Service 63ea89
	if (ret) {
Packit Service 63ea89
		fprintf(stderr, "test_buf_select_short nonvec failed\n");
Packit Service 63ea89
		goto err;
Packit Service 63ea89
	}
Packit Service 63ea89
Packit Service 63ea89
	ret = test_buf_select_short(fname, 0);
Packit Service 63ea89
	if (ret) {
Packit Service 63ea89
		fprintf(stderr, "test_buf_select_short vec failed\n");
Packit Service 63ea89
		goto err;
Packit Service 63ea89
	}
Packit Service 63ea89
Packit Service 63ea89
	ret = test_eventfd_read();
Packit Service 63ea89
	if (ret) {
Packit Service 63ea89
		fprintf(stderr, "test_eventfd_read failed\n");
Packit Service 63ea89
		goto err;
Packit Service 63ea89
	}
Packit Service 63ea89
Packit Service 63ea89
	ret = read_poll_link(fname);
Packit Service 63ea89
	if (ret) {
Packit Service 63ea89
		fprintf(stderr, "read_poll_link failed\n");
Packit Service 63ea89
		goto err;
Packit Service 63ea89
	}
Packit Service 63ea89
Packit Service 63ea89
	ret = test_io_link(fname);
Packit Service 63ea89
	if (ret) {
Packit Service 63ea89
		fprintf(stderr, "test_io_link failed\n");
Packit Service 63ea89
		goto err;
Packit Service 63ea89
	}
Packit Service 63ea89
Packit Service 63ea89
	ret = test_write_efbig();
Packit Service 63ea89
	if (ret) {
Packit Service 63ea89
		fprintf(stderr, "test_write_efbig failed\n");
Packit Service 63ea89
		goto err;
Packit Service 63ea89
	}
Packit Service 63ea89
Packit Service 63ea89
	if (fname != argv[1])
Packit Service 63ea89
		unlink(fname);
Packit Service 63ea89
	return 0;
Packit Service 63ea89
err:
Packit Service 63ea89
	if (fname != argv[1])
Packit Service 63ea89
		unlink(fname);
Packit Service 63ea89
	return 1;
Packit Service 63ea89
}