1*25da2beaSAndroid Build Coastguard Worker /* SPDX-License-Identifier: MIT */
2*25da2beaSAndroid Build Coastguard Worker /*
3*25da2beaSAndroid Build Coastguard Worker * Description: link <open file><read from file><close file> with an existing
4*25da2beaSAndroid Build Coastguard Worker * file present in the opened slot, verifying that we get the new file
5*25da2beaSAndroid Build Coastguard Worker * rather than the old one.
6*25da2beaSAndroid Build Coastguard Worker *
7*25da2beaSAndroid Build Coastguard Worker */
8*25da2beaSAndroid Build Coastguard Worker #include <errno.h>
9*25da2beaSAndroid Build Coastguard Worker #include <stdio.h>
10*25da2beaSAndroid Build Coastguard Worker #include <unistd.h>
11*25da2beaSAndroid Build Coastguard Worker #include <stdlib.h>
12*25da2beaSAndroid Build Coastguard Worker #include <string.h>
13*25da2beaSAndroid Build Coastguard Worker #include <fcntl.h>
14*25da2beaSAndroid Build Coastguard Worker
15*25da2beaSAndroid Build Coastguard Worker #include "liburing.h"
16*25da2beaSAndroid Build Coastguard Worker #include "helpers.h"
17*25da2beaSAndroid Build Coastguard Worker
18*25da2beaSAndroid Build Coastguard Worker #define MAX_FILES 8
19*25da2beaSAndroid Build Coastguard Worker #define FNAME1 ".slot.reuse.1"
20*25da2beaSAndroid Build Coastguard Worker #define FNAME2 ".slot.reuse.2"
21*25da2beaSAndroid Build Coastguard Worker #define PAT1 0xaa
22*25da2beaSAndroid Build Coastguard Worker #define PAT2 0x55
23*25da2beaSAndroid Build Coastguard Worker #define BSIZE 4096
24*25da2beaSAndroid Build Coastguard Worker
test(struct io_uring * ring)25*25da2beaSAndroid Build Coastguard Worker static int test(struct io_uring *ring)
26*25da2beaSAndroid Build Coastguard Worker {
27*25da2beaSAndroid Build Coastguard Worker struct io_uring_cqe *cqe;
28*25da2beaSAndroid Build Coastguard Worker struct io_uring_sqe *sqe;
29*25da2beaSAndroid Build Coastguard Worker char buf[BSIZE];
30*25da2beaSAndroid Build Coastguard Worker int ret, i;
31*25da2beaSAndroid Build Coastguard Worker
32*25da2beaSAndroid Build Coastguard Worker /* open FNAME1 in slot 0 */
33*25da2beaSAndroid Build Coastguard Worker sqe = io_uring_get_sqe(ring);
34*25da2beaSAndroid Build Coastguard Worker io_uring_prep_openat_direct(sqe, AT_FDCWD, FNAME1, O_RDONLY, 0, 0);
35*25da2beaSAndroid Build Coastguard Worker sqe->user_data = 1;
36*25da2beaSAndroid Build Coastguard Worker
37*25da2beaSAndroid Build Coastguard Worker ret = io_uring_submit(ring);
38*25da2beaSAndroid Build Coastguard Worker if (ret != 1) {
39*25da2beaSAndroid Build Coastguard Worker fprintf(stderr, "sqe submit failed: %d\n", ret);
40*25da2beaSAndroid Build Coastguard Worker goto err;
41*25da2beaSAndroid Build Coastguard Worker }
42*25da2beaSAndroid Build Coastguard Worker
43*25da2beaSAndroid Build Coastguard Worker ret = io_uring_wait_cqe(ring, &cqe);
44*25da2beaSAndroid Build Coastguard Worker if (ret < 0) {
45*25da2beaSAndroid Build Coastguard Worker fprintf(stderr, "wait completion %d\n", ret);
46*25da2beaSAndroid Build Coastguard Worker goto err;
47*25da2beaSAndroid Build Coastguard Worker }
48*25da2beaSAndroid Build Coastguard Worker if (cqe->res != 0) {
49*25da2beaSAndroid Build Coastguard Worker fprintf(stderr, "open res %d\n", ret);
50*25da2beaSAndroid Build Coastguard Worker goto err;
51*25da2beaSAndroid Build Coastguard Worker }
52*25da2beaSAndroid Build Coastguard Worker io_uring_cqe_seen(ring, cqe);
53*25da2beaSAndroid Build Coastguard Worker
54*25da2beaSAndroid Build Coastguard Worker /*
55*25da2beaSAndroid Build Coastguard Worker * Now open FNAME2 in that same slot, verifying we get data from
56*25da2beaSAndroid Build Coastguard Worker * FNAME2 and not FNAME1.
57*25da2beaSAndroid Build Coastguard Worker */
58*25da2beaSAndroid Build Coastguard Worker sqe = io_uring_get_sqe(ring);
59*25da2beaSAndroid Build Coastguard Worker io_uring_prep_openat_direct(sqe, AT_FDCWD, FNAME2, O_RDONLY, 0, 0);
60*25da2beaSAndroid Build Coastguard Worker sqe->flags |= IOSQE_IO_LINK;
61*25da2beaSAndroid Build Coastguard Worker sqe->user_data = 2;
62*25da2beaSAndroid Build Coastguard Worker
63*25da2beaSAndroid Build Coastguard Worker sqe = io_uring_get_sqe(ring);
64*25da2beaSAndroid Build Coastguard Worker io_uring_prep_read(sqe, 0, buf, sizeof(buf), 0);
65*25da2beaSAndroid Build Coastguard Worker sqe->flags |= IOSQE_FIXED_FILE;
66*25da2beaSAndroid Build Coastguard Worker sqe->flags |= IOSQE_IO_LINK;
67*25da2beaSAndroid Build Coastguard Worker sqe->user_data = 3;
68*25da2beaSAndroid Build Coastguard Worker
69*25da2beaSAndroid Build Coastguard Worker sqe = io_uring_get_sqe(ring);
70*25da2beaSAndroid Build Coastguard Worker io_uring_prep_close_direct(sqe, 0);
71*25da2beaSAndroid Build Coastguard Worker sqe->user_data = 4;
72*25da2beaSAndroid Build Coastguard Worker
73*25da2beaSAndroid Build Coastguard Worker ret = io_uring_submit(ring);
74*25da2beaSAndroid Build Coastguard Worker if (ret != 3) {
75*25da2beaSAndroid Build Coastguard Worker fprintf(stderr, "sqe submit failed: %d\n", ret);
76*25da2beaSAndroid Build Coastguard Worker goto err;
77*25da2beaSAndroid Build Coastguard Worker }
78*25da2beaSAndroid Build Coastguard Worker
79*25da2beaSAndroid Build Coastguard Worker for (i = 0; i < 3; i++) {
80*25da2beaSAndroid Build Coastguard Worker ret = io_uring_wait_cqe(ring, &cqe);
81*25da2beaSAndroid Build Coastguard Worker if (ret < 0) {
82*25da2beaSAndroid Build Coastguard Worker fprintf(stderr, "wait completion %d\n", ret);
83*25da2beaSAndroid Build Coastguard Worker goto err;
84*25da2beaSAndroid Build Coastguard Worker }
85*25da2beaSAndroid Build Coastguard Worker switch (cqe->user_data) {
86*25da2beaSAndroid Build Coastguard Worker case 2:
87*25da2beaSAndroid Build Coastguard Worker if (cqe->res) {
88*25da2beaSAndroid Build Coastguard Worker fprintf(stderr, "bad open %d\n", cqe->res);
89*25da2beaSAndroid Build Coastguard Worker goto err;
90*25da2beaSAndroid Build Coastguard Worker }
91*25da2beaSAndroid Build Coastguard Worker break;
92*25da2beaSAndroid Build Coastguard Worker case 3:
93*25da2beaSAndroid Build Coastguard Worker if (cqe->res != sizeof(buf)) {
94*25da2beaSAndroid Build Coastguard Worker fprintf(stderr, "bad read %d\n", cqe->res);
95*25da2beaSAndroid Build Coastguard Worker goto err;
96*25da2beaSAndroid Build Coastguard Worker }
97*25da2beaSAndroid Build Coastguard Worker break;
98*25da2beaSAndroid Build Coastguard Worker case 4:
99*25da2beaSAndroid Build Coastguard Worker if (cqe->res) {
100*25da2beaSAndroid Build Coastguard Worker fprintf(stderr, "bad close %d\n", cqe->res);
101*25da2beaSAndroid Build Coastguard Worker goto err;
102*25da2beaSAndroid Build Coastguard Worker }
103*25da2beaSAndroid Build Coastguard Worker break;
104*25da2beaSAndroid Build Coastguard Worker }
105*25da2beaSAndroid Build Coastguard Worker io_uring_cqe_seen(ring, cqe);
106*25da2beaSAndroid Build Coastguard Worker }
107*25da2beaSAndroid Build Coastguard Worker
108*25da2beaSAndroid Build Coastguard Worker for (i = 0; i < sizeof(buf); i++) {
109*25da2beaSAndroid Build Coastguard Worker if (buf[i] == PAT2)
110*25da2beaSAndroid Build Coastguard Worker continue;
111*25da2beaSAndroid Build Coastguard Worker fprintf(stderr, "Bad pattern %x at %d\n", buf[i], i);
112*25da2beaSAndroid Build Coastguard Worker goto err;
113*25da2beaSAndroid Build Coastguard Worker }
114*25da2beaSAndroid Build Coastguard Worker
115*25da2beaSAndroid Build Coastguard Worker return 0;
116*25da2beaSAndroid Build Coastguard Worker err:
117*25da2beaSAndroid Build Coastguard Worker return 1;
118*25da2beaSAndroid Build Coastguard Worker }
119*25da2beaSAndroid Build Coastguard Worker
main(int argc,char * argv[])120*25da2beaSAndroid Build Coastguard Worker int main(int argc, char *argv[])
121*25da2beaSAndroid Build Coastguard Worker {
122*25da2beaSAndroid Build Coastguard Worker struct io_uring ring;
123*25da2beaSAndroid Build Coastguard Worker struct io_uring_params p = { };
124*25da2beaSAndroid Build Coastguard Worker int ret, files[MAX_FILES];
125*25da2beaSAndroid Build Coastguard Worker
126*25da2beaSAndroid Build Coastguard Worker if (argc > 1)
127*25da2beaSAndroid Build Coastguard Worker return 0;
128*25da2beaSAndroid Build Coastguard Worker
129*25da2beaSAndroid Build Coastguard Worker ret = io_uring_queue_init_params(8, &ring, &p);
130*25da2beaSAndroid Build Coastguard Worker if (ret) {
131*25da2beaSAndroid Build Coastguard Worker fprintf(stderr, "ring setup failed: %d\n", ret);
132*25da2beaSAndroid Build Coastguard Worker return 1;
133*25da2beaSAndroid Build Coastguard Worker }
134*25da2beaSAndroid Build Coastguard Worker if (!(p.features & IORING_FEAT_CQE_SKIP))
135*25da2beaSAndroid Build Coastguard Worker return 0;
136*25da2beaSAndroid Build Coastguard Worker
137*25da2beaSAndroid Build Coastguard Worker memset(files, -1, sizeof(files));
138*25da2beaSAndroid Build Coastguard Worker ret = io_uring_register_files(&ring, files, ARRAY_SIZE(files));
139*25da2beaSAndroid Build Coastguard Worker if (ret) {
140*25da2beaSAndroid Build Coastguard Worker fprintf(stderr, "Failed registering files\n");
141*25da2beaSAndroid Build Coastguard Worker return 1;
142*25da2beaSAndroid Build Coastguard Worker }
143*25da2beaSAndroid Build Coastguard Worker
144*25da2beaSAndroid Build Coastguard Worker t_create_file_pattern(FNAME1, 4096, PAT1);
145*25da2beaSAndroid Build Coastguard Worker t_create_file_pattern(FNAME2, 4096, PAT2);
146*25da2beaSAndroid Build Coastguard Worker
147*25da2beaSAndroid Build Coastguard Worker ret = test(&ring);
148*25da2beaSAndroid Build Coastguard Worker if (ret) {
149*25da2beaSAndroid Build Coastguard Worker fprintf(stderr, "test failed\n");
150*25da2beaSAndroid Build Coastguard Worker goto err;
151*25da2beaSAndroid Build Coastguard Worker }
152*25da2beaSAndroid Build Coastguard Worker
153*25da2beaSAndroid Build Coastguard Worker unlink(FNAME1);
154*25da2beaSAndroid Build Coastguard Worker unlink(FNAME2);
155*25da2beaSAndroid Build Coastguard Worker return 0;
156*25da2beaSAndroid Build Coastguard Worker err:
157*25da2beaSAndroid Build Coastguard Worker unlink(FNAME1);
158*25da2beaSAndroid Build Coastguard Worker unlink(FNAME2);
159*25da2beaSAndroid Build Coastguard Worker return 1;
160*25da2beaSAndroid Build Coastguard Worker }
161