polyphony 0.84 → 0.86

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 (241) hide show
  1. checksums.yaml +4 -4
  2. data/CHANGELOG.md +14 -0
  3. data/Gemfile.lock +1 -1
  4. data/Rakefile +1 -1
  5. data/examples/core/multi_suspend.rb +39 -0
  6. data/examples/core/shutdown_all_children.rb +41 -0
  7. data/examples/io/gzip.rb +8 -0
  8. data/examples/io/splice_echo_server.rb +15 -0
  9. data/ext/polyphony/backend_io_uring.c +57 -31
  10. data/ext/polyphony/io_extensions.c +137 -26
  11. data/lib/polyphony/extensions/fiber.rb +3 -1
  12. data/lib/polyphony/extensions/io.rb +4 -0
  13. data/lib/polyphony/extensions/pipe.rb +4 -0
  14. data/lib/polyphony/extensions/socket.rb +4 -0
  15. data/lib/polyphony/version.rb +1 -1
  16. data/polyphony.gemspec +1 -1
  17. data/test/test_backend.rb +1 -1
  18. data/test/test_fiber.rb +5 -2
  19. data/test/test_signal.rb +3 -3
  20. data/vendor/liburing/.github/pull_request_template.md +86 -0
  21. data/vendor/liburing/.github/workflows/build.yml +85 -0
  22. data/vendor/liburing/.github/workflows/shellcheck.yml +20 -0
  23. data/vendor/liburing/.gitignore +149 -0
  24. data/vendor/liburing/COPYING +502 -0
  25. data/vendor/liburing/COPYING.GPL +339 -0
  26. data/vendor/liburing/LICENSE +7 -0
  27. data/vendor/liburing/Makefile +82 -0
  28. data/vendor/liburing/Makefile.common +5 -0
  29. data/vendor/liburing/Makefile.quiet +11 -0
  30. data/vendor/liburing/README +46 -0
  31. data/vendor/liburing/configure +486 -0
  32. data/vendor/liburing/debian/README.Debian +7 -0
  33. data/vendor/liburing/debian/changelog +27 -0
  34. data/vendor/liburing/debian/compat +1 -0
  35. data/vendor/liburing/debian/control +48 -0
  36. data/vendor/liburing/debian/copyright +49 -0
  37. data/vendor/liburing/debian/liburing-dev.install +4 -0
  38. data/vendor/liburing/debian/liburing-dev.manpages +6 -0
  39. data/vendor/liburing/debian/liburing1-udeb.install +1 -0
  40. data/vendor/liburing/debian/liburing1.install +1 -0
  41. data/vendor/liburing/debian/liburing1.symbols +32 -0
  42. data/vendor/liburing/debian/patches/series +1 -0
  43. data/vendor/liburing/debian/rules +81 -0
  44. data/vendor/liburing/debian/source/format +1 -0
  45. data/vendor/liburing/debian/source/local-options +2 -0
  46. data/vendor/liburing/debian/source/options +1 -0
  47. data/vendor/liburing/debian/watch +3 -0
  48. data/vendor/liburing/examples/Makefile +38 -0
  49. data/vendor/liburing/examples/io_uring-cp.c +282 -0
  50. data/vendor/liburing/examples/io_uring-test.c +112 -0
  51. data/vendor/liburing/examples/link-cp.c +193 -0
  52. data/vendor/liburing/examples/ucontext-cp.c +273 -0
  53. data/vendor/liburing/liburing.pc.in +12 -0
  54. data/vendor/liburing/liburing.spec +66 -0
  55. data/vendor/liburing/make-debs.sh +53 -0
  56. data/vendor/liburing/man/io_uring.7 +754 -0
  57. data/vendor/liburing/man/io_uring_cq_advance.3 +35 -0
  58. data/vendor/liburing/man/io_uring_cq_ready.3 +25 -0
  59. data/vendor/liburing/man/io_uring_cqe_get_data.3 +34 -0
  60. data/vendor/liburing/man/io_uring_cqe_seen.3 +32 -0
  61. data/vendor/liburing/man/io_uring_enter.2 +1483 -0
  62. data/vendor/liburing/man/io_uring_free_probe.3 +24 -0
  63. data/vendor/liburing/man/io_uring_get_probe.3 +29 -0
  64. data/vendor/liburing/man/io_uring_get_sqe.3 +38 -0
  65. data/vendor/liburing/man/io_uring_opcode_supported.3 +29 -0
  66. data/vendor/liburing/man/io_uring_prep_msg_ring.3 +58 -0
  67. data/vendor/liburing/man/io_uring_prep_read.3 +50 -0
  68. data/vendor/liburing/man/io_uring_prep_read_fixed.3 +54 -0
  69. data/vendor/liburing/man/io_uring_prep_readv.3 +51 -0
  70. data/vendor/liburing/man/io_uring_prep_readv2.3 +79 -0
  71. data/vendor/liburing/man/io_uring_prep_write.3 +50 -0
  72. data/vendor/liburing/man/io_uring_prep_write_fixed.3 +54 -0
  73. data/vendor/liburing/man/io_uring_prep_writev.3 +51 -0
  74. data/vendor/liburing/man/io_uring_prep_writev2.3 +78 -0
  75. data/vendor/liburing/man/io_uring_queue_exit.3 +27 -0
  76. data/vendor/liburing/man/io_uring_queue_init.3 +44 -0
  77. data/vendor/liburing/man/io_uring_register.2 +688 -0
  78. data/vendor/liburing/man/io_uring_register_buffers.3 +41 -0
  79. data/vendor/liburing/man/io_uring_register_files.3 +35 -0
  80. data/vendor/liburing/man/io_uring_setup.2 +534 -0
  81. data/vendor/liburing/man/io_uring_sq_ready.3 +25 -0
  82. data/vendor/liburing/man/io_uring_sq_space_left.3 +25 -0
  83. data/vendor/liburing/man/io_uring_sqe_set_data.3 +30 -0
  84. data/vendor/liburing/man/io_uring_sqe_set_flags.3 +60 -0
  85. data/vendor/liburing/man/io_uring_sqring_wait.3 +30 -0
  86. data/vendor/liburing/man/io_uring_submit.3 +29 -0
  87. data/vendor/liburing/man/io_uring_submit_and_wait.3 +34 -0
  88. data/vendor/liburing/man/io_uring_submit_and_wait_timeout.3 +49 -0
  89. data/vendor/liburing/man/io_uring_unregister_buffers.3 +26 -0
  90. data/vendor/liburing/man/io_uring_unregister_files.3 +26 -0
  91. data/vendor/liburing/man/io_uring_wait_cqe.3 +33 -0
  92. data/vendor/liburing/man/io_uring_wait_cqe_nr.3 +36 -0
  93. data/vendor/liburing/man/io_uring_wait_cqe_timeout.3 +39 -0
  94. data/vendor/liburing/man/io_uring_wait_cqes.3 +46 -0
  95. data/vendor/liburing/src/Makefile +89 -0
  96. data/vendor/liburing/src/arch/aarch64/syscall.h +95 -0
  97. data/vendor/liburing/src/arch/generic/lib.h +21 -0
  98. data/vendor/liburing/src/arch/generic/syscall.h +87 -0
  99. data/vendor/liburing/src/arch/syscall-defs.h +67 -0
  100. data/vendor/liburing/src/arch/x86/lib.h +32 -0
  101. data/vendor/liburing/src/arch/x86/syscall.h +160 -0
  102. data/vendor/liburing/src/include/liburing/barrier.h +81 -0
  103. data/vendor/liburing/src/include/liburing/io_uring.h +442 -0
  104. data/vendor/liburing/src/include/liburing.h +921 -0
  105. data/vendor/liburing/src/int_flags.h +8 -0
  106. data/vendor/liburing/src/lib.h +57 -0
  107. data/vendor/liburing/src/liburing.map +53 -0
  108. data/vendor/liburing/src/nolibc.c +48 -0
  109. data/vendor/liburing/src/queue.c +403 -0
  110. data/vendor/liburing/src/register.c +293 -0
  111. data/vendor/liburing/src/setup.c +332 -0
  112. data/vendor/liburing/src/syscall.c +47 -0
  113. data/vendor/liburing/src/syscall.h +103 -0
  114. data/vendor/liburing/test/232c93d07b74-test.c +306 -0
  115. data/vendor/liburing/test/35fa71a030ca-test.c +329 -0
  116. data/vendor/liburing/test/500f9fbadef8-test.c +89 -0
  117. data/vendor/liburing/test/7ad0e4b2f83c-test.c +93 -0
  118. data/vendor/liburing/test/8a9973408177-test.c +106 -0
  119. data/vendor/liburing/test/917257daa0fe-test.c +53 -0
  120. data/vendor/liburing/test/Makefile +244 -0
  121. data/vendor/liburing/test/a0908ae19763-test.c +58 -0
  122. data/vendor/liburing/test/a4c0b3decb33-test.c +180 -0
  123. data/vendor/liburing/test/accept-link.c +254 -0
  124. data/vendor/liburing/test/accept-reuse.c +164 -0
  125. data/vendor/liburing/test/accept-test.c +79 -0
  126. data/vendor/liburing/test/accept.c +477 -0
  127. data/vendor/liburing/test/across-fork.c +283 -0
  128. data/vendor/liburing/test/b19062a56726-test.c +53 -0
  129. data/vendor/liburing/test/b5837bd5311d-test.c +77 -0
  130. data/vendor/liburing/test/ce593a6c480a-test.c +136 -0
  131. data/vendor/liburing/test/close-opath.c +122 -0
  132. data/vendor/liburing/test/config +10 -0
  133. data/vendor/liburing/test/connect.c +398 -0
  134. data/vendor/liburing/test/cq-full.c +96 -0
  135. data/vendor/liburing/test/cq-overflow.c +294 -0
  136. data/vendor/liburing/test/cq-peek-batch.c +102 -0
  137. data/vendor/liburing/test/cq-ready.c +94 -0
  138. data/vendor/liburing/test/cq-size.c +64 -0
  139. data/vendor/liburing/test/d4ae271dfaae-test.c +96 -0
  140. data/vendor/liburing/test/d77a67ed5f27-test.c +65 -0
  141. data/vendor/liburing/test/defer.c +307 -0
  142. data/vendor/liburing/test/double-poll-crash.c +185 -0
  143. data/vendor/liburing/test/drop-submit.c +92 -0
  144. data/vendor/liburing/test/eeed8b54e0df-test.c +114 -0
  145. data/vendor/liburing/test/empty-eownerdead.c +45 -0
  146. data/vendor/liburing/test/eventfd-disable.c +151 -0
  147. data/vendor/liburing/test/eventfd-reg.c +76 -0
  148. data/vendor/liburing/test/eventfd-ring.c +97 -0
  149. data/vendor/liburing/test/eventfd.c +112 -0
  150. data/vendor/liburing/test/exec-target.c +6 -0
  151. data/vendor/liburing/test/exit-no-cleanup.c +117 -0
  152. data/vendor/liburing/test/fadvise.c +202 -0
  153. data/vendor/liburing/test/fallocate.c +249 -0
  154. data/vendor/liburing/test/fc2a85cb02ef-test.c +131 -0
  155. data/vendor/liburing/test/file-register.c +858 -0
  156. data/vendor/liburing/test/file-update.c +173 -0
  157. data/vendor/liburing/test/file-verify.c +629 -0
  158. data/vendor/liburing/test/files-exit-hang-poll.c +128 -0
  159. data/vendor/liburing/test/files-exit-hang-timeout.c +134 -0
  160. data/vendor/liburing/test/fixed-link.c +90 -0
  161. data/vendor/liburing/test/fpos.c +252 -0
  162. data/vendor/liburing/test/fsync.c +224 -0
  163. data/vendor/liburing/test/hardlink.c +136 -0
  164. data/vendor/liburing/test/helpers.c +135 -0
  165. data/vendor/liburing/test/helpers.h +67 -0
  166. data/vendor/liburing/test/io-cancel.c +550 -0
  167. data/vendor/liburing/test/io_uring_enter.c +296 -0
  168. data/vendor/liburing/test/io_uring_register.c +676 -0
  169. data/vendor/liburing/test/io_uring_setup.c +192 -0
  170. data/vendor/liburing/test/iopoll.c +372 -0
  171. data/vendor/liburing/test/lfs-openat-write.c +119 -0
  172. data/vendor/liburing/test/lfs-openat.c +275 -0
  173. data/vendor/liburing/test/link-timeout.c +1107 -0
  174. data/vendor/liburing/test/link.c +496 -0
  175. data/vendor/liburing/test/link_drain.c +229 -0
  176. data/vendor/liburing/test/madvise.c +195 -0
  177. data/vendor/liburing/test/mkdir.c +108 -0
  178. data/vendor/liburing/test/msg-ring.c +234 -0
  179. data/vendor/liburing/test/multicqes_drain.c +387 -0
  180. data/vendor/liburing/test/nop-all-sizes.c +99 -0
  181. data/vendor/liburing/test/nop.c +115 -0
  182. data/vendor/liburing/test/open-close.c +261 -0
  183. data/vendor/liburing/test/openat2.c +308 -0
  184. data/vendor/liburing/test/personality.c +204 -0
  185. data/vendor/liburing/test/pipe-eof.c +83 -0
  186. data/vendor/liburing/test/pipe-reuse.c +105 -0
  187. data/vendor/liburing/test/poll-cancel-ton.c +135 -0
  188. data/vendor/liburing/test/poll-cancel.c +228 -0
  189. data/vendor/liburing/test/poll-link.c +230 -0
  190. data/vendor/liburing/test/poll-many.c +208 -0
  191. data/vendor/liburing/test/poll-mshot-update.c +273 -0
  192. data/vendor/liburing/test/poll-ring.c +48 -0
  193. data/vendor/liburing/test/poll-v-poll.c +353 -0
  194. data/vendor/liburing/test/poll.c +109 -0
  195. data/vendor/liburing/test/pollfree.c +426 -0
  196. data/vendor/liburing/test/probe.c +135 -0
  197. data/vendor/liburing/test/read-write.c +876 -0
  198. data/vendor/liburing/test/register-restrictions.c +633 -0
  199. data/vendor/liburing/test/rename.c +135 -0
  200. data/vendor/liburing/test/ring-leak.c +173 -0
  201. data/vendor/liburing/test/ring-leak2.c +249 -0
  202. data/vendor/liburing/test/rsrc_tags.c +449 -0
  203. data/vendor/liburing/test/runtests-loop.sh +16 -0
  204. data/vendor/liburing/test/runtests.sh +170 -0
  205. data/vendor/liburing/test/rw_merge_test.c +97 -0
  206. data/vendor/liburing/test/self.c +91 -0
  207. data/vendor/liburing/test/send_recv.c +286 -0
  208. data/vendor/liburing/test/send_recvmsg.c +345 -0
  209. data/vendor/liburing/test/sendmsg_fs_cve.c +200 -0
  210. data/vendor/liburing/test/shared-wq.c +84 -0
  211. data/vendor/liburing/test/short-read.c +75 -0
  212. data/vendor/liburing/test/shutdown.c +165 -0
  213. data/vendor/liburing/test/sigfd-deadlock.c +74 -0
  214. data/vendor/liburing/test/skip-cqe.c +429 -0
  215. data/vendor/liburing/test/socket-rw-eagain.c +158 -0
  216. data/vendor/liburing/test/socket-rw-offset.c +157 -0
  217. data/vendor/liburing/test/socket-rw.c +145 -0
  218. data/vendor/liburing/test/splice.c +512 -0
  219. data/vendor/liburing/test/sq-full-cpp.cc +45 -0
  220. data/vendor/liburing/test/sq-full.c +45 -0
  221. data/vendor/liburing/test/sq-poll-dup.c +204 -0
  222. data/vendor/liburing/test/sq-poll-kthread.c +169 -0
  223. data/vendor/liburing/test/sq-poll-share.c +137 -0
  224. data/vendor/liburing/test/sq-space_left.c +159 -0
  225. data/vendor/liburing/test/sqpoll-cancel-hang.c +157 -0
  226. data/vendor/liburing/test/sqpoll-disable-exit.c +196 -0
  227. data/vendor/liburing/test/sqpoll-exit-hang.c +78 -0
  228. data/vendor/liburing/test/sqpoll-sleep.c +69 -0
  229. data/vendor/liburing/test/statx.c +172 -0
  230. data/vendor/liburing/test/stdout.c +232 -0
  231. data/vendor/liburing/test/submit-link-fail.c +154 -0
  232. data/vendor/liburing/test/submit-reuse.c +239 -0
  233. data/vendor/liburing/test/symlink.c +116 -0
  234. data/vendor/liburing/test/teardowns.c +58 -0
  235. data/vendor/liburing/test/thread-exit.c +143 -0
  236. data/vendor/liburing/test/timeout-new.c +252 -0
  237. data/vendor/liburing/test/timeout-overflow.c +204 -0
  238. data/vendor/liburing/test/timeout.c +1523 -0
  239. data/vendor/liburing/test/unlink.c +112 -0
  240. data/vendor/liburing/test/wakeup-hang.c +162 -0
  241. metadata +227 -2
