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,411 @@
|
|
|
1
|
+
/* SPDX-License-Identifier: MIT */
|
|
2
|
+
/*
|
|
3
|
+
* Test fixed buffers consisting of hugepages.
|
|
4
|
+
*/
|
|
5
|
+
#include <stdio.h>
|
|
6
|
+
#include <string.h>
|
|
7
|
+
#include <fcntl.h>
|
|
8
|
+
#include <stdlib.h>
|
|
9
|
+
#include <sys/mman.h>
|
|
10
|
+
#include <linux/mman.h>
|
|
11
|
+
|
|
12
|
+
#include "liburing.h"
|
|
13
|
+
#include "helpers.h"
|
|
14
|
+
|
|
15
|
+
/*
|
|
16
|
+
* Before testing
|
|
17
|
+
* echo (>=4) > /proc/sys/vm/nr_hugepages
|
|
18
|
+
* echo madvise > /sys/kernel/mm/transparent_hugepage/enabled
|
|
19
|
+
* echo always > /sys/kernel/mm/transparent_hugepage/hugepages-16kB/enabled
|
|
20
|
+
*
|
|
21
|
+
* Not 100% guaranteed to get THP-backed memory, but in general it does.
|
|
22
|
+
*/
|
|
23
|
+
#define MTHP_16KB (16UL * 1024)
|
|
24
|
+
#define HUGEPAGE_SIZE (2UL * 1024 * 1024)
|
|
25
|
+
#define NR_BUFS 1
|
|
26
|
+
#define IN_FD "/dev/urandom"
|
|
27
|
+
#define OUT_FD "/dev/zero"
|
|
28
|
+
|
|
29
|
+
static int open_files(char *fname_in, int *fd_in, int *fd_out)
|
|
30
|
+
{
|
|
31
|
+
*fd_in = open(fname_in, O_RDONLY, 0644);
|
|
32
|
+
if (*fd_in < 0) {
|
|
33
|
+
printf("open %s failed\n", fname_in);
|
|
34
|
+
return -1;
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
*fd_out = open(OUT_FD, O_RDWR, 0644);
|
|
38
|
+
if (*fd_out < 0) {
|
|
39
|
+
printf("open %s failed\n", OUT_FD);
|
|
40
|
+
return -1;
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
return 0;
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
static void unmap(struct iovec *iov, int nr_bufs, size_t offset)
|
|
47
|
+
{
|
|
48
|
+
int i;
|
|
49
|
+
|
|
50
|
+
for (i = 0; i < nr_bufs; i++)
|
|
51
|
+
munmap(iov[i].iov_base - offset, iov[i].iov_len + offset);
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
static int mmap_hugebufs(struct iovec *iov, int nr_bufs, size_t buf_size, size_t offset)
|
|
55
|
+
{
|
|
56
|
+
int i;
|
|
57
|
+
|
|
58
|
+
for (i = 0; i < nr_bufs; i++) {
|
|
59
|
+
void *base = NULL;
|
|
60
|
+
|
|
61
|
+
base = mmap(NULL, buf_size, PROT_READ | PROT_WRITE,
|
|
62
|
+
MAP_PRIVATE | MAP_ANONYMOUS | MAP_HUGETLB, -1, 0);
|
|
63
|
+
if (base == MAP_FAILED) {
|
|
64
|
+
printf("Unable to map hugetlb page. Try increasing the "
|
|
65
|
+
"value in /proc/sys/vm/nr_hugepages\n");
|
|
66
|
+
unmap(iov, i, offset);
|
|
67
|
+
return -1;
|
|
68
|
+
}
|
|
69
|
+
|
|
70
|
+
memset(base, 0, buf_size);
|
|
71
|
+
iov[i].iov_base = base + offset;
|
|
72
|
+
iov[i].iov_len = buf_size - offset;
|
|
73
|
+
}
|
|
74
|
+
|
|
75
|
+
return 0;
|
|
76
|
+
}
|
|
77
|
+
|
|
78
|
+
/* map a hugepage and smaller page to a contiguous memory */
|
|
79
|
+
static int mmap_mixture(struct iovec *iov, int nr_bufs, size_t buf_size, bool huge_on_left)
|
|
80
|
+
{
|
|
81
|
+
int i;
|
|
82
|
+
void *small_base = NULL, *huge_base = NULL, *start = NULL,
|
|
83
|
+
*huge_start = NULL, *small_start = NULL;
|
|
84
|
+
size_t small_size = buf_size - HUGEPAGE_SIZE;
|
|
85
|
+
size_t seg_size = ((buf_size / HUGEPAGE_SIZE) + 1) * HUGEPAGE_SIZE;
|
|
86
|
+
|
|
87
|
+
start = mmap(NULL, seg_size * nr_bufs, PROT_NONE,
|
|
88
|
+
MAP_PRIVATE | MAP_ANONYMOUS | MAP_NORESERVE, -1, 0);
|
|
89
|
+
if (start == MAP_FAILED) {
|
|
90
|
+
printf("Unable to preserve the page mixture memory. "
|
|
91
|
+
"Try increasing the RLIMIT_MEMLOCK resource limit\n");
|
|
92
|
+
return -1;
|
|
93
|
+
}
|
|
94
|
+
|
|
95
|
+
for (i = 0; i < nr_bufs; i++) {
|
|
96
|
+
if (huge_on_left) {
|
|
97
|
+
huge_start = start;
|
|
98
|
+
small_start = start + HUGEPAGE_SIZE;
|
|
99
|
+
} else {
|
|
100
|
+
huge_start = start + HUGEPAGE_SIZE;
|
|
101
|
+
small_start = start + HUGEPAGE_SIZE - small_size;
|
|
102
|
+
}
|
|
103
|
+
|
|
104
|
+
huge_base = mmap(huge_start, HUGEPAGE_SIZE, PROT_READ | PROT_WRITE,
|
|
105
|
+
MAP_PRIVATE | MAP_ANONYMOUS | MAP_HUGETLB | MAP_FIXED, -1, 0);
|
|
106
|
+
if (huge_base == MAP_FAILED) {
|
|
107
|
+
printf("Unable to map hugetlb page in the page mixture. "
|
|
108
|
+
"Try increasing the value in /proc/sys/vm/nr_hugepages\n");
|
|
109
|
+
unmap(iov, nr_bufs, 0);
|
|
110
|
+
return -1;
|
|
111
|
+
}
|
|
112
|
+
|
|
113
|
+
small_base = mmap(small_start, small_size, PROT_READ | PROT_WRITE,
|
|
114
|
+
MAP_PRIVATE | MAP_ANONYMOUS | MAP_FIXED, -1, 0);
|
|
115
|
+
if (small_base == MAP_FAILED) {
|
|
116
|
+
printf("Unable to map small page in the page mixture. "
|
|
117
|
+
"Try increasing the RLIMIT_MEMLOCK resource limit\n");
|
|
118
|
+
unmap(iov, nr_bufs, 0);
|
|
119
|
+
return -1;
|
|
120
|
+
}
|
|
121
|
+
|
|
122
|
+
if (huge_on_left) {
|
|
123
|
+
iov[i].iov_base = huge_base;
|
|
124
|
+
memset(huge_base, 0, buf_size);
|
|
125
|
+
}
|
|
126
|
+
else {
|
|
127
|
+
iov[i].iov_base = small_base;
|
|
128
|
+
memset(small_base, 0, buf_size);
|
|
129
|
+
}
|
|
130
|
+
iov[i].iov_len = buf_size;
|
|
131
|
+
start += seg_size;
|
|
132
|
+
}
|
|
133
|
+
|
|
134
|
+
return 0;
|
|
135
|
+
}
|
|
136
|
+
|
|
137
|
+
static void free_bufs(struct iovec *iov, int nr_bufs, size_t offset)
|
|
138
|
+
{
|
|
139
|
+
int i;
|
|
140
|
+
|
|
141
|
+
for (i = 0; i < nr_bufs; i++)
|
|
142
|
+
free(iov[i].iov_base - offset);
|
|
143
|
+
}
|
|
144
|
+
|
|
145
|
+
static int get_mthp_bufs(struct iovec *iov, int nr_bufs, size_t buf_size,
|
|
146
|
+
size_t alignment, size_t offset)
|
|
147
|
+
{
|
|
148
|
+
int i;
|
|
149
|
+
|
|
150
|
+
for (i = 0; i < nr_bufs; i++) {
|
|
151
|
+
void *base = NULL;
|
|
152
|
+
|
|
153
|
+
if (posix_memalign(&base, alignment, buf_size)) {
|
|
154
|
+
printf("Unable to allocate mthp pages. "
|
|
155
|
+
"Try increasing the RLIMIT_MEMLOCK resource limit\n");
|
|
156
|
+
free_bufs(iov, i, offset);
|
|
157
|
+
return -1;
|
|
158
|
+
}
|
|
159
|
+
|
|
160
|
+
memset(base, 0, buf_size);
|
|
161
|
+
iov[i].iov_base = base + offset;
|
|
162
|
+
iov[i].iov_len = buf_size - offset;
|
|
163
|
+
}
|
|
164
|
+
|
|
165
|
+
return 0;
|
|
166
|
+
}
|
|
167
|
+
|
|
168
|
+
static int do_read(struct io_uring *ring, int fd, struct iovec *iov, int nr_bufs)
|
|
169
|
+
{
|
|
170
|
+
struct io_uring_sqe *sqe;
|
|
171
|
+
struct io_uring_cqe *cqe;
|
|
172
|
+
int i, ret;
|
|
173
|
+
|
|
174
|
+
for (i = 0; i < nr_bufs; i++) {
|
|
175
|
+
sqe = io_uring_get_sqe(ring);
|
|
176
|
+
if (!sqe) {
|
|
177
|
+
fprintf(stderr, "Could not get SQE.\n");
|
|
178
|
+
return -1;
|
|
179
|
+
}
|
|
180
|
+
|
|
181
|
+
io_uring_prep_read_fixed(sqe, fd, iov[i].iov_base, iov[i].iov_len, 0, i);
|
|
182
|
+
io_uring_submit(ring);
|
|
183
|
+
|
|
184
|
+
ret = io_uring_wait_cqe(ring, &cqe);
|
|
185
|
+
if (ret < 0) {
|
|
186
|
+
fprintf(stderr, "Error waiting for completion: %s\n", strerror(-ret));
|
|
187
|
+
return -1;
|
|
188
|
+
}
|
|
189
|
+
|
|
190
|
+
if (cqe->res < 0) {
|
|
191
|
+
fprintf(stderr, "Error in async read operation: %s\n", strerror(-cqe->res));
|
|
192
|
+
return -1;
|
|
193
|
+
}
|
|
194
|
+
if (cqe->res != iov[i].iov_len) {
|
|
195
|
+
fprintf(stderr, "cqe res: %d, expected: %lu\n", cqe->res, (unsigned long) iov[i].iov_len);
|
|
196
|
+
return -1;
|
|
197
|
+
}
|
|
198
|
+
|
|
199
|
+
io_uring_cqe_seen(ring, cqe);
|
|
200
|
+
}
|
|
201
|
+
|
|
202
|
+
return 0;
|
|
203
|
+
}
|
|
204
|
+
|
|
205
|
+
static int do_write(struct io_uring *ring, int fd, struct iovec *iov, int nr_bufs)
|
|
206
|
+
{
|
|
207
|
+
struct io_uring_sqe *sqe;
|
|
208
|
+
struct io_uring_cqe *cqe;
|
|
209
|
+
int i, ret;
|
|
210
|
+
|
|
211
|
+
for (i = 0; i < nr_bufs; i++) {
|
|
212
|
+
sqe = io_uring_get_sqe(ring);
|
|
213
|
+
if (!sqe) {
|
|
214
|
+
fprintf(stderr, "Could not get SQE.\n");
|
|
215
|
+
return -1;
|
|
216
|
+
}
|
|
217
|
+
|
|
218
|
+
io_uring_prep_write_fixed(sqe, fd, iov[i].iov_base, iov[i].iov_len, 0, i);
|
|
219
|
+
io_uring_submit(ring);
|
|
220
|
+
|
|
221
|
+
ret = io_uring_wait_cqe(ring, &cqe);
|
|
222
|
+
if (ret < 0) {
|
|
223
|
+
fprintf(stderr, "Error waiting for completion: %s\n", strerror(-ret));
|
|
224
|
+
return -1;
|
|
225
|
+
}
|
|
226
|
+
|
|
227
|
+
if (cqe->res < 0) {
|
|
228
|
+
fprintf(stderr, "Error in async write operation: %s\n", strerror(-cqe->res));
|
|
229
|
+
return -1;
|
|
230
|
+
}
|
|
231
|
+
if (cqe->res != iov[i].iov_len) {
|
|
232
|
+
fprintf(stderr, "cqe res: %d, expected: %lu\n", cqe->res, (unsigned long) iov[i].iov_len);
|
|
233
|
+
return -1;
|
|
234
|
+
}
|
|
235
|
+
|
|
236
|
+
io_uring_cqe_seen(ring, cqe);
|
|
237
|
+
}
|
|
238
|
+
|
|
239
|
+
return 0;
|
|
240
|
+
}
|
|
241
|
+
|
|
242
|
+
static int register_submit(struct io_uring *ring, struct iovec *iov,
|
|
243
|
+
int nr_bufs, int fd_in, int fd_out)
|
|
244
|
+
{
|
|
245
|
+
int ret;
|
|
246
|
+
|
|
247
|
+
ret = io_uring_register_buffers(ring, iov, nr_bufs);
|
|
248
|
+
if (ret) {
|
|
249
|
+
fprintf(stderr, "Error registering buffers: %s\n", strerror(-ret));
|
|
250
|
+
return ret;
|
|
251
|
+
}
|
|
252
|
+
|
|
253
|
+
ret = do_read(ring, fd_in, iov, nr_bufs);
|
|
254
|
+
if (ret) {
|
|
255
|
+
fprintf(stderr, "Read test failed\n");
|
|
256
|
+
return ret;
|
|
257
|
+
}
|
|
258
|
+
|
|
259
|
+
ret = do_write(ring, fd_out, iov, nr_bufs);
|
|
260
|
+
if (ret) {
|
|
261
|
+
fprintf(stderr, "Write test failed\n");
|
|
262
|
+
return ret;
|
|
263
|
+
}
|
|
264
|
+
|
|
265
|
+
ret = io_uring_unregister_buffers(ring);
|
|
266
|
+
if (ret) {
|
|
267
|
+
fprintf(stderr, "Error unregistering buffers for one hugepage test: %s", strerror(-ret));
|
|
268
|
+
return ret;
|
|
269
|
+
}
|
|
270
|
+
|
|
271
|
+
return 0;
|
|
272
|
+
}
|
|
273
|
+
|
|
274
|
+
static int test_one_hugepage(struct io_uring *ring, int fd_in, int fd_out)
|
|
275
|
+
{
|
|
276
|
+
struct iovec iov[NR_BUFS];
|
|
277
|
+
size_t buf_size = HUGEPAGE_SIZE;
|
|
278
|
+
int ret;
|
|
279
|
+
|
|
280
|
+
if (mmap_hugebufs(iov, NR_BUFS, buf_size, 0))
|
|
281
|
+
return T_EXIT_SKIP;
|
|
282
|
+
|
|
283
|
+
ret = register_submit(ring, iov, NR_BUFS, fd_in, fd_out);
|
|
284
|
+
unmap(iov, NR_BUFS, 0);
|
|
285
|
+
return ret ? T_EXIT_FAIL : T_EXIT_PASS;
|
|
286
|
+
}
|
|
287
|
+
|
|
288
|
+
static int test_multi_hugepages(struct io_uring *ring, int fd_in, int fd_out)
|
|
289
|
+
{
|
|
290
|
+
struct iovec iov[NR_BUFS];
|
|
291
|
+
size_t buf_size = 4 * HUGEPAGE_SIZE;
|
|
292
|
+
int ret;
|
|
293
|
+
|
|
294
|
+
if (mmap_hugebufs(iov, NR_BUFS, buf_size, 0))
|
|
295
|
+
return T_EXIT_SKIP;
|
|
296
|
+
|
|
297
|
+
ret = register_submit(ring, iov, NR_BUFS, fd_in, fd_out);
|
|
298
|
+
unmap(iov, NR_BUFS, 0);
|
|
299
|
+
return ret ? T_EXIT_FAIL : T_EXIT_PASS;
|
|
300
|
+
}
|
|
301
|
+
|
|
302
|
+
static int test_unaligned_hugepage(struct io_uring *ring, int fd_in, int fd_out)
|
|
303
|
+
{
|
|
304
|
+
struct iovec iov[NR_BUFS];
|
|
305
|
+
size_t buf_size = 3 * HUGEPAGE_SIZE;
|
|
306
|
+
size_t offset = 0x1234;
|
|
307
|
+
int ret;
|
|
308
|
+
|
|
309
|
+
if (mmap_hugebufs(iov, NR_BUFS, buf_size, offset))
|
|
310
|
+
return T_EXIT_SKIP;
|
|
311
|
+
|
|
312
|
+
ret = register_submit(ring, iov, NR_BUFS, fd_in, fd_out);
|
|
313
|
+
unmap(iov, NR_BUFS, offset);
|
|
314
|
+
return ret ? T_EXIT_FAIL : T_EXIT_PASS;
|
|
315
|
+
}
|
|
316
|
+
|
|
317
|
+
static int test_multi_unaligned_mthps(struct io_uring *ring, int fd_in, int fd_out)
|
|
318
|
+
{
|
|
319
|
+
struct iovec iov[NR_BUFS];
|
|
320
|
+
int ret;
|
|
321
|
+
size_t buf_size = 3 * MTHP_16KB;
|
|
322
|
+
size_t offset = 0x1234;
|
|
323
|
+
|
|
324
|
+
if (get_mthp_bufs(iov, NR_BUFS, buf_size, MTHP_16KB, offset))
|
|
325
|
+
return T_EXIT_SKIP;
|
|
326
|
+
|
|
327
|
+
ret = register_submit(ring, iov, NR_BUFS, fd_in, fd_out);
|
|
328
|
+
free_bufs(iov, NR_BUFS, offset);
|
|
329
|
+
return ret ? T_EXIT_FAIL : T_EXIT_PASS;
|
|
330
|
+
}
|
|
331
|
+
|
|
332
|
+
/* Should not coalesce */
|
|
333
|
+
static int test_page_mixture(struct io_uring *ring, int fd_in, int fd_out, int huge_on_left)
|
|
334
|
+
{
|
|
335
|
+
struct iovec iov[NR_BUFS];
|
|
336
|
+
size_t buf_size = HUGEPAGE_SIZE + MTHP_16KB;
|
|
337
|
+
int ret;
|
|
338
|
+
|
|
339
|
+
if (mmap_mixture(iov, NR_BUFS, buf_size, huge_on_left))
|
|
340
|
+
return T_EXIT_SKIP;
|
|
341
|
+
|
|
342
|
+
ret = register_submit(ring, iov, NR_BUFS, fd_in, fd_out);
|
|
343
|
+
unmap(iov, NR_BUFS, 0);
|
|
344
|
+
return ret ? T_EXIT_FAIL : T_EXIT_PASS;
|
|
345
|
+
}
|
|
346
|
+
|
|
347
|
+
int main(int argc, char *argv[])
|
|
348
|
+
{
|
|
349
|
+
struct io_uring ring;
|
|
350
|
+
int ret, fd_in, fd_out;
|
|
351
|
+
char *fname_in;
|
|
352
|
+
|
|
353
|
+
if (argc > 1)
|
|
354
|
+
fname_in = argv[1];
|
|
355
|
+
else
|
|
356
|
+
fname_in = IN_FD;
|
|
357
|
+
|
|
358
|
+
if (open_files(fname_in, &fd_in, &fd_out))
|
|
359
|
+
return T_EXIT_SKIP;
|
|
360
|
+
|
|
361
|
+
ret = t_create_ring(8, &ring, 0);
|
|
362
|
+
if (ret == T_SETUP_SKIP)
|
|
363
|
+
return T_EXIT_SKIP;
|
|
364
|
+
else if (ret < 0)
|
|
365
|
+
return T_EXIT_FAIL;
|
|
366
|
+
|
|
367
|
+
ret = test_one_hugepage(&ring, fd_in, fd_out);
|
|
368
|
+
if (ret != T_EXIT_PASS) {
|
|
369
|
+
if (ret != T_EXIT_SKIP)
|
|
370
|
+
fprintf(stderr, "Test one hugepage failed.\n");
|
|
371
|
+
return ret;
|
|
372
|
+
}
|
|
373
|
+
|
|
374
|
+
ret = test_multi_hugepages(&ring, fd_in, fd_out);
|
|
375
|
+
if (ret != T_EXIT_PASS) {
|
|
376
|
+
if (ret != T_EXIT_SKIP)
|
|
377
|
+
fprintf(stderr, "Test multi hugepages failed.\n");
|
|
378
|
+
return ret;
|
|
379
|
+
}
|
|
380
|
+
|
|
381
|
+
ret = test_unaligned_hugepage(&ring, fd_in, fd_out);
|
|
382
|
+
if (ret != T_EXIT_PASS) {
|
|
383
|
+
if (ret != T_EXIT_SKIP)
|
|
384
|
+
fprintf(stderr, "Test unaligned hugepage failed.\n");
|
|
385
|
+
return ret;
|
|
386
|
+
}
|
|
387
|
+
|
|
388
|
+
ret = test_multi_unaligned_mthps(&ring, fd_in, fd_out);
|
|
389
|
+
if (ret != T_EXIT_PASS) {
|
|
390
|
+
if (ret != T_EXIT_SKIP)
|
|
391
|
+
fprintf(stderr, "Test unaligned multi-size'd THPs failed.\n");
|
|
392
|
+
return ret;
|
|
393
|
+
}
|
|
394
|
+
|
|
395
|
+
ret = test_page_mixture(&ring, fd_in, fd_out, true);
|
|
396
|
+
if (ret != T_EXIT_PASS) {
|
|
397
|
+
if (ret != T_EXIT_SKIP)
|
|
398
|
+
fprintf(stderr, "Test huge small page mixture (start with huge) failed.\n");
|
|
399
|
+
return ret;
|
|
400
|
+
}
|
|
401
|
+
|
|
402
|
+
ret = test_page_mixture(&ring, fd_in, fd_out, false);
|
|
403
|
+
if (ret != T_EXIT_PASS) {
|
|
404
|
+
if (ret != T_EXIT_SKIP)
|
|
405
|
+
fprintf(stderr, "Test huge small page mixture (start with small) failed.\n");
|
|
406
|
+
return ret;
|
|
407
|
+
}
|
|
408
|
+
|
|
409
|
+
io_uring_queue_exit(&ring);
|
|
410
|
+
return T_EXIT_PASS;
|
|
411
|
+
}
|
|
@@ -0,0 +1,90 @@
|
|
|
1
|
+
/* SPDX-License-Identifier: MIT */
|
|
2
|
+
#include <errno.h>
|
|
3
|
+
#include <stdio.h>
|
|
4
|
+
#include <unistd.h>
|
|
5
|
+
#include <stdlib.h>
|
|
6
|
+
#include <string.h>
|
|
7
|
+
#include <fcntl.h>
|
|
8
|
+
#include <sys/types.h>
|
|
9
|
+
|
|
10
|
+
#include "helpers.h"
|
|
11
|
+
#include "liburing.h"
|
|
12
|
+
|
|
13
|
+
#define IOVECS_LEN 2
|
|
14
|
+
|
|
15
|
+
int main(int argc, char *argv[])
|
|
16
|
+
{
|
|
17
|
+
struct iovec iovecs[IOVECS_LEN];
|
|
18
|
+
struct io_uring ring;
|
|
19
|
+
int i, fd, ret;
|
|
20
|
+
|
|
21
|
+
if (argc > 1)
|
|
22
|
+
return T_EXIT_SKIP;
|
|
23
|
+
|
|
24
|
+
fd = open("/dev/zero", O_RDONLY);
|
|
25
|
+
if (fd < 0) {
|
|
26
|
+
fprintf(stderr, "Failed to open /dev/zero\n");
|
|
27
|
+
return T_EXIT_FAIL;
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
if (io_uring_queue_init(32, &ring, 0) < 0) {
|
|
31
|
+
fprintf(stderr, "Failed to init io_uring\n");
|
|
32
|
+
close(fd);
|
|
33
|
+
return T_EXIT_FAIL;
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
for (i = 0; i < IOVECS_LEN; ++i) {
|
|
37
|
+
iovecs[i].iov_base = t_malloc(64);
|
|
38
|
+
iovecs[i].iov_len = 64;
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
ret = io_uring_register_buffers(&ring, iovecs, IOVECS_LEN);
|
|
42
|
+
if (ret) {
|
|
43
|
+
fprintf(stderr, "Failed to register buffers\n");
|
|
44
|
+
return T_EXIT_FAIL;
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
for (i = 0; i < IOVECS_LEN; ++i) {
|
|
48
|
+
struct io_uring_sqe *sqe = io_uring_get_sqe(&ring);
|
|
49
|
+
const char *str = "#include <errno.h>";
|
|
50
|
+
|
|
51
|
+
iovecs[i].iov_len = strlen(str);
|
|
52
|
+
io_uring_prep_read_fixed(sqe, fd, iovecs[i].iov_base, strlen(str), 0, i);
|
|
53
|
+
if (i == 0)
|
|
54
|
+
io_uring_sqe_set_flags(sqe, IOSQE_IO_LINK);
|
|
55
|
+
io_uring_sqe_set_data(sqe, (void *)str);
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
ret = io_uring_submit_and_wait(&ring, IOVECS_LEN);
|
|
59
|
+
if (ret < 0) {
|
|
60
|
+
fprintf(stderr, "Failed to submit IO\n");
|
|
61
|
+
return T_EXIT_FAIL;
|
|
62
|
+
} else if (ret < 2) {
|
|
63
|
+
fprintf(stderr, "Submitted %d, wanted %d\n", ret, IOVECS_LEN);
|
|
64
|
+
return T_EXIT_FAIL;
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
for (i = 0; i < IOVECS_LEN; i++) {
|
|
68
|
+
struct io_uring_cqe *cqe;
|
|
69
|
+
|
|
70
|
+
ret = io_uring_wait_cqe(&ring, &cqe);
|
|
71
|
+
if (ret) {
|
|
72
|
+
fprintf(stderr, "wait_cqe=%d\n", ret);
|
|
73
|
+
return T_EXIT_FAIL;
|
|
74
|
+
}
|
|
75
|
+
if (cqe->res != iovecs[i].iov_len) {
|
|
76
|
+
fprintf(stderr, "read: wanted %ld, got %d\n",
|
|
77
|
+
(long) iovecs[i].iov_len, cqe->res);
|
|
78
|
+
return T_EXIT_FAIL;
|
|
79
|
+
}
|
|
80
|
+
io_uring_cqe_seen(&ring, cqe);
|
|
81
|
+
}
|
|
82
|
+
|
|
83
|
+
close(fd);
|
|
84
|
+
io_uring_queue_exit(&ring);
|
|
85
|
+
|
|
86
|
+
for (i = 0; i < IOVECS_LEN; ++i)
|
|
87
|
+
free(iovecs[i].iov_base);
|
|
88
|
+
|
|
89
|
+
return T_EXIT_PASS;
|
|
90
|
+
}
|
|
@@ -0,0 +1,160 @@
|
|
|
1
|
+
/* SPDX-License-Identifier: MIT */
|
|
2
|
+
/*
|
|
3
|
+
* Description: link <open file><read from file><close file> with an existing
|
|
4
|
+
* file present in the opened slot, verifying that we get the new file
|
|
5
|
+
* rather than the old one.
|
|
6
|
+
*
|
|
7
|
+
*/
|
|
8
|
+
#include <errno.h>
|
|
9
|
+
#include <stdio.h>
|
|
10
|
+
#include <unistd.h>
|
|
11
|
+
#include <stdlib.h>
|
|
12
|
+
#include <string.h>
|
|
13
|
+
#include <fcntl.h>
|
|
14
|
+
|
|
15
|
+
#include "liburing.h"
|
|
16
|
+
#include "helpers.h"
|
|
17
|
+
|
|
18
|
+
#define MAX_FILES 8
|
|
19
|
+
#define FNAME1 ".slot.reuse.1"
|
|
20
|
+
#define FNAME2 ".slot.reuse.2"
|
|
21
|
+
#define PAT1 0xaa
|
|
22
|
+
#define PAT2 0x55
|
|
23
|
+
#define BSIZE 4096
|
|
24
|
+
|
|
25
|
+
static int test(struct io_uring *ring)
|
|
26
|
+
{
|
|
27
|
+
struct io_uring_cqe *cqe;
|
|
28
|
+
struct io_uring_sqe *sqe;
|
|
29
|
+
char buf[BSIZE];
|
|
30
|
+
int ret, i;
|
|
31
|
+
|
|
32
|
+
/* open FNAME1 in slot 0 */
|
|
33
|
+
sqe = io_uring_get_sqe(ring);
|
|
34
|
+
io_uring_prep_openat_direct(sqe, AT_FDCWD, FNAME1, O_RDONLY, 0, 0);
|
|
35
|
+
sqe->user_data = 1;
|
|
36
|
+
|
|
37
|
+
ret = io_uring_submit(ring);
|
|
38
|
+
if (ret != 1) {
|
|
39
|
+
fprintf(stderr, "sqe submit failed: %d\n", ret);
|
|
40
|
+
goto err;
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
ret = io_uring_wait_cqe(ring, &cqe);
|
|
44
|
+
if (ret < 0) {
|
|
45
|
+
fprintf(stderr, "wait completion %d\n", ret);
|
|
46
|
+
goto err;
|
|
47
|
+
}
|
|
48
|
+
if (cqe->res != 0) {
|
|
49
|
+
fprintf(stderr, "open res %d\n", ret);
|
|
50
|
+
goto err;
|
|
51
|
+
}
|
|
52
|
+
io_uring_cqe_seen(ring, cqe);
|
|
53
|
+
|
|
54
|
+
/*
|
|
55
|
+
* Now open FNAME2 in that same slot, verifying we get data from
|
|
56
|
+
* FNAME2 and not FNAME1.
|
|
57
|
+
*/
|
|
58
|
+
sqe = io_uring_get_sqe(ring);
|
|
59
|
+
io_uring_prep_openat_direct(sqe, AT_FDCWD, FNAME2, O_RDONLY, 0, 0);
|
|
60
|
+
sqe->flags |= IOSQE_IO_LINK;
|
|
61
|
+
sqe->user_data = 2;
|
|
62
|
+
|
|
63
|
+
sqe = io_uring_get_sqe(ring);
|
|
64
|
+
io_uring_prep_read(sqe, 0, buf, sizeof(buf), 0);
|
|
65
|
+
sqe->flags |= IOSQE_FIXED_FILE;
|
|
66
|
+
sqe->flags |= IOSQE_IO_LINK;
|
|
67
|
+
sqe->user_data = 3;
|
|
68
|
+
|
|
69
|
+
sqe = io_uring_get_sqe(ring);
|
|
70
|
+
io_uring_prep_close_direct(sqe, 0);
|
|
71
|
+
sqe->user_data = 4;
|
|
72
|
+
|
|
73
|
+
ret = io_uring_submit(ring);
|
|
74
|
+
if (ret != 3) {
|
|
75
|
+
fprintf(stderr, "sqe submit failed: %d\n", ret);
|
|
76
|
+
goto err;
|
|
77
|
+
}
|
|
78
|
+
|
|
79
|
+
for (i = 0; i < 3; i++) {
|
|
80
|
+
ret = io_uring_wait_cqe(ring, &cqe);
|
|
81
|
+
if (ret < 0) {
|
|
82
|
+
fprintf(stderr, "wait completion %d\n", ret);
|
|
83
|
+
goto err;
|
|
84
|
+
}
|
|
85
|
+
switch (cqe->user_data) {
|
|
86
|
+
case 2:
|
|
87
|
+
if (cqe->res) {
|
|
88
|
+
fprintf(stderr, "bad open %d\n", cqe->res);
|
|
89
|
+
goto err;
|
|
90
|
+
}
|
|
91
|
+
break;
|
|
92
|
+
case 3:
|
|
93
|
+
if (cqe->res != sizeof(buf)) {
|
|
94
|
+
fprintf(stderr, "bad read %d\n", cqe->res);
|
|
95
|
+
goto err;
|
|
96
|
+
}
|
|
97
|
+
break;
|
|
98
|
+
case 4:
|
|
99
|
+
if (cqe->res) {
|
|
100
|
+
fprintf(stderr, "bad close %d\n", cqe->res);
|
|
101
|
+
goto err;
|
|
102
|
+
}
|
|
103
|
+
break;
|
|
104
|
+
}
|
|
105
|
+
io_uring_cqe_seen(ring, cqe);
|
|
106
|
+
}
|
|
107
|
+
|
|
108
|
+
for (i = 0; i < sizeof(buf); i++) {
|
|
109
|
+
if (buf[i] == PAT2)
|
|
110
|
+
continue;
|
|
111
|
+
fprintf(stderr, "Bad pattern %x at %d\n", buf[i], i);
|
|
112
|
+
goto err;
|
|
113
|
+
}
|
|
114
|
+
|
|
115
|
+
return 0;
|
|
116
|
+
err:
|
|
117
|
+
return 1;
|
|
118
|
+
}
|
|
119
|
+
|
|
120
|
+
int main(int argc, char *argv[])
|
|
121
|
+
{
|
|
122
|
+
struct io_uring ring;
|
|
123
|
+
struct io_uring_params p = { };
|
|
124
|
+
int ret, files[MAX_FILES];
|
|
125
|
+
|
|
126
|
+
if (argc > 1)
|
|
127
|
+
return T_EXIT_SKIP;
|
|
128
|
+
|
|
129
|
+
ret = io_uring_queue_init_params(8, &ring, &p);
|
|
130
|
+
if (ret) {
|
|
131
|
+
fprintf(stderr, "ring setup failed: %d\n", ret);
|
|
132
|
+
return T_EXIT_FAIL;
|
|
133
|
+
}
|
|
134
|
+
if (!(p.features & IORING_FEAT_CQE_SKIP))
|
|
135
|
+
return T_EXIT_SKIP;
|
|
136
|
+
|
|
137
|
+
memset(files, -1, sizeof(files));
|
|
138
|
+
ret = io_uring_register_files(&ring, files, ARRAY_SIZE(files));
|
|
139
|
+
if (ret) {
|
|
140
|
+
fprintf(stderr, "Failed registering files\n");
|
|
141
|
+
return T_EXIT_FAIL;
|
|
142
|
+
}
|
|
143
|
+
|
|
144
|
+
t_create_file_pattern(FNAME1, 4096, PAT1);
|
|
145
|
+
t_create_file_pattern(FNAME2, 4096, PAT2);
|
|
146
|
+
|
|
147
|
+
ret = test(&ring);
|
|
148
|
+
if (ret) {
|
|
149
|
+
fprintf(stderr, "test failed\n");
|
|
150
|
+
goto err;
|
|
151
|
+
}
|
|
152
|
+
|
|
153
|
+
unlink(FNAME1);
|
|
154
|
+
unlink(FNAME2);
|
|
155
|
+
return T_EXIT_PASS;
|
|
156
|
+
err:
|
|
157
|
+
unlink(FNAME1);
|
|
158
|
+
unlink(FNAME2);
|
|
159
|
+
return T_EXIT_FAIL;
|
|
160
|
+
}
|