fsync.c 4.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225
  1. #include "../config-host.h"
  2. /* SPDX-License-Identifier: MIT */
  3. /*
  4. * Description: test io_uring fsync handling
  5. *
  6. */
  7. #include <errno.h>
  8. #include <stdio.h>
  9. #include <unistd.h>
  10. #include <stdlib.h>
  11. #include <string.h>
  12. #include <fcntl.h>
  13. #include "helpers.h"
  14. #include "liburing.h"
  15. static int test_single_fsync(struct io_uring *ring)
  16. {
  17. struct io_uring_cqe *cqe;
  18. struct io_uring_sqe *sqe;
  19. char buf[32];
  20. int fd, ret;
  21. sprintf(buf, "./XXXXXX");
  22. fd = mkstemp(buf);
  23. if (fd < 0) {
  24. perror("open");
  25. return 1;
  26. }
  27. sqe = io_uring_get_sqe(ring);
  28. if (!sqe) {
  29. fprintf(stderr, "get sqe failed\n");
  30. goto err;
  31. }
  32. io_uring_prep_fsync(sqe, fd, 0);
  33. ret = io_uring_submit(ring);
  34. if (ret <= 0) {
  35. fprintf(stderr, "sqe submit failed: %d\n", ret);
  36. goto err;
  37. }
  38. ret = io_uring_wait_cqe(ring, &cqe);
  39. if (ret < 0) {
  40. fprintf(stderr, "wait completion %d\n", ret);
  41. goto err;
  42. }
  43. io_uring_cqe_seen(ring, cqe);
  44. unlink(buf);
  45. return 0;
  46. err:
  47. unlink(buf);
  48. return 1;
  49. }
  50. static int test_barrier_fsync(struct io_uring *ring)
  51. {
  52. struct io_uring_cqe *cqe;
  53. struct io_uring_sqe *sqe;
  54. struct iovec iovecs[4];
  55. int i, fd, ret;
  56. off_t off;
  57. fd = open("fsync-testfile", O_WRONLY | O_CREAT, 0644);
  58. if (fd < 0) {
  59. perror("open");
  60. return 1;
  61. }
  62. unlink("fsync-testfile");
  63. for (i = 0; i < ARRAY_SIZE(iovecs); i++) {
  64. iovecs[i].iov_base = t_malloc(4096);
  65. iovecs[i].iov_len = 4096;
  66. }
  67. off = 0;
  68. for (i = 0; i < 4; i++) {
  69. sqe = io_uring_get_sqe(ring);
  70. if (!sqe) {
  71. fprintf(stderr, "get sqe failed\n");
  72. goto err;
  73. }
  74. io_uring_prep_writev(sqe, fd, &iovecs[i], 1, off);
  75. sqe->user_data = 0;
  76. off += 4096;
  77. }
  78. sqe = io_uring_get_sqe(ring);
  79. if (!sqe) {
  80. fprintf(stderr, "get sqe failed\n");
  81. goto err;
  82. }
  83. io_uring_prep_fsync(sqe, fd, IORING_FSYNC_DATASYNC);
  84. sqe->user_data = 1;
  85. io_uring_sqe_set_flags(sqe, IOSQE_IO_DRAIN);
  86. ret = io_uring_submit(ring);
  87. if (ret < 0) {
  88. fprintf(stderr, "sqe submit failed: %d\n", ret);
  89. goto err;
  90. } else if (ret < 5) {
  91. fprintf(stderr, "Submitted only %d\n", ret);
  92. goto err;
  93. }
  94. for (i = 0; i < 5; i++) {
  95. ret = io_uring_wait_cqe(ring, &cqe);
  96. if (ret < 0) {
  97. fprintf(stderr, "wait completion %d\n", ret);
  98. goto err;
  99. }
  100. /* kernel doesn't support IOSQE_IO_DRAIN */
  101. if (cqe->res == -EINVAL)
  102. break;
  103. if (i <= 3) {
  104. if (cqe->user_data) {
  105. fprintf(stderr, "Got fsync early?\n");
  106. goto err;
  107. }
  108. } else {
  109. if (!cqe->user_data) {
  110. fprintf(stderr, "Got write late?\n");
  111. goto err;
  112. }
  113. }
  114. io_uring_cqe_seen(ring, cqe);
  115. }
  116. ret = 0;
  117. goto out;
  118. err:
  119. ret = 1;
  120. out:
  121. for (i = 0; i < ARRAY_SIZE(iovecs); i++)
  122. free(iovecs[i].iov_base);
  123. return ret;
  124. }
  125. #define FILE_SIZE 1024
  126. static int test_sync_file_range(struct io_uring *ring)
  127. {
  128. int ret, fd, save_errno;
  129. struct io_uring_sqe *sqe;
  130. struct io_uring_cqe *cqe;
  131. t_create_file(".sync_file_range", FILE_SIZE);
  132. fd = open(".sync_file_range", O_RDWR);
  133. save_errno = errno;
  134. unlink(".sync_file_range");
  135. errno = save_errno;
  136. if (fd < 0) {
  137. perror("file open");
  138. return 1;
  139. }
  140. sqe = io_uring_get_sqe(ring);
  141. if (!sqe) {
  142. fprintf(stderr, "sqe get failed\n");
  143. return 1;
  144. }
  145. io_uring_prep_sync_file_range(sqe, fd, 0, 0, 0);
  146. sqe->user_data = 1;
  147. ret = io_uring_submit(ring);
  148. if (ret != 1) {
  149. fprintf(stderr, "submit failed: %d\n", ret);
  150. return 1;
  151. }
  152. ret = io_uring_wait_cqe(ring, &cqe);
  153. if (ret) {
  154. fprintf(stderr, "wait_cqe failed: %d\n", ret);
  155. return 1;
  156. }
  157. if (cqe->res) {
  158. fprintf(stderr, "sfr failed: %d\n", cqe->res);
  159. return 1;
  160. }
  161. io_uring_cqe_seen(ring, cqe);
  162. return 0;
  163. }
  164. int main(int argc, char *argv[])
  165. {
  166. struct io_uring ring;
  167. int ret;
  168. if (argc > 1)
  169. return T_EXIT_SKIP;
  170. ret = io_uring_queue_init(8, &ring, 0);
  171. if (ret) {
  172. fprintf(stderr, "ring setup failed\n");
  173. return T_EXIT_FAIL;
  174. }
  175. ret = test_single_fsync(&ring);
  176. if (ret) {
  177. fprintf(stderr, "test_single_fsync failed\n");
  178. return ret;
  179. }
  180. ret = test_barrier_fsync(&ring);
  181. if (ret) {
  182. fprintf(stderr, "test_barrier_fsync failed\n");
  183. return ret;
  184. }
  185. ret = test_sync_file_range(&ring);
  186. if (ret) {
  187. fprintf(stderr, "test_sync_file_range failed\n");
  188. return ret;
  189. }
  190. return T_EXIT_PASS;
  191. }