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,132 @@
1
+ /* SPDX-License-Identifier: MIT */
2
+ /*
3
+ * Description: run various rename tests
4
+ *
5
+ */
6
+ #include <errno.h>
7
+ #include <stdio.h>
8
+ #include <unistd.h>
9
+ #include <stdlib.h>
10
+ #include <string.h>
11
+ #include <fcntl.h>
12
+ #include <sys/stat.h>
13
+
14
+ #include "liburing.h"
15
+
16
+ static int test_rename(struct io_uring *ring, const char *old, const char *new)
17
+ {
18
+ struct io_uring_cqe *cqe;
19
+ struct io_uring_sqe *sqe;
20
+ int ret;
21
+
22
+ sqe = io_uring_get_sqe(ring);
23
+ if (!sqe) {
24
+ fprintf(stderr, "get sqe failed\n");
25
+ goto err;
26
+ }
27
+
28
+ memset(sqe, 0, sizeof(*sqe));
29
+
30
+ io_uring_prep_rename(sqe, old, new);
31
+
32
+ ret = io_uring_submit(ring);
33
+ if (ret <= 0) {
34
+ fprintf(stderr, "sqe submit failed: %d\n", ret);
35
+ goto err;
36
+ }
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
+ ret = cqe->res;
44
+ io_uring_cqe_seen(ring, cqe);
45
+ return ret;
46
+ err:
47
+ return 1;
48
+ }
49
+
50
+ static int stat_file(const char *buf)
51
+ {
52
+ struct stat sb;
53
+
54
+ if (!stat(buf, &sb))
55
+ return 0;
56
+
57
+ return errno;
58
+ }
59
+
60
+ int main(int argc, char *argv[])
61
+ {
62
+ struct io_uring ring;
63
+ char src[32] = "./XXXXXX";
64
+ char dst[32] = "./XXXXXX";
65
+ int ret;
66
+
67
+ if (argc > 1)
68
+ return 0;
69
+
70
+ ret = io_uring_queue_init(1, &ring, 0);
71
+ if (ret) {
72
+ fprintf(stderr, "ring setup failed: %d\n", ret);
73
+ return 1;
74
+ }
75
+
76
+ ret = mkstemp(src);
77
+ if (ret < 0) {
78
+ perror("mkstemp");
79
+ return 1;
80
+ }
81
+ close(ret);
82
+
83
+ ret = mkstemp(dst);
84
+ if (ret < 0) {
85
+ perror("mkstemp");
86
+ return 1;
87
+ }
88
+ close(ret);
89
+
90
+ if (stat_file(src) != 0) {
91
+ perror("stat");
92
+ return 1;
93
+ }
94
+ if (stat_file(dst) != 0) {
95
+ perror("stat");
96
+ return 1;
97
+ }
98
+
99
+ ret = test_rename(&ring, src, dst);
100
+ if (ret < 0) {
101
+ if (ret == -EBADF || ret == -EINVAL) {
102
+ fprintf(stdout, "Rename not supported, skipping\n");
103
+ goto out;
104
+ }
105
+ fprintf(stderr, "rename: %s\n", strerror(-ret));
106
+ goto err;
107
+ } else if (ret)
108
+ goto err;
109
+
110
+ if (stat_file(src) != ENOENT) {
111
+ fprintf(stderr, "stat got %s\n", strerror(ret));
112
+ return 1;
113
+ }
114
+
115
+ if (stat_file(dst) != 0) {
116
+ perror("stat");
117
+ return 1;
118
+ }
119
+
120
+ ret = test_rename(&ring, "/x/y/1/2", "/2/1/y/x");
121
+ if (ret != -ENOENT) {
122
+ fprintf(stderr, "test_rename invalid failed: %d\n", ret);
123
+ return ret;
124
+ }
125
+ out:
126
+ unlink(dst);
127
+ return 0;
128
+ err:
129
+ unlink(src);
130
+ unlink(dst);
131
+ return 1;
132
+ }
@@ -0,0 +1,283 @@
1
+ /* SPDX-License-Identifier: MIT */
2
+ /*
3
+ * Based on description from Al Viro - this demonstrates a leak of the
4
+ * io_uring instance, by sending the io_uring fd over a UNIX socket.
5
+ *
6
+ * See:
7
+ *
8
+ * https://lore.kernel.org/linux-block/20190129192702.3605-1-axboe@kernel.dk/T/#m6c87fc64e4d063786af6ec6fadce3ac1e95d3184
9
+ *
10
+ */
11
+ #include <stdio.h>
12
+ #include <stdlib.h>
13
+ #include <stddef.h>
14
+ #include <signal.h>
15
+ #include <inttypes.h>
16
+ #include <sys/types.h>
17
+ #include <sys/syscall.h>
18
+ #include <sys/socket.h>
19
+ #include <sys/wait.h>
20
+ #include <fcntl.h>
21
+ #include <unistd.h>
22
+ #include <string.h>
23
+ #include <linux/fs.h>
24
+
25
+ #include "liburing.h"
26
+ #include "helpers.h"
27
+ #include "../src/syscall.h"
28
+
29
+ static int __io_uring_register_files(int ring_fd, int fd1, int fd2)
30
+ {
31
+ __s32 fds[2] = { fd1, fd2 };
32
+
33
+ return __sys_io_uring_register(ring_fd, IORING_REGISTER_FILES, fds, 2);
34
+ }
35
+
36
+ static int get_ring_fd(void)
37
+ {
38
+ struct io_uring_params p;
39
+ int fd;
40
+
41
+ memset(&p, 0, sizeof(p));
42
+
43
+ fd = __sys_io_uring_setup(2, &p);
44
+ if (fd < 0) {
45
+ perror("io_uring_setup");
46
+ return -1;
47
+ }
48
+
49
+ return fd;
50
+ }
51
+
52
+ static int send_fd(int socket, int fd)
53
+ {
54
+ char buf[CMSG_SPACE(sizeof(fd))];
55
+ struct cmsghdr *cmsg;
56
+ struct msghdr msg;
57
+
58
+ memset(buf, 0, sizeof(buf));
59
+ memset(&msg, 0, sizeof(msg));
60
+
61
+ msg.msg_control = buf;
62
+ msg.msg_controllen = sizeof(buf);
63
+
64
+ cmsg = CMSG_FIRSTHDR(&msg);
65
+ cmsg->cmsg_level = SOL_SOCKET;
66
+ cmsg->cmsg_type = SCM_RIGHTS;
67
+ cmsg->cmsg_len = CMSG_LEN(sizeof(fd));
68
+
69
+ memmove(CMSG_DATA(cmsg), &fd, sizeof(fd));
70
+
71
+ msg.msg_controllen = CMSG_SPACE(sizeof(fd));
72
+
73
+ if (sendmsg(socket, &msg, 0) < 0) {
74
+ if (errno == EINVAL)
75
+ return T_EXIT_SKIP;
76
+ perror("sendmsg");
77
+ return T_EXIT_FAIL;
78
+ }
79
+
80
+ return T_EXIT_PASS;
81
+ }
82
+
83
+ static int test_iowq_request_cancel(void)
84
+ {
85
+ char buffer[128];
86
+ struct io_uring ring;
87
+ struct io_uring_sqe *sqe;
88
+ int ret, fds[2];
89
+
90
+ ret = io_uring_queue_init(8, &ring, 0);
91
+ if (ret < 0) {
92
+ fprintf(stderr, "failed to init io_uring: %s\n", strerror(-ret));
93
+ return ret;
94
+ }
95
+ if (pipe(fds)) {
96
+ perror("pipe");
97
+ return -1;
98
+ }
99
+ ret = io_uring_register_files(&ring, fds, 2);
100
+ if (ret) {
101
+ fprintf(stderr, "file_register: %d\n", ret);
102
+ return ret;
103
+ }
104
+ close(fds[1]);
105
+
106
+ sqe = io_uring_get_sqe(&ring);
107
+ if (!sqe) {
108
+ fprintf(stderr, "%s: failed to get sqe\n", __FUNCTION__);
109
+ return 1;
110
+ }
111
+ /* potentially sitting in internal polling */
112
+ io_uring_prep_read(sqe, 0, buffer, 10, 0);
113
+ sqe->flags |= IOSQE_FIXED_FILE;
114
+
115
+ sqe = io_uring_get_sqe(&ring);
116
+ if (!sqe) {
117
+ fprintf(stderr, "%s: failed to get sqe\n", __FUNCTION__);
118
+ return 1;
119
+ }
120
+ /* staying in io-wq */
121
+ io_uring_prep_read(sqe, 0, buffer, 10, 0);
122
+ sqe->flags |= IOSQE_FIXED_FILE | IOSQE_ASYNC;
123
+
124
+ ret = io_uring_submit(&ring);
125
+ if (ret != 2) {
126
+ fprintf(stderr, "%s: got %d, wanted 1\n", __FUNCTION__, ret);
127
+ return 1;
128
+ }
129
+
130
+ /* should unregister files and close the write fd */
131
+ io_uring_queue_exit(&ring);
132
+
133
+ /*
134
+ * We're trying to wait for the ring to "really" exit, that will be
135
+ * done async. For that rely on the registered write end to be closed
136
+ * after ring quiesce, so failing read from the other pipe end.
137
+ */
138
+ ret = read(fds[0], buffer, 10);
139
+ if (ret < 0)
140
+ perror("read");
141
+ close(fds[0]);
142
+ return 0;
143
+ }
144
+
145
+ static void trigger_unix_gc(void)
146
+ {
147
+ int fd;
148
+
149
+ fd = socket(AF_UNIX, SOCK_DGRAM, 0);
150
+ if (fd < 0)
151
+ perror("socket dgram");
152
+ else
153
+ close(fd);
154
+ }
155
+
156
+ static int test_scm_cycles(bool update)
157
+ {
158
+ char buffer[128];
159
+ struct io_uring ring;
160
+ int i, ret;
161
+ int sp[2], fds[2], reg_fds[4];
162
+
163
+ if (socketpair(AF_UNIX, SOCK_DGRAM, 0, sp) != 0) {
164
+ perror("Failed to create Unix-domain socket pair\n");
165
+ return 1;
166
+ }
167
+ ret = io_uring_queue_init(8, &ring, 0);
168
+ if (ret < 0) {
169
+ fprintf(stderr, "failed to init io_uring: %s\n", strerror(-ret));
170
+ return ret;
171
+ }
172
+ if (pipe(fds)) {
173
+ perror("pipe");
174
+ return -1;
175
+ }
176
+ ret = send_fd(sp[0], ring.ring_fd);
177
+ if (ret != T_EXIT_PASS)
178
+ return ret;
179
+
180
+ /* register an empty set for updates */
181
+ if (update) {
182
+ for (i = 0; i < 4; i++)
183
+ reg_fds[i] = -1;
184
+ ret = io_uring_register_files(&ring, reg_fds, 4);
185
+ if (ret) {
186
+ fprintf(stderr, "file_register: %d\n", ret);
187
+ return ret;
188
+ }
189
+ }
190
+
191
+ reg_fds[0] = fds[0];
192
+ reg_fds[1] = fds[1];
193
+ reg_fds[2] = sp[0];
194
+ reg_fds[3] = sp[1];
195
+ if (update) {
196
+ ret = io_uring_register_files_update(&ring, 0, reg_fds, 4);
197
+ if (ret != 4) {
198
+ fprintf(stderr, "file_register: %d\n", ret);
199
+ return ret;
200
+ }
201
+ } else {
202
+ ret = io_uring_register_files(&ring, reg_fds, 4);
203
+ if (ret) {
204
+ fprintf(stderr, "file_register: %d\n", ret);
205
+ return ret;
206
+ }
207
+ }
208
+
209
+ close(fds[1]);
210
+ close(sp[0]);
211
+ close(sp[1]);
212
+
213
+ /* should unregister files and close the write fd */
214
+ io_uring_queue_exit(&ring);
215
+
216
+ trigger_unix_gc();
217
+
218
+ /*
219
+ * We're trying to wait for the ring to "really" exit, that will be
220
+ * done async. For that rely on the registered write end to be closed
221
+ * after ring quiesce, so failing read from the other pipe end.
222
+ */
223
+ ret = read(fds[0], buffer, 10);
224
+ if (ret < 0)
225
+ perror("read");
226
+ close(fds[0]);
227
+ return 0;
228
+ }
229
+
230
+ int main(int argc, char *argv[])
231
+ {
232
+ int sp[2], pid, ring_fd, ret;
233
+ int i;
234
+
235
+ if (argc > 1)
236
+ return 0;
237
+
238
+ ret = test_iowq_request_cancel();
239
+ if (ret) {
240
+ fprintf(stderr, "test_iowq_request_cancel() failed\n");
241
+ return 1;
242
+ }
243
+
244
+ for (i = 0; i < 2; i++) {
245
+ bool update = !!(i & 1);
246
+
247
+ ret = test_scm_cycles(update);
248
+ if (ret == T_EXIT_SKIP)
249
+ return T_EXIT_SKIP;
250
+ if (ret) {
251
+ fprintf(stderr, "test_scm_cycles() failed %i\n",
252
+ update);
253
+ return 1;
254
+ }
255
+ }
256
+
257
+ if (socketpair(AF_UNIX, SOCK_DGRAM, 0, sp) != 0) {
258
+ perror("Failed to create Unix-domain socket pair\n");
259
+ return 1;
260
+ }
261
+
262
+ ring_fd = get_ring_fd();
263
+ if (ring_fd < 0)
264
+ return 1;
265
+
266
+ ret = __io_uring_register_files(ring_fd, sp[0], sp[1]);
267
+ if (ret < 0) {
268
+ perror("register files");
269
+ return 1;
270
+ }
271
+
272
+ pid = fork();
273
+ if (pid) {
274
+ ret = send_fd(sp[0], ring_fd);
275
+ if (ret != T_EXIT_PASS)
276
+ return ret;
277
+ }
278
+
279
+ close(ring_fd);
280
+ close(sp[0]);
281
+ close(sp[1]);
282
+ return 0;
283
+ }
@@ -0,0 +1,249 @@
1
+ /* SPDX-License-Identifier: MIT */
2
+ /*
3
+ * Description: Test two ring deadlock. A buggy kernel will end up
4
+ * having io_wq_* workers pending, as the circular reference
5
+ * will prevent full exit.
6
+ *
7
+ * Based on a test case from Josef <josef.grieb@gmail.com>
8
+ *
9
+ */
10
+ #include <errno.h>
11
+ #include <fcntl.h>
12
+ #include <netinet/in.h>
13
+ #include <stdio.h>
14
+ #include <stdlib.h>
15
+ #include <string.h>
16
+ #include <strings.h>
17
+ #include <poll.h>
18
+ #include <sys/socket.h>
19
+ #include <unistd.h>
20
+ #include <sys/eventfd.h>
21
+ #include <pthread.h>
22
+
23
+ #include "liburing.h"
24
+ #include "../src/syscall.h"
25
+
26
+ enum {
27
+ ACCEPT,
28
+ READ,
29
+ WRITE,
30
+ POLLING_IN,
31
+ POLLING_RDHUP,
32
+ CLOSE,
33
+ EVENTFD_READ,
34
+ };
35
+
36
+ typedef struct conn_info {
37
+ __u32 fd;
38
+ __u16 type;
39
+ __u16 bid;
40
+ } conn_info;
41
+
42
+ static char read_eventfd_buffer[8];
43
+
44
+ static pthread_mutex_t lock;
45
+ static struct io_uring *client_ring;
46
+
47
+ static int client_eventfd = -1;
48
+
49
+ static int setup_io_uring(struct io_uring *ring)
50
+ {
51
+ struct io_uring_params p = { };
52
+ int ret;
53
+
54
+ ret = io_uring_queue_init_params(8, ring, &p);
55
+ if (ret) {
56
+ fprintf(stderr, "Unable to setup io_uring: %s\n",
57
+ strerror(-ret));
58
+ return 1;
59
+ }
60
+ return 0;
61
+ }
62
+
63
+ static void add_socket_eventfd_read(struct io_uring *ring, int fd)
64
+ {
65
+ struct io_uring_sqe *sqe;
66
+ conn_info conn_i = {
67
+ .fd = fd,
68
+ .type = EVENTFD_READ,
69
+ };
70
+
71
+ sqe = io_uring_get_sqe(ring);
72
+ io_uring_prep_read(sqe, fd, &read_eventfd_buffer, 8, 0);
73
+ io_uring_sqe_set_flags(sqe, IOSQE_ASYNC);
74
+
75
+ memcpy(&sqe->user_data, &conn_i, sizeof(conn_i));
76
+ }
77
+
78
+ static void add_socket_pollin(struct io_uring *ring, int fd)
79
+ {
80
+ struct io_uring_sqe *sqe;
81
+ conn_info conn_i = {
82
+ .fd = fd,
83
+ .type = POLLING_IN,
84
+ };
85
+
86
+ sqe = io_uring_get_sqe(ring);
87
+ io_uring_prep_poll_add(sqe, fd, POLL_IN);
88
+
89
+ memcpy(&sqe->user_data, &conn_i, sizeof(conn_i));
90
+ }
91
+
92
+ static void *server_thread(void *arg)
93
+ {
94
+ struct sockaddr_in serv_addr;
95
+ int port = 0;
96
+ int sock_listen_fd, evfd;
97
+ const int val = 1;
98
+ struct io_uring ring;
99
+
100
+ sock_listen_fd = socket(AF_INET, SOCK_STREAM | SOCK_NONBLOCK, 0);
101
+ setsockopt(sock_listen_fd, SOL_SOCKET, SO_REUSEADDR, &val, sizeof(val));
102
+
103
+ memset(&serv_addr, 0, sizeof(serv_addr));
104
+ serv_addr.sin_family = AF_INET;
105
+ serv_addr.sin_port = htons(port);
106
+ serv_addr.sin_addr.s_addr = INADDR_ANY;
107
+
108
+ evfd = eventfd(0, EFD_CLOEXEC);
109
+
110
+ // bind and listen
111
+ if (bind(sock_listen_fd, (struct sockaddr *)&serv_addr, sizeof(serv_addr)) < 0) {
112
+ perror("Error binding socket...\n");
113
+ exit(1);
114
+ }
115
+ if (listen(sock_listen_fd, 1) < 0) {
116
+ perror("Error listening on socket...\n");
117
+ exit(1);
118
+ }
119
+
120
+ setup_io_uring(&ring);
121
+ add_socket_eventfd_read(&ring, evfd);
122
+ add_socket_pollin(&ring, sock_listen_fd);
123
+
124
+ while (1) {
125
+ struct io_uring_cqe *cqe;
126
+ unsigned head;
127
+ unsigned count = 0;
128
+
129
+ io_uring_submit_and_wait(&ring, 1);
130
+
131
+ io_uring_for_each_cqe(&ring, head, cqe) {
132
+ struct conn_info conn_i;
133
+
134
+ count++;
135
+ memcpy(&conn_i, &cqe->user_data, sizeof(conn_i));
136
+
137
+ if (conn_i.type == ACCEPT) {
138
+ int sock_conn_fd = cqe->res;
139
+ // only read when there is no error, >= 0
140
+ if (sock_conn_fd > 0) {
141
+ add_socket_pollin(&ring, sock_listen_fd);
142
+
143
+ pthread_mutex_lock(&lock);
144
+ io_uring_submit(client_ring);
145
+ pthread_mutex_unlock(&lock);
146
+
147
+ }
148
+ } else if (conn_i.type == POLLING_IN) {
149
+ break;
150
+ }
151
+ }
152
+ io_uring_cq_advance(&ring, count);
153
+ }
154
+ }
155
+
156
+ static void *client_thread(void *arg)
157
+ {
158
+ struct io_uring ring;
159
+ int ret;
160
+
161
+ setup_io_uring(&ring);
162
+ client_ring = &ring;
163
+
164
+ client_eventfd = eventfd(0, EFD_CLOEXEC);
165
+ pthread_mutex_lock(&lock);
166
+ add_socket_eventfd_read(&ring, client_eventfd);
167
+ pthread_mutex_unlock(&lock);
168
+
169
+ while (1) {
170
+ struct io_uring_cqe *cqe;
171
+ unsigned head;
172
+ unsigned count = 0;
173
+
174
+ pthread_mutex_lock(&lock);
175
+ io_uring_submit(&ring);
176
+ pthread_mutex_unlock(&lock);
177
+
178
+ ret = __sys_io_uring_enter(ring.ring_fd, 0, 1, IORING_ENTER_GETEVENTS, NULL);
179
+ if (ret < 0) {
180
+ perror("Error io_uring_enter...\n");
181
+ exit(1);
182
+ }
183
+
184
+ // go through all CQEs
185
+ io_uring_for_each_cqe(&ring, head, cqe) {
186
+ struct conn_info conn_i;
187
+ int type;
188
+
189
+ count++;
190
+ memcpy(&conn_i, &cqe->user_data, sizeof(conn_i));
191
+
192
+ type = conn_i.type;
193
+ if (type == READ) {
194
+ pthread_mutex_lock(&lock);
195
+
196
+ if (cqe->res <= 0) {
197
+ // connection closed or error
198
+ shutdown(conn_i.fd, SHUT_RDWR);
199
+ } else {
200
+ pthread_mutex_unlock(&lock);
201
+ break;
202
+ }
203
+ add_socket_pollin(&ring, conn_i.fd);
204
+ pthread_mutex_unlock(&lock);
205
+ } else if (type == WRITE) {
206
+ } else if (type == POLLING_IN) {
207
+ break;
208
+ } else if (type == POLLING_RDHUP) {
209
+ break;
210
+ } else if (type == CLOSE) {
211
+ } else if (type == EVENTFD_READ) {
212
+ add_socket_eventfd_read(&ring, client_eventfd);
213
+ }
214
+ }
215
+
216
+ io_uring_cq_advance(&ring, count);
217
+ }
218
+ }
219
+
220
+ static void sig_alrm(int sig)
221
+ {
222
+ exit(0);
223
+ }
224
+
225
+ int main(int argc, char *argv[])
226
+ {
227
+ pthread_t server_thread_t, client_thread_t;
228
+ struct sigaction act;
229
+
230
+ if (argc > 1)
231
+ return 0;
232
+
233
+ if (pthread_mutex_init(&lock, NULL) != 0) {
234
+ printf("\n mutex init failed\n");
235
+ return 1;
236
+ }
237
+
238
+ pthread_create(&server_thread_t, NULL, &server_thread, NULL);
239
+ pthread_create(&client_thread_t, NULL, &client_thread, NULL);
240
+
241
+ memset(&act, 0, sizeof(act));
242
+ act.sa_handler = sig_alrm;
243
+ act.sa_flags = SA_RESTART;
244
+ sigaction(SIGALRM, &act, NULL);
245
+ alarm(1);
246
+
247
+ pthread_join(server_thread_t, NULL);
248
+ return 0;
249
+ }