@@ -0,0 +1,261 @@
1
+ /* SPDX-License-Identifier: MIT */
2
+ /*
3
+ * Description: run various openat(2) 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 <assert.h>
13
+
14
+ #include "helpers.h"
15
+ #include "liburing.h"
16
+
17
+ static int submit_wait(struct io_uring *ring)
18
+ {
19
+ struct io_uring_cqe *cqe;
20
+ int ret;
21
+
22
+ ret = io_uring_submit(ring);
23
+ if (ret <= 0) {
24
+ fprintf(stderr, "sqe submit failed: %d\n", ret);
25
+ return 1;
26
+ }
27
+ ret = io_uring_wait_cqe(ring, &cqe);
28
+ if (ret < 0) {
29
+ fprintf(stderr, "wait completion %d\n", ret);
30
+ return 1;
31
+ }
32
+
33
+ ret = cqe->res;
34
+ io_uring_cqe_seen(ring, cqe);
35
+ return ret;
36
+ }
37
+
38
+ static inline int try_close(struct io_uring *ring, int fd, int slot)
39
+ {
40
+ struct io_uring_sqe *sqe;
41
+
42
+ sqe = io_uring_get_sqe(ring);
43
+ io_uring_prep_close(sqe, fd);
44
+ __io_uring_set_target_fixed_file(sqe, slot);
45
+ return submit_wait(ring);
46
+ }
47
+
48
+ static int test_close_fixed(void)
49
+ {
50
+ struct io_uring ring;
51
+ struct io_uring_sqe *sqe;
52
+ int ret, fds[2];
53
+ char buf[1];
54
+
55
+ ret = io_uring_queue_init(8, &ring, 0);
56
+ if (ret) {
57
+ fprintf(stderr, "ring setup failed\n");
58
+ return -1;
59
+ }
60
+ if (pipe(fds)) {
61
+ perror("pipe");
62
+ return -1;
63
+ }
64
+
65
+ ret = try_close(&ring, 0, 0);
66
+ if (ret == -EINVAL) {
67
+ fprintf(stderr, "close for fixed files is not supported\n");
68
+ return 0;
69
+ } else if (ret != -ENXIO) {
70
+ fprintf(stderr, "no table failed %i\n", ret);
71
+ return -1;
72
+ }
73
+
74
+ ret = try_close(&ring, 1, 0);
75
+ if (ret != -EINVAL) {
76
+ fprintf(stderr, "set fd failed %i\n", ret);
77
+ return -1;
78
+ }
79
+
80
+ ret = io_uring_register_files(&ring, fds, 2);
81
+ if (ret) {
82
+ fprintf(stderr, "file_register: %d\n", ret);
83
+ return ret;
84
+ }
85
+
86
+ ret = try_close(&ring, 0, 2);
87
+ if (ret != -EINVAL) {
88
+ fprintf(stderr, "out of table failed %i\n", ret);
89
+ return -1;
90
+ }
91
+
92
+ ret = try_close(&ring, 0, 0);
93
+ if (ret != 0) {
94
+ fprintf(stderr, "close failed %i\n", ret);
95
+ return -1;
96
+ }
97
+
98
+ sqe = io_uring_get_sqe(&ring);
99
+ io_uring_prep_read(sqe, 0, buf, sizeof(buf), 0);
100
+ sqe->flags |= IOSQE_FIXED_FILE;
101
+ ret = submit_wait(&ring);
102
+ if (ret != -EBADF) {
103
+ fprintf(stderr, "read failed %i\n", ret);
104
+ return -1;
105
+ }
106
+
107
+ ret = try_close(&ring, 0, 1);
108
+ if (ret != 0) {
109
+ fprintf(stderr, "close 2 failed %i\n", ret);
110
+ return -1;
111
+ }
112
+
113
+ ret = try_close(&ring, 0, 0);
114
+ if (ret != -EBADF) {
115
+ fprintf(stderr, "empty slot failed %i\n", ret);
116
+ return -1;
117
+ }
118
+
119
+ close(fds[0]);
120
+ close(fds[1]);
121
+ io_uring_queue_exit(&ring);
122
+ return 0;
123
+ }
124
+
125
+ static int test_close(struct io_uring *ring, int fd, int is_ring_fd)
126
+ {
127
+ struct io_uring_cqe *cqe;
128
+ struct io_uring_sqe *sqe;
129
+ int ret;
130
+
131
+ sqe = io_uring_get_sqe(ring);
132
+ if (!sqe) {
133
+ fprintf(stderr, "get sqe failed\n");
134
+ goto err;
135
+ }
136
+ io_uring_prep_close(sqe, fd);
137
+
138
+ ret = io_uring_submit(ring);
139
+ if (ret <= 0) {
140
+ fprintf(stderr, "sqe submit failed: %d\n", ret);
141
+ goto err;
142
+ }
143
+
144
+ ret = io_uring_wait_cqe(ring, &cqe);
145
+ if (ret < 0) {
146
+ if (!(is_ring_fd && ret == -EBADF)) {
147
+ fprintf(stderr, "wait completion %d\n", ret);
148
+ goto err;
149
+ }
150
+ return ret;
151
+ }
152
+ ret = cqe->res;
153
+ io_uring_cqe_seen(ring, cqe);
154
+ return ret;
155
+ err:
156
+ return -1;
157
+ }
158
+
159
+ static int test_openat(struct io_uring *ring, const char *path, int dfd)
160
+ {
161
+ struct io_uring_cqe *cqe;
162
+ struct io_uring_sqe *sqe;
163
+ int ret;
164
+
165
+ sqe = io_uring_get_sqe(ring);
166
+ if (!sqe) {
167
+ fprintf(stderr, "get sqe failed\n");
168
+ goto err;
169
+ }
170
+ io_uring_prep_openat(sqe, dfd, path, O_RDONLY, 0);
171
+
172
+ ret = io_uring_submit(ring);
173
+ if (ret <= 0) {
174
+ fprintf(stderr, "sqe submit failed: %d\n", ret);
175
+ goto err;
176
+ }
177
+
178
+ ret = io_uring_wait_cqe(ring, &cqe);
179
+ if (ret < 0) {
180
+ fprintf(stderr, "wait completion %d\n", ret);
181
+ goto err;
182
+ }
183
+ ret = cqe->res;
184
+ io_uring_cqe_seen(ring, cqe);
185
+ return ret;
186
+ err:
187
+ return -1;
188
+ }
189
+
190
+ int main(int argc, char *argv[])
191
+ {
192
+ struct io_uring ring;
193
+ const char *path, *path_rel;
194
+ int ret, do_unlink;
195
+
196
+ ret = io_uring_queue_init(8, &ring, 0);
197
+ if (ret) {
198
+ fprintf(stderr, "ring setup failed\n");
199
+ return 1;
200
+ }
201
+
202
+ if (argc > 1) {
203
+ path = "/tmp/.open.close";
204
+ path_rel = argv[1];
205
+ do_unlink = 0;
206
+ } else {
207
+ path = "/tmp/.open.close";
208
+ path_rel = ".open.close";
209
+ do_unlink = 1;
210
+ }
211
+
212
+ t_create_file(path, 4096);
213
+
214
+ if (do_unlink)
215
+ t_create_file(path_rel, 4096);
216
+
217
+ ret = test_openat(&ring, path, -1);
218
+ if (ret < 0) {
219
+ if (ret == -EINVAL) {
220
+ fprintf(stdout, "Open not supported, skipping\n");
221
+ goto done;
222
+ }
223
+ fprintf(stderr, "test_openat absolute failed: %d\n", ret);
224
+ goto err;
225
+ }
226
+
227
+ ret = test_openat(&ring, path_rel, AT_FDCWD);
228
+ if (ret < 0) {
229
+ fprintf(stderr, "test_openat relative failed: %d\n", ret);
230
+ goto err;
231
+ }
232
+
233
+ ret = test_close(&ring, ret, 0);
234
+ if (ret) {
235
+ fprintf(stderr, "test_close normal failed\n");
236
+ goto err;
237
+ }
238
+
239
+ ret = test_close(&ring, ring.ring_fd, 1);
240
+ if (ret != -EBADF) {
241
+ fprintf(stderr, "test_close ring_fd failed\n");
242
+ goto err;
243
+ }
244
+
245
+ ret = test_close_fixed();
246
+ if (ret) {
247
+ fprintf(stderr, "test_close_fixed failed\n");
248
+ goto err;
249
+ }
250
+
251
+ done:
252
+ unlink(path);
253
+ if (do_unlink)
254
+ unlink(path_rel);
255
+ return 0;
256
+ err:
257
+ unlink(path);
258
+ if (do_unlink)
259
+ unlink(path_rel);
260
+ return 1;
261
+ }
@@ -0,0 +1,308 @@
1
+ /* SPDX-License-Identifier: MIT */
2
+ /*
3
+ * Description: run various openat(2) 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/uio.h>
13
+
14
+ #include "helpers.h"
15
+ #include "liburing.h"
16
+
17
+ static int test_openat2(struct io_uring *ring, const char *path, int dfd,
18
+ bool direct, int fixed_index)
19
+ {
20
+ struct io_uring_cqe *cqe;
21
+ struct io_uring_sqe *sqe;
22
+ struct open_how how;
23
+ int ret;
24
+
25
+ sqe = io_uring_get_sqe(ring);
26
+ if (!sqe) {
27
+ fprintf(stderr, "get sqe failed\n");
28
+ return -1;
29
+ }
30
+ memset(&how, 0, sizeof(how));
31
+ how.flags = O_RDWR;
32
+
33
+ if (!direct)
34
+ io_uring_prep_openat2(sqe, dfd, path, &how);
35
+ else
36
+ io_uring_prep_openat2_direct(sqe, dfd, path, &how, fixed_index);
37
+
38
+ ret = io_uring_submit(ring);
39
+ if (ret <= 0) {
40
+ fprintf(stderr, "sqe submit failed: %d\n", ret);
41
+ return -1;
42
+ }
43
+
44
+ ret = io_uring_wait_cqe(ring, &cqe);
45
+ if (ret < 0) {
46
+ fprintf(stderr, "wait completion %d\n", ret);
47
+ return -1;
48
+ }
49
+ ret = cqe->res;
50
+ io_uring_cqe_seen(ring, cqe);
51
+
52
+ if (direct && ret > 0) {
53
+ close(ret);
54
+ return -EINVAL;
55
+ }
56
+ return ret;
57
+ }
58
+
59
+ static int test_open_fixed(const char *path, int dfd)
60
+ {
61
+ struct io_uring_cqe *cqe;
62
+ struct io_uring_sqe *sqe;
63
+ struct io_uring ring;
64
+ const char pattern = 0xac;
65
+ char buffer[] = { 0, 0 };
66
+ int i, ret, fd = -1;
67
+
68
+ ret = io_uring_queue_init(8, &ring, 0);
69
+ if (ret) {
70
+ fprintf(stderr, "ring setup failed\n");
71
+ return -1;
72
+ }
73
+ ret = io_uring_register_files(&ring, &fd, 1);
74
+ if (ret) {
75
+ fprintf(stderr, "%s: register ret=%d\n", __FUNCTION__, ret);
76
+ return -1;
77
+ }
78
+
79
+ ret = test_openat2(&ring, path, dfd, true, 0);
80
+ if (ret == -EINVAL) {
81
+ printf("fixed open isn't supported\n");
82
+ return 1;
83
+ } else if (ret) {
84
+ fprintf(stderr, "direct open failed %d\n", ret);
85
+ return -1;
86
+ }
87
+
88
+ sqe = io_uring_get_sqe(&ring);
89
+ io_uring_prep_write(sqe, 0, &pattern, 1, 0);
90
+ sqe->user_data = 1;
91
+ sqe->flags |= IOSQE_FIXED_FILE | IOSQE_IO_LINK;
92
+
93
+ sqe = io_uring_get_sqe(&ring);
94
+ io_uring_prep_read(sqe, 0, buffer, 1, 0);
95
+ sqe->user_data = 2;
96
+ sqe->flags |= IOSQE_FIXED_FILE;
97
+
98
+ ret = io_uring_submit(&ring);
99
+ if (ret != 2) {
100
+ fprintf(stderr, "%s: got %d, wanted 2\n", __FUNCTION__, ret);
101
+ return -1;
102
+ }
103
+
104
+ for (i = 0; i < 2; i++) {
105
+ ret = io_uring_wait_cqe(&ring, &cqe);
106
+ if (ret < 0) {
107
+ fprintf(stderr, "wait completion %d\n", ret);
108
+ return -1;
109
+ }
110
+ if (cqe->res != 1) {
111
+ fprintf(stderr, "unexpectetd ret %d\n", cqe->res);
112
+ return -1;
113
+ }
114
+ io_uring_cqe_seen(&ring, cqe);
115
+ }
116
+ if (memcmp(&pattern, buffer, 1) != 0) {
117
+ fprintf(stderr, "buf validation failed\n");
118
+ return -1;
119
+ }
120
+
121
+ io_uring_queue_exit(&ring);
122
+ return 0;
123
+ }
124
+
125
+ static int test_open_fixed_fail(const char *path, int dfd)
126
+ {
127
+ struct io_uring ring;
128
+ int ret, fd = -1;
129
+
130
+ ret = io_uring_queue_init(8, &ring, 0);
131
+ if (ret) {
132
+ fprintf(stderr, "ring setup failed\n");
133
+ return -1;
134
+ }
135
+
136
+ ret = test_openat2(&ring, path, dfd, true, 0);
137
+ if (ret != -ENXIO) {
138
+ fprintf(stderr, "install into not existing table, %i\n", ret);
139
+ return 1;
140
+ }
141
+
142
+ ret = io_uring_register_files(&ring, &fd, 1);
143
+ if (ret) {
144
+ fprintf(stderr, "%s: register ret=%d\n", __FUNCTION__, ret);
145
+ return -1;
146
+ }
147
+
148
+ ret = test_openat2(&ring, path, dfd, true, 1);
149
+ if (ret != -EINVAL) {
150
+ fprintf(stderr, "install out of bounds, %i\n", ret);
151
+ return -1;
152
+ }
153
+
154
+ ret = test_openat2(&ring, path, dfd, true, (1u << 16));
155
+ if (ret != -EINVAL) {
156
+ fprintf(stderr, "install out of bounds or u16 overflow, %i\n", ret);
157
+ return -1;
158
+ }
159
+
160
+ ret = test_openat2(&ring, path, dfd, true, (1u << 16) + 1);
161
+ if (ret != -EINVAL) {
162
+ fprintf(stderr, "install out of bounds or u16 overflow, %i\n", ret);
163
+ return -1;
164
+ }
165
+
166
+ io_uring_queue_exit(&ring);
167
+ return 0;
168
+ }
169
+
170
+ static int test_direct_reinstall(const char *path, int dfd)
171
+ {
172
+ struct io_uring_cqe *cqe;
173
+ struct io_uring_sqe *sqe;
174
+ char buf[1] = { 0xfa };
175
+ struct io_uring ring;
176
+ int ret, pipe_fds[2];
177
+ ssize_t ret2;
178
+
179
+ if (pipe2(pipe_fds, O_NONBLOCK)) {
180
+ fprintf(stderr, "pipe() failed\n");
181
+ return -1;
182
+ }
183
+ ret = io_uring_queue_init(8, &ring, 0);
184
+ if (ret) {
185
+ fprintf(stderr, "ring setup failed\n");
186
+ return -1;
187
+ }
188
+ ret = io_uring_register_files(&ring, pipe_fds, 2);
189
+ if (ret) {
190
+ fprintf(stderr, "%s: register ret=%d\n", __FUNCTION__, ret);
191
+ return -1;
192
+ }
193
+
194
+ /* reinstall into the second slot */
195
+ ret = test_openat2(&ring, path, dfd, true, 1);
196
+ if (ret != 0) {
197
+ fprintf(stderr, "reinstall failed, %i\n", ret);
198
+ return -1;
199
+ }
200
+
201
+ /* verify it's reinstalled, first write into the slot... */
202
+ sqe = io_uring_get_sqe(&ring);
203
+ io_uring_prep_write(sqe, 1, buf, sizeof(buf), 0);
204
+ sqe->flags |= IOSQE_FIXED_FILE;
205
+
206
+ ret = io_uring_submit(&ring);
207
+ if (ret != 1) {
208
+ fprintf(stderr, "sqe submit failed: %d\n", ret);
209
+ return -1;
210
+ }
211
+ ret = io_uring_wait_cqe(&ring, &cqe);
212
+ if (ret < 0) {
213
+ fprintf(stderr, "wait completion %d\n", ret);
214
+ return ret;
215
+ }
216
+ ret = cqe->res;
217
+ io_uring_cqe_seen(&ring, cqe);
218
+ if (ret != 1) {
219
+ fprintf(stderr, "invalid write %i\n", ret);
220
+ return -1;
221
+ }
222
+
223
+ /* ... and make sure nothing has been written to the pipe */
224
+ ret2 = read(pipe_fds[0], buf, 1);
225
+ if (ret2 != 0 && !(ret2 < 0 && errno == EAGAIN)) {
226
+ fprintf(stderr, "invalid pipe read, %d %d\n", errno, (int)ret2);
227
+ return -1;
228
+ }
229
+
230
+ close(pipe_fds[0]);
231
+ close(pipe_fds[1]);
232
+ io_uring_queue_exit(&ring);
233
+ return 0;
234
+ }
235
+
236
+ int main(int argc, char *argv[])
237
+ {
238
+ struct io_uring ring;
239
+ const char *path, *path_rel;
240
+ int ret, do_unlink;
241
+
242
+ ret = io_uring_queue_init(8, &ring, 0);
243
+ if (ret) {
244
+ fprintf(stderr, "ring setup failed\n");
245
+ return 1;
246
+ }
247
+
248
+ if (argc > 1) {
249
+ path = "/tmp/.open.close";
250
+ path_rel = argv[1];
251
+ do_unlink = 0;
252
+ } else {
253
+ path = "/tmp/.open.close";
254
+ path_rel = ".open.close";
255
+ do_unlink = 1;
256
+ }
257
+
258
+ t_create_file(path, 4096);
259
+
260
+ if (do_unlink)
261
+ t_create_file(path_rel, 4096);
262
+
263
+ ret = test_openat2(&ring, path, -1, false, 0);
264
+ if (ret < 0) {
265
+ if (ret == -EINVAL) {
266
+ fprintf(stdout, "openat2 not supported, skipping\n");
267
+ goto done;
268
+ }
269
+ fprintf(stderr, "test_openat2 absolute failed: %d\n", ret);
270
+ goto err;
271
+ }
272
+
273
+ ret = test_openat2(&ring, path_rel, AT_FDCWD, false, 0);
274
+ if (ret < 0) {
275
+ fprintf(stderr, "test_openat2 relative failed: %d\n", ret);
276
+ goto err;
277
+ }
278
+
279
+ ret = test_open_fixed(path, -1);
280
+ if (ret > 0)
281
+ goto done;
282
+ if (ret) {
283
+ fprintf(stderr, "test_open_fixed failed\n");
284
+ goto err;
285
+ }
286
+ ret = test_open_fixed_fail(path, -1);
287
+ if (ret) {
288
+ fprintf(stderr, "test_open_fixed_fail failed\n");
289
+ goto err;
290
+ }
291
+
292
+ ret = test_direct_reinstall(path, -1);
293
+ if (ret) {
294
+ fprintf(stderr, "test_direct_reinstall failed\n");
295
+ goto err;
296
+ }
297
+
298
+ done:
299
+ unlink(path);
300
+ if (do_unlink)
301
+ unlink(path_rel);
302
+ return 0;
303
+ err:
304
+ unlink(path);
305
+ if (do_unlink)
306
+ unlink(path_rel);
307
+ return 1;
308
+ }