polyphony 0.85 → 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 (230) hide show
  1. checksums.yaml +4 -4
  2. data/CHANGELOG.md +4 -0
  3. data/Gemfile.lock +1 -1
  4. data/ext/polyphony/io_extensions.c +2 -3
  5. data/lib/polyphony/version.rb +1 -1
  6. data/polyphony.gemspec +1 -1
  7. data/test/test_backend.rb +1 -1
  8. data/test/test_signal.rb +3 -3
  9. data/vendor/liburing/.github/pull_request_template.md +86 -0
  10. data/vendor/liburing/.github/workflows/build.yml +85 -0
  11. data/vendor/liburing/.github/workflows/shellcheck.yml +20 -0
  12. data/vendor/liburing/.gitignore +149 -0
  13. data/vendor/liburing/COPYING +502 -0
  14. data/vendor/liburing/COPYING.GPL +339 -0
  15. data/vendor/liburing/LICENSE +7 -0
  16. data/vendor/liburing/Makefile +82 -0
  17. data/vendor/liburing/Makefile.common +5 -0
  18. data/vendor/liburing/Makefile.quiet +11 -0
  19. data/vendor/liburing/README +46 -0
  20. data/vendor/liburing/configure +486 -0
  21. data/vendor/liburing/debian/README.Debian +7 -0
  22. data/vendor/liburing/debian/changelog +27 -0
  23. data/vendor/liburing/debian/compat +1 -0
  24. data/vendor/liburing/debian/control +48 -0
  25. data/vendor/liburing/debian/copyright +49 -0
  26. data/vendor/liburing/debian/liburing-dev.install +4 -0
  27. data/vendor/liburing/debian/liburing-dev.manpages +6 -0
  28. data/vendor/liburing/debian/liburing1-udeb.install +1 -0
  29. data/vendor/liburing/debian/liburing1.install +1 -0
  30. data/vendor/liburing/debian/liburing1.symbols +32 -0
  31. data/vendor/liburing/debian/patches/series +1 -0
  32. data/vendor/liburing/debian/rules +81 -0
  33. data/vendor/liburing/debian/source/format +1 -0
  34. data/vendor/liburing/debian/source/local-options +2 -0
  35. data/vendor/liburing/debian/source/options +1 -0
  36. data/vendor/liburing/debian/watch +3 -0
  37. data/vendor/liburing/examples/Makefile +38 -0
  38. data/vendor/liburing/examples/io_uring-cp.c +282 -0
  39. data/vendor/liburing/examples/io_uring-test.c +112 -0
  40. data/vendor/liburing/examples/link-cp.c +193 -0
  41. data/vendor/liburing/examples/ucontext-cp.c +273 -0
  42. data/vendor/liburing/liburing.pc.in +12 -0
  43. data/vendor/liburing/liburing.spec +66 -0
  44. data/vendor/liburing/make-debs.sh +53 -0
  45. data/vendor/liburing/man/io_uring.7 +754 -0
  46. data/vendor/liburing/man/io_uring_cq_advance.3 +35 -0
  47. data/vendor/liburing/man/io_uring_cq_ready.3 +25 -0
  48. data/vendor/liburing/man/io_uring_cqe_get_data.3 +34 -0
  49. data/vendor/liburing/man/io_uring_cqe_seen.3 +32 -0
  50. data/vendor/liburing/man/io_uring_enter.2 +1483 -0
  51. data/vendor/liburing/man/io_uring_free_probe.3 +24 -0
  52. data/vendor/liburing/man/io_uring_get_probe.3 +29 -0
  53. data/vendor/liburing/man/io_uring_get_sqe.3 +38 -0
  54. data/vendor/liburing/man/io_uring_opcode_supported.3 +29 -0
  55. data/vendor/liburing/man/io_uring_prep_msg_ring.3 +58 -0
  56. data/vendor/liburing/man/io_uring_prep_read.3 +50 -0
  57. data/vendor/liburing/man/io_uring_prep_read_fixed.3 +54 -0
  58. data/vendor/liburing/man/io_uring_prep_readv.3 +51 -0
  59. data/vendor/liburing/man/io_uring_prep_readv2.3 +79 -0
  60. data/vendor/liburing/man/io_uring_prep_write.3 +50 -0
  61. data/vendor/liburing/man/io_uring_prep_write_fixed.3 +54 -0
  62. data/vendor/liburing/man/io_uring_prep_writev.3 +51 -0
  63. data/vendor/liburing/man/io_uring_prep_writev2.3 +78 -0
  64. data/vendor/liburing/man/io_uring_queue_exit.3 +27 -0
  65. data/vendor/liburing/man/io_uring_queue_init.3 +44 -0
  66. data/vendor/liburing/man/io_uring_register.2 +688 -0
  67. data/vendor/liburing/man/io_uring_register_buffers.3 +41 -0
  68. data/vendor/liburing/man/io_uring_register_files.3 +35 -0
  69. data/vendor/liburing/man/io_uring_setup.2 +534 -0
  70. data/vendor/liburing/man/io_uring_sq_ready.3 +25 -0
  71. data/vendor/liburing/man/io_uring_sq_space_left.3 +25 -0
  72. data/vendor/liburing/man/io_uring_sqe_set_data.3 +30 -0
  73. data/vendor/liburing/man/io_uring_sqe_set_flags.3 +60 -0
  74. data/vendor/liburing/man/io_uring_sqring_wait.3 +30 -0
  75. data/vendor/liburing/man/io_uring_submit.3 +29 -0
  76. data/vendor/liburing/man/io_uring_submit_and_wait.3 +34 -0
  77. data/vendor/liburing/man/io_uring_submit_and_wait_timeout.3 +49 -0
  78. data/vendor/liburing/man/io_uring_unregister_buffers.3 +26 -0
  79. data/vendor/liburing/man/io_uring_unregister_files.3 +26 -0
  80. data/vendor/liburing/man/io_uring_wait_cqe.3 +33 -0
  81. data/vendor/liburing/man/io_uring_wait_cqe_nr.3 +36 -0
  82. data/vendor/liburing/man/io_uring_wait_cqe_timeout.3 +39 -0
  83. data/vendor/liburing/man/io_uring_wait_cqes.3 +46 -0
  84. data/vendor/liburing/src/Makefile +89 -0
  85. data/vendor/liburing/src/arch/aarch64/syscall.h +95 -0
  86. data/vendor/liburing/src/arch/generic/lib.h +21 -0
  87. data/vendor/liburing/src/arch/generic/syscall.h +87 -0
  88. data/vendor/liburing/src/arch/syscall-defs.h +67 -0
  89. data/vendor/liburing/src/arch/x86/lib.h +32 -0
  90. data/vendor/liburing/src/arch/x86/syscall.h +160 -0
  91. data/vendor/liburing/src/include/liburing/barrier.h +81 -0
  92. data/vendor/liburing/src/include/liburing/io_uring.h +442 -0
  93. data/vendor/liburing/src/include/liburing.h +921 -0
  94. data/vendor/liburing/src/int_flags.h +8 -0
  95. data/vendor/liburing/src/lib.h +57 -0
  96. data/vendor/liburing/src/liburing.map +53 -0
  97. data/vendor/liburing/src/nolibc.c +48 -0
  98. data/vendor/liburing/src/queue.c +403 -0
  99. data/vendor/liburing/src/register.c +293 -0
  100. data/vendor/liburing/src/setup.c +332 -0
  101. data/vendor/liburing/src/syscall.c +47 -0
  102. data/vendor/liburing/src/syscall.h +103 -0
  103. data/vendor/liburing/test/232c93d07b74-test.c +306 -0
  104. data/vendor/liburing/test/35fa71a030ca-test.c +329 -0
  105. data/vendor/liburing/test/500f9fbadef8-test.c +89 -0
  106. data/vendor/liburing/test/7ad0e4b2f83c-test.c +93 -0
  107. data/vendor/liburing/test/8a9973408177-test.c +106 -0
  108. data/vendor/liburing/test/917257daa0fe-test.c +53 -0
  109. data/vendor/liburing/test/Makefile +244 -0
  110. data/vendor/liburing/test/a0908ae19763-test.c +58 -0
  111. data/vendor/liburing/test/a4c0b3decb33-test.c +180 -0
  112. data/vendor/liburing/test/accept-link.c +254 -0
  113. data/vendor/liburing/test/accept-reuse.c +164 -0
  114. data/vendor/liburing/test/accept-test.c +79 -0
  115. data/vendor/liburing/test/accept.c +477 -0
  116. data/vendor/liburing/test/across-fork.c +283 -0
  117. data/vendor/liburing/test/b19062a56726-test.c +53 -0
  118. data/vendor/liburing/test/b5837bd5311d-test.c +77 -0
  119. data/vendor/liburing/test/ce593a6c480a-test.c +136 -0
  120. data/vendor/liburing/test/close-opath.c +122 -0
  121. data/vendor/liburing/test/config +10 -0
  122. data/vendor/liburing/test/connect.c +398 -0
  123. data/vendor/liburing/test/cq-full.c +96 -0
  124. data/vendor/liburing/test/cq-overflow.c +294 -0
  125. data/vendor/liburing/test/cq-peek-batch.c +102 -0
  126. data/vendor/liburing/test/cq-ready.c +94 -0
  127. data/vendor/liburing/test/cq-size.c +64 -0
  128. data/vendor/liburing/test/d4ae271dfaae-test.c +96 -0
  129. data/vendor/liburing/test/d77a67ed5f27-test.c +65 -0
  130. data/vendor/liburing/test/defer.c +307 -0
  131. data/vendor/liburing/test/double-poll-crash.c +185 -0
  132. data/vendor/liburing/test/drop-submit.c +92 -0
  133. data/vendor/liburing/test/eeed8b54e0df-test.c +114 -0
  134. data/vendor/liburing/test/empty-eownerdead.c +45 -0
  135. data/vendor/liburing/test/eventfd-disable.c +151 -0
  136. data/vendor/liburing/test/eventfd-reg.c +76 -0
  137. data/vendor/liburing/test/eventfd-ring.c +97 -0
  138. data/vendor/liburing/test/eventfd.c +112 -0
  139. data/vendor/liburing/test/exec-target.c +6 -0
  140. data/vendor/liburing/test/exit-no-cleanup.c +117 -0
  141. data/vendor/liburing/test/fadvise.c +202 -0
  142. data/vendor/liburing/test/fallocate.c +249 -0
  143. data/vendor/liburing/test/fc2a85cb02ef-test.c +131 -0
  144. data/vendor/liburing/test/file-register.c +858 -0
  145. data/vendor/liburing/test/file-update.c +173 -0
  146. data/vendor/liburing/test/file-verify.c +629 -0
  147. data/vendor/liburing/test/files-exit-hang-poll.c +128 -0
  148. data/vendor/liburing/test/files-exit-hang-timeout.c +134 -0
  149. data/vendor/liburing/test/fixed-link.c +90 -0
  150. data/vendor/liburing/test/fpos.c +252 -0
  151. data/vendor/liburing/test/fsync.c +224 -0
  152. data/vendor/liburing/test/hardlink.c +136 -0
  153. data/vendor/liburing/test/helpers.c +135 -0
  154. data/vendor/liburing/test/helpers.h +67 -0
  155. data/vendor/liburing/test/io-cancel.c +550 -0
  156. data/vendor/liburing/test/io_uring_enter.c +296 -0
  157. data/vendor/liburing/test/io_uring_register.c +676 -0
  158. data/vendor/liburing/test/io_uring_setup.c +192 -0
  159. data/vendor/liburing/test/iopoll.c +372 -0
  160. data/vendor/liburing/test/lfs-openat-write.c +119 -0
  161. data/vendor/liburing/test/lfs-openat.c +275 -0
  162. data/vendor/liburing/test/link-timeout.c +1107 -0
  163. data/vendor/liburing/test/link.c +496 -0
  164. data/vendor/liburing/test/link_drain.c +229 -0
  165. data/vendor/liburing/test/madvise.c +195 -0
  166. data/vendor/liburing/test/mkdir.c +108 -0
  167. data/vendor/liburing/test/msg-ring.c +234 -0
  168. data/vendor/liburing/test/multicqes_drain.c +387 -0
  169. data/vendor/liburing/test/nop-all-sizes.c +99 -0
  170. data/vendor/liburing/test/nop.c +115 -0
  171. data/vendor/liburing/test/open-close.c +261 -0
  172. data/vendor/liburing/test/openat2.c +308 -0
  173. data/vendor/liburing/test/personality.c +204 -0
  174. data/vendor/liburing/test/pipe-eof.c +83 -0
  175. data/vendor/liburing/test/pipe-reuse.c +105 -0
  176. data/vendor/liburing/test/poll-cancel-ton.c +135 -0
  177. data/vendor/liburing/test/poll-cancel.c +228 -0
  178. data/vendor/liburing/test/poll-link.c +230 -0
  179. data/vendor/liburing/test/poll-many.c +208 -0
  180. data/vendor/liburing/test/poll-mshot-update.c +273 -0
  181. data/vendor/liburing/test/poll-ring.c +48 -0
  182. data/vendor/liburing/test/poll-v-poll.c +353 -0
  183. data/vendor/liburing/test/poll.c +109 -0
  184. data/vendor/liburing/test/pollfree.c +426 -0
  185. data/vendor/liburing/test/probe.c +135 -0
  186. data/vendor/liburing/test/read-write.c +876 -0
  187. data/vendor/liburing/test/register-restrictions.c +633 -0
  188. data/vendor/liburing/test/rename.c +135 -0
  189. data/vendor/liburing/test/ring-leak.c +173 -0
  190. data/vendor/liburing/test/ring-leak2.c +249 -0
  191. data/vendor/liburing/test/rsrc_tags.c +449 -0
  192. data/vendor/liburing/test/runtests-loop.sh +16 -0
  193. data/vendor/liburing/test/runtests.sh +170 -0
  194. data/vendor/liburing/test/rw_merge_test.c +97 -0
  195. data/vendor/liburing/test/self.c +91 -0
  196. data/vendor/liburing/test/send_recv.c +286 -0
  197. data/vendor/liburing/test/send_recvmsg.c +345 -0
  198. data/vendor/liburing/test/sendmsg_fs_cve.c +200 -0
  199. data/vendor/liburing/test/shared-wq.c +84 -0
  200. data/vendor/liburing/test/short-read.c +75 -0
  201. data/vendor/liburing/test/shutdown.c +165 -0
  202. data/vendor/liburing/test/sigfd-deadlock.c +74 -0
  203. data/vendor/liburing/test/skip-cqe.c +429 -0
  204. data/vendor/liburing/test/socket-rw-eagain.c +158 -0
  205. data/vendor/liburing/test/socket-rw-offset.c +157 -0
  206. data/vendor/liburing/test/socket-rw.c +145 -0
  207. data/vendor/liburing/test/splice.c +512 -0
  208. data/vendor/liburing/test/sq-full-cpp.cc +45 -0
  209. data/vendor/liburing/test/sq-full.c +45 -0
  210. data/vendor/liburing/test/sq-poll-dup.c +204 -0
  211. data/vendor/liburing/test/sq-poll-kthread.c +169 -0
  212. data/vendor/liburing/test/sq-poll-share.c +137 -0
  213. data/vendor/liburing/test/sq-space_left.c +159 -0
  214. data/vendor/liburing/test/sqpoll-cancel-hang.c +157 -0
  215. data/vendor/liburing/test/sqpoll-disable-exit.c +196 -0
  216. data/vendor/liburing/test/sqpoll-exit-hang.c +78 -0
  217. data/vendor/liburing/test/sqpoll-sleep.c +69 -0
  218. data/vendor/liburing/test/statx.c +172 -0
  219. data/vendor/liburing/test/stdout.c +232 -0
  220. data/vendor/liburing/test/submit-link-fail.c +154 -0
  221. data/vendor/liburing/test/submit-reuse.c +239 -0
  222. data/vendor/liburing/test/symlink.c +116 -0
  223. data/vendor/liburing/test/teardowns.c +58 -0
  224. data/vendor/liburing/test/thread-exit.c +143 -0
  225. data/vendor/liburing/test/timeout-new.c +252 -0
  226. data/vendor/liburing/test/timeout-overflow.c +204 -0
  227. data/vendor/liburing/test/timeout.c +1523 -0
  228. data/vendor/liburing/test/unlink.c +112 -0
  229. data/vendor/liburing/test/wakeup-hang.c +162 -0
  230. metadata +223 -2
