summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorJens Axboe <axboe@kernel.dk>2019-04-06 21:48:39 -0600
committerJens Axboe <axboe@kernel.dk>2019-04-06 21:50:15 -0600
commit61c27f73f8897bb65b2d52942011e4db5306d87a (patch)
tree98840d13734fb607fa7ddefaaadeb7cad7518198
parentf8b12e48bffc2c74d40a67c8af49b5a32c0f25ac (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>
-rw-r--r--src/io_uring.h1
-rw-r--r--src/liburing.h5
-rw-r--r--test/fsync.c126
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;
}