blob: 3d94dfecabfb05a19e0a989142a01e4adb197793 [file] [log] [blame]
/* SPDX-License-Identifier: MIT */
/*
* Description: basic read/write tests with polled IO
*/
#include <errno.h>
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <string.h>
#include <fcntl.h>
#include <sys/types.h>
#include <sys/poll.h>
#include <sys/eventfd.h>
#include <sys/resource.h>
#include "helpers.h"
#include "liburing.h"
#include "../src/syscall.h"
#define FILE_SIZE (128 * 1024)
#define BS 4096
#define BUFFERS (FILE_SIZE / BS)
static struct iovec *vecs;
static int no_buf_select;
static int no_iopoll;
static int provide_buffers(struct io_uring *ring)
{
struct io_uring_sqe *sqe;
struct io_uring_cqe *cqe;
int ret, i;
for (i = 0; i < BUFFERS; i++) {
sqe = io_uring_get_sqe(ring);
io_uring_prep_provide_buffers(sqe, vecs[i].iov_base,
vecs[i].iov_len, 1, 1, i);
}
ret = io_uring_submit(ring);
if (ret != BUFFERS) {
fprintf(stderr, "submit: %d\n", ret);
return 1;
}
for (i = 0; i < BUFFERS; i++) {
ret = io_uring_wait_cqe(ring, &cqe);
if (cqe->res < 0) {
fprintf(stderr, "cqe->res=%d\n", cqe->res);
return 1;
}
io_uring_cqe_seen(ring, cqe);
}
return 0;
}
static int __test_io(const char *file, struct io_uring *ring, int write, int sqthread,
int fixed, int buf_select)
{
struct io_uring_sqe *sqe;
struct io_uring_cqe *cqe;
int open_flags;
int i, fd, ret;
off_t offset;
if (buf_select && write)
write = 0;
if (buf_select && fixed)
fixed = 0;
if (buf_select && provide_buffers(ring))
return 1;
if (write)
open_flags = O_WRONLY;
else
open_flags = O_RDONLY;
open_flags |= O_DIRECT;
fd = open(file, open_flags);
if (fd < 0) {
perror("file open");
goto err;
}
if (fixed) {
ret = io_uring_register_buffers(ring, vecs, BUFFERS);
if (ret) {
fprintf(stderr, "buffer reg failed: %d\n", ret);
goto err;
}
}
if (sqthread) {
ret = io_uring_register_files(ring, &fd, 1);
if (ret) {
fprintf(stderr, "file reg failed: %d\n", ret);
goto err;
}
}
offset = 0;
for (i = 0; i < BUFFERS; i++) {
sqe = io_uring_get_sqe(ring);
if (!sqe) {
fprintf(stderr, "sqe get failed\n");
goto err;
}
offset = BS * (rand() % BUFFERS);
if (write) {
int do_fixed = fixed;
int use_fd = fd;
if (sqthread)
use_fd = 0;
if (fixed && (i & 1))
do_fixed = 0;
if (do_fixed) {
io_uring_prep_write_fixed(sqe, use_fd, vecs[i].iov_base,
vecs[i].iov_len,
offset, i);
} else {
io_uring_prep_writev(sqe, use_fd, &vecs[i], 1,
offset);
}
} else {
int do_fixed = fixed;
int use_fd = fd;
if (sqthread)
use_fd = 0;
if (fixed && (i & 1))
do_fixed = 0;
if (do_fixed) {
io_uring_prep_read_fixed(sqe, use_fd, vecs[i].iov_base,
vecs[i].iov_len,
offset, i);
} else {
io_uring_prep_readv(sqe, use_fd, &vecs[i], 1,
offset);
}
}
if (sqthread)
sqe->flags |= IOSQE_FIXED_FILE;
if (buf_select) {
sqe->flags |= IOSQE_BUFFER_SELECT;
sqe->buf_group = buf_select;
sqe->user_data = i;
}
}
ret = io_uring_submit(ring);
if (ret != BUFFERS) {
fprintf(stderr, "submit got %d, wanted %d\n", ret, BUFFERS);
goto err;
}
for (i = 0; i < BUFFERS; i++) {
ret = io_uring_wait_cqe(ring, &cqe);
if (ret) {
fprintf(stderr, "wait_cqe=%d\n", ret);
goto err;
} else if (cqe->res == -EOPNOTSUPP) {
fprintf(stdout, "File/device/fs doesn't support polled IO\n");
no_iopoll = 1;
goto out;
} else if (cqe->res != BS) {
fprintf(stderr, "cqe res %d, wanted %d\n", cqe->res, BS);
goto err;
}
io_uring_cqe_seen(ring, cqe);
}
if (fixed) {
ret = io_uring_unregister_buffers(ring);
if (ret) {
fprintf(stderr, "buffer unreg failed: %d\n", ret);
goto err;
}
}
if (sqthread) {
ret = io_uring_unregister_files(ring);
if (ret) {
fprintf(stderr, "file unreg failed: %d\n", ret);
goto err;
}
}
out:
close(fd);
return 0;
err:
if (fd != -1)
close(fd);
return 1;
}
extern int __io_uring_flush_sq(struct io_uring *ring);
/*
* if we are polling io_uring_submit needs to always enter the
* kernel to fetch events
*/
static int test_io_uring_submit_enters(const char *file)
{
struct io_uring ring;
int fd, i, ret, ring_flags, open_flags;
unsigned head;
struct io_uring_cqe *cqe;
if (no_iopoll)
return 0;
ring_flags = IORING_SETUP_IOPOLL;
ret = io_uring_queue_init(64, &ring, ring_flags);
if (ret) {
fprintf(stderr, "ring create failed: %d\n", ret);
return 1;
}
open_flags = O_WRONLY | O_DIRECT;
fd = open(file, open_flags);
if (fd < 0) {
perror("file open");
goto err;
}
for (i = 0; i < BUFFERS; i++) {
struct io_uring_sqe *sqe;
off_t offset = BS * (rand() % BUFFERS);
sqe = io_uring_get_sqe(&ring);
io_uring_prep_writev(sqe, fd, &vecs[i], 1, offset);
sqe->user_data = 1;
}
/* submit manually to avoid adding IORING_ENTER_GETEVENTS */
ret = __sys_io_uring_enter(ring.ring_fd, __io_uring_flush_sq(&ring), 0,
0, NULL);
if (ret < 0)
goto err;
for (i = 0; i < 500; i++) {
ret = io_uring_submit(&ring);
if (ret != 0) {
fprintf(stderr, "still had %d sqes to submit, this is unexpected", ret);
goto err;
}
io_uring_for_each_cqe(&ring, head, cqe) {
/* runs after test_io so should not have happened */
if (cqe->res == -EOPNOTSUPP) {
fprintf(stdout, "File/device/fs doesn't support polled IO\n");
goto err;
}
goto ok;
}
usleep(10000);
}
err:
ret = 1;
if (fd != -1)
close(fd);
ok:
io_uring_queue_exit(&ring);
return ret;
}
static int test_io(const char *file, int write, int sqthread, int fixed,
int buf_select)
{
struct io_uring ring;
int ret, ring_flags;
if (no_iopoll)
return 0;
ring_flags = IORING_SETUP_IOPOLL;
if (sqthread) {
static int warned;
if (geteuid()) {
if (!warned)
fprintf(stdout, "SQPOLL requires root, skipping\n");
warned = 1;
return 0;
}
}
ret = io_uring_queue_init(64, &ring, ring_flags);
if (ret) {
fprintf(stderr, "ring create failed: %d\n", ret);
return 1;
}
ret = __test_io(file, &ring, write, sqthread, fixed, buf_select);
io_uring_queue_exit(&ring);
return ret;
}
static int probe_buf_select(void)
{
struct io_uring_probe *p;
struct io_uring ring;
int ret;
ret = io_uring_queue_init(1, &ring, 0);
if (ret) {
fprintf(stderr, "ring create failed: %d\n", ret);
return 1;
}
p = io_uring_get_probe_ring(&ring);
if (!p || !io_uring_opcode_supported(p, IORING_OP_PROVIDE_BUFFERS)) {
no_buf_select = 1;
fprintf(stdout, "Buffer select not supported, skipping\n");
return 0;
}
free(p);
return 0;
}
int main(int argc, char *argv[])
{
int i, ret, nr;
char *fname;
if (probe_buf_select())
return 1;
if (argc > 1) {
fname = argv[1];
} else {
fname = ".iopoll-rw";
t_create_file(fname, FILE_SIZE);
}
vecs = t_create_buffers(BUFFERS, BS);
nr = 16;
if (no_buf_select)
nr = 8;
for (i = 0; i < nr; i++) {
int v1, v2, v3, v4;
v1 = (i & 1) != 0;
v2 = (i & 2) != 0;
v3 = (i & 4) != 0;
v4 = (i & 8) != 0;
ret = test_io(fname, v1, v2, v3, v4);
if (ret) {
fprintf(stderr, "test_io failed %d/%d/%d/%d\n", v1, v2, v3, v4);
goto err;
}
if (no_iopoll)
break;
}
ret = test_io_uring_submit_enters(fname);
if (ret) {
fprintf(stderr, "test_io_uring_submit_enters failed\n");
goto err;
}
if (fname != argv[1])
unlink(fname);
return 0;
err:
if (fname != argv[1])
unlink(fname);
return 1;
}