uringmachine 0.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (307) hide show
  1. checksums.yaml +7 -0
  2. data/.github/dependabot.yml +12 -0
  3. data/.github/workflows/test.yml +35 -0
  4. data/.gitignore +59 -0
  5. data/.gitmodules +3 -0
  6. data/CHANGELOG.md +7 -0
  7. data/Gemfile +3 -0
  8. data/LICENSE +21 -0
  9. data/README.md +11 -0
  10. data/Rakefile +39 -0
  11. data/TODO.md +0 -0
  12. data/examples/echo_server.rb +52 -0
  13. data/examples/event_loop.rb +69 -0
  14. data/examples/fibers.rb +105 -0
  15. data/examples/http_server.rb +56 -0
  16. data/examples/http_server_multishot.rb +57 -0
  17. data/examples/http_server_simpler.rb +34 -0
  18. data/ext/um/extconf.rb +71 -0
  19. data/ext/um/iou.h +101 -0
  20. data/ext/um/op_ctx.c +138 -0
  21. data/ext/um/ring.c +755 -0
  22. data/ext/um/um.c +267 -0
  23. data/ext/um/um.h +97 -0
  24. data/ext/um/um_class.c +175 -0
  25. data/ext/um/um_ext.c +11 -0
  26. data/ext/um/um_op.c +87 -0
  27. data/ext/um/um_utils.c +23 -0
  28. data/lib/uringmachine/version.rb +3 -0
  29. data/lib/uringmachine.rb +8 -0
  30. data/test/helper.rb +70 -0
  31. data/test/test_iou.rb +876 -0
  32. data/test/test_um.rb +168 -0
  33. data/uringmachine.gemspec +27 -0
  34. data/vendor/liburing/.github/actions/codespell/stopwords +7 -0
  35. data/vendor/liburing/.github/pull_request_template.md +86 -0
  36. data/vendor/liburing/.github/workflows/build.yml +137 -0
  37. data/vendor/liburing/.github/workflows/codespell.yml +25 -0
  38. data/vendor/liburing/.github/workflows/shellcheck.yml +20 -0
  39. data/vendor/liburing/.gitignore +41 -0
  40. data/vendor/liburing/CHANGELOG +111 -0
  41. data/vendor/liburing/CITATION.cff +11 -0
  42. data/vendor/liburing/COPYING +502 -0
  43. data/vendor/liburing/COPYING.GPL +339 -0
  44. data/vendor/liburing/LICENSE +20 -0
  45. data/vendor/liburing/Makefile +96 -0
  46. data/vendor/liburing/Makefile.common +7 -0
  47. data/vendor/liburing/Makefile.quiet +11 -0
  48. data/vendor/liburing/README +106 -0
  49. data/vendor/liburing/SECURITY.md +6 -0
  50. data/vendor/liburing/configure +624 -0
  51. data/vendor/liburing/debian/README.Debian +7 -0
  52. data/vendor/liburing/debian/changelog +38 -0
  53. data/vendor/liburing/debian/control +39 -0
  54. data/vendor/liburing/debian/copyright +49 -0
  55. data/vendor/liburing/debian/liburing-dev.install +4 -0
  56. data/vendor/liburing/debian/liburing-dev.manpages +5 -0
  57. data/vendor/liburing/debian/liburing2.install +1 -0
  58. data/vendor/liburing/debian/liburing2.symbols +56 -0
  59. data/vendor/liburing/debian/patches/series +1 -0
  60. data/vendor/liburing/debian/rules +29 -0
  61. data/vendor/liburing/debian/source/format +1 -0
  62. data/vendor/liburing/debian/source/local-options +2 -0
  63. data/vendor/liburing/debian/source/options +1 -0
  64. data/vendor/liburing/debian/watch +3 -0
  65. data/vendor/liburing/examples/Makefile +53 -0
  66. data/vendor/liburing/examples/helpers.c +62 -0
  67. data/vendor/liburing/examples/helpers.h +7 -0
  68. data/vendor/liburing/examples/io_uring-close-test.c +123 -0
  69. data/vendor/liburing/examples/io_uring-cp.c +282 -0
  70. data/vendor/liburing/examples/io_uring-test.c +112 -0
  71. data/vendor/liburing/examples/io_uring-udp.c +403 -0
  72. data/vendor/liburing/examples/link-cp.c +193 -0
  73. data/vendor/liburing/examples/napi-busy-poll-client.c +509 -0
  74. data/vendor/liburing/examples/napi-busy-poll-server.c +450 -0
  75. data/vendor/liburing/examples/poll-bench.c +101 -0
  76. data/vendor/liburing/examples/proxy.c +2461 -0
  77. data/vendor/liburing/examples/proxy.h +102 -0
  78. data/vendor/liburing/examples/rsrc-update-bench.c +100 -0
  79. data/vendor/liburing/examples/send-zerocopy.c +658 -0
  80. data/vendor/liburing/examples/ucontext-cp.c +258 -0
  81. data/vendor/liburing/liburing-ffi.pc.in +12 -0
  82. data/vendor/liburing/liburing.pc.in +12 -0
  83. data/vendor/liburing/liburing.spec +66 -0
  84. data/vendor/liburing/make-debs.sh +55 -0
  85. data/vendor/liburing/src/Makefile +129 -0
  86. data/vendor/liburing/src/arch/aarch64/lib.h +47 -0
  87. data/vendor/liburing/src/arch/aarch64/syscall.h +91 -0
  88. data/vendor/liburing/src/arch/generic/lib.h +17 -0
  89. data/vendor/liburing/src/arch/generic/syscall.h +100 -0
  90. data/vendor/liburing/src/arch/riscv64/lib.h +48 -0
  91. data/vendor/liburing/src/arch/riscv64/syscall.h +100 -0
  92. data/vendor/liburing/src/arch/syscall-defs.h +94 -0
  93. data/vendor/liburing/src/arch/x86/lib.h +11 -0
  94. data/vendor/liburing/src/arch/x86/syscall.h +296 -0
  95. data/vendor/liburing/src/ffi.c +15 -0
  96. data/vendor/liburing/src/include/liburing/barrier.h +81 -0
  97. data/vendor/liburing/src/include/liburing/io_uring.h +818 -0
  98. data/vendor/liburing/src/include/liburing.h +1602 -0
  99. data/vendor/liburing/src/int_flags.h +11 -0
  100. data/vendor/liburing/src/lib.h +52 -0
  101. data/vendor/liburing/src/liburing-ffi.map +211 -0
  102. data/vendor/liburing/src/liburing.map +104 -0
  103. data/vendor/liburing/src/nolibc.c +55 -0
  104. data/vendor/liburing/src/queue.c +468 -0
  105. data/vendor/liburing/src/register.c +374 -0
  106. data/vendor/liburing/src/setup.c +689 -0
  107. data/vendor/liburing/src/setup.h +9 -0
  108. data/vendor/liburing/src/syscall.c +29 -0
  109. data/vendor/liburing/src/syscall.h +53 -0
  110. data/vendor/liburing/src/version.c +21 -0
  111. data/vendor/liburing/test/232c93d07b74.c +305 -0
  112. data/vendor/liburing/test/35fa71a030ca.c +329 -0
  113. data/vendor/liburing/test/500f9fbadef8.c +91 -0
  114. data/vendor/liburing/test/7ad0e4b2f83c.c +94 -0
  115. data/vendor/liburing/test/8a9973408177.c +107 -0
  116. data/vendor/liburing/test/917257daa0fe.c +54 -0
  117. data/vendor/liburing/test/Makefile +297 -0
  118. data/vendor/liburing/test/a0908ae19763.c +59 -0
  119. data/vendor/liburing/test/a4c0b3decb33.c +181 -0
  120. data/vendor/liburing/test/accept-link.c +255 -0
  121. data/vendor/liburing/test/accept-non-empty.c +256 -0
  122. data/vendor/liburing/test/accept-reuse.c +163 -0
  123. data/vendor/liburing/test/accept-test.c +83 -0
  124. data/vendor/liburing/test/accept.c +919 -0
  125. data/vendor/liburing/test/across-fork.c +284 -0
  126. data/vendor/liburing/test/b19062a56726.c +54 -0
  127. data/vendor/liburing/test/b5837bd5311d.c +78 -0
  128. data/vendor/liburing/test/bind-listen.c +408 -0
  129. data/vendor/liburing/test/buf-ring-nommap.c +123 -0
  130. data/vendor/liburing/test/buf-ring-put.c +83 -0
  131. data/vendor/liburing/test/buf-ring.c +473 -0
  132. data/vendor/liburing/test/ce593a6c480a.c +139 -0
  133. data/vendor/liburing/test/close-opath.c +123 -0
  134. data/vendor/liburing/test/config +14 -0
  135. data/vendor/liburing/test/connect-rep.c +204 -0
  136. data/vendor/liburing/test/connect.c +442 -0
  137. data/vendor/liburing/test/coredump.c +60 -0
  138. data/vendor/liburing/test/cq-full.c +97 -0
  139. data/vendor/liburing/test/cq-overflow.c +530 -0
  140. data/vendor/liburing/test/cq-peek-batch.c +103 -0
  141. data/vendor/liburing/test/cq-ready.c +95 -0
  142. data/vendor/liburing/test/cq-size.c +65 -0
  143. data/vendor/liburing/test/d4ae271dfaae.c +96 -0
  144. data/vendor/liburing/test/d77a67ed5f27.c +65 -0
  145. data/vendor/liburing/test/defer-taskrun.c +391 -0
  146. data/vendor/liburing/test/defer-tw-timeout.c +173 -0
  147. data/vendor/liburing/test/defer.c +319 -0
  148. data/vendor/liburing/test/double-poll-crash.c +195 -0
  149. data/vendor/liburing/test/drop-submit.c +94 -0
  150. data/vendor/liburing/test/eeed8b54e0df.c +120 -0
  151. data/vendor/liburing/test/empty-eownerdead.c +45 -0
  152. data/vendor/liburing/test/eploop.c +74 -0
  153. data/vendor/liburing/test/eventfd-disable.c +179 -0
  154. data/vendor/liburing/test/eventfd-reg.c +77 -0
  155. data/vendor/liburing/test/eventfd-ring.c +98 -0
  156. data/vendor/liburing/test/eventfd.c +113 -0
  157. data/vendor/liburing/test/evloop.c +73 -0
  158. data/vendor/liburing/test/exec-target.c +6 -0
  159. data/vendor/liburing/test/exit-no-cleanup.c +117 -0
  160. data/vendor/liburing/test/fadvise.c +202 -0
  161. data/vendor/liburing/test/fallocate.c +265 -0
  162. data/vendor/liburing/test/fc2a85cb02ef.c +132 -0
  163. data/vendor/liburing/test/fd-install.c +500 -0
  164. data/vendor/liburing/test/fd-pass.c +237 -0
  165. data/vendor/liburing/test/fdinfo.c +419 -0
  166. data/vendor/liburing/test/file-register.c +1189 -0
  167. data/vendor/liburing/test/file-update.c +231 -0
  168. data/vendor/liburing/test/file-verify.c +654 -0
  169. data/vendor/liburing/test/files-exit-hang-poll.c +114 -0
  170. data/vendor/liburing/test/files-exit-hang-timeout.c +137 -0
  171. data/vendor/liburing/test/fixed-buf-iter.c +115 -0
  172. data/vendor/liburing/test/fixed-buf-merge.c +101 -0
  173. data/vendor/liburing/test/fixed-hugepage.c +411 -0
  174. data/vendor/liburing/test/fixed-link.c +90 -0
  175. data/vendor/liburing/test/fixed-reuse.c +160 -0
  176. data/vendor/liburing/test/fpos.c +255 -0
  177. data/vendor/liburing/test/fsnotify.c +118 -0
  178. data/vendor/liburing/test/fsync.c +224 -0
  179. data/vendor/liburing/test/futex.c +571 -0
  180. data/vendor/liburing/test/hardlink.c +170 -0
  181. data/vendor/liburing/test/helpers.c +318 -0
  182. data/vendor/liburing/test/helpers.h +108 -0
  183. data/vendor/liburing/test/ignore-single-mmap.c +48 -0
  184. data/vendor/liburing/test/init-mem.c +164 -0
  185. data/vendor/liburing/test/io-cancel.c +561 -0
  186. data/vendor/liburing/test/io_uring_enter.c +264 -0
  187. data/vendor/liburing/test/io_uring_passthrough.c +482 -0
  188. data/vendor/liburing/test/io_uring_register.c +503 -0
  189. data/vendor/liburing/test/io_uring_setup.c +110 -0
  190. data/vendor/liburing/test/iopoll-leak.c +85 -0
  191. data/vendor/liburing/test/iopoll-overflow.c +118 -0
  192. data/vendor/liburing/test/iopoll.c +465 -0
  193. data/vendor/liburing/test/lfs-openat-write.c +119 -0
  194. data/vendor/liburing/test/lfs-openat.c +273 -0
  195. data/vendor/liburing/test/link-timeout.c +1108 -0
  196. data/vendor/liburing/test/link.c +497 -0
  197. data/vendor/liburing/test/link_drain.c +255 -0
  198. data/vendor/liburing/test/madvise.c +195 -0
  199. data/vendor/liburing/test/min-timeout-wait.c +354 -0
  200. data/vendor/liburing/test/min-timeout.c +233 -0
  201. data/vendor/liburing/test/mkdir.c +112 -0
  202. data/vendor/liburing/test/msg-ring-fd.c +331 -0
  203. data/vendor/liburing/test/msg-ring-flags.c +212 -0
  204. data/vendor/liburing/test/msg-ring-overflow.c +159 -0
  205. data/vendor/liburing/test/msg-ring.c +467 -0
  206. data/vendor/liburing/test/multicqes_drain.c +429 -0
  207. data/vendor/liburing/test/napi-test.c +215 -0
  208. data/vendor/liburing/test/napi-test.sh +48 -0
  209. data/vendor/liburing/test/no-mmap-inval.c +42 -0
  210. data/vendor/liburing/test/nolibc.c +62 -0
  211. data/vendor/liburing/test/nop-all-sizes.c +99 -0
  212. data/vendor/liburing/test/nop.c +177 -0
  213. data/vendor/liburing/test/nvme.h +169 -0
  214. data/vendor/liburing/test/ooo-file-unreg.c +82 -0
  215. data/vendor/liburing/test/open-close.c +261 -0
  216. data/vendor/liburing/test/open-direct-link.c +188 -0
  217. data/vendor/liburing/test/open-direct-pick.c +180 -0
  218. data/vendor/liburing/test/openat2.c +312 -0
  219. data/vendor/liburing/test/personality.c +204 -0
  220. data/vendor/liburing/test/pipe-bug.c +95 -0
  221. data/vendor/liburing/test/pipe-eof.c +83 -0
  222. data/vendor/liburing/test/pipe-reuse.c +105 -0
  223. data/vendor/liburing/test/poll-cancel-all.c +496 -0
  224. data/vendor/liburing/test/poll-cancel-ton.c +135 -0
  225. data/vendor/liburing/test/poll-cancel.c +228 -0
  226. data/vendor/liburing/test/poll-link.c +221 -0
  227. data/vendor/liburing/test/poll-many.c +230 -0
  228. data/vendor/liburing/test/poll-mshot-overflow.c +265 -0
  229. data/vendor/liburing/test/poll-mshot-update.c +323 -0
  230. data/vendor/liburing/test/poll-race-mshot.c +276 -0
  231. data/vendor/liburing/test/poll-race.c +105 -0
  232. data/vendor/liburing/test/poll-ring.c +48 -0
  233. data/vendor/liburing/test/poll-v-poll.c +353 -0
  234. data/vendor/liburing/test/poll.c +327 -0
  235. data/vendor/liburing/test/probe.c +135 -0
  236. data/vendor/liburing/test/read-before-exit.c +129 -0
  237. data/vendor/liburing/test/read-mshot-empty.c +153 -0
  238. data/vendor/liburing/test/read-mshot.c +404 -0
  239. data/vendor/liburing/test/read-write.c +1013 -0
  240. data/vendor/liburing/test/recv-msgall-stream.c +398 -0
  241. data/vendor/liburing/test/recv-msgall.c +263 -0
  242. data/vendor/liburing/test/recv-multishot.c +602 -0
  243. data/vendor/liburing/test/recvsend_bundle.c +691 -0
  244. data/vendor/liburing/test/reg-fd-only.c +131 -0
  245. data/vendor/liburing/test/reg-hint.c +56 -0
  246. data/vendor/liburing/test/reg-reg-ring.c +90 -0
  247. data/vendor/liburing/test/regbuf-merge.c +91 -0
  248. data/vendor/liburing/test/register-restrictions.c +633 -0
  249. data/vendor/liburing/test/rename.c +132 -0
  250. data/vendor/liburing/test/ring-leak.c +283 -0
  251. data/vendor/liburing/test/ring-leak2.c +249 -0
  252. data/vendor/liburing/test/ringbuf-read.c +196 -0
  253. data/vendor/liburing/test/ringbuf-status.c +242 -0
  254. data/vendor/liburing/test/rsrc_tags.c +461 -0
  255. data/vendor/liburing/test/runtests-loop.sh +16 -0
  256. data/vendor/liburing/test/runtests-quiet.sh +11 -0
  257. data/vendor/liburing/test/runtests.sh +168 -0
  258. data/vendor/liburing/test/rw_merge_test.c +98 -0
  259. data/vendor/liburing/test/self.c +91 -0
  260. data/vendor/liburing/test/send-zerocopy.c +971 -0
  261. data/vendor/liburing/test/send_recv.c +412 -0
  262. data/vendor/liburing/test/send_recvmsg.c +444 -0
  263. data/vendor/liburing/test/shared-wq.c +84 -0
  264. data/vendor/liburing/test/short-read.c +75 -0
  265. data/vendor/liburing/test/shutdown.c +165 -0
  266. data/vendor/liburing/test/sigfd-deadlock.c +88 -0
  267. data/vendor/liburing/test/single-issuer.c +169 -0
  268. data/vendor/liburing/test/skip-cqe.c +428 -0
  269. data/vendor/liburing/test/socket-getsetsock-cmd.c +346 -0
  270. data/vendor/liburing/test/socket-io-cmd.c +237 -0
  271. data/vendor/liburing/test/socket-rw-eagain.c +149 -0
  272. data/vendor/liburing/test/socket-rw-offset.c +149 -0
  273. data/vendor/liburing/test/socket-rw.c +137 -0
  274. data/vendor/liburing/test/socket.c +408 -0
  275. data/vendor/liburing/test/splice.c +512 -0
  276. data/vendor/liburing/test/sq-full-cpp.cc +45 -0
  277. data/vendor/liburing/test/sq-full.c +45 -0
  278. data/vendor/liburing/test/sq-poll-dup.c +211 -0
  279. data/vendor/liburing/test/sq-poll-kthread.c +169 -0
  280. data/vendor/liburing/test/sq-poll-share.c +138 -0
  281. data/vendor/liburing/test/sq-space_left.c +159 -0
  282. data/vendor/liburing/test/sqpoll-disable-exit.c +196 -0
  283. data/vendor/liburing/test/sqpoll-exec.c +132 -0
  284. data/vendor/liburing/test/sqpoll-exit-hang.c +78 -0
  285. data/vendor/liburing/test/sqpoll-sleep.c +69 -0
  286. data/vendor/liburing/test/statx.c +172 -0
  287. data/vendor/liburing/test/stdout.c +232 -0
  288. data/vendor/liburing/test/submit-and-wait.c +108 -0
  289. data/vendor/liburing/test/submit-link-fail.c +156 -0
  290. data/vendor/liburing/test/submit-reuse.c +237 -0
  291. data/vendor/liburing/test/symlink.c +117 -0
  292. data/vendor/liburing/test/sync-cancel.c +235 -0
  293. data/vendor/liburing/test/teardowns.c +58 -0
  294. data/vendor/liburing/test/test.h +36 -0
  295. data/vendor/liburing/test/thread-exit.c +143 -0
  296. data/vendor/liburing/test/timeout-new.c +256 -0
  297. data/vendor/liburing/test/timeout.c +1798 -0
  298. data/vendor/liburing/test/truncate.c +186 -0
  299. data/vendor/liburing/test/tty-write-dpoll.c +60 -0
  300. data/vendor/liburing/test/unlink.c +112 -0
  301. data/vendor/liburing/test/version.c +25 -0
  302. data/vendor/liburing/test/wait-timeout.c +287 -0
  303. data/vendor/liburing/test/waitid.c +373 -0
  304. data/vendor/liburing/test/wakeup-hang.c +162 -0
  305. data/vendor/liburing/test/wq-aff.c +146 -0
  306. data/vendor/liburing/test/xattr.c +442 -0
  307. metadata +412 -0
