blob: c5c55075a8787a4c0fbab3fd32b141299c595f15 [file] [log] [blame]
/* SPDX-License-Identifier: MIT */
/*
* Description: run various file registration tests
*
*/
#include <errno.h>
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <string.h>
#include <fcntl.h>
#include "helpers.h"
#include "liburing.h"
static int no_update = 0;
static void close_files(int *files, int nr_files, int add)
{
char fname[32];
int i;
for (i = 0; i < nr_files; i++) {
if (files)
close(files[i]);
if (!add)
sprintf(fname, ".reg.%d", i);
else
sprintf(fname, ".add.%d", i + add);
unlink(fname);
}
if (files)
free(files);
}
static int *open_files(int nr_files, int extra, int add)
{
char fname[32];
int *files;
int i;
files = t_calloc(nr_files + extra, sizeof(int));
for (i = 0; i < nr_files; i++) {
if (!add)
sprintf(fname, ".reg.%d", i);
else
sprintf(fname, ".add.%d", i + add);
files[i] = open(fname, O_RDWR | O_CREAT, 0644);
if (files[i] < 0) {
perror("open");
free(files);
files = NULL;
break;
}
}
if (extra) {
for (i = nr_files; i < nr_files + extra; i++)
files[i] = -1;
}
return files;
}
static int test_shrink(struct io_uring *ring)
{
int ret, off, fd;
int *files;
files = open_files(50, 0, 0);
ret = io_uring_register_files(ring, files, 50);
if (ret) {
fprintf(stderr, "%s: register ret=%d\n", __FUNCTION__, ret);
goto err;
}
off = 0;
do {
fd = -1;
ret = io_uring_register_files_update(ring, off, &fd, 1);
if (ret != 1) {
if (off == 50 && ret == -EINVAL)
break;
fprintf(stderr, "%s: update ret=%d\n", __FUNCTION__, ret);
break;
}
off++;
} while (1);
ret = io_uring_unregister_files(ring);
if (ret) {
fprintf(stderr, "%s: unregister ret=%d\n", __FUNCTION__, ret);
goto err;
}
close_files(files, 50, 0);
return 0;
err:
close_files(files, 50, 0);
return 1;
}
static int test_grow(struct io_uring *ring)
{
int ret, off;
int *files, *fds = NULL;
files = open_files(50, 250, 0);
ret = io_uring_register_files(ring, files, 300);
if (ret) {
fprintf(stderr, "%s: register ret=%d\n", __FUNCTION__, ret);
goto err;
}
off = 50;
do {
fds = open_files(1, 0, off);
ret = io_uring_register_files_update(ring, off, fds, 1);
if (ret != 1) {
if (off == 300 && ret == -EINVAL)
break;
fprintf(stderr, "%s: update ret=%d\n", __FUNCTION__, ret);
break;
}
if (off >= 300) {
fprintf(stderr, "%s: Succeeded beyond end-of-list?\n", __FUNCTION__);
goto err;
}
off++;
} while (1);
ret = io_uring_unregister_files(ring);
if (ret) {
fprintf(stderr, "%s: unregister ret=%d\n", __FUNCTION__, ret);
goto err;
}
close_files(files, 100, 0);
close_files(NULL, 251, 50);
return 0;
err:
close_files(files, 100, 0);
close_files(NULL, 251, 50);
return 1;
}
static int test_replace_all(struct io_uring *ring)
{
int *files, *fds = NULL;
int ret, i;
files = open_files(100, 0, 0);
ret = io_uring_register_files(ring, files, 100);
if (ret) {
fprintf(stderr, "%s: register ret=%d\n", __FUNCTION__, ret);
goto err;
}
fds = t_malloc(100 * sizeof(int));
for (i = 0; i < 100; i++)
fds[i] = -1;
ret = io_uring_register_files_update(ring, 0, fds, 100);
if (ret != 100) {
fprintf(stderr, "%s: update ret=%d\n", __FUNCTION__, ret);
goto err;
}
ret = io_uring_unregister_files(ring);
if (ret) {
fprintf(stderr, "%s: unregister ret=%d\n", __FUNCTION__, ret);
goto err;
}
close_files(files, 100, 0);
if (fds)
free(fds);
return 0;
err:
close_files(files, 100, 0);
if (fds)
free(fds);
return 1;
}
static int test_replace(struct io_uring *ring)
{
int *files, *fds = NULL;
int ret;
files = open_files(100, 0, 0);
ret = io_uring_register_files(ring, files, 100);
if (ret) {
fprintf(stderr, "%s: register ret=%d\n", __FUNCTION__, ret);
goto err;
}
fds = open_files(10, 0, 1);
ret = io_uring_register_files_update(ring, 90, fds, 10);
if (ret != 10) {
fprintf(stderr, "%s: update ret=%d\n", __FUNCTION__, ret);
goto err;
}
ret = io_uring_unregister_files(ring);
if (ret) {
fprintf(stderr, "%s: unregister ret=%d\n", __FUNCTION__, ret);
goto err;
}
close_files(files, 100, 0);
if (fds)
close_files(fds, 10, 1);
return 0;
err:
close_files(files, 100, 0);
if (fds)
close_files(fds, 10, 1);
return 1;
}
static int test_removals(struct io_uring *ring)
{
int *files, *fds = NULL;
int ret, i;
files = open_files(100, 0, 0);
ret = io_uring_register_files(ring, files, 100);
if (ret) {
fprintf(stderr, "%s: register ret=%d\n", __FUNCTION__, ret);
goto err;
}
fds = t_calloc(10, sizeof(int));
for (i = 0; i < 10; i++)
fds[i] = -1;
ret = io_uring_register_files_update(ring, 50, fds, 10);
if (ret != 10) {
fprintf(stderr, "%s: update ret=%d\n", __FUNCTION__, ret);
goto err;
}
ret = io_uring_unregister_files(ring);
if (ret) {
fprintf(stderr, "%s: unregister ret=%d\n", __FUNCTION__, ret);
goto err;
}
close_files(files, 100, 0);
if (fds)
free(fds);
return 0;
err:
close_files(files, 100, 0);
if (fds)
free(fds);
return 1;
}
static int test_additions(struct io_uring *ring)
{
int *files, *fds = NULL;
int ret;
files = open_files(100, 100, 0);
ret = io_uring_register_files(ring, files, 200);
if (ret) {
fprintf(stderr, "%s: register ret=%d\n", __FUNCTION__, ret);
goto err;
}
fds = open_files(2, 0, 1);
ret = io_uring_register_files_update(ring, 100, fds, 2);
if (ret != 2) {
fprintf(stderr, "%s: update ret=%d\n", __FUNCTION__, ret);
goto err;
}
ret = io_uring_unregister_files(ring);
if (ret) {
fprintf(stderr, "%s: unregister ret=%d\n", __FUNCTION__, ret);
goto err;
}
close_files(files, 100, 0);
if (fds)
close_files(fds, 2, 1);
return 0;
err:
close_files(files, 100, 0);
if (fds)
close_files(fds, 2, 1);
return 1;
}
static int test_sparse(struct io_uring *ring)
{
int *files;
int ret;
files = open_files(100, 100, 0);
ret = io_uring_register_files(ring, files, 200);
if (ret) {
if (ret == -EBADF) {
fprintf(stdout, "Sparse files not supported\n");
no_update = 1;
goto done;
}
fprintf(stderr, "%s: register ret=%d\n", __FUNCTION__, ret);
goto err;
}
ret = io_uring_unregister_files(ring);
if (ret) {
fprintf(stderr, "%s: unregister ret=%d\n", __FUNCTION__, ret);
goto err;
}
done:
close_files(files, 100, 0);
return 0;
err:
close_files(files, 100, 0);
return 1;
}
static int test_basic_many(struct io_uring *ring)
{
int *files;
int ret;
files = open_files(768, 0, 0);
ret = io_uring_register_files(ring, files, 768);
if (ret) {
fprintf(stderr, "%s: register %d\n", __FUNCTION__, ret);
goto err;
}
ret = io_uring_unregister_files(ring);
if (ret) {
fprintf(stderr, "%s: unregister %d\n", __FUNCTION__, ret);
goto err;
}
close_files(files, 768, 0);
return 0;
err:
close_files(files, 768, 0);
return 1;
}
static int test_basic(struct io_uring *ring, int fail)
{
int *files;
int ret;
files = open_files(fail ? 10 : 100, 0, 0);
ret = io_uring_register_files(ring, files, 100);
if (ret) {
if (fail) {
if (ret == -EBADF || ret == -EFAULT)
return 0;
}
fprintf(stderr, "%s: register %d\n", __FUNCTION__, ret);
goto err;
}
if (fail) {
fprintf(stderr, "Registration succeeded, but expected fail\n");
goto err;
}
ret = io_uring_unregister_files(ring);
if (ret) {
fprintf(stderr, "%s: unregister %d\n", __FUNCTION__, ret);
goto err;
}
close_files(files, 100, 0);
return 0;
err:
close_files(files, 100, 0);
return 1;
}
/*
* Register 0 files, but reserve space for 10. Then add one file.
*/
static int test_zero(struct io_uring *ring)
{
int *files, *fds = NULL;
int ret;
files = open_files(0, 10, 0);
ret = io_uring_register_files(ring, files, 10);
if (ret) {
fprintf(stderr, "%s: register ret=%d\n", __FUNCTION__, ret);
goto err;
}
fds = open_files(1, 0, 1);
ret = io_uring_register_files_update(ring, 0, fds, 1);
if (ret != 1) {
fprintf(stderr, "%s: update ret=%d\n", __FUNCTION__, ret);
goto err;
}
ret = io_uring_unregister_files(ring);
if (ret) {
fprintf(stderr, "%s: unregister ret=%d\n", __FUNCTION__, ret);
goto err;
}
if (fds)
close_files(fds, 1, 1);
free(files);
return 0;
err:
if (fds)
close_files(fds, 1, 1);
free(files);
return 1;
}
static int test_fixed_read_write(struct io_uring *ring, int index)
{
struct io_uring_sqe *sqe;
struct io_uring_cqe *cqe;
struct iovec iov[2];
int ret;
iov[0].iov_base = t_malloc(4096);
iov[0].iov_len = 4096;
memset(iov[0].iov_base, 0x5a, 4096);
iov[1].iov_base = t_malloc(4096);
iov[1].iov_len = 4096;
sqe = io_uring_get_sqe(ring);
if (!sqe) {
fprintf(stderr, "%s: failed to get sqe\n", __FUNCTION__);
return 1;
}
io_uring_prep_writev(sqe, index, &iov[0], 1, 0);
sqe->flags |= IOSQE_FIXED_FILE;
sqe->user_data = 1;
ret = io_uring_submit(ring);
if (ret != 1) {
fprintf(stderr, "%s: got %d, wanted 1\n", __FUNCTION__, ret);
return 1;
}
ret = io_uring_wait_cqe(ring, &cqe);
if (ret < 0) {
fprintf(stderr, "%s: io_uring_wait_cqe=%d\n", __FUNCTION__, ret);
return 1;
}
if (cqe->res != 4096) {
fprintf(stderr, "%s: write cqe->res=%d\n", __FUNCTION__, cqe->res);
return 1;
}
io_uring_cqe_seen(ring, cqe);
sqe = io_uring_get_sqe(ring);
if (!sqe) {
fprintf(stderr, "%s: failed to get sqe\n", __FUNCTION__);
return 1;
}
io_uring_prep_readv(sqe, index, &iov[1], 1, 0);
sqe->flags |= IOSQE_FIXED_FILE;
sqe->user_data = 2;
ret = io_uring_submit(ring);
if (ret != 1) {
fprintf(stderr, "%s: got %d, wanted 1\n", __FUNCTION__, ret);
return 1;
}
ret = io_uring_wait_cqe(ring, &cqe);
if (ret < 0) {
fprintf(stderr, "%s: io_uring_wait_cqe=%d\n", __FUNCTION__, ret);
return 1;
}
if (cqe->res != 4096) {
fprintf(stderr, "%s: read cqe->res=%d\n", __FUNCTION__, cqe->res);
return 1;
}
io_uring_cqe_seen(ring, cqe);
if (memcmp(iov[1].iov_base, iov[0].iov_base, 4096)) {
fprintf(stderr, "%s: data mismatch\n", __FUNCTION__);
return 1;
}
free(iov[0].iov_base);
free(iov[1].iov_base);
return 0;
}
/*
* Register 8K of sparse files, update one at a random spot, then do some
* file IO to verify it works.
*/
static int test_huge(struct io_uring *ring)
{
int *files;
int ret;
files = open_files(0, 8192, 0);
ret = io_uring_register_files(ring, files, 8192);
if (ret) {
/* huge sets not supported */
if (ret == -EMFILE) {
fprintf(stdout, "%s: No huge file set support, skipping\n", __FUNCTION__);
goto out;
}
fprintf(stderr, "%s: register ret=%d\n", __FUNCTION__, ret);
goto err;
}
files[7193] = open(".reg.7193", O_RDWR | O_CREAT, 0644);
if (files[7193] < 0) {
fprintf(stderr, "%s: open=%d\n", __FUNCTION__, errno);
goto err;
}
ret = io_uring_register_files_update(ring, 7193, &files[7193], 1);
if (ret != 1) {
fprintf(stderr, "%s: update ret=%d\n", __FUNCTION__, ret);
goto err;
}
if (test_fixed_read_write(ring, 7193))
goto err;
ret = io_uring_unregister_files(ring);
if (ret) {
fprintf(stderr, "%s: unregister ret=%d\n", __FUNCTION__, ret);
goto err;
}
if (files[7193] != -1) {
close(files[7193]);
unlink(".reg.7193");
}
out:
free(files);
return 0;
err:
if (files[7193] != -1) {
close(files[7193]);
unlink(".reg.7193");
}
free(files);
return 1;
}
static int test_skip(struct io_uring *ring)
{
int *files;
int ret;
files = open_files(100, 0, 0);
ret = io_uring_register_files(ring, files, 100);
if (ret) {
fprintf(stderr, "%s: register ret=%d\n", __FUNCTION__, ret);
goto err;
}
files[90] = IORING_REGISTER_FILES_SKIP;
ret = io_uring_register_files_update(ring, 90, &files[90], 1);
if (ret != 1) {
if (ret == -EBADF) {
fprintf(stdout, "Skipping files not supported\n");
goto done;
}
fprintf(stderr, "%s: update ret=%d\n", __FUNCTION__, ret);
goto err;
}
/* verify can still use file index 90 */
if (test_fixed_read_write(ring, 90))
goto err;
ret = io_uring_unregister_files(ring);
if (ret) {
fprintf(stderr, "%s: unregister ret=%d\n", __FUNCTION__, ret);
goto err;
}
done:
close_files(files, 100, 0);
return 0;
err:
close_files(files, 100, 0);
return 1;
}
static int test_sparse_updates(void)
{
struct io_uring ring;
int ret, i, *fds, newfd;
ret = io_uring_queue_init(8, &ring, 0);
if (ret) {
fprintf(stderr, "queue_init: %d\n", ret);
return ret;
}
fds = t_malloc(256 * sizeof(int));
for (i = 0; i < 256; i++)
fds[i] = -1;
ret = io_uring_register_files(&ring, fds, 256);
if (ret) {
fprintf(stderr, "file_register: %d\n", ret);
return ret;
}
newfd = 1;
for (i = 0; i < 256; i++) {
ret = io_uring_register_files_update(&ring, i, &newfd, 1);
if (ret != 1) {
fprintf(stderr, "file_update: %d\n", ret);
return ret;
}
}
io_uring_unregister_files(&ring);
for (i = 0; i < 256; i++)
fds[i] = 1;
ret = io_uring_register_files(&ring, fds, 256);
if (ret) {
fprintf(stderr, "file_register: %d\n", ret);
return ret;
}
newfd = -1;
for (i = 0; i < 256; i++) {
ret = io_uring_register_files_update(&ring, i, &newfd, 1);
if (ret != 1) {
fprintf(stderr, "file_update: %d\n", ret);
return ret;
}
}
io_uring_unregister_files(&ring);
io_uring_queue_exit(&ring);
return 0;
}
static int test_fixed_removal_ordering(void)
{
char buffer[128];
struct io_uring ring;
struct io_uring_sqe *sqe;
struct io_uring_cqe *cqe;
struct __kernel_timespec ts;
int ret, fd, i, fds[2];
ret = io_uring_queue_init(8, &ring, 0);
if (ret < 0) {
fprintf(stderr, "failed to init io_uring: %s\n", strerror(-ret));
return ret;
}
if (pipe(fds)) {
perror("pipe");
return -1;
}
ret = io_uring_register_files(&ring, fds, 2);
if (ret) {
fprintf(stderr, "file_register: %d\n", ret);
return ret;
}
/* ring should have fds referenced, can close them */
close(fds[0]);
close(fds[1]);
sqe = io_uring_get_sqe(&ring);
if (!sqe) {
fprintf(stderr, "%s: get sqe failed\n", __FUNCTION__);
return 1;
}
/* outwait file recycling delay */
ts.tv_sec = 3;
ts.tv_nsec = 0;
io_uring_prep_timeout(sqe, &ts, 0, 0);
sqe->flags |= IOSQE_IO_LINK | IOSQE_IO_HARDLINK;
sqe->user_data = 1;
sqe = io_uring_get_sqe(&ring);
if (!sqe) {
printf("get sqe failed\n");
return -1;
}
io_uring_prep_write(sqe, 1, buffer, sizeof(buffer), 0);
sqe->flags |= IOSQE_FIXED_FILE;
sqe->user_data = 2;
ret = io_uring_submit(&ring);
if (ret != 2) {
fprintf(stderr, "%s: got %d, wanted 2\n", __FUNCTION__, ret);
return -1;
}
/* remove unused pipe end */
fd = -1;
ret = io_uring_register_files_update(&ring, 0, &fd, 1);
if (ret != 1) {
fprintf(stderr, "update off=0 failed\n");
return -1;
}
/* remove used pipe end */
fd = -1;
ret = io_uring_register_files_update(&ring, 1, &fd, 1);
if (ret != 1) {
fprintf(stderr, "update off=1 failed\n");
return -1;
}
for (i = 0; i < 2; ++i) {
ret = io_uring_wait_cqe(&ring, &cqe);
if (ret < 0) {
fprintf(stderr, "%s: io_uring_wait_cqe=%d\n", __FUNCTION__, ret);
return 1;
}
io_uring_cqe_seen(&ring, cqe);
}
io_uring_queue_exit(&ring);
return 0;
}
int main(int argc, char *argv[])
{
struct io_uring ring;
int ret;
if (argc > 1)
return 0;
ret = io_uring_queue_init(8, &ring, 0);
if (ret) {
printf("ring setup failed\n");
return 1;
}
ret = test_basic(&ring, 0);
if (ret) {
printf("test_basic failed\n");
return ret;
}
ret = test_basic(&ring, 1);
if (ret) {
printf("test_basic failed\n");
return ret;
}
ret = test_basic_many(&ring);
if (ret) {
printf("test_basic_many failed\n");
return ret;
}
ret = test_sparse(&ring);
if (ret) {
printf("test_sparse failed\n");
return ret;
}
if (no_update)
return 0;
ret = test_additions(&ring);
if (ret) {
printf("test_additions failed\n");
return ret;
}
ret = test_removals(&ring);
if (ret) {
printf("test_removals failed\n");
return ret;
}
ret = test_replace(&ring);
if (ret) {
printf("test_replace failed\n");
return ret;
}
ret = test_replace_all(&ring);
if (ret) {
printf("test_replace_all failed\n");
return ret;
}
ret = test_grow(&ring);
if (ret) {
printf("test_grow failed\n");
return ret;
}
ret = test_shrink(&ring);
if (ret) {
printf("test_shrink failed\n");
return ret;
}
ret = test_zero(&ring);
if (ret) {
printf("test_zero failed\n");
return ret;
}
ret = test_huge(&ring);
if (ret) {
printf("test_huge failed\n");
return ret;
}
ret = test_skip(&ring);
if (ret) {
printf("test_skip failed\n");
return 1;
}
ret = test_sparse_updates();
if (ret) {
printf("test_sparse_updates failed\n");
return ret;
}
ret = test_fixed_removal_ordering();
if (ret) {
printf("test_fixed_removal_ordering failed\n");
return 1;
}
return 0;
}