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.
- checksums.yaml +7 -0
- data/.github/dependabot.yml +12 -0
- data/.github/workflows/test.yml +35 -0
- data/.gitignore +59 -0
- data/.gitmodules +3 -0
- data/CHANGELOG.md +7 -0
- data/Gemfile +3 -0
- data/LICENSE +21 -0
- data/README.md +11 -0
- data/Rakefile +39 -0
- data/TODO.md +0 -0
- data/examples/echo_server.rb +52 -0
- data/examples/event_loop.rb +69 -0
- data/examples/fibers.rb +105 -0
- data/examples/http_server.rb +56 -0
- data/examples/http_server_multishot.rb +57 -0
- data/examples/http_server_simpler.rb +34 -0
- data/ext/um/extconf.rb +71 -0
- data/ext/um/iou.h +101 -0
- data/ext/um/op_ctx.c +138 -0
- data/ext/um/ring.c +755 -0
- data/ext/um/um.c +267 -0
- data/ext/um/um.h +97 -0
- data/ext/um/um_class.c +175 -0
- data/ext/um/um_ext.c +11 -0
- data/ext/um/um_op.c +87 -0
- data/ext/um/um_utils.c +23 -0
- data/lib/uringmachine/version.rb +3 -0
- data/lib/uringmachine.rb +8 -0
- data/test/helper.rb +70 -0
- data/test/test_iou.rb +876 -0
- data/test/test_um.rb +168 -0
- data/uringmachine.gemspec +27 -0
- data/vendor/liburing/.github/actions/codespell/stopwords +7 -0
- data/vendor/liburing/.github/pull_request_template.md +86 -0
- data/vendor/liburing/.github/workflows/build.yml +137 -0
- data/vendor/liburing/.github/workflows/codespell.yml +25 -0
- data/vendor/liburing/.github/workflows/shellcheck.yml +20 -0
- data/vendor/liburing/.gitignore +41 -0
- data/vendor/liburing/CHANGELOG +111 -0
- data/vendor/liburing/CITATION.cff +11 -0
- data/vendor/liburing/COPYING +502 -0
- data/vendor/liburing/COPYING.GPL +339 -0
- data/vendor/liburing/LICENSE +20 -0
- data/vendor/liburing/Makefile +96 -0
- data/vendor/liburing/Makefile.common +7 -0
- data/vendor/liburing/Makefile.quiet +11 -0
- data/vendor/liburing/README +106 -0
- data/vendor/liburing/SECURITY.md +6 -0
- data/vendor/liburing/configure +624 -0
- data/vendor/liburing/debian/README.Debian +7 -0
- data/vendor/liburing/debian/changelog +38 -0
- data/vendor/liburing/debian/control +39 -0
- data/vendor/liburing/debian/copyright +49 -0
- data/vendor/liburing/debian/liburing-dev.install +4 -0
- data/vendor/liburing/debian/liburing-dev.manpages +5 -0
- data/vendor/liburing/debian/liburing2.install +1 -0
- data/vendor/liburing/debian/liburing2.symbols +56 -0
- data/vendor/liburing/debian/patches/series +1 -0
- data/vendor/liburing/debian/rules +29 -0
- data/vendor/liburing/debian/source/format +1 -0
- data/vendor/liburing/debian/source/local-options +2 -0
- data/vendor/liburing/debian/source/options +1 -0
- data/vendor/liburing/debian/watch +3 -0
- data/vendor/liburing/examples/Makefile +53 -0
- data/vendor/liburing/examples/helpers.c +62 -0
- data/vendor/liburing/examples/helpers.h +7 -0
- data/vendor/liburing/examples/io_uring-close-test.c +123 -0
- data/vendor/liburing/examples/io_uring-cp.c +282 -0
- data/vendor/liburing/examples/io_uring-test.c +112 -0
- data/vendor/liburing/examples/io_uring-udp.c +403 -0
- data/vendor/liburing/examples/link-cp.c +193 -0
- data/vendor/liburing/examples/napi-busy-poll-client.c +509 -0
- data/vendor/liburing/examples/napi-busy-poll-server.c +450 -0
- data/vendor/liburing/examples/poll-bench.c +101 -0
- data/vendor/liburing/examples/proxy.c +2461 -0
- data/vendor/liburing/examples/proxy.h +102 -0
- data/vendor/liburing/examples/rsrc-update-bench.c +100 -0
- data/vendor/liburing/examples/send-zerocopy.c +658 -0
- data/vendor/liburing/examples/ucontext-cp.c +258 -0
- data/vendor/liburing/liburing-ffi.pc.in +12 -0
- data/vendor/liburing/liburing.pc.in +12 -0
- data/vendor/liburing/liburing.spec +66 -0
- data/vendor/liburing/make-debs.sh +55 -0
- data/vendor/liburing/src/Makefile +129 -0
- data/vendor/liburing/src/arch/aarch64/lib.h +47 -0
- data/vendor/liburing/src/arch/aarch64/syscall.h +91 -0
- data/vendor/liburing/src/arch/generic/lib.h +17 -0
- data/vendor/liburing/src/arch/generic/syscall.h +100 -0
- data/vendor/liburing/src/arch/riscv64/lib.h +48 -0
- data/vendor/liburing/src/arch/riscv64/syscall.h +100 -0
- data/vendor/liburing/src/arch/syscall-defs.h +94 -0
- data/vendor/liburing/src/arch/x86/lib.h +11 -0
- data/vendor/liburing/src/arch/x86/syscall.h +296 -0
- data/vendor/liburing/src/ffi.c +15 -0
- data/vendor/liburing/src/include/liburing/barrier.h +81 -0
- data/vendor/liburing/src/include/liburing/io_uring.h +818 -0
- data/vendor/liburing/src/include/liburing.h +1602 -0
- data/vendor/liburing/src/int_flags.h +11 -0
- data/vendor/liburing/src/lib.h +52 -0
- data/vendor/liburing/src/liburing-ffi.map +211 -0
- data/vendor/liburing/src/liburing.map +104 -0
- data/vendor/liburing/src/nolibc.c +55 -0
- data/vendor/liburing/src/queue.c +468 -0
- data/vendor/liburing/src/register.c +374 -0
- data/vendor/liburing/src/setup.c +689 -0
- data/vendor/liburing/src/setup.h +9 -0
- data/vendor/liburing/src/syscall.c +29 -0
- data/vendor/liburing/src/syscall.h +53 -0
- data/vendor/liburing/src/version.c +21 -0
- data/vendor/liburing/test/232c93d07b74.c +305 -0
- data/vendor/liburing/test/35fa71a030ca.c +329 -0
- data/vendor/liburing/test/500f9fbadef8.c +91 -0
- data/vendor/liburing/test/7ad0e4b2f83c.c +94 -0
- data/vendor/liburing/test/8a9973408177.c +107 -0
- data/vendor/liburing/test/917257daa0fe.c +54 -0
- data/vendor/liburing/test/Makefile +297 -0
- data/vendor/liburing/test/a0908ae19763.c +59 -0
- data/vendor/liburing/test/a4c0b3decb33.c +181 -0
- data/vendor/liburing/test/accept-link.c +255 -0
- data/vendor/liburing/test/accept-non-empty.c +256 -0
- data/vendor/liburing/test/accept-reuse.c +163 -0
- data/vendor/liburing/test/accept-test.c +83 -0
- data/vendor/liburing/test/accept.c +919 -0
- data/vendor/liburing/test/across-fork.c +284 -0
- data/vendor/liburing/test/b19062a56726.c +54 -0
- data/vendor/liburing/test/b5837bd5311d.c +78 -0
- data/vendor/liburing/test/bind-listen.c +408 -0
- data/vendor/liburing/test/buf-ring-nommap.c +123 -0
- data/vendor/liburing/test/buf-ring-put.c +83 -0
- data/vendor/liburing/test/buf-ring.c +473 -0
- data/vendor/liburing/test/ce593a6c480a.c +139 -0
- data/vendor/liburing/test/close-opath.c +123 -0
- data/vendor/liburing/test/config +14 -0
- data/vendor/liburing/test/connect-rep.c +204 -0
- data/vendor/liburing/test/connect.c +442 -0
- data/vendor/liburing/test/coredump.c +60 -0
- data/vendor/liburing/test/cq-full.c +97 -0
- data/vendor/liburing/test/cq-overflow.c +530 -0
- data/vendor/liburing/test/cq-peek-batch.c +103 -0
- data/vendor/liburing/test/cq-ready.c +95 -0
- data/vendor/liburing/test/cq-size.c +65 -0
- data/vendor/liburing/test/d4ae271dfaae.c +96 -0
- data/vendor/liburing/test/d77a67ed5f27.c +65 -0
- data/vendor/liburing/test/defer-taskrun.c +391 -0
- data/vendor/liburing/test/defer-tw-timeout.c +173 -0
- data/vendor/liburing/test/defer.c +319 -0
- data/vendor/liburing/test/double-poll-crash.c +195 -0
- data/vendor/liburing/test/drop-submit.c +94 -0
- data/vendor/liburing/test/eeed8b54e0df.c +120 -0
- data/vendor/liburing/test/empty-eownerdead.c +45 -0
- data/vendor/liburing/test/eploop.c +74 -0
- data/vendor/liburing/test/eventfd-disable.c +179 -0
- data/vendor/liburing/test/eventfd-reg.c +77 -0
- data/vendor/liburing/test/eventfd-ring.c +98 -0
- data/vendor/liburing/test/eventfd.c +113 -0
- data/vendor/liburing/test/evloop.c +73 -0
- data/vendor/liburing/test/exec-target.c +6 -0
- data/vendor/liburing/test/exit-no-cleanup.c +117 -0
- data/vendor/liburing/test/fadvise.c +202 -0
- data/vendor/liburing/test/fallocate.c +265 -0
- data/vendor/liburing/test/fc2a85cb02ef.c +132 -0
- data/vendor/liburing/test/fd-install.c +500 -0
- data/vendor/liburing/test/fd-pass.c +237 -0
- data/vendor/liburing/test/fdinfo.c +419 -0
- data/vendor/liburing/test/file-register.c +1189 -0
- data/vendor/liburing/test/file-update.c +231 -0
- data/vendor/liburing/test/file-verify.c +654 -0
- data/vendor/liburing/test/files-exit-hang-poll.c +114 -0
- data/vendor/liburing/test/files-exit-hang-timeout.c +137 -0
- data/vendor/liburing/test/fixed-buf-iter.c +115 -0
- data/vendor/liburing/test/fixed-buf-merge.c +101 -0
- data/vendor/liburing/test/fixed-hugepage.c +411 -0
- data/vendor/liburing/test/fixed-link.c +90 -0
- data/vendor/liburing/test/fixed-reuse.c +160 -0
- data/vendor/liburing/test/fpos.c +255 -0
- data/vendor/liburing/test/fsnotify.c +118 -0
- data/vendor/liburing/test/fsync.c +224 -0
- data/vendor/liburing/test/futex.c +571 -0
- data/vendor/liburing/test/hardlink.c +170 -0
- data/vendor/liburing/test/helpers.c +318 -0
- data/vendor/liburing/test/helpers.h +108 -0
- data/vendor/liburing/test/ignore-single-mmap.c +48 -0
- data/vendor/liburing/test/init-mem.c +164 -0
- data/vendor/liburing/test/io-cancel.c +561 -0
- data/vendor/liburing/test/io_uring_enter.c +264 -0
- data/vendor/liburing/test/io_uring_passthrough.c +482 -0
- data/vendor/liburing/test/io_uring_register.c +503 -0
- data/vendor/liburing/test/io_uring_setup.c +110 -0
- data/vendor/liburing/test/iopoll-leak.c +85 -0
- data/vendor/liburing/test/iopoll-overflow.c +118 -0
- data/vendor/liburing/test/iopoll.c +465 -0
- data/vendor/liburing/test/lfs-openat-write.c +119 -0
- data/vendor/liburing/test/lfs-openat.c +273 -0
- data/vendor/liburing/test/link-timeout.c +1108 -0
- data/vendor/liburing/test/link.c +497 -0
- data/vendor/liburing/test/link_drain.c +255 -0
- data/vendor/liburing/test/madvise.c +195 -0
- data/vendor/liburing/test/min-timeout-wait.c +354 -0
- data/vendor/liburing/test/min-timeout.c +233 -0
- data/vendor/liburing/test/mkdir.c +112 -0
- data/vendor/liburing/test/msg-ring-fd.c +331 -0
- data/vendor/liburing/test/msg-ring-flags.c +212 -0
- data/vendor/liburing/test/msg-ring-overflow.c +159 -0
- data/vendor/liburing/test/msg-ring.c +467 -0
- data/vendor/liburing/test/multicqes_drain.c +429 -0
- data/vendor/liburing/test/napi-test.c +215 -0
- data/vendor/liburing/test/napi-test.sh +48 -0
- data/vendor/liburing/test/no-mmap-inval.c +42 -0
- data/vendor/liburing/test/nolibc.c +62 -0
- data/vendor/liburing/test/nop-all-sizes.c +99 -0
- data/vendor/liburing/test/nop.c +177 -0
- data/vendor/liburing/test/nvme.h +169 -0
- data/vendor/liburing/test/ooo-file-unreg.c +82 -0
- data/vendor/liburing/test/open-close.c +261 -0
- data/vendor/liburing/test/open-direct-link.c +188 -0
- data/vendor/liburing/test/open-direct-pick.c +180 -0
- data/vendor/liburing/test/openat2.c +312 -0
- data/vendor/liburing/test/personality.c +204 -0
- data/vendor/liburing/test/pipe-bug.c +95 -0
- data/vendor/liburing/test/pipe-eof.c +83 -0
- data/vendor/liburing/test/pipe-reuse.c +105 -0
- data/vendor/liburing/test/poll-cancel-all.c +496 -0
- data/vendor/liburing/test/poll-cancel-ton.c +135 -0
- data/vendor/liburing/test/poll-cancel.c +228 -0
- data/vendor/liburing/test/poll-link.c +221 -0
- data/vendor/liburing/test/poll-many.c +230 -0
- data/vendor/liburing/test/poll-mshot-overflow.c +265 -0
- data/vendor/liburing/test/poll-mshot-update.c +323 -0
- data/vendor/liburing/test/poll-race-mshot.c +276 -0
- data/vendor/liburing/test/poll-race.c +105 -0
- data/vendor/liburing/test/poll-ring.c +48 -0
- data/vendor/liburing/test/poll-v-poll.c +353 -0
- data/vendor/liburing/test/poll.c +327 -0
- data/vendor/liburing/test/probe.c +135 -0
- data/vendor/liburing/test/read-before-exit.c +129 -0
- data/vendor/liburing/test/read-mshot-empty.c +153 -0
- data/vendor/liburing/test/read-mshot.c +404 -0
- data/vendor/liburing/test/read-write.c +1013 -0
- data/vendor/liburing/test/recv-msgall-stream.c +398 -0
- data/vendor/liburing/test/recv-msgall.c +263 -0
- data/vendor/liburing/test/recv-multishot.c +602 -0
- data/vendor/liburing/test/recvsend_bundle.c +691 -0
- data/vendor/liburing/test/reg-fd-only.c +131 -0
- data/vendor/liburing/test/reg-hint.c +56 -0
- data/vendor/liburing/test/reg-reg-ring.c +90 -0
- data/vendor/liburing/test/regbuf-merge.c +91 -0
- data/vendor/liburing/test/register-restrictions.c +633 -0
- data/vendor/liburing/test/rename.c +132 -0
- data/vendor/liburing/test/ring-leak.c +283 -0
- data/vendor/liburing/test/ring-leak2.c +249 -0
- data/vendor/liburing/test/ringbuf-read.c +196 -0
- data/vendor/liburing/test/ringbuf-status.c +242 -0
- data/vendor/liburing/test/rsrc_tags.c +461 -0
- data/vendor/liburing/test/runtests-loop.sh +16 -0
- data/vendor/liburing/test/runtests-quiet.sh +11 -0
- data/vendor/liburing/test/runtests.sh +168 -0
- data/vendor/liburing/test/rw_merge_test.c +98 -0
- data/vendor/liburing/test/self.c +91 -0
- data/vendor/liburing/test/send-zerocopy.c +971 -0
- data/vendor/liburing/test/send_recv.c +412 -0
- data/vendor/liburing/test/send_recvmsg.c +444 -0
- data/vendor/liburing/test/shared-wq.c +84 -0
- data/vendor/liburing/test/short-read.c +75 -0
- data/vendor/liburing/test/shutdown.c +165 -0
- data/vendor/liburing/test/sigfd-deadlock.c +88 -0
- data/vendor/liburing/test/single-issuer.c +169 -0
- data/vendor/liburing/test/skip-cqe.c +428 -0
- data/vendor/liburing/test/socket-getsetsock-cmd.c +346 -0
- data/vendor/liburing/test/socket-io-cmd.c +237 -0
- data/vendor/liburing/test/socket-rw-eagain.c +149 -0
- data/vendor/liburing/test/socket-rw-offset.c +149 -0
- data/vendor/liburing/test/socket-rw.c +137 -0
- data/vendor/liburing/test/socket.c +408 -0
- data/vendor/liburing/test/splice.c +512 -0
- data/vendor/liburing/test/sq-full-cpp.cc +45 -0
- data/vendor/liburing/test/sq-full.c +45 -0
- data/vendor/liburing/test/sq-poll-dup.c +211 -0
- data/vendor/liburing/test/sq-poll-kthread.c +169 -0
- data/vendor/liburing/test/sq-poll-share.c +138 -0
- data/vendor/liburing/test/sq-space_left.c +159 -0
- data/vendor/liburing/test/sqpoll-disable-exit.c +196 -0
- data/vendor/liburing/test/sqpoll-exec.c +132 -0
- data/vendor/liburing/test/sqpoll-exit-hang.c +78 -0
- data/vendor/liburing/test/sqpoll-sleep.c +69 -0
- data/vendor/liburing/test/statx.c +172 -0
- data/vendor/liburing/test/stdout.c +232 -0
- data/vendor/liburing/test/submit-and-wait.c +108 -0
- data/vendor/liburing/test/submit-link-fail.c +156 -0
- data/vendor/liburing/test/submit-reuse.c +237 -0
- data/vendor/liburing/test/symlink.c +117 -0
- data/vendor/liburing/test/sync-cancel.c +235 -0
- data/vendor/liburing/test/teardowns.c +58 -0
- data/vendor/liburing/test/test.h +36 -0
- data/vendor/liburing/test/thread-exit.c +143 -0
- data/vendor/liburing/test/timeout-new.c +256 -0
- data/vendor/liburing/test/timeout.c +1798 -0
- data/vendor/liburing/test/truncate.c +186 -0
- data/vendor/liburing/test/tty-write-dpoll.c +60 -0
- data/vendor/liburing/test/unlink.c +112 -0
- data/vendor/liburing/test/version.c +25 -0
- data/vendor/liburing/test/wait-timeout.c +287 -0
- data/vendor/liburing/test/waitid.c +373 -0
- data/vendor/liburing/test/wakeup-hang.c +162 -0
- data/vendor/liburing/test/wq-aff.c +146 -0
- data/vendor/liburing/test/xattr.c +442 -0
- metadata +412 -0
|
@@ -0,0 +1,232 @@
|
|
|
1
|
+
/* SPDX-License-Identifier: MIT */
|
|
2
|
+
/*
|
|
3
|
+
* Description: check that STDOUT write works
|
|
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
|
+
|
|
12
|
+
#include "helpers.h"
|
|
13
|
+
#include "liburing.h"
|
|
14
|
+
|
|
15
|
+
static int test_pipe_io_fixed(struct io_uring *ring)
|
|
16
|
+
{
|
|
17
|
+
const char str[] = "This is a fixed pipe test\n";
|
|
18
|
+
struct io_uring_cqe *cqe;
|
|
19
|
+
struct io_uring_sqe *sqe;
|
|
20
|
+
struct iovec vecs[2];
|
|
21
|
+
char buffer[128];
|
|
22
|
+
int i, ret, fds[2];
|
|
23
|
+
|
|
24
|
+
t_posix_memalign(&vecs[0].iov_base, 4096, 4096);
|
|
25
|
+
memcpy(vecs[0].iov_base, str, strlen(str));
|
|
26
|
+
vecs[0].iov_len = strlen(str);
|
|
27
|
+
|
|
28
|
+
if (pipe(fds) < 0) {
|
|
29
|
+
perror("pipe");
|
|
30
|
+
return 1;
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
ret = io_uring_register_buffers(ring, vecs, 1);
|
|
34
|
+
if (ret) {
|
|
35
|
+
fprintf(stderr, "Failed to register buffers: %d\n", ret);
|
|
36
|
+
return 1;
|
|
37
|
+
}
|
|
38
|
+
|
|
39
|
+
sqe = io_uring_get_sqe(ring);
|
|
40
|
+
if (!sqe) {
|
|
41
|
+
fprintf(stderr, "get sqe failed\n");
|
|
42
|
+
goto err;
|
|
43
|
+
}
|
|
44
|
+
io_uring_prep_write_fixed(sqe, fds[1], vecs[0].iov_base,
|
|
45
|
+
vecs[0].iov_len, 0, 0);
|
|
46
|
+
sqe->user_data = 1;
|
|
47
|
+
|
|
48
|
+
sqe = io_uring_get_sqe(ring);
|
|
49
|
+
if (!sqe) {
|
|
50
|
+
fprintf(stderr, "get sqe failed\n");
|
|
51
|
+
goto err;
|
|
52
|
+
}
|
|
53
|
+
vecs[1].iov_base = buffer;
|
|
54
|
+
vecs[1].iov_len = sizeof(buffer);
|
|
55
|
+
io_uring_prep_readv(sqe, fds[0], &vecs[1], 1, 0);
|
|
56
|
+
sqe->user_data = 2;
|
|
57
|
+
|
|
58
|
+
ret = io_uring_submit(ring);
|
|
59
|
+
if (ret < 0) {
|
|
60
|
+
fprintf(stderr, "sqe submit failed: %d\n", ret);
|
|
61
|
+
goto err;
|
|
62
|
+
} else if (ret != 2) {
|
|
63
|
+
fprintf(stderr, "Submitted only %d\n", ret);
|
|
64
|
+
goto err;
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
for (i = 0; i < 2; i++) {
|
|
68
|
+
ret = io_uring_wait_cqe(ring, &cqe);
|
|
69
|
+
if (ret < 0) {
|
|
70
|
+
fprintf(stderr, "wait completion %d\n", ret);
|
|
71
|
+
goto err;
|
|
72
|
+
}
|
|
73
|
+
if (cqe->res < 0) {
|
|
74
|
+
fprintf(stderr, "I/O write error on %lu: %s\n",
|
|
75
|
+
(unsigned long) cqe->user_data,
|
|
76
|
+
strerror(-cqe->res));
|
|
77
|
+
goto err;
|
|
78
|
+
}
|
|
79
|
+
if (cqe->res != strlen(str)) {
|
|
80
|
+
fprintf(stderr, "Got %d bytes, wanted %d on %lu\n",
|
|
81
|
+
cqe->res, (int)strlen(str),
|
|
82
|
+
(unsigned long) cqe->user_data);
|
|
83
|
+
goto err;
|
|
84
|
+
}
|
|
85
|
+
if (cqe->user_data == 2 && memcmp(str, buffer, strlen(str))) {
|
|
86
|
+
fprintf(stderr, "read data mismatch\n");
|
|
87
|
+
goto err;
|
|
88
|
+
}
|
|
89
|
+
io_uring_cqe_seen(ring, cqe);
|
|
90
|
+
}
|
|
91
|
+
io_uring_unregister_buffers(ring);
|
|
92
|
+
return 0;
|
|
93
|
+
err:
|
|
94
|
+
return 1;
|
|
95
|
+
}
|
|
96
|
+
|
|
97
|
+
static int test_stdout_io_fixed(struct io_uring *ring)
|
|
98
|
+
{
|
|
99
|
+
const char str[] = "This is a fixed pipe test\n";
|
|
100
|
+
struct io_uring_cqe *cqe;
|
|
101
|
+
struct io_uring_sqe *sqe;
|
|
102
|
+
struct iovec vecs;
|
|
103
|
+
int ret;
|
|
104
|
+
|
|
105
|
+
t_posix_memalign(&vecs.iov_base, 4096, 4096);
|
|
106
|
+
memcpy(vecs.iov_base, str, strlen(str));
|
|
107
|
+
vecs.iov_len = strlen(str);
|
|
108
|
+
|
|
109
|
+
ret = io_uring_register_buffers(ring, &vecs, 1);
|
|
110
|
+
if (ret) {
|
|
111
|
+
fprintf(stderr, "Failed to register buffers: %d\n", ret);
|
|
112
|
+
return 1;
|
|
113
|
+
}
|
|
114
|
+
|
|
115
|
+
sqe = io_uring_get_sqe(ring);
|
|
116
|
+
if (!sqe) {
|
|
117
|
+
fprintf(stderr, "get sqe failed\n");
|
|
118
|
+
goto err;
|
|
119
|
+
}
|
|
120
|
+
io_uring_prep_write_fixed(sqe, STDOUT_FILENO, vecs.iov_base, vecs.iov_len, 0, 0);
|
|
121
|
+
|
|
122
|
+
ret = io_uring_submit(ring);
|
|
123
|
+
if (ret < 0) {
|
|
124
|
+
fprintf(stderr, "sqe submit failed: %d\n", ret);
|
|
125
|
+
goto err;
|
|
126
|
+
} else if (ret < 1) {
|
|
127
|
+
fprintf(stderr, "Submitted only %d\n", ret);
|
|
128
|
+
goto err;
|
|
129
|
+
}
|
|
130
|
+
|
|
131
|
+
ret = io_uring_wait_cqe(ring, &cqe);
|
|
132
|
+
if (ret < 0) {
|
|
133
|
+
fprintf(stderr, "wait completion %d\n", ret);
|
|
134
|
+
goto err;
|
|
135
|
+
}
|
|
136
|
+
if (cqe->res < 0) {
|
|
137
|
+
fprintf(stderr, "STDOUT write error: %s\n", strerror(-cqe->res));
|
|
138
|
+
goto err;
|
|
139
|
+
}
|
|
140
|
+
if (cqe->res != vecs.iov_len) {
|
|
141
|
+
fprintf(stderr, "Got %d write, wanted %d\n", cqe->res, (int)vecs.iov_len);
|
|
142
|
+
goto err;
|
|
143
|
+
}
|
|
144
|
+
io_uring_cqe_seen(ring, cqe);
|
|
145
|
+
io_uring_unregister_buffers(ring);
|
|
146
|
+
return 0;
|
|
147
|
+
err:
|
|
148
|
+
return 1;
|
|
149
|
+
}
|
|
150
|
+
|
|
151
|
+
static int test_stdout_io(struct io_uring *ring)
|
|
152
|
+
{
|
|
153
|
+
struct io_uring_cqe *cqe;
|
|
154
|
+
struct io_uring_sqe *sqe;
|
|
155
|
+
struct iovec vecs;
|
|
156
|
+
int ret;
|
|
157
|
+
|
|
158
|
+
vecs.iov_base = "This is a pipe test\n";
|
|
159
|
+
vecs.iov_len = strlen(vecs.iov_base);
|
|
160
|
+
|
|
161
|
+
sqe = io_uring_get_sqe(ring);
|
|
162
|
+
if (!sqe) {
|
|
163
|
+
fprintf(stderr, "get sqe failed\n");
|
|
164
|
+
goto err;
|
|
165
|
+
}
|
|
166
|
+
io_uring_prep_writev(sqe, STDOUT_FILENO, &vecs, 1, 0);
|
|
167
|
+
|
|
168
|
+
ret = io_uring_submit(ring);
|
|
169
|
+
if (ret < 0) {
|
|
170
|
+
fprintf(stderr, "sqe submit failed: %d\n", ret);
|
|
171
|
+
goto err;
|
|
172
|
+
} else if (ret < 1) {
|
|
173
|
+
fprintf(stderr, "Submitted only %d\n", ret);
|
|
174
|
+
goto err;
|
|
175
|
+
}
|
|
176
|
+
|
|
177
|
+
ret = io_uring_wait_cqe(ring, &cqe);
|
|
178
|
+
if (ret < 0) {
|
|
179
|
+
fprintf(stderr, "wait completion %d\n", ret);
|
|
180
|
+
goto err;
|
|
181
|
+
}
|
|
182
|
+
if (cqe->res < 0) {
|
|
183
|
+
fprintf(stderr, "STDOUT write error: %s\n",
|
|
184
|
+
strerror(-cqe->res));
|
|
185
|
+
goto err;
|
|
186
|
+
}
|
|
187
|
+
if (cqe->res != vecs.iov_len) {
|
|
188
|
+
fprintf(stderr, "Got %d write, wanted %d\n", cqe->res,
|
|
189
|
+
(int)vecs.iov_len);
|
|
190
|
+
goto err;
|
|
191
|
+
}
|
|
192
|
+
io_uring_cqe_seen(ring, cqe);
|
|
193
|
+
|
|
194
|
+
return 0;
|
|
195
|
+
err:
|
|
196
|
+
return 1;
|
|
197
|
+
}
|
|
198
|
+
|
|
199
|
+
int main(int argc, char *argv[])
|
|
200
|
+
{
|
|
201
|
+
struct io_uring ring;
|
|
202
|
+
int ret;
|
|
203
|
+
|
|
204
|
+
if (argc > 1)
|
|
205
|
+
return 0;
|
|
206
|
+
|
|
207
|
+
ret = io_uring_queue_init(8, &ring, 0);
|
|
208
|
+
if (ret) {
|
|
209
|
+
fprintf(stderr, "ring setup failed\n");
|
|
210
|
+
return 1;
|
|
211
|
+
}
|
|
212
|
+
|
|
213
|
+
ret = test_stdout_io(&ring);
|
|
214
|
+
if (ret) {
|
|
215
|
+
fprintf(stderr, "test_pipe_io failed\n");
|
|
216
|
+
return ret;
|
|
217
|
+
}
|
|
218
|
+
|
|
219
|
+
ret = test_stdout_io_fixed(&ring);
|
|
220
|
+
if (ret) {
|
|
221
|
+
fprintf(stderr, "test_pipe_io_fixed failed\n");
|
|
222
|
+
return ret;
|
|
223
|
+
}
|
|
224
|
+
|
|
225
|
+
ret = test_pipe_io_fixed(&ring);
|
|
226
|
+
if (ret) {
|
|
227
|
+
fprintf(stderr, "test_pipe_io_fixed failed\n");
|
|
228
|
+
return ret;
|
|
229
|
+
}
|
|
230
|
+
|
|
231
|
+
return 0;
|
|
232
|
+
}
|
|
@@ -0,0 +1,108 @@
|
|
|
1
|
+
/* SPDX-License-Identifier: MIT */
|
|
2
|
+
/*
|
|
3
|
+
* Description: Test that io_uring_submit_and_wait_timeout() returns the
|
|
4
|
+
* right value (submit count) and that it doesn't end up waiting twice.
|
|
5
|
+
*
|
|
6
|
+
*/
|
|
7
|
+
#include <errno.h>
|
|
8
|
+
#include <stdio.h>
|
|
9
|
+
#include <unistd.h>
|
|
10
|
+
#include <stdlib.h>
|
|
11
|
+
#include <string.h>
|
|
12
|
+
#include <fcntl.h>
|
|
13
|
+
#include <sys/time.h>
|
|
14
|
+
|
|
15
|
+
#include "liburing.h"
|
|
16
|
+
#include "test.h"
|
|
17
|
+
|
|
18
|
+
static unsigned long long mtime_since(const struct timeval *s,
|
|
19
|
+
const struct timeval *e)
|
|
20
|
+
{
|
|
21
|
+
long long sec, usec;
|
|
22
|
+
|
|
23
|
+
sec = e->tv_sec - s->tv_sec;
|
|
24
|
+
usec = (e->tv_usec - s->tv_usec);
|
|
25
|
+
if (sec > 0 && usec < 0) {
|
|
26
|
+
sec--;
|
|
27
|
+
usec += 1000000;
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
sec *= 1000;
|
|
31
|
+
usec /= 1000;
|
|
32
|
+
return sec + usec;
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
static unsigned long long mtime_since_now(struct timeval *tv)
|
|
36
|
+
{
|
|
37
|
+
struct timeval end;
|
|
38
|
+
|
|
39
|
+
gettimeofday(&end, NULL);
|
|
40
|
+
return mtime_since(tv, &end);
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
static int test(struct io_uring *ring)
|
|
44
|
+
{
|
|
45
|
+
struct io_uring_cqe *cqe;
|
|
46
|
+
struct io_uring_sqe *sqe;
|
|
47
|
+
struct __kernel_timespec ts;
|
|
48
|
+
struct timeval tv;
|
|
49
|
+
int ret, i;
|
|
50
|
+
|
|
51
|
+
for (i = 0; i < 1; i++) {
|
|
52
|
+
sqe = io_uring_get_sqe(ring);
|
|
53
|
+
if (!sqe) {
|
|
54
|
+
fprintf(stderr, "get sqe failed at %d\n", i);
|
|
55
|
+
goto err;
|
|
56
|
+
}
|
|
57
|
+
io_uring_prep_nop(sqe);
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
ts.tv_sec = 1;
|
|
61
|
+
ts.tv_nsec = 0;
|
|
62
|
+
gettimeofday(&tv, NULL);
|
|
63
|
+
ret = io_uring_submit_and_wait_timeout(ring, &cqe, 2, &ts, NULL);
|
|
64
|
+
if (ret < 0) {
|
|
65
|
+
fprintf(stderr, "submit_and_wait_timeout: %d\n", ret);
|
|
66
|
+
goto err;
|
|
67
|
+
}
|
|
68
|
+
ret = mtime_since_now(&tv);
|
|
69
|
+
/* allow some slack, should be around 1s */
|
|
70
|
+
if (ret > 1200) {
|
|
71
|
+
fprintf(stderr, "wait took too long: %d\n", ret);
|
|
72
|
+
goto err;
|
|
73
|
+
}
|
|
74
|
+
return 0;
|
|
75
|
+
err:
|
|
76
|
+
return 1;
|
|
77
|
+
}
|
|
78
|
+
|
|
79
|
+
static int test_ring(void)
|
|
80
|
+
{
|
|
81
|
+
struct io_uring ring;
|
|
82
|
+
struct io_uring_params p = { };
|
|
83
|
+
int ret;
|
|
84
|
+
|
|
85
|
+
p.flags = 0;
|
|
86
|
+
ret = io_uring_queue_init_params(8, &ring, &p);
|
|
87
|
+
if (ret) {
|
|
88
|
+
fprintf(stderr, "ring setup failed: %d\n", ret);
|
|
89
|
+
return 1;
|
|
90
|
+
}
|
|
91
|
+
|
|
92
|
+
ret = test(&ring);
|
|
93
|
+
if (ret) {
|
|
94
|
+
fprintf(stderr, "test failed\n");
|
|
95
|
+
goto err;
|
|
96
|
+
}
|
|
97
|
+
err:
|
|
98
|
+
io_uring_queue_exit(&ring);
|
|
99
|
+
return ret;
|
|
100
|
+
}
|
|
101
|
+
|
|
102
|
+
int main(int argc, char *argv[])
|
|
103
|
+
{
|
|
104
|
+
if (argc > 1)
|
|
105
|
+
return 0;
|
|
106
|
+
|
|
107
|
+
return test_ring();
|
|
108
|
+
}
|
|
@@ -0,0 +1,156 @@
|
|
|
1
|
+
/* SPDX-License-Identifier: MIT */
|
|
2
|
+
/*
|
|
3
|
+
* Description: tests linked requests failing during submission
|
|
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 <assert.h>
|
|
12
|
+
|
|
13
|
+
#include "liburing.h"
|
|
14
|
+
|
|
15
|
+
#define DRAIN_USER_DATA 42
|
|
16
|
+
|
|
17
|
+
static int test_underprep_fail(bool hardlink, bool drain, bool link_last,
|
|
18
|
+
int link_size, int fail_idx)
|
|
19
|
+
{
|
|
20
|
+
const int invalid_fd = 42;
|
|
21
|
+
int link_flags = IOSQE_IO_LINK;
|
|
22
|
+
int total_submit = link_size;
|
|
23
|
+
struct io_uring ring;
|
|
24
|
+
struct io_uring_sqe *sqe;
|
|
25
|
+
struct io_uring_cqe *cqe;
|
|
26
|
+
char buffer[1] = { };
|
|
27
|
+
int i, ret, fds[2];
|
|
28
|
+
|
|
29
|
+
if (drain)
|
|
30
|
+
link_flags |= IOSQE_IO_DRAIN;
|
|
31
|
+
if (hardlink)
|
|
32
|
+
link_flags |= IOSQE_IO_HARDLINK;
|
|
33
|
+
|
|
34
|
+
assert(fail_idx < link_size);
|
|
35
|
+
assert(link_size < 40);
|
|
36
|
+
|
|
37
|
+
/* create a new ring as it leaves it dirty */
|
|
38
|
+
ret = io_uring_queue_init(8, &ring, 0);
|
|
39
|
+
if (ret) {
|
|
40
|
+
printf("ring setup failed\n");
|
|
41
|
+
return -1;
|
|
42
|
+
}
|
|
43
|
+
if (pipe(fds)) {
|
|
44
|
+
perror("pipe");
|
|
45
|
+
return -1;
|
|
46
|
+
}
|
|
47
|
+
|
|
48
|
+
if (drain) {
|
|
49
|
+
/* clog drain, so following reqs sent to draining */
|
|
50
|
+
sqe = io_uring_get_sqe(&ring);
|
|
51
|
+
io_uring_prep_read(sqe, fds[0], buffer, sizeof(buffer), 0);
|
|
52
|
+
sqe->user_data = DRAIN_USER_DATA;
|
|
53
|
+
sqe->flags |= IOSQE_IO_DRAIN;
|
|
54
|
+
total_submit++;
|
|
55
|
+
}
|
|
56
|
+
|
|
57
|
+
for (i = 0; i < link_size; i++) {
|
|
58
|
+
sqe = io_uring_get_sqe(&ring);
|
|
59
|
+
if (i == fail_idx) {
|
|
60
|
+
io_uring_prep_read(sqe, invalid_fd, buffer, 1, 0);
|
|
61
|
+
sqe->ioprio = (short) -1;
|
|
62
|
+
} else {
|
|
63
|
+
io_uring_prep_nop(sqe);
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
if (i != link_size - 1 || !link_last)
|
|
67
|
+
sqe->flags |= link_flags;
|
|
68
|
+
sqe->user_data = i;
|
|
69
|
+
}
|
|
70
|
+
|
|
71
|
+
ret = io_uring_submit(&ring);
|
|
72
|
+
if (ret != total_submit) {
|
|
73
|
+
/* Old behaviour, failed early and under-submitted */
|
|
74
|
+
if (ret == fail_idx + 1 + drain)
|
|
75
|
+
goto out;
|
|
76
|
+
fprintf(stderr, "submit failed: %d\n", ret);
|
|
77
|
+
return -1;
|
|
78
|
+
}
|
|
79
|
+
|
|
80
|
+
if (drain) {
|
|
81
|
+
/* unclog drain */
|
|
82
|
+
ret = write(fds[1], buffer, sizeof(buffer));
|
|
83
|
+
if (ret < 0) {
|
|
84
|
+
perror("write");
|
|
85
|
+
return 1;
|
|
86
|
+
}
|
|
87
|
+
}
|
|
88
|
+
|
|
89
|
+
for (i = 0; i < total_submit; i++) {
|
|
90
|
+
ret = io_uring_wait_cqe(&ring, &cqe);
|
|
91
|
+
if (ret) {
|
|
92
|
+
fprintf(stderr, "wait_cqe=%d\n", ret);
|
|
93
|
+
return 1;
|
|
94
|
+
}
|
|
95
|
+
|
|
96
|
+
ret = cqe->res;
|
|
97
|
+
if (cqe->user_data == DRAIN_USER_DATA) {
|
|
98
|
+
if (ret != 1) {
|
|
99
|
+
fprintf(stderr, "drain failed %d\n", ret);
|
|
100
|
+
return 1;
|
|
101
|
+
}
|
|
102
|
+
} else if (cqe->user_data == fail_idx) {
|
|
103
|
+
if (ret == 0 || ret == -ECANCELED) {
|
|
104
|
+
fprintf(stderr, "half-prep req unexpected return %d\n", ret);
|
|
105
|
+
return 1;
|
|
106
|
+
}
|
|
107
|
+
} else {
|
|
108
|
+
if (ret != -ECANCELED) {
|
|
109
|
+
fprintf(stderr, "cancel failed %d, ud %d\n", ret, (int)cqe->user_data);
|
|
110
|
+
return 1;
|
|
111
|
+
}
|
|
112
|
+
}
|
|
113
|
+
io_uring_cqe_seen(&ring, cqe);
|
|
114
|
+
}
|
|
115
|
+
out:
|
|
116
|
+
close(fds[0]);
|
|
117
|
+
close(fds[1]);
|
|
118
|
+
io_uring_queue_exit(&ring);
|
|
119
|
+
return 0;
|
|
120
|
+
}
|
|
121
|
+
|
|
122
|
+
int main(int argc, char *argv[])
|
|
123
|
+
{
|
|
124
|
+
int ret, link_size, fail_idx, i;
|
|
125
|
+
|
|
126
|
+
if (argc > 1)
|
|
127
|
+
return 0;
|
|
128
|
+
|
|
129
|
+
/*
|
|
130
|
+
* hardlink, size=3, fail_idx=1, drain=false -- kernel fault
|
|
131
|
+
* link, size=3, fail_idx=0, drain=true -- kernel fault
|
|
132
|
+
* link, size=3, fail_idx=1, drain=true -- invalid cqe->res
|
|
133
|
+
*/
|
|
134
|
+
for (link_size = 0; link_size < 3; link_size++) {
|
|
135
|
+
for (fail_idx = 0; fail_idx < link_size; fail_idx++) {
|
|
136
|
+
for (i = 0; i < 8; i++) {
|
|
137
|
+
bool hardlink = (i & 1) != 0;
|
|
138
|
+
bool drain = (i & 2) != 0;
|
|
139
|
+
bool link_last = (i & 4) != 0;
|
|
140
|
+
|
|
141
|
+
ret = test_underprep_fail(hardlink, drain, link_last,
|
|
142
|
+
link_size, fail_idx);
|
|
143
|
+
if (!ret)
|
|
144
|
+
continue;
|
|
145
|
+
|
|
146
|
+
fprintf(stderr, "failed %d, hard %d, drain %d,"
|
|
147
|
+
"link_last %d, size %d, idx %d\n",
|
|
148
|
+
ret, hardlink, drain, link_last,
|
|
149
|
+
link_size, fail_idx);
|
|
150
|
+
return 1;
|
|
151
|
+
}
|
|
152
|
+
}
|
|
153
|
+
}
|
|
154
|
+
|
|
155
|
+
return 0;
|
|
156
|
+
}
|
|
@@ -0,0 +1,237 @@
|
|
|
1
|
+
/* SPDX-License-Identifier: MIT */
|
|
2
|
+
/*
|
|
3
|
+
* Test reads that will punt to blocking context, with immediate overwrite
|
|
4
|
+
* of iovec->iov_base to NULL. If the kernel doesn't properly handle
|
|
5
|
+
* reuse of the iovec, we should get -EFAULT.
|
|
6
|
+
*/
|
|
7
|
+
#include <unistd.h>
|
|
8
|
+
#include <fcntl.h>
|
|
9
|
+
#include <stdio.h>
|
|
10
|
+
#include <string.h>
|
|
11
|
+
#include <stdlib.h>
|
|
12
|
+
#include <pthread.h>
|
|
13
|
+
#include <sys/time.h>
|
|
14
|
+
|
|
15
|
+
#include "helpers.h"
|
|
16
|
+
#include "liburing.h"
|
|
17
|
+
|
|
18
|
+
#define STR_SIZE 32768
|
|
19
|
+
#define FILE_SIZE 65536
|
|
20
|
+
|
|
21
|
+
struct thread_data {
|
|
22
|
+
int fd1, fd2;
|
|
23
|
+
volatile int do_exit;
|
|
24
|
+
};
|
|
25
|
+
|
|
26
|
+
static void *flusher(void *__data)
|
|
27
|
+
{
|
|
28
|
+
struct thread_data *data = __data;
|
|
29
|
+
|
|
30
|
+
while (!data->do_exit) {
|
|
31
|
+
posix_fadvise(data->fd1, 0, FILE_SIZE, POSIX_FADV_DONTNEED);
|
|
32
|
+
posix_fadvise(data->fd2, 0, FILE_SIZE, POSIX_FADV_DONTNEED);
|
|
33
|
+
usleep(10);
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
return NULL;
|
|
37
|
+
}
|
|
38
|
+
|
|
39
|
+
static char str1[STR_SIZE];
|
|
40
|
+
static char str2[STR_SIZE];
|
|
41
|
+
|
|
42
|
+
static struct io_uring ring;
|
|
43
|
+
|
|
44
|
+
static int no_stable;
|
|
45
|
+
|
|
46
|
+
static int prep(int fd, char *str, int split, int async)
|
|
47
|
+
{
|
|
48
|
+
struct io_uring_sqe *sqe;
|
|
49
|
+
struct iovec iovs[16];
|
|
50
|
+
int ret, i;
|
|
51
|
+
|
|
52
|
+
if (split) {
|
|
53
|
+
int vsize = STR_SIZE / 16;
|
|
54
|
+
void *ptr = str;
|
|
55
|
+
|
|
56
|
+
for (i = 0; i < 16; i++) {
|
|
57
|
+
iovs[i].iov_base = ptr;
|
|
58
|
+
iovs[i].iov_len = vsize;
|
|
59
|
+
ptr += vsize;
|
|
60
|
+
}
|
|
61
|
+
} else {
|
|
62
|
+
iovs[0].iov_base = str;
|
|
63
|
+
iovs[0].iov_len = STR_SIZE;
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
sqe = io_uring_get_sqe(&ring);
|
|
67
|
+
io_uring_prep_readv(sqe, fd, iovs, split ? 16 : 1, 0);
|
|
68
|
+
sqe->user_data = fd;
|
|
69
|
+
if (async)
|
|
70
|
+
sqe->flags = IOSQE_ASYNC;
|
|
71
|
+
ret = io_uring_submit(&ring);
|
|
72
|
+
if (ret != 1) {
|
|
73
|
+
fprintf(stderr, "submit got %d\n", ret);
|
|
74
|
+
return 1;
|
|
75
|
+
}
|
|
76
|
+
if (split) {
|
|
77
|
+
for (i = 0; i < 16; i++)
|
|
78
|
+
iovs[i].iov_base = NULL;
|
|
79
|
+
} else {
|
|
80
|
+
iovs[0].iov_base = NULL;
|
|
81
|
+
}
|
|
82
|
+
return 0;
|
|
83
|
+
}
|
|
84
|
+
|
|
85
|
+
static int wait_nr(int nr)
|
|
86
|
+
{
|
|
87
|
+
int i, ret;
|
|
88
|
+
|
|
89
|
+
for (i = 0; i < nr; i++) {
|
|
90
|
+
struct io_uring_cqe *cqe;
|
|
91
|
+
|
|
92
|
+
ret = io_uring_wait_cqe(&ring, &cqe);
|
|
93
|
+
if (ret)
|
|
94
|
+
return ret;
|
|
95
|
+
if (cqe->res < 0) {
|
|
96
|
+
fprintf(stderr, "cqe->res=%d\n", cqe->res);
|
|
97
|
+
return 1;
|
|
98
|
+
}
|
|
99
|
+
io_uring_cqe_seen(&ring, cqe);
|
|
100
|
+
}
|
|
101
|
+
|
|
102
|
+
return 0;
|
|
103
|
+
}
|
|
104
|
+
|
|
105
|
+
static unsigned long long mtime_since(const struct timeval *s,
|
|
106
|
+
const struct timeval *e)
|
|
107
|
+
{
|
|
108
|
+
long long sec, usec;
|
|
109
|
+
|
|
110
|
+
sec = e->tv_sec - s->tv_sec;
|
|
111
|
+
usec = (e->tv_usec - s->tv_usec);
|
|
112
|
+
if (sec > 0 && usec < 0) {
|
|
113
|
+
sec--;
|
|
114
|
+
usec += 1000000;
|
|
115
|
+
}
|
|
116
|
+
|
|
117
|
+
sec *= 1000;
|
|
118
|
+
usec /= 1000;
|
|
119
|
+
return sec + usec;
|
|
120
|
+
}
|
|
121
|
+
|
|
122
|
+
static unsigned long long mtime_since_now(struct timeval *tv)
|
|
123
|
+
{
|
|
124
|
+
struct timeval end;
|
|
125
|
+
|
|
126
|
+
gettimeofday(&end, NULL);
|
|
127
|
+
return mtime_since(tv, &end);
|
|
128
|
+
}
|
|
129
|
+
|
|
130
|
+
static int test_reuse(int argc, char *argv[], int split, int async)
|
|
131
|
+
{
|
|
132
|
+
struct thread_data data;
|
|
133
|
+
struct io_uring_params p = { };
|
|
134
|
+
int fd1, fd2, ret, i;
|
|
135
|
+
struct timeval tv;
|
|
136
|
+
pthread_t thread;
|
|
137
|
+
char *fname1 = ".reuse.1";
|
|
138
|
+
int do_unlink = 1;
|
|
139
|
+
void *tret;
|
|
140
|
+
|
|
141
|
+
ret = io_uring_queue_init_params(32, &ring, &p);
|
|
142
|
+
if (ret) {
|
|
143
|
+
fprintf(stderr, "io_uring_queue_init: %d\n", ret);
|
|
144
|
+
return 1;
|
|
145
|
+
}
|
|
146
|
+
|
|
147
|
+
if (!(p.features & IORING_FEAT_SUBMIT_STABLE)) {
|
|
148
|
+
fprintf(stdout, "FEAT_SUBMIT_STABLE not there, skipping\n");
|
|
149
|
+
io_uring_queue_exit(&ring);
|
|
150
|
+
no_stable = 1;
|
|
151
|
+
return 0;
|
|
152
|
+
}
|
|
153
|
+
|
|
154
|
+
if (argc > 1) {
|
|
155
|
+
fname1 = argv[1];
|
|
156
|
+
do_unlink = 0;
|
|
157
|
+
} else {
|
|
158
|
+
t_create_file(fname1, FILE_SIZE);
|
|
159
|
+
}
|
|
160
|
+
|
|
161
|
+
fd1 = open(fname1, O_RDONLY);
|
|
162
|
+
if (do_unlink)
|
|
163
|
+
unlink(fname1);
|
|
164
|
+
if (fd1 < 0) {
|
|
165
|
+
perror("open fname1");
|
|
166
|
+
goto err;
|
|
167
|
+
}
|
|
168
|
+
|
|
169
|
+
t_create_file(".reuse.2", FILE_SIZE);
|
|
170
|
+
fd2 = open(".reuse.2", O_RDONLY);
|
|
171
|
+
unlink(".reuse.2");
|
|
172
|
+
if (fd2 < 0) {
|
|
173
|
+
perror("open .reuse.2");
|
|
174
|
+
goto err;
|
|
175
|
+
}
|
|
176
|
+
|
|
177
|
+
data.fd1 = fd1;
|
|
178
|
+
data.fd2 = fd2;
|
|
179
|
+
data.do_exit = 0;
|
|
180
|
+
pthread_create(&thread, NULL, flusher, &data);
|
|
181
|
+
usleep(10000);
|
|
182
|
+
|
|
183
|
+
gettimeofday(&tv, NULL);
|
|
184
|
+
for (i = 0; i < 1000; i++) {
|
|
185
|
+
ret = prep(fd1, str1, split, async);
|
|
186
|
+
if (ret) {
|
|
187
|
+
fprintf(stderr, "prep1 failed: %d\n", ret);
|
|
188
|
+
goto err;
|
|
189
|
+
}
|
|
190
|
+
ret = prep(fd2, str2, split, async);
|
|
191
|
+
if (ret) {
|
|
192
|
+
fprintf(stderr, "prep1 failed: %d\n", ret);
|
|
193
|
+
goto err;
|
|
194
|
+
}
|
|
195
|
+
ret = wait_nr(2);
|
|
196
|
+
if (ret) {
|
|
197
|
+
fprintf(stderr, "wait_nr: %d\n", ret);
|
|
198
|
+
goto err;
|
|
199
|
+
}
|
|
200
|
+
if (mtime_since_now(&tv) > 5000)
|
|
201
|
+
break;
|
|
202
|
+
}
|
|
203
|
+
|
|
204
|
+
data.do_exit = 1;
|
|
205
|
+
pthread_join(thread, &tret);
|
|
206
|
+
|
|
207
|
+
close(fd2);
|
|
208
|
+
close(fd1);
|
|
209
|
+
io_uring_queue_exit(&ring);
|
|
210
|
+
return 0;
|
|
211
|
+
err:
|
|
212
|
+
io_uring_queue_exit(&ring);
|
|
213
|
+
return 1;
|
|
214
|
+
|
|
215
|
+
}
|
|
216
|
+
|
|
217
|
+
int main(int argc, char *argv[])
|
|
218
|
+
{
|
|
219
|
+
int ret, i;
|
|
220
|
+
|
|
221
|
+
for (i = 0; i < 4; i++) {
|
|
222
|
+
int split, async;
|
|
223
|
+
|
|
224
|
+
split = (i & 1) != 0;
|
|
225
|
+
async = (i & 2) != 0;
|
|
226
|
+
|
|
227
|
+
ret = test_reuse(argc, argv, split, async);
|
|
228
|
+
if (ret) {
|
|
229
|
+
fprintf(stderr, "test_reuse %d %d failed\n", split, async);
|
|
230
|
+
return ret;
|
|
231
|
+
}
|
|
232
|
+
if (no_stable)
|
|
233
|
+
break;
|
|
234
|
+
}
|
|
235
|
+
|
|
236
|
+
return 0;
|
|
237
|
+
}
|