@@ -0,0 +1,118 @@
1
+ /* SPDX-License-Identifier: MIT */
2
+ /*
3
+ * Description: IOPOLL with overflow test case
4
+ */
5
+ #include <errno.h>
6
+ #include <stdio.h>
7
+ #include <unistd.h>
8
+ #include <stdlib.h>
9
+ #include <string.h>
10
+ #include <fcntl.h>
11
+ #include <sys/types.h>
12
+ #include <poll.h>
13
+ #include <sys/eventfd.h>
14
+ #include <sys/resource.h>
15
+ #include "helpers.h"
16
+ #include "liburing.h"
17
+ #include "../src/syscall.h"
18
+
19
+ #define FILE_SIZE (128 * 1024)
20
+ #define BS 4096
21
+ #define BUFFERS (FILE_SIZE / BS)
22
+
23
+ static struct iovec *vecs;
24
+
25
+ static int test(struct io_uring *ring, int fd)
26
+ {
27
+ struct io_uring_sqe *sqe;
28
+ int i, j, ret;
29
+ loff_t off;
30
+
31
+ off = FILE_SIZE - BS;
32
+ for (j = 0; j < 8; j++) {
33
+ for (i = 0; i < BUFFERS; i++) {
34
+ sqe = io_uring_get_sqe(ring);
35
+ io_uring_prep_read(sqe, fd, vecs[i].iov_base,
36
+ vecs[i].iov_len, off);
37
+ if (!off)
38
+ off = FILE_SIZE - BS;
39
+ else
40
+ off -= BS;
41
+ }
42
+ ret = io_uring_submit(ring);
43
+ if (ret != BUFFERS) {
44
+ fprintf(stderr, "submitted %d\n", ret);
45
+ return T_EXIT_FAIL;
46
+ }
47
+ }
48
+
49
+ sleep(1);
50
+
51
+ ret = __sys_io_uring_enter(ring->ring_fd, 0, BUFFERS * 8,
52
+ IORING_ENTER_GETEVENTS, NULL);
53
+
54
+ for (i = 0; i < BUFFERS * 8; i++) {
55
+ struct io_uring_cqe *cqe;
56
+
57
+ ret = io_uring_wait_cqe(ring, &cqe);
58
+ if (ret) {
59
+ fprintf(stderr, "wait=%d\n", ret);
60
+ return T_EXIT_FAIL;
61
+ }
62
+ io_uring_cqe_seen(ring, cqe);
63
+ }
64
+
65
+ return T_EXIT_PASS;
66
+ }
67
+
68
+ int main(int argc, char *argv[])
69
+ {
70
+ struct io_uring_params p = { };
71
+ struct io_uring ring;
72
+ char buf[256];
73
+ char *fname;
74
+ int ret, fd;
75
+
76
+ p.flags = IORING_SETUP_IOPOLL | IORING_SETUP_CQSIZE;
77
+ p.cq_entries = 64;
78
+ ret = t_create_ring_params(64, &ring, &p);
79
+ if (ret == T_SETUP_SKIP)
80
+ return 0;
81
+ if (ret != T_SETUP_OK) {
82
+ fprintf(stderr, "ring create failed: %d\n", ret);
83
+ return 1;
84
+ }
85
+
86
+ if (argc > 1) {
87
+ fname = argv[1];
88
+ } else {
89
+ srand((unsigned)time(NULL));
90
+ snprintf(buf, sizeof(buf), ".basic-rw-%u-%u",
91
+ (unsigned)rand(), (unsigned)getpid());
92
+ fname = buf;
93
+ t_create_file(fname, FILE_SIZE);
94
+ }
95
+
96
+ fd = open(fname, O_RDONLY | O_DIRECT);
97
+ if (fd < 0) {
98
+ if (errno == EINVAL) {
99
+ if (fname != argv[1])
100
+ unlink(fname);
101
+ return T_EXIT_SKIP;
102
+ }
103
+ perror("open");
104
+ goto err;
105
+ }
106
+
107
+ vecs = t_create_buffers(BUFFERS, BS);
108
+
109
+ ret = test(&ring, fd);
110
+
111
+ if (fname != argv[1])
112
+ unlink(fname);
113
+ return ret;
114
+ err:
115
+ if (fname != argv[1])
116
+ unlink(fname);
117
+ return T_EXIT_FAIL;
118
+ }
@@ -0,0 +1,465 @@
1
+ /* SPDX-License-Identifier: MIT */
2
+ /*
3
+ * Description: basic read/write tests with polled IO
4
+ */
5
+ #include <errno.h>
6
+ #include <stdio.h>
7
+ #include <unistd.h>
8
+ #include <stdlib.h>
9
+ #include <string.h>
10
+ #include <fcntl.h>
11
+ #include <sys/types.h>
12
+ #include <poll.h>
13
+ #include <sys/eventfd.h>
14
+ #include <sys/resource.h>
15
+ #include "helpers.h"
16
+ #include "liburing.h"
17
+ #include "../src/syscall.h"
18
+
19
+ #define FILE_SIZE (128 * 1024)
20
+ #define BS 4096
21
+ #define BUFFERS (FILE_SIZE / BS)
22
+
23
+ static struct iovec *vecs;
24
+ static int no_buf_select;
25
+ static int no_iopoll;
26
+
27
+ static int provide_buffers(struct io_uring *ring)
28
+ {
29
+ struct io_uring_sqe *sqe;
30
+ struct io_uring_cqe *cqe;
31
+ int ret, i;
32
+
33
+ for (i = 0; i < BUFFERS; i++) {
34
+ sqe = io_uring_get_sqe(ring);
35
+ io_uring_prep_provide_buffers(sqe, vecs[i].iov_base,
36
+ vecs[i].iov_len, 1, 1, i);
37
+ }
38
+
39
+ ret = io_uring_submit(ring);
40
+ if (ret != BUFFERS) {
41
+ fprintf(stderr, "submit: %d\n", ret);
42
+ return 1;
43
+ }
44
+
45
+ for (i = 0; i < BUFFERS; i++) {
46
+ ret = io_uring_wait_cqe(ring, &cqe);
47
+ if (cqe->res < 0) {
48
+ fprintf(stderr, "cqe->res=%d\n", cqe->res);
49
+ return 1;
50
+ }
51
+ io_uring_cqe_seen(ring, cqe);
52
+ }
53
+
54
+ return 0;
55
+ }
56
+
57
+ static int __test_io(const char *file, struct io_uring *ring, int write, int sqthread,
58
+ int fixed, int buf_select)
59
+ {
60
+ struct io_uring_sqe *sqe;
61
+ struct io_uring_cqe *cqe;
62
+ int open_flags;
63
+ int i, fd = -1, ret;
64
+ off_t offset;
65
+
66
+ if (buf_select) {
67
+ write = 0;
68
+ fixed = 0;
69
+ }
70
+ if (buf_select && provide_buffers(ring))
71
+ return 1;
72
+
73
+ if (write)
74
+ open_flags = O_WRONLY;
75
+ else
76
+ open_flags = O_RDONLY;
77
+ open_flags |= O_DIRECT;
78
+
79
+ if (fixed) {
80
+ ret = t_register_buffers(ring, vecs, BUFFERS);
81
+ if (ret == T_SETUP_SKIP)
82
+ return 0;
83
+ if (ret != T_SETUP_OK) {
84
+ fprintf(stderr, "buffer reg failed: %d\n", ret);
85
+ goto err;
86
+ }
87
+ }
88
+ fd = open(file, open_flags);
89
+ if (fd < 0) {
90
+ if (errno == EINVAL)
91
+ return 0;
92
+ perror("file open");
93
+ goto err;
94
+ }
95
+ if (sqthread) {
96
+ ret = io_uring_register_files(ring, &fd, 1);
97
+ if (ret) {
98
+ fprintf(stderr, "file reg failed: %d\n", ret);
99
+ goto err;
100
+ }
101
+ }
102
+
103
+ offset = 0;
104
+ for (i = 0; i < BUFFERS; i++) {
105
+ sqe = io_uring_get_sqe(ring);
106
+ if (!sqe) {
107
+ fprintf(stderr, "sqe get failed\n");
108
+ goto err;
109
+ }
110
+ offset = BS * (rand() % BUFFERS);
111
+ if (write) {
112
+ int do_fixed = fixed;
113
+ int use_fd = fd;
114
+
115
+ if (sqthread)
116
+ use_fd = 0;
117
+ if (fixed && (i & 1))
118
+ do_fixed = 0;
119
+ if (do_fixed) {
120
+ io_uring_prep_write_fixed(sqe, use_fd, vecs[i].iov_base,
121
+ vecs[i].iov_len,
122
+ offset, i);
123
+ } else {
124
+ io_uring_prep_writev(sqe, use_fd, &vecs[i], 1,
125
+ offset);
126
+ }
127
+ } else {
128
+ int do_fixed = fixed;
129
+ int use_fd = fd;
130
+
131
+ if (sqthread)
132
+ use_fd = 0;
133
+ if (fixed && (i & 1))
134
+ do_fixed = 0;
135
+ if (do_fixed) {
136
+ io_uring_prep_read_fixed(sqe, use_fd, vecs[i].iov_base,
137
+ vecs[i].iov_len,
138
+ offset, i);
139
+ } else {
140
+ io_uring_prep_readv(sqe, use_fd, &vecs[i], 1,
141
+ offset);
142
+ }
143
+
144
+ }
145
+ if (sqthread)
146
+ sqe->flags |= IOSQE_FIXED_FILE;
147
+ if (buf_select) {
148
+ sqe->flags |= IOSQE_BUFFER_SELECT;
149
+ sqe->buf_group = buf_select;
150
+ sqe->user_data = i;
151
+ }
152
+ }
153
+
154
+ ret = io_uring_submit(ring);
155
+ if (ret != BUFFERS) {
156
+ ret = io_uring_peek_cqe(ring, &cqe);
157
+ if (!ret && cqe->res == -EOPNOTSUPP) {
158
+ no_iopoll = 1;
159
+ io_uring_cqe_seen(ring, cqe);
160
+ goto out;
161
+ }
162
+ fprintf(stderr, "submit got %d, wanted %d\n", ret, BUFFERS);
163
+ goto err;
164
+ }
165
+
166
+ for (i = 0; i < BUFFERS; i++) {
167
+ ret = io_uring_wait_cqe(ring, &cqe);
168
+ if (ret) {
169
+ fprintf(stderr, "wait_cqe=%d\n", ret);
170
+ goto err;
171
+ } else if (cqe->res == -EOPNOTSUPP) {
172
+ fprintf(stdout, "File/device/fs doesn't support polled IO\n");
173
+ no_iopoll = 1;
174
+ goto out;
175
+ } else if (cqe->res != BS) {
176
+ fprintf(stderr, "cqe res %d, wanted %d\n", cqe->res, BS);
177
+ goto err;
178
+ }
179
+ io_uring_cqe_seen(ring, cqe);
180
+ }
181
+
182
+ if (fixed) {
183
+ ret = io_uring_unregister_buffers(ring);
184
+ if (ret) {
185
+ fprintf(stderr, "buffer unreg failed: %d\n", ret);
186
+ goto err;
187
+ }
188
+ }
189
+ if (sqthread) {
190
+ ret = io_uring_unregister_files(ring);
191
+ if (ret) {
192
+ fprintf(stderr, "file unreg failed: %d\n", ret);
193
+ goto err;
194
+ }
195
+ }
196
+
197
+ out:
198
+ close(fd);
199
+ return 0;
200
+ err:
201
+ if (fd != -1)
202
+ close(fd);
203
+ return 1;
204
+ }
205
+
206
+ static void sig_alrm(int sig)
207
+ {
208
+ fprintf(stderr, "Ran out of time for peek test!\n");
209
+ exit(T_EXIT_FAIL);
210
+ }
211
+
212
+ /*
213
+ * if we are polling, io_uring_cqe_peek() always needs to enter the kernel
214
+ */
215
+ static int test_io_uring_cqe_peek(const char *file)
216
+ {
217
+ struct io_uring_cqe *cqe;
218
+ struct io_uring ring;
219
+ struct sigaction act;
220
+ int fd, i, ret = T_EXIT_FAIL;
221
+
222
+ if (no_iopoll)
223
+ return 0;
224
+
225
+ ret = io_uring_queue_init(64, &ring, IORING_SETUP_IOPOLL);
226
+ if (ret) {
227
+ fprintf(stderr, "ring create failed: %d\n", ret);
228
+ return 1;
229
+ }
230
+
231
+ fd = open(file, O_RDONLY | O_DIRECT);
232
+ if (fd < 0) {
233
+ if (errno == EINVAL) {
234
+ io_uring_queue_exit(&ring);
235
+ return T_EXIT_SKIP;
236
+ }
237
+ perror("file open");
238
+ goto err;
239
+ }
240
+
241
+ for (i = 0; i < BUFFERS; i++) {
242
+ struct io_uring_sqe *sqe;
243
+ off_t offset = BS * (rand() % BUFFERS);
244
+
245
+ sqe = io_uring_get_sqe(&ring);
246
+ io_uring_prep_readv(sqe, fd, &vecs[i], 1, offset);
247
+ sqe->user_data = 1;
248
+ }
249
+
250
+ /*
251
+ * Set alarm for 5 seconds, we should be done way before that
252
+ */
253
+ memset(&act, 0, sizeof(act));
254
+ act.sa_handler = sig_alrm;
255
+ sigaction(SIGALRM, &act, NULL);
256
+ alarm(5);
257
+
258
+ ret = io_uring_submit(&ring);
259
+ if (ret != BUFFERS) {
260
+ fprintf(stderr, "submit=%d\n", ret);
261
+ goto err;
262
+ }
263
+
264
+ ret = T_EXIT_PASS;
265
+ i = 0;
266
+ do {
267
+ ret = io_uring_peek_cqe(&ring, &cqe);
268
+ if (ret)
269
+ continue;
270
+ io_uring_cqe_seen(&ring, cqe);
271
+ i++;
272
+ } while (i < BUFFERS);
273
+
274
+ err:
275
+ if (fd != -1)
276
+ close(fd);
277
+ io_uring_queue_exit(&ring);
278
+ return ret;
279
+ }
280
+
281
+ /*
282
+ * if we are polling io_uring_submit needs to always enter the
283
+ * kernel to fetch events
284
+ */
285
+ static int test_io_uring_submit_enters(const char *file)
286
+ {
287
+ struct io_uring ring;
288
+ int fd, i, ret, ring_flags, open_flags;
289
+ unsigned head;
290
+ struct io_uring_cqe *cqe;
291
+
292
+ if (no_iopoll)
293
+ return 0;
294
+
295
+ ring_flags = IORING_SETUP_IOPOLL;
296
+ ret = io_uring_queue_init(64, &ring, ring_flags);
297
+ if (ret) {
298
+ fprintf(stderr, "ring create failed: %d\n", ret);
299
+ return 1;
300
+ }
301
+
302
+ open_flags = O_WRONLY | O_DIRECT;
303
+ fd = open(file, open_flags);
304
+ if (fd < 0) {
305
+ if (errno == EINVAL)
306
+ return T_EXIT_SKIP;
307
+ perror("file open");
308
+ goto err;
309
+ }
310
+
311
+ for (i = 0; i < BUFFERS; i++) {
312
+ struct io_uring_sqe *sqe;
313
+ off_t offset = BS * (rand() % BUFFERS);
314
+
315
+ sqe = io_uring_get_sqe(&ring);
316
+ io_uring_prep_writev(sqe, fd, &vecs[i], 1, offset);
317
+ sqe->user_data = 1;
318
+ }
319
+
320
+ /* submit manually to avoid adding IORING_ENTER_GETEVENTS */
321
+ ret = __sys_io_uring_enter(ring.ring_fd, __io_uring_flush_sq(&ring), 0,
322
+ 0, NULL);
323
+ if (ret < 0)
324
+ goto err;
325
+
326
+ for (i = 0; i < 500; i++) {
327
+ ret = io_uring_submit(&ring);
328
+ if (ret != 0) {
329
+ fprintf(stderr, "still had %d sqes to submit, this is unexpected", ret);
330
+ goto err;
331
+ }
332
+
333
+ io_uring_for_each_cqe(&ring, head, cqe) {
334
+ /* runs after test_io so should not have happened */
335
+ if (cqe->res == -EOPNOTSUPP) {
336
+ fprintf(stdout, "File/device/fs doesn't support polled IO\n");
337
+ goto err;
338
+ }
339
+ goto ok;
340
+ }
341
+ usleep(10000);
342
+ }
343
+ err:
344
+ ret = 1;
345
+ if (fd != -1)
346
+ close(fd);
347
+
348
+ ok:
349
+ io_uring_queue_exit(&ring);
350
+ return ret;
351
+ }
352
+
353
+ static int test_io(const char *file, int write, int sqthread, int fixed,
354
+ int buf_select, int defer)
355
+ {
356
+ struct io_uring ring;
357
+ int ret, ring_flags = IORING_SETUP_IOPOLL;
358
+
359
+ if (no_iopoll)
360
+ return 0;
361
+
362
+ if (defer)
363
+ ring_flags |= IORING_SETUP_SINGLE_ISSUER |
364
+ IORING_SETUP_DEFER_TASKRUN;
365
+
366
+ ret = t_create_ring(64, &ring, ring_flags);
367
+ if (ret == T_SETUP_SKIP)
368
+ return 0;
369
+ if (ret != T_SETUP_OK) {
370
+ fprintf(stderr, "ring create failed: %d\n", ret);
371
+ return 1;
372
+ }
373
+ ret = __test_io(file, &ring, write, sqthread, fixed, buf_select);
374
+ io_uring_queue_exit(&ring);
375
+ return ret;
376
+ }
377
+
378
+ static int probe_buf_select(void)
379
+ {
380
+ struct io_uring_probe *p;
381
+ struct io_uring ring;
382
+ int ret;
383
+
384
+ ret = io_uring_queue_init(1, &ring, 0);
385
+ if (ret) {
386
+ fprintf(stderr, "ring create failed: %d\n", ret);
387
+ return 1;
388
+ }
389
+
390
+ p = io_uring_get_probe_ring(&ring);
391
+ if (!p || !io_uring_opcode_supported(p, IORING_OP_PROVIDE_BUFFERS)) {
392
+ no_buf_select = 1;
393
+ fprintf(stdout, "Buffer select not supported, skipping\n");
394
+ return 0;
395
+ }
396
+ io_uring_free_probe(p);
397
+ return 0;
398
+ }
399
+
400
+ int main(int argc, char *argv[])
401
+ {
402
+ int i, ret, nr;
403
+ char buf[256];
404
+ char *fname;
405
+
406
+ if (probe_buf_select())
407
+ return T_EXIT_FAIL;
408
+
409
+ if (argc > 1) {
410
+ fname = argv[1];
411
+ } else {
412
+ srand((unsigned)time(NULL));
413
+ snprintf(buf, sizeof(buf), ".basic-rw-%u-%u",
414
+ (unsigned)rand(), (unsigned)getpid());
415
+ fname = buf;
416
+ t_create_file(fname, FILE_SIZE);
417
+ }
418
+
419
+ vecs = t_create_buffers(BUFFERS, BS);
420
+
421
+ nr = 32;
422
+ if (no_buf_select)
423
+ nr = 8;
424
+ else if (!t_probe_defer_taskrun())
425
+ nr = 16;
426
+ for (i = 0; i < nr; i++) {
427
+ int write = (i & 1) != 0;
428
+ int sqthread = (i & 2) != 0;
429
+ int fixed = (i & 4) != 0;
430
+ int buf_select = (i & 8) != 0;
431
+ int defer = (i & 16) != 0;
432
+
433
+ ret = test_io(fname, write, sqthread, fixed, buf_select, defer);
434
+ if (ret) {
435
+ fprintf(stderr, "test_io failed %d/%d/%d/%d/%d\n",
436
+ write, sqthread, fixed, buf_select, defer);
437
+ goto err;
438
+ }
439
+ if (no_iopoll)
440
+ break;
441
+ }
442
+
443
+ ret = test_io_uring_submit_enters(fname);
444
+ if (ret == T_EXIT_FAIL) {
445
+ fprintf(stderr, "test_io_uring_submit_enters failed\n");
446
+ goto err;
447
+ }
448
+
449
+ /*
450
+ * Keep this last, it exits on failure
451
+ */
452
+ ret = test_io_uring_cqe_peek(fname);
453
+ if (ret == T_EXIT_FAIL) {
454
+ fprintf(stderr, "test_io_uring_cqe_peek failed\n");
455
+ goto err;
456
+ }
457
+
458
+ if (fname != argv[1])
459
+ unlink(fname);
460
+ return T_EXIT_PASS;
461
+ err:
462
+ if (fname != argv[1])
463
+ unlink(fname);
464
+ return T_EXIT_FAIL;
465
+ }
@@ -0,0 +1,119 @@
1
+ /* SPDX-License-Identifier: MIT */
2
+
3
+ #include <string.h>
4
+ #include <stdio.h>
5
+ #include <stdlib.h>
6
+ #include <sys/types.h>
7
+ #include <sys/stat.h>
8
+ #include <fcntl.h>
9
+ #include <errno.h>
10
+ #include <sys/resource.h>
11
+ #include <unistd.h>
12
+
13
+ #include "liburing.h"
14
+ #include "helpers.h"
15
+
16
+ static const int RSIZE = 2;
17
+ static const int OPEN_FLAGS = O_RDWR | O_CREAT | O_LARGEFILE;
18
+ static const mode_t OPEN_MODE = S_IRUSR | S_IWUSR;
19
+
20
+ #define DIE(...) \
21
+ do { \
22
+ fprintf(stderr, __VA_ARGS__); \
23
+ abort(); \
24
+ } while(0)
25
+
26
+ static int do_write(struct io_uring *ring, int fd, off_t offset)
27
+ {
28
+ char buf[] = "some test write buf";
29
+ struct io_uring_sqe *sqe;
30
+ struct io_uring_cqe *cqe;
31
+ int res, ret;
32
+
33
+ sqe = io_uring_get_sqe(ring);
34
+ if (!sqe) {
35
+ fprintf(stderr, "failed to get sqe\n");
36
+ return 1;
37
+ }
38
+ io_uring_prep_write(sqe, fd, buf, sizeof(buf), offset);
39
+
40
+ ret = io_uring_submit(ring);
41
+ if (ret < 0) {
42
+ fprintf(stderr, "failed to submit write: %s\n", strerror(-ret));
43
+ return 1;
44
+ }
45
+
46
+ ret = io_uring_wait_cqe(ring, &cqe);
47
+ if (ret < 0) {
48
+ fprintf(stderr, "wait_cqe failed: %s\n", strerror(-ret));
49
+ return 1;
50
+ }
51
+
52
+ res = cqe->res;
53
+ io_uring_cqe_seen(ring, cqe);
54
+ if (res < 0) {
55
+ fprintf(stderr, "write failed: %s\n", strerror(-res));
56
+ return 1;
57
+ }
58
+
59
+ return 0;
60
+ }
61
+
62
+ static int test_open_write(struct io_uring *ring, int dfd, const char *fn)
63
+ {
64
+ struct io_uring_sqe *sqe;
65
+ struct io_uring_cqe *cqe;
66
+ int ret, fd = -1;
67
+
68
+ sqe = io_uring_get_sqe(ring);
69
+ if (!sqe) {
70
+ fprintf(stderr, "failed to get sqe\n");
71
+ return 1;
72
+ }
73
+ io_uring_prep_openat(sqe, dfd, fn, OPEN_FLAGS, OPEN_MODE);
74
+
75
+ ret = io_uring_submit(ring);
76
+ if (ret < 0) {
77
+ fprintf(stderr, "failed to submit openat: %s\n", strerror(-ret));
78
+ return 1;
79
+ }
80
+
81
+ ret = io_uring_wait_cqe(ring, &cqe);
82
+ if (ret < 0) {
83
+ fprintf(stderr, "wait_cqe failed: %s\n", strerror(-ret));
84
+ return 1;
85
+ }
86
+
87
+ fd = cqe->res;
88
+ io_uring_cqe_seen(ring, cqe);
89
+ if (fd < 0) {
90
+ fprintf(stderr, "openat failed: %s\n", strerror(-fd));
91
+ return 1;
92
+ }
93
+
94
+ return do_write(ring, fd, 1ULL << 32);
95
+ }
96
+
97
+ int main(int argc, char *argv[])
98
+ {
99
+ struct io_uring ring;
100
+ int dfd, ret;
101
+
102
+ if (argc > 1)
103
+ return T_EXIT_SKIP;
104
+
105
+ dfd = open("/tmp", O_RDONLY | O_DIRECTORY);
106
+ if (dfd < 0)
107
+ DIE("open /tmp: %s\n", strerror(errno));
108
+
109
+ ret = io_uring_queue_init(RSIZE, &ring, 0);
110
+ if (ret < 0)
111
+ DIE("failed to init io_uring: %s\n", strerror(-ret));
112
+
113
+ ret = test_open_write(&ring, dfd, "io_uring_openat_write_test1");
114
+
115
+ io_uring_queue_exit(&ring);
116
+ close(dfd);
117
+ unlink("/tmp/io_uring_openat_write_test1");
118
+ return ret;
119
+ }