@@ -0,0 +1,273 @@
1
+ /* SPDX-License-Identifier: MIT */
2
+ /*
3
+ * Description: test many files being polled for and updated
4
+ *
5
+ */
6
+ #include <errno.h>
7
+ #include <stdio.h>
8
+ #include <unistd.h>
9
+ #include <stdlib.h>
10
+ #include <string.h>
11
+ #include <signal.h>
12
+ #include <poll.h>
13
+ #include <sys/resource.h>
14
+ #include <fcntl.h>
15
+ #include <pthread.h>
16
+
17
+ #include "liburing.h"
18
+
19
+ #define NFILES 5000
20
+ #define BATCH 500
21
+ #define NLOOPS 1000
22
+
23
+ #define RING_SIZE 512
24
+
25
+ struct p {
26
+ int fd[2];
27
+ int triggered;
28
+ };
29
+
30
+ static struct p p[NFILES];
31
+
32
+ static int has_poll_update(void)
33
+ {
34
+ struct io_uring ring;
35
+ struct io_uring_cqe *cqe;
36
+ struct io_uring_sqe *sqe;
37
+ bool has_update = false;
38
+ int ret;
39
+
40
+ ret = io_uring_queue_init(8, &ring, 0);
41
+ if (ret)
42
+ return -1;
43
+
44
+ sqe = io_uring_get_sqe(&ring);
45
+ io_uring_prep_poll_update(sqe, 0, 0, POLLIN, IORING_TIMEOUT_UPDATE);
46
+
47
+ ret = io_uring_submit(&ring);
48
+ if (ret != 1)
49
+ return -1;
50
+
51
+ ret = io_uring_wait_cqe(&ring, &cqe);
52
+ if (!ret) {
53
+ if (cqe->res == -ENOENT)
54
+ has_update = true;
55
+ else if (cqe->res != -EINVAL)
56
+ return -1;
57
+ io_uring_cqe_seen(&ring, cqe);
58
+ }
59
+ io_uring_queue_exit(&ring);
60
+ return has_update;
61
+ }
62
+
63
+ static int arm_poll(struct io_uring *ring, int off)
64
+ {
65
+ struct io_uring_sqe *sqe;
66
+
67
+ sqe = io_uring_get_sqe(ring);
68
+ if (!sqe) {
69
+ fprintf(stderr, "failed getting sqe\n");
70
+ return 1;
71
+ }
72
+
73
+ io_uring_prep_poll_multishot(sqe, p[off].fd[0], POLLIN);
74
+ sqe->user_data = off;
75
+ return 0;
76
+ }
77
+
78
+ static int reap_polls(struct io_uring *ring)
79
+ {
80
+ struct io_uring_cqe *cqe;
81
+ int i, ret, off;
82
+ char c;
83
+
84
+ for (i = 0; i < BATCH; i++) {
85
+ struct io_uring_sqe *sqe;
86
+
87
+ sqe = io_uring_get_sqe(ring);
88
+ /* update event */
89
+ io_uring_prep_poll_update(sqe, i, 0, POLLIN,
90
+ IORING_POLL_UPDATE_EVENTS);
91
+ sqe->user_data = 0x12345678;
92
+ }
93
+
94
+ ret = io_uring_submit(ring);
95
+ if (ret != BATCH) {
96
+ fprintf(stderr, "submitted %d, %d\n", ret, BATCH);
97
+ return 1;
98
+ }
99
+
100
+ for (i = 0; i < 2 * BATCH; i++) {
101
+ ret = io_uring_wait_cqe(ring, &cqe);
102
+ if (ret) {
103
+ fprintf(stderr, "wait cqe %d\n", ret);
104
+ return ret;
105
+ }
106
+ off = cqe->user_data;
107
+ if (off == 0x12345678)
108
+ goto seen;
109
+ ret = read(p[off].fd[0], &c, 1);
110
+ if (ret != 1) {
111
+ if (ret == -1 && errno == EAGAIN)
112
+ goto seen;
113
+ fprintf(stderr, "read got %d/%d\n", ret, errno);
114
+ break;
115
+ }
116
+ seen:
117
+ io_uring_cqe_seen(ring, cqe);
118
+ }
119
+
120
+ if (i != 2 * BATCH) {
121
+ fprintf(stderr, "gave up at %d\n", i);
122
+ return 1;
123
+ }
124
+
125
+ return 0;
126
+ }
127
+
128
+ static int trigger_polls(void)
129
+ {
130
+ char c = 89;
131
+ int i, ret;
132
+
133
+ for (i = 0; i < BATCH; i++) {
134
+ int off;
135
+
136
+ do {
137
+ off = rand() % NFILES;
138
+ if (!p[off].triggered)
139
+ break;
140
+ } while (1);
141
+
142
+ p[off].triggered = 1;
143
+ ret = write(p[off].fd[1], &c, 1);
144
+ if (ret != 1) {
145
+ fprintf(stderr, "write got %d/%d\n", ret, errno);
146
+ return 1;
147
+ }
148
+ }
149
+
150
+ return 0;
151
+ }
152
+
153
+ static void *trigger_polls_fn(void *data)
154
+ {
155
+ trigger_polls();
156
+ return NULL;
157
+ }
158
+
159
+ static int arm_polls(struct io_uring *ring)
160
+ {
161
+ int ret, to_arm = NFILES, i, off;
162
+
163
+ off = 0;
164
+ while (to_arm) {
165
+ int this_arm;
166
+
167
+ this_arm = to_arm;
168
+ if (this_arm > RING_SIZE)
169
+ this_arm = RING_SIZE;
170
+
171
+ for (i = 0; i < this_arm; i++) {
172
+ if (arm_poll(ring, off)) {
173
+ fprintf(stderr, "arm failed at %d\n", off);
174
+ return 1;
175
+ }
176
+ off++;
177
+ }
178
+
179
+ ret = io_uring_submit(ring);
180
+ if (ret != this_arm) {
181
+ fprintf(stderr, "submitted %d, %d\n", ret, this_arm);
182
+ return 1;
183
+ }
184
+ to_arm -= this_arm;
185
+ }
186
+
187
+ return 0;
188
+ }
189
+
190
+ int main(int argc, char *argv[])
191
+ {
192
+ struct io_uring ring;
193
+ struct io_uring_params params = { };
194
+ struct rlimit rlim;
195
+ pthread_t thread;
196
+ int i, j, ret;
197
+
198
+ if (argc > 1)
199
+ return 0;
200
+
201
+ ret = has_poll_update();
202
+ if (ret < 0) {
203
+ fprintf(stderr, "poll update check failed %i\n", ret);
204
+ return -1;
205
+ } else if (!ret) {
206
+ fprintf(stderr, "no poll update, skip\n");
207
+ return 0;
208
+ }
209
+
210
+ if (getrlimit(RLIMIT_NOFILE, &rlim) < 0) {
211
+ perror("getrlimit");
212
+ goto err_noring;
213
+ }
214
+
215
+ if (rlim.rlim_cur < (2 * NFILES + 5)) {
216
+ rlim.rlim_cur = (2 * NFILES + 5);
217
+ rlim.rlim_max = rlim.rlim_cur;
218
+ if (setrlimit(RLIMIT_NOFILE, &rlim) < 0) {
219
+ if (errno == EPERM)
220
+ goto err_nofail;
221
+ perror("setrlimit");
222
+ goto err_noring;
223
+ }
224
+ }
225
+
226
+ for (i = 0; i < NFILES; i++) {
227
+ if (pipe(p[i].fd) < 0) {
228
+ perror("pipe");
229
+ goto err_noring;
230
+ }
231
+ fcntl(p[i].fd[0], F_SETFL, O_NONBLOCK);
232
+ }
233
+
234
+ params.flags = IORING_SETUP_CQSIZE;
235
+ params.cq_entries = 4096;
236
+ ret = io_uring_queue_init_params(RING_SIZE, &ring, &params);
237
+ if (ret) {
238
+ if (ret == -EINVAL) {
239
+ fprintf(stdout, "No CQSIZE, trying without\n");
240
+ ret = io_uring_queue_init(RING_SIZE, &ring, 0);
241
+ if (ret) {
242
+ fprintf(stderr, "ring setup failed: %d\n", ret);
243
+ return 1;
244
+ }
245
+ }
246
+ }
247
+
248
+ if (arm_polls(&ring))
249
+ goto err;
250
+
251
+ for (i = 0; i < NLOOPS; i++) {
252
+ pthread_create(&thread, NULL, trigger_polls_fn, NULL);
253
+ ret = reap_polls(&ring);
254
+ if (ret)
255
+ goto err;
256
+ pthread_join(thread, NULL);
257
+
258
+ for (j = 0; j < NFILES; j++)
259
+ p[j].triggered = 0;
260
+ }
261
+
262
+ io_uring_queue_exit(&ring);
263
+ return 0;
264
+ err:
265
+ io_uring_queue_exit(&ring);
266
+ err_noring:
267
+ fprintf(stderr, "poll-many failed\n");
268
+ return 1;
269
+ err_nofail:
270
+ fprintf(stderr, "poll-many: not enough files available (and not root), "
271
+ "skipped\n");
272
+ return 0;
273
+ }
@@ -0,0 +1,48 @@
1
+ /* SPDX-License-Identifier: MIT */
2
+ /*
3
+ * Description: Test poll against ring itself. A buggy kernel will end up
4
+ * having io_wq_* workers pending, as the circular reference
5
+ * will prevent full exit.
6
+ *
7
+ */
8
+ #include <errno.h>
9
+ #include <stdio.h>
10
+ #include <unistd.h>
11
+ #include <stdlib.h>
12
+ #include <string.h>
13
+ #include <poll.h>
14
+
15
+ #include "liburing.h"
16
+
17
+ int main(int argc, char *argv[])
18
+ {
19
+ struct io_uring_sqe *sqe;
20
+ struct io_uring ring;
21
+ int ret;
22
+
23
+ if (argc > 1)
24
+ return 0;
25
+
26
+ ret = io_uring_queue_init(1, &ring, 0);
27
+ if (ret) {
28
+ fprintf(stderr, "child: ring setup failed: %d\n", ret);
29
+ return 1;
30
+ }
31
+
32
+ sqe = io_uring_get_sqe(&ring);
33
+ if (!sqe) {
34
+ fprintf(stderr, "get sqe failed\n");
35
+ return 1;
36
+ }
37
+
38
+ io_uring_prep_poll_add(sqe, ring.ring_fd, POLLIN);
39
+ io_uring_sqe_set_data(sqe, sqe);
40
+
41
+ ret = io_uring_submit(&ring);
42
+ if (ret <= 0) {
43
+ fprintf(stderr, "child: sqe submit failed: %d\n", ret);
44
+ return 1;
45
+ }
46
+
47
+ return 0;
48
+ }
@@ -0,0 +1,353 @@
1
+ /* SPDX-License-Identifier: MIT */
2
+ /*
3
+ * Description: test io_uring poll handling
4
+ *
5
+ */
6
+ #include <errno.h>
7
+ #include <stdio.h>
8
+ #include <unistd.h>
9
+ #include <stdlib.h>
10
+ #include <string.h>
11
+ #include <signal.h>
12
+ #include <fcntl.h>
13
+ #include <poll.h>
14
+ #include <sys/wait.h>
15
+ #include <sys/select.h>
16
+ #include <pthread.h>
17
+ #include <sys/epoll.h>
18
+
19
+ #include "liburing.h"
20
+
21
+ struct thread_data {
22
+ struct io_uring *ring;
23
+ int fd;
24
+ int events;
25
+ const char *test;
26
+ int out[2];
27
+ };
28
+
29
+ static void *epoll_wait_fn(void *data)
30
+ {
31
+ struct thread_data *td = data;
32
+ struct epoll_event ev;
33
+
34
+ if (epoll_wait(td->fd, &ev, 1, -1) < 0) {
35
+ perror("epoll_wait");
36
+ goto err;
37
+ }
38
+
39
+ return NULL;
40
+ err:
41
+ return (void *) 1;
42
+ }
43
+
44
+ static void *iou_poll(void *data)
45
+ {
46
+ struct thread_data *td = data;
47
+ struct io_uring_sqe *sqe;
48
+ struct io_uring_cqe *cqe;
49
+ int ret;
50
+
51
+ sqe = io_uring_get_sqe(td->ring);
52
+ io_uring_prep_poll_add(sqe, td->fd, td->events);
53
+
54
+ ret = io_uring_submit(td->ring);
55
+ if (ret != 1) {
56
+ fprintf(stderr, "submit got %d\n", ret);
57
+ goto err;
58
+ }
59
+
60
+ ret = io_uring_wait_cqe(td->ring, &cqe);
61
+ if (ret) {
62
+ fprintf(stderr, "wait_cqe: %d\n", ret);
63
+ goto err;
64
+ }
65
+
66
+ td->out[0] = cqe->res & 0x3f;
67
+ io_uring_cqe_seen(td->ring, cqe);
68
+ return NULL;
69
+ err:
70
+ return (void *) 1;
71
+ }
72
+
73
+ static void *poll_pipe(void *data)
74
+ {
75
+ struct thread_data *td = data;
76
+ struct pollfd pfd;
77
+ int ret;
78
+
79
+ pfd.fd = td->fd;
80
+ pfd.events = td->events;
81
+
82
+ ret = poll(&pfd, 1, -1);
83
+ if (ret < 0)
84
+ perror("poll");
85
+
86
+ td->out[1] = pfd.revents;
87
+ return NULL;
88
+ }
89
+
90
+ static int do_pipe_pollin_test(struct io_uring *ring)
91
+ {
92
+ struct thread_data td;
93
+ pthread_t threads[2];
94
+ int ret, pipe1[2];
95
+ char buf;
96
+
97
+ if (pipe(pipe1) < 0) {
98
+ perror("pipe");
99
+ return 1;
100
+ }
101
+
102
+ td.ring = ring;
103
+ td.fd = pipe1[0];
104
+ td.events = POLLIN;
105
+ td.test = __FUNCTION__;
106
+
107
+ pthread_create(&threads[1], NULL, iou_poll, &td);
108
+ pthread_create(&threads[0], NULL, poll_pipe, &td);
109
+ usleep(100000);
110
+
111
+ buf = 0x89;
112
+ ret = write(pipe1[1], &buf, sizeof(buf));
113
+ if (ret != sizeof(buf)) {
114
+ fprintf(stderr, "write failed: %d\n", ret);
115
+ return 1;
116
+ }
117
+
118
+ pthread_join(threads[0], NULL);
119
+ pthread_join(threads[1], NULL);
120
+
121
+ if (td.out[0] != td.out[1]) {
122
+ fprintf(stderr, "%s: res %x/%x differ\n", __FUNCTION__,
123
+ td.out[0], td.out[1]);
124
+ return 1;
125
+ }
126
+ return 0;
127
+ }
128
+
129
+ static int do_pipe_pollout_test(struct io_uring *ring)
130
+ {
131
+ struct thread_data td;
132
+ pthread_t threads[2];
133
+ int ret, pipe1[2];
134
+ char buf;
135
+
136
+ if (pipe(pipe1) < 0) {
137
+ perror("pipe");
138
+ return 1;
139
+ }
140
+
141
+ td.ring = ring;
142
+ td.fd = pipe1[1];
143
+ td.events = POLLOUT;
144
+ td.test = __FUNCTION__;
145
+
146
+ pthread_create(&threads[0], NULL, poll_pipe, &td);
147
+ pthread_create(&threads[1], NULL, iou_poll, &td);
148
+ usleep(100000);
149
+
150
+ buf = 0x89;
151
+ ret = write(pipe1[1], &buf, sizeof(buf));
152
+ if (ret != sizeof(buf)) {
153
+ fprintf(stderr, "write failed: %d\n", ret);
154
+ return 1;
155
+ }
156
+
157
+ pthread_join(threads[0], NULL);
158
+ pthread_join(threads[1], NULL);
159
+
160
+ if (td.out[0] != td.out[1]) {
161
+ fprintf(stderr, "%s: res %x/%x differ\n", __FUNCTION__,
162
+ td.out[0], td.out[1]);
163
+ return 1;
164
+ }
165
+
166
+ return 0;
167
+ }
168
+
169
+ static int do_fd_test(struct io_uring *ring, const char *fname, int events)
170
+ {
171
+ struct thread_data td;
172
+ pthread_t threads[2];
173
+ int fd;
174
+
175
+ fd = open(fname, O_RDONLY);
176
+ if (fd < 0) {
177
+ perror("open");
178
+ return 1;
179
+ }
180
+
181
+ td.ring = ring;
182
+ td.fd = fd;
183
+ td.events = events;
184
+ td.test = __FUNCTION__;
185
+
186
+ pthread_create(&threads[0], NULL, poll_pipe, &td);
187
+ pthread_create(&threads[1], NULL, iou_poll, &td);
188
+
189
+ pthread_join(threads[0], NULL);
190
+ pthread_join(threads[1], NULL);
191
+
192
+ if (td.out[0] != td.out[1]) {
193
+ fprintf(stderr, "%s: res %x/%x differ\n", __FUNCTION__,
194
+ td.out[0], td.out[1]);
195
+ return 1;
196
+ }
197
+
198
+ return 0;
199
+ }
200
+
201
+ static int iou_epoll_ctl(struct io_uring *ring, int epfd, int fd,
202
+ struct epoll_event *ev)
203
+ {
204
+ struct io_uring_sqe *sqe;
205
+ struct io_uring_cqe *cqe;
206
+ int ret;
207
+
208
+ sqe = io_uring_get_sqe(ring);
209
+ if (!sqe) {
210
+ fprintf(stderr, "Failed to get sqe\n");
211
+ return 1;
212
+ }
213
+
214
+ io_uring_prep_epoll_ctl(sqe, epfd, fd, EPOLL_CTL_ADD, ev);
215
+
216
+ ret = io_uring_submit(ring);
217
+ if (ret != 1) {
218
+ fprintf(stderr, "submit: %d\n", ret);
219
+ return 1;
220
+ }
221
+
222
+ ret = io_uring_wait_cqe(ring, &cqe);
223
+ if (ret) {
224
+ fprintf(stderr, "wait_cqe: %d\n", ret);
225
+ return 1;
226
+ }
227
+
228
+ ret = cqe->res;
229
+ io_uring_cqe_seen(ring, cqe);
230
+ return ret;
231
+ }
232
+
233
+ static int do_test_epoll(struct io_uring *ring, int iou_epoll_add)
234
+ {
235
+ struct epoll_event ev;
236
+ struct thread_data td;
237
+ pthread_t threads[2];
238
+ int ret, pipe1[2];
239
+ char buf;
240
+ int fd;
241
+
242
+ fd = epoll_create1(0);
243
+ if (fd < 0) {
244
+ perror("epoll_create");
245
+ return 1;
246
+ }
247
+
248
+ if (pipe(pipe1) < 0) {
249
+ perror("pipe");
250
+ return 1;
251
+ }
252
+
253
+ ev.events = EPOLLIN;
254
+ ev.data.fd = pipe1[0];
255
+
256
+ if (!iou_epoll_add) {
257
+ if (epoll_ctl(fd, EPOLL_CTL_ADD, pipe1[0], &ev) < 0) {
258
+ perror("epoll_ctrl");
259
+ return 1;
260
+ }
261
+ } else {
262
+ ret = iou_epoll_ctl(ring, fd, pipe1[0], &ev);
263
+ if (ret == -EINVAL) {
264
+ fprintf(stdout, "epoll not supported, skipping\n");
265
+ return 0;
266
+ } else if (ret < 0) {
267
+ return 1;
268
+ }
269
+ }
270
+
271
+ td.ring = ring;
272
+ td.fd = fd;
273
+ td.events = POLLIN;
274
+ td.test = __FUNCTION__;
275
+
276
+ pthread_create(&threads[0], NULL, iou_poll, &td);
277
+ pthread_create(&threads[1], NULL, epoll_wait_fn, &td);
278
+ usleep(100000);
279
+
280
+ buf = 0x89;
281
+ ret = write(pipe1[1], &buf, sizeof(buf));
282
+ if (ret != sizeof(buf)) {
283
+ fprintf(stderr, "write failed: %d\n", ret);
284
+ return 1;
285
+ }
286
+
287
+ pthread_join(threads[0], NULL);
288
+ pthread_join(threads[1], NULL);
289
+ return 0;
290
+ }
291
+
292
+ int main(int argc, char *argv[])
293
+ {
294
+ struct io_uring ring;
295
+ const char *fname;
296
+ int ret;
297
+
298
+ ret = io_uring_queue_init(1, &ring, 0);
299
+ if (ret) {
300
+ fprintf(stderr, "ring setup failed\n");
301
+ return 1;
302
+ }
303
+
304
+ ret = do_pipe_pollin_test(&ring);
305
+ if (ret) {
306
+ fprintf(stderr, "pipe pollin test failed\n");
307
+ return ret;
308
+ }
309
+
310
+ ret = do_pipe_pollout_test(&ring);
311
+ if (ret) {
312
+ fprintf(stderr, "pipe pollout test failed\n");
313
+ return ret;
314
+ }
315
+
316
+ ret = do_test_epoll(&ring, 0);
317
+ if (ret) {
318
+ fprintf(stderr, "epoll test 0 failed\n");
319
+ return ret;
320
+ }
321
+
322
+ ret = do_test_epoll(&ring, 1);
323
+ if (ret) {
324
+ fprintf(stderr, "epoll test 1 failed\n");
325
+ return ret;
326
+ }
327
+
328
+ if (argc > 1)
329
+ fname = argv[1];
330
+ else
331
+ fname = argv[0];
332
+
333
+ ret = do_fd_test(&ring, fname, POLLIN);
334
+ if (ret) {
335
+ fprintf(stderr, "fd test IN failed\n");
336
+ return ret;
337
+ }
338
+
339
+ ret = do_fd_test(&ring, fname, POLLOUT);
340
+ if (ret) {
341
+ fprintf(stderr, "fd test OUT failed\n");
342
+ return ret;
343
+ }
344
+
345
+ ret = do_fd_test(&ring, fname, POLLOUT | POLLIN);
346
+ if (ret) {
347
+ fprintf(stderr, "fd test IN|OUT failed\n");
348
+ return ret;
349
+ }
350
+
351
+ return 0;
352
+
353
+ }