/* SPDX-License-Identifier: MIT */ /* * Description: run various truncate tests * */ #include #include #include #include #include #include #include #include #include "liburing.h" #include "helpers.h" #define TWO_GIG_SIZE ((loff_t)2 * 1024 * 1024 * 1024) #define ONE_GIG_SIZE ((loff_t)1024 * 1024 * 1024) #define HALF_GIG_SIZE ((loff_t)512 * 1024 * 1024) static int test_truncate(struct io_uring *ring, int fd) { struct io_uring_cqe *cqe; struct io_uring_sqe *sqe; int ret = -1; sqe = io_uring_get_sqe(ring); if (!sqe) { fprintf(stderr, "get sqe failed\n"); return T_EXIT_FAIL; } memset(sqe, 0, sizeof(*sqe)); io_uring_prep_rw(IORING_OP_FTRUNCATE, sqe, fd, "fail", 0, 4); ret = io_uring_submit(ring); if (ret <= 0) { fprintf(stderr, "sqe submit failed: %d\n", ret); return T_EXIT_FAIL; } ret = io_uring_wait_cqe(ring, &cqe); if (ret < 0) { fprintf(stderr, "wait completion %d\n", ret); return T_EXIT_FAIL; } ret = cqe->res; io_uring_cqe_seen(ring, cqe); if (ret == -EINVAL) return T_EXIT_PASS; fprintf(stderr, "unexpected truncate res %d\n", ret); return T_EXIT_FAIL; } static int test_ftruncate(struct io_uring *ring, int fd, loff_t len) { struct io_uring_cqe *cqe; struct io_uring_sqe *sqe; int ret; sqe = io_uring_get_sqe(ring); if (!sqe) { fprintf(stderr, "get sqe failed\n"); goto err; } memset(sqe, 0, sizeof(*sqe)); io_uring_prep_ftruncate(sqe, fd, len); ret = io_uring_submit(ring); if (ret <= 0) { fprintf(stderr, "sqe submit failed: %d\n", ret); goto err; } ret = io_uring_wait_cqe(ring, &cqe); if (ret < 0) { fprintf(stderr, "wait completion %d\n", ret); goto err; } ret = cqe->res; io_uring_cqe_seen(ring, cqe); return ret; err: return 1; } static int get_file_size(int fd, loff_t *size) { struct stat st; if (fstat(fd, &st) < 0) { perror("fstat"); return -1; } if (S_ISREG(st.st_mode)) { *size = st.st_size; return 0; } else if (S_ISBLK(st.st_mode)) { unsigned long long bytes; if (ioctl(fd, BLKGETSIZE64, &bytes) != 0) { perror("ioctl"); return -1; } *size = bytes; return 0; } return -1; } int main(int argc, char *argv[]) { struct io_uring ring; char path[32] = ".truncate.XXXXXX"; int ret; int fd; int i; loff_t size; loff_t test_sizes[3]; if (argc > 1) return T_EXIT_SKIP; ret = io_uring_queue_init(1, &ring, 0); if (ret) { fprintf(stderr, "ring setup failed: %d\n", ret); return T_EXIT_FAIL; } fd = mkostemp(path, O_WRONLY | O_CREAT | O_TRUNC); if (fd < 0) { perror("mkostemp"); return T_EXIT_FAIL; } test_sizes[0] = TWO_GIG_SIZE; test_sizes[1] = ONE_GIG_SIZE; test_sizes[2] = HALF_GIG_SIZE; for (i = 0; i < 3; i++) { ret = test_ftruncate(&ring, fd, test_sizes[i]); if (ret < 0) { if (ret == -EBADF || ret == -EINVAL) { if (i == 0) { fprintf(stdout, "Ftruncate not supported, skipping\n"); ret = T_EXIT_SKIP; goto out; } goto err; } fprintf(stderr, "ftruncate: %s\n", strerror(-ret)); goto err; } else if (ret) { fprintf(stderr, "unexpected cqe->res %d\n", ret); goto err; } if (get_file_size(fd, &size)) goto err; if (size != test_sizes[i]) { fprintf(stderr, "fail %d size=%llu, %llu\n", i, (unsigned long long) size, (unsigned long long) test_sizes[i]); goto err; } } ret = test_truncate(&ring, fd); if (ret != T_EXIT_PASS) goto err; out: unlink(path); close(fd); return T_EXIT_PASS; err: unlink(path); close(fd); return T_EXIT_FAIL; }