From 61c27f73f8897bb65b2d52942011e4db5306d87a Mon Sep 17 00:00:00 2001 From: Jens Axboe Date: Sat, 6 Apr 2019 21:48:39 -0600 Subject: Add barrier fsync test case Also changes the fsync prep helper to require passing in the actual flag, not just a boolean for fsync vs fdatasync. Signed-off-by: Jens Axboe --- src/io_uring.h | 1 + src/liburing.h | 5 +-- test/fsync.c | 126 +++++++++++++++++++++++++++++++++++++++++++++++++-------- 3 files changed, 113 insertions(+), 19 deletions(-) diff --git a/src/io_uring.h b/src/io_uring.h index e234086..57b8f4d 100644 --- a/src/io_uring.h +++ b/src/io_uring.h @@ -59,6 +59,7 @@ struct io_uring_sqe { * sqe->fsync_flags */ #define IORING_FSYNC_DATASYNC (1U << 0) +#define IORING_FSYNC_BARRIER (1U << 1) /* * IO completion data structure (Completion Queue Entry) diff --git a/src/liburing.h b/src/liburing.h index 1cea6a5..5c8b01a 100644 --- a/src/liburing.h +++ b/src/liburing.h @@ -131,13 +131,12 @@ static inline void io_uring_prep_poll_remove(struct io_uring_sqe *sqe, } static inline void io_uring_prep_fsync(struct io_uring_sqe *sqe, int fd, - int datasync) + unsigned fsync_flags) { memset(sqe, 0, sizeof(*sqe)); sqe->opcode = IORING_OP_FSYNC; sqe->fd = fd; - if (datasync) - sqe->fsync_flags = IORING_FSYNC_DATASYNC; + sqe->fsync_flags = fsync_flags; } #endif diff --git a/test/fsync.c b/test/fsync.c index 064f027..771e0d5 100644 --- a/test/fsync.c +++ b/test/fsync.c @@ -11,11 +11,10 @@ #include "../src/liburing.h" -int main(int argc, char *argv[]) +static int test_single_fsync(struct io_uring *ring) { struct io_uring_cqe *cqe; struct io_uring_sqe *sqe; - struct io_uring ring; char buf[32]; int fd, ret; @@ -25,34 +24,129 @@ int main(int argc, char *argv[]) perror("open"); return 1; } - - ret = io_uring_queue_init(1, &ring, 0); - if (ret) { - printf("ring setup failed\n"); + + sqe = io_uring_get_sqe(ring); + if (!sqe) { + printf("get sqe failed\n"); + goto err; + } + + io_uring_prep_fsync(sqe, fd, 0); + + ret = io_uring_submit(ring); + if (ret <= 0) { + printf("sqe submit failed\n"); + goto err; + } + + ret = io_uring_wait_completion(ring, &cqe); + if (ret < 0) { + printf("wait completion %d\n", ret); + goto err; + } + + unlink(buf); + return 0; +err: + unlink(buf); + return 1; +} + +static int test_barrier_fsync(struct io_uring *ring) +{ + struct io_uring_cqe *cqe; + struct io_uring_sqe *sqe; + struct iovec iovecs[4]; + int i, fd, ret; + off_t off; + + fd = open("testfile", O_WRONLY | O_CREAT, 0644); + if (fd < 0) { + perror("open"); return 1; + } + for (i = 0; i < 4; i++) { + iovecs[i].iov_base = malloc(4096); + iovecs[i].iov_len = 4096; } - sqe = io_uring_get_sqe(&ring); + off = 0; + for (i = 0; i < 4; i++) { + sqe = io_uring_get_sqe(ring); + if (!sqe) { + printf("get sqe failed\n"); + goto err; + } + + io_uring_prep_writev(sqe, fd, &iovecs[i], 1, off); + sqe->user_data = 0; + off += 4096; + } + + sqe = io_uring_get_sqe(ring); if (!sqe) { printf("get sqe failed\n"); - return 1; + goto err; } - io_uring_prep_fsync(sqe, fd, 0); + io_uring_prep_fsync(sqe, fd, + IORING_FSYNC_DATASYNC | IORING_FSYNC_BARRIER); + sqe->user_data = 1; - ret = io_uring_submit(&ring); + ret = io_uring_submit(ring); if (ret <= 0) { - printf("child: sqe submit failed\n"); - return 1; + printf("sqe submit failed\n"); + if (ret == EINVAL) + printf("kernel may not support barrier fsync yet\n"); + goto err; } - ret = io_uring_wait_completion(&ring, &cqe); - if (ret < 0) { - printf("child: wait completion %d\n", ret); + for (i = 0; i < 5; i++) { + ret = io_uring_wait_completion(ring, &cqe); + if (ret < 0) { + printf("child: wait completion %d\n", ret); + goto err; + } + if (i <= 3) { + if (cqe->user_data) { + printf("Got fsync early?\n"); + goto err; + } + } else { + if (!cqe->user_data) { + printf("Got write late?\n"); + goto err; + } + } + } + + unlink("testfile"); + return 0; +err: + unlink("testfile"); + return 1; +} + +int main(int argc, char *argv[]) +{ + struct io_uring ring; + int ret; + + ret = io_uring_queue_init(8, &ring, 0); + if (ret) { + printf("ring setup failed\n"); return 1; + } - unlink(buf); + ret = test_single_fsync(&ring); + if (ret) + return ret; + + ret = test_barrier_fsync(&ring); + if (ret) + return ret; + return 0; } -- cgit