diff options
author | Jens Axboe <axboe@kernel.dk> | 2019-04-06 21:48:39 -0600 |
---|---|---|
committer | Jens Axboe <axboe@kernel.dk> | 2019-04-06 21:50:15 -0600 |
commit | 61c27f73f8897bb65b2d52942011e4db5306d87a (patch) | |
tree | 98840d13734fb607fa7ddefaaadeb7cad7518198 /test | |
parent | f8b12e48bffc2c74d40a67c8af49b5a32c0f25ac (diff) |
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 <axboe@kernel.dk>
Diffstat (limited to 'test')
-rw-r--r-- | test/fsync.c | 126 |
1 files changed, 110 insertions, 16 deletions
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; } |