/* SPDX-License-Identifier: MIT */ /* * Test io_uring_setup with IORING_SETUP_REGISTERED_FD_ONLY * */ #include #include "helpers.h" #define NORMAL_PAGE_ENTRIES 8 #define HUGE_PAGE_ENTRIES 512 static int no_mmap; static int test_nops(struct io_uring *ring, int sq_size, int nr_nops) { struct io_uring_sqe *sqe; struct io_uring_cqe *cqe; int i, ret; do { int todo = nr_nops; if (todo > sq_size) todo = sq_size; for (i = 0; i < todo; i++) { sqe = io_uring_get_sqe(ring); io_uring_prep_nop(sqe); } ret = io_uring_submit(ring); if (ret != todo) { fprintf(stderr, "short submit %d\n", ret); return T_EXIT_FAIL; } for (i = 0; i < todo; i++) { ret = io_uring_wait_cqe(ring, &cqe); if (ret) { fprintf(stderr, "wait err %d\n", ret); return T_EXIT_FAIL; } io_uring_cqe_seen(ring, cqe); } nr_nops -= todo; } while (nr_nops); return T_EXIT_PASS; } static int test(int nentries, int ring_flags) { struct io_uring ring; unsigned values[2]; int ret; ret = io_uring_queue_init(nentries, &ring, IORING_SETUP_REGISTERED_FD_ONLY | IORING_SETUP_NO_MMAP | ring_flags); if (ret == -EINVAL) { no_mmap = 1; return T_EXIT_SKIP; } else if (ret == -ENOMEM) { fprintf(stdout, "Enable huge pages to test big rings\n"); return T_EXIT_SKIP; } else if (ret) { fprintf(stderr, "ring setup failed: %d\n", ret); return T_EXIT_FAIL; } ret = io_uring_register_ring_fd(&ring); if (ret != -EEXIST) { fprintf(stderr, "registering already-registered ring fd should fail\n"); goto err; } ret = io_uring_close_ring_fd(&ring); if (ret != -EBADF) { fprintf(stderr, "closing already-closed ring fd should fail\n"); goto err; } /* Test a simple io_uring_register operation expected to work. * io_uring_register_iowq_max_workers is arbitrary. */ values[0] = values[1] = 0; ret = io_uring_register_iowq_max_workers(&ring, values); if (ret || (values[0] == 0 && values[1] == 0)) { fprintf(stderr, "io_uring_register operation failed after closing ring fd\n"); goto err; } ret = test_nops(&ring, nentries, nentries * 4); if (ret) goto err; io_uring_queue_exit(&ring); return T_EXIT_PASS; err: io_uring_queue_exit(&ring); return T_EXIT_FAIL; } int main(int argc, char *argv[]) { int ret; if (argc > 1) return T_EXIT_SKIP; /* test single normal page */ ret = test(NORMAL_PAGE_ENTRIES, 0); if (ret == T_EXIT_SKIP || no_mmap) { return T_EXIT_SKIP; } else if (ret != T_EXIT_PASS) { fprintf(stderr, "test 8 failed\n"); return T_EXIT_FAIL; } /* test single normal page */ ret = test(NORMAL_PAGE_ENTRIES, IORING_SETUP_SQPOLL); if (ret == T_EXIT_SKIP || no_mmap) { return T_EXIT_SKIP; } else if (ret != T_EXIT_PASS) { fprintf(stderr, "test 8 failed\n"); return T_EXIT_FAIL; } /* test with entries requiring a huge page */ ret = test(HUGE_PAGE_ENTRIES, 0); if (ret == T_EXIT_SKIP) { return T_EXIT_SKIP; } else if (ret != T_EXIT_PASS) { fprintf(stderr, "test 512 failed\n"); return T_EXIT_FAIL; } return T_EXIT_PASS; }