/* SPDX-License-Identifier: MIT */ /* * Description: run various fixed file fd passing tests * */ #include #include #include #include #include #include #include "liburing.h" #include "helpers.h" #define FSIZE 128 #define PAT 0x9a #define USER_DATA 0x89 static int no_fd_pass; static int verify_fixed_read(struct io_uring *ring, int fixed_fd, int fail) { struct io_uring_sqe *sqe; struct io_uring_cqe *cqe; unsigned char buf[FSIZE]; int i; sqe = io_uring_get_sqe(ring); io_uring_prep_read(sqe, fixed_fd, buf, FSIZE, 0); sqe->flags |= IOSQE_FIXED_FILE; io_uring_submit(ring); io_uring_wait_cqe(ring, &cqe); if (cqe->res != FSIZE) { if (fail && cqe->res == -EBADF) return 0; fprintf(stderr, "Read: %d\n", cqe->res); return 1; } io_uring_cqe_seen(ring, cqe); for (i = 0; i < FSIZE; i++) { if (buf[i] != PAT) { fprintf(stderr, "got %x, wanted %x\n", buf[i], PAT); return 1; } } return 0; } static int test(const char *filename, int source_fd, int target_fd) { struct io_uring sring, dring; struct io_uring_sqe *sqe; struct io_uring_cqe *cqe; int ret; ret = io_uring_queue_init(8, &sring, 0); if (ret) { fprintf(stderr, "ring setup failed: %d\n", ret); return T_EXIT_FAIL; } ret = io_uring_queue_init(8, &dring, 0); if (ret) { fprintf(stderr, "ring setup failed: %d\n", ret); return T_EXIT_FAIL; } ret = io_uring_register_files_sparse(&sring, 8); if (ret) { if (ret == -EINVAL) return T_EXIT_SKIP; fprintf(stderr, "register files failed %d\n", ret); return T_EXIT_FAIL; } ret = io_uring_register_files_sparse(&dring, 8); if (ret) { fprintf(stderr, "register files failed %d\n", ret); return T_EXIT_FAIL; } if (target_fd == IORING_FILE_INDEX_ALLOC) { /* we want to test installing into a non-zero slot */ ret = io_uring_register_file_alloc_range(&dring, 1, 1); if (ret) { fprintf(stderr, "io_uring_register_file_alloc_range %d\n", ret); return T_EXIT_FAIL; } } /* open direct descriptor */ sqe = io_uring_get_sqe(&sring); io_uring_prep_openat_direct(sqe, AT_FDCWD, filename, 0, 0644, source_fd); io_uring_submit(&sring); ret = io_uring_wait_cqe(&sring, &cqe); if (ret) { fprintf(stderr, "wait cqe failed %d\n", ret); return T_EXIT_FAIL; } if (cqe->res) { fprintf(stderr, "cqe res %d\n", cqe->res); return T_EXIT_FAIL; } io_uring_cqe_seen(&sring, cqe); /* verify data is sane for source ring */ if (verify_fixed_read(&sring, source_fd, 0)) return T_EXIT_FAIL; /* send direct descriptor to destination ring */ sqe = io_uring_get_sqe(&sring); if (target_fd == IORING_FILE_INDEX_ALLOC) { io_uring_prep_msg_ring_fd_alloc(sqe, dring.ring_fd, source_fd, USER_DATA, 0); } else { io_uring_prep_msg_ring_fd(sqe, dring.ring_fd, source_fd, target_fd, USER_DATA, 0); } io_uring_submit(&sring); ret = io_uring_wait_cqe(&sring, &cqe); if (ret) { fprintf(stderr, "wait cqe failed %d\n", ret); return T_EXIT_FAIL; } if (cqe->res < 0) { if (cqe->res == -EINVAL && !no_fd_pass) { no_fd_pass = 1; return T_EXIT_SKIP; } fprintf(stderr, "msg_ring failed %d\n", cqe->res); return T_EXIT_FAIL; } io_uring_cqe_seen(&sring, cqe); /* get posted completion for the passing */ ret = io_uring_wait_cqe(&dring, &cqe); if (ret) { fprintf(stderr, "wait cqe failed %d\n", ret); return T_EXIT_FAIL; } if (cqe->user_data != USER_DATA) { fprintf(stderr, "bad user_data %ld\n", (long) cqe->res); return T_EXIT_FAIL; } if (cqe->res < 0) { fprintf(stderr, "bad result %i\n", cqe->res); return T_EXIT_FAIL; } if (target_fd == IORING_FILE_INDEX_ALLOC) { if (cqe->res != 1) { fprintf(stderr, "invalid allocated index %i\n", cqe->res); return T_EXIT_FAIL; } target_fd = cqe->res; } io_uring_cqe_seen(&dring, cqe); /* now verify we can read the sane data from the destination ring */ if (verify_fixed_read(&dring, target_fd, 0)) return T_EXIT_FAIL; /* close descriptor in source ring */ sqe = io_uring_get_sqe(&sring); io_uring_prep_close_direct(sqe, source_fd); io_uring_submit(&sring); ret = io_uring_wait_cqe(&sring, &cqe); if (ret) { fprintf(stderr, "wait cqe failed %d\n", ret); return T_EXIT_FAIL; } if (cqe->res) { fprintf(stderr, "direct close failed %d\n", cqe->res); return T_EXIT_FAIL; } io_uring_cqe_seen(&sring, cqe); /* check that source ring fails after close */ if (verify_fixed_read(&sring, source_fd, 1)) return T_EXIT_FAIL; /* check we can still read from destination ring */ if (verify_fixed_read(&dring, target_fd, 0)) return T_EXIT_FAIL; io_uring_queue_exit(&sring); io_uring_queue_exit(&dring); return T_EXIT_PASS; } int main(int argc, char *argv[]) { char fname[80]; int ret; if (argc > 1) return T_EXIT_SKIP; sprintf(fname, ".fd-pass.%d", getpid()); t_create_file_pattern(fname, FSIZE, PAT); ret = test(fname, 0, 1); if (ret == T_EXIT_FAIL) { fprintf(stderr, "test failed 0 1\n"); ret = T_EXIT_FAIL; } ret = test(fname, 0, 2); if (ret == T_EXIT_FAIL) { fprintf(stderr, "test failed 0 2\n"); ret = T_EXIT_FAIL; } ret = test(fname, 1, 1); if (ret == T_EXIT_FAIL) { fprintf(stderr, "test failed 1 1\n"); ret = T_EXIT_FAIL; } ret = test(fname, 1, 0); if (ret == T_EXIT_FAIL) { fprintf(stderr, "test failed 1 0\n"); ret = T_EXIT_FAIL; } ret = test(fname, 1, IORING_FILE_INDEX_ALLOC); if (ret == T_EXIT_FAIL) { fprintf(stderr, "test failed 1 ALLOC\n"); ret = T_EXIT_FAIL; } unlink(fname); return ret; }