xref: /aosp_15_r20/external/liburing/test/iopoll.c (revision 25da2bea747f3a93b4c30fd9708b0618ef55a0e6)
1*25da2beaSAndroid Build Coastguard Worker /* SPDX-License-Identifier: MIT */
2*25da2beaSAndroid Build Coastguard Worker /*
3*25da2beaSAndroid Build Coastguard Worker  * Description: basic read/write tests with polled IO
4*25da2beaSAndroid Build Coastguard Worker  */
5*25da2beaSAndroid Build Coastguard Worker #include <errno.h>
6*25da2beaSAndroid Build Coastguard Worker #include <stdio.h>
7*25da2beaSAndroid Build Coastguard Worker #include <unistd.h>
8*25da2beaSAndroid Build Coastguard Worker #include <stdlib.h>
9*25da2beaSAndroid Build Coastguard Worker #include <string.h>
10*25da2beaSAndroid Build Coastguard Worker #include <fcntl.h>
11*25da2beaSAndroid Build Coastguard Worker #include <sys/types.h>
12*25da2beaSAndroid Build Coastguard Worker #include <poll.h>
13*25da2beaSAndroid Build Coastguard Worker #include <sys/eventfd.h>
14*25da2beaSAndroid Build Coastguard Worker #include <sys/resource.h>
15*25da2beaSAndroid Build Coastguard Worker #include "helpers.h"
16*25da2beaSAndroid Build Coastguard Worker #include "liburing.h"
17*25da2beaSAndroid Build Coastguard Worker #include "../src/syscall.h"
18*25da2beaSAndroid Build Coastguard Worker 
19*25da2beaSAndroid Build Coastguard Worker #define FILE_SIZE	(128 * 1024)
20*25da2beaSAndroid Build Coastguard Worker #define BS		4096
21*25da2beaSAndroid Build Coastguard Worker #define BUFFERS		(FILE_SIZE / BS)
22*25da2beaSAndroid Build Coastguard Worker 
23*25da2beaSAndroid Build Coastguard Worker static struct iovec *vecs;
24*25da2beaSAndroid Build Coastguard Worker static int no_buf_select;
25*25da2beaSAndroid Build Coastguard Worker static int no_iopoll;
26*25da2beaSAndroid Build Coastguard Worker 
provide_buffers(struct io_uring * ring)27*25da2beaSAndroid Build Coastguard Worker static int provide_buffers(struct io_uring *ring)
28*25da2beaSAndroid Build Coastguard Worker {
29*25da2beaSAndroid Build Coastguard Worker 	struct io_uring_sqe *sqe;
30*25da2beaSAndroid Build Coastguard Worker 	struct io_uring_cqe *cqe;
31*25da2beaSAndroid Build Coastguard Worker 	int ret, i;
32*25da2beaSAndroid Build Coastguard Worker 
33*25da2beaSAndroid Build Coastguard Worker 	for (i = 0; i < BUFFERS; i++) {
34*25da2beaSAndroid Build Coastguard Worker 		sqe = io_uring_get_sqe(ring);
35*25da2beaSAndroid Build Coastguard Worker 		io_uring_prep_provide_buffers(sqe, vecs[i].iov_base,
36*25da2beaSAndroid Build Coastguard Worker 						vecs[i].iov_len, 1, 1, i);
37*25da2beaSAndroid Build Coastguard Worker 	}
38*25da2beaSAndroid Build Coastguard Worker 
39*25da2beaSAndroid Build Coastguard Worker 	ret = io_uring_submit(ring);
40*25da2beaSAndroid Build Coastguard Worker 	if (ret != BUFFERS) {
41*25da2beaSAndroid Build Coastguard Worker 		fprintf(stderr, "submit: %d\n", ret);
42*25da2beaSAndroid Build Coastguard Worker 		return 1;
43*25da2beaSAndroid Build Coastguard Worker 	}
44*25da2beaSAndroid Build Coastguard Worker 
45*25da2beaSAndroid Build Coastguard Worker 	for (i = 0; i < BUFFERS; i++) {
46*25da2beaSAndroid Build Coastguard Worker 		ret = io_uring_wait_cqe(ring, &cqe);
47*25da2beaSAndroid Build Coastguard Worker 		if (cqe->res < 0) {
48*25da2beaSAndroid Build Coastguard Worker 			fprintf(stderr, "cqe->res=%d\n", cqe->res);
49*25da2beaSAndroid Build Coastguard Worker 			return 1;
50*25da2beaSAndroid Build Coastguard Worker 		}
51*25da2beaSAndroid Build Coastguard Worker 		io_uring_cqe_seen(ring, cqe);
52*25da2beaSAndroid Build Coastguard Worker 	}
53*25da2beaSAndroid Build Coastguard Worker 
54*25da2beaSAndroid Build Coastguard Worker 	return 0;
55*25da2beaSAndroid Build Coastguard Worker }
56*25da2beaSAndroid Build Coastguard Worker 
__test_io(const char * file,struct io_uring * ring,int write,int sqthread,int fixed,int buf_select)57*25da2beaSAndroid Build Coastguard Worker static int __test_io(const char *file, struct io_uring *ring, int write, int sqthread,
58*25da2beaSAndroid Build Coastguard Worker 		     int fixed, int buf_select)
59*25da2beaSAndroid Build Coastguard Worker {
60*25da2beaSAndroid Build Coastguard Worker 	struct io_uring_sqe *sqe;
61*25da2beaSAndroid Build Coastguard Worker 	struct io_uring_cqe *cqe;
62*25da2beaSAndroid Build Coastguard Worker 	int open_flags;
63*25da2beaSAndroid Build Coastguard Worker 	int i, fd = -1, ret;
64*25da2beaSAndroid Build Coastguard Worker 	off_t offset;
65*25da2beaSAndroid Build Coastguard Worker 
66*25da2beaSAndroid Build Coastguard Worker 	if (buf_select) {
67*25da2beaSAndroid Build Coastguard Worker 		write = 0;
68*25da2beaSAndroid Build Coastguard Worker 		fixed = 0;
69*25da2beaSAndroid Build Coastguard Worker 	}
70*25da2beaSAndroid Build Coastguard Worker 	if (buf_select && provide_buffers(ring))
71*25da2beaSAndroid Build Coastguard Worker 		return 1;
72*25da2beaSAndroid Build Coastguard Worker 
73*25da2beaSAndroid Build Coastguard Worker 	if (write)
74*25da2beaSAndroid Build Coastguard Worker 		open_flags = O_WRONLY;
75*25da2beaSAndroid Build Coastguard Worker 	else
76*25da2beaSAndroid Build Coastguard Worker 		open_flags = O_RDONLY;
77*25da2beaSAndroid Build Coastguard Worker 	open_flags |= O_DIRECT;
78*25da2beaSAndroid Build Coastguard Worker 
79*25da2beaSAndroid Build Coastguard Worker 	if (fixed) {
80*25da2beaSAndroid Build Coastguard Worker 		ret = t_register_buffers(ring, vecs, BUFFERS);
81*25da2beaSAndroid Build Coastguard Worker 		if (ret == T_SETUP_SKIP)
82*25da2beaSAndroid Build Coastguard Worker 			return 0;
83*25da2beaSAndroid Build Coastguard Worker 		if (ret != T_SETUP_OK) {
84*25da2beaSAndroid Build Coastguard Worker 			fprintf(stderr, "buffer reg failed: %d\n", ret);
85*25da2beaSAndroid Build Coastguard Worker 			goto err;
86*25da2beaSAndroid Build Coastguard Worker 		}
87*25da2beaSAndroid Build Coastguard Worker 	}
88*25da2beaSAndroid Build Coastguard Worker 	fd = open(file, open_flags);
89*25da2beaSAndroid Build Coastguard Worker 	if (fd < 0) {
90*25da2beaSAndroid Build Coastguard Worker 		perror("file open");
91*25da2beaSAndroid Build Coastguard Worker 		goto err;
92*25da2beaSAndroid Build Coastguard Worker 	}
93*25da2beaSAndroid Build Coastguard Worker 	if (sqthread) {
94*25da2beaSAndroid Build Coastguard Worker 		ret = io_uring_register_files(ring, &fd, 1);
95*25da2beaSAndroid Build Coastguard Worker 		if (ret) {
96*25da2beaSAndroid Build Coastguard Worker 			fprintf(stderr, "file reg failed: %d\n", ret);
97*25da2beaSAndroid Build Coastguard Worker 			goto err;
98*25da2beaSAndroid Build Coastguard Worker 		}
99*25da2beaSAndroid Build Coastguard Worker 	}
100*25da2beaSAndroid Build Coastguard Worker 
101*25da2beaSAndroid Build Coastguard Worker 	offset = 0;
102*25da2beaSAndroid Build Coastguard Worker 	for (i = 0; i < BUFFERS; i++) {
103*25da2beaSAndroid Build Coastguard Worker 		sqe = io_uring_get_sqe(ring);
104*25da2beaSAndroid Build Coastguard Worker 		if (!sqe) {
105*25da2beaSAndroid Build Coastguard Worker 			fprintf(stderr, "sqe get failed\n");
106*25da2beaSAndroid Build Coastguard Worker 			goto err;
107*25da2beaSAndroid Build Coastguard Worker 		}
108*25da2beaSAndroid Build Coastguard Worker 		offset = BS * (rand() % BUFFERS);
109*25da2beaSAndroid Build Coastguard Worker 		if (write) {
110*25da2beaSAndroid Build Coastguard Worker 			int do_fixed = fixed;
111*25da2beaSAndroid Build Coastguard Worker 			int use_fd = fd;
112*25da2beaSAndroid Build Coastguard Worker 
113*25da2beaSAndroid Build Coastguard Worker 			if (sqthread)
114*25da2beaSAndroid Build Coastguard Worker 				use_fd = 0;
115*25da2beaSAndroid Build Coastguard Worker 			if (fixed && (i & 1))
116*25da2beaSAndroid Build Coastguard Worker 				do_fixed = 0;
117*25da2beaSAndroid Build Coastguard Worker 			if (do_fixed) {
118*25da2beaSAndroid Build Coastguard Worker 				io_uring_prep_write_fixed(sqe, use_fd, vecs[i].iov_base,
119*25da2beaSAndroid Build Coastguard Worker 								vecs[i].iov_len,
120*25da2beaSAndroid Build Coastguard Worker 								offset, i);
121*25da2beaSAndroid Build Coastguard Worker 			} else {
122*25da2beaSAndroid Build Coastguard Worker 				io_uring_prep_writev(sqe, use_fd, &vecs[i], 1,
123*25da2beaSAndroid Build Coastguard Worker 								offset);
124*25da2beaSAndroid Build Coastguard Worker 			}
125*25da2beaSAndroid Build Coastguard Worker 		} else {
126*25da2beaSAndroid Build Coastguard Worker 			int do_fixed = fixed;
127*25da2beaSAndroid Build Coastguard Worker 			int use_fd = fd;
128*25da2beaSAndroid Build Coastguard Worker 
129*25da2beaSAndroid Build Coastguard Worker 			if (sqthread)
130*25da2beaSAndroid Build Coastguard Worker 				use_fd = 0;
131*25da2beaSAndroid Build Coastguard Worker 			if (fixed && (i & 1))
132*25da2beaSAndroid Build Coastguard Worker 				do_fixed = 0;
133*25da2beaSAndroid Build Coastguard Worker 			if (do_fixed) {
134*25da2beaSAndroid Build Coastguard Worker 				io_uring_prep_read_fixed(sqe, use_fd, vecs[i].iov_base,
135*25da2beaSAndroid Build Coastguard Worker 								vecs[i].iov_len,
136*25da2beaSAndroid Build Coastguard Worker 								offset, i);
137*25da2beaSAndroid Build Coastguard Worker 			} else {
138*25da2beaSAndroid Build Coastguard Worker 				io_uring_prep_readv(sqe, use_fd, &vecs[i], 1,
139*25da2beaSAndroid Build Coastguard Worker 								offset);
140*25da2beaSAndroid Build Coastguard Worker 			}
141*25da2beaSAndroid Build Coastguard Worker 
142*25da2beaSAndroid Build Coastguard Worker 		}
143*25da2beaSAndroid Build Coastguard Worker 		if (sqthread)
144*25da2beaSAndroid Build Coastguard Worker 			sqe->flags |= IOSQE_FIXED_FILE;
145*25da2beaSAndroid Build Coastguard Worker 		if (buf_select) {
146*25da2beaSAndroid Build Coastguard Worker 			sqe->flags |= IOSQE_BUFFER_SELECT;
147*25da2beaSAndroid Build Coastguard Worker 			sqe->buf_group = buf_select;
148*25da2beaSAndroid Build Coastguard Worker 			sqe->user_data = i;
149*25da2beaSAndroid Build Coastguard Worker 		}
150*25da2beaSAndroid Build Coastguard Worker 	}
151*25da2beaSAndroid Build Coastguard Worker 
152*25da2beaSAndroid Build Coastguard Worker 	ret = io_uring_submit(ring);
153*25da2beaSAndroid Build Coastguard Worker 	if (ret != BUFFERS) {
154*25da2beaSAndroid Build Coastguard Worker 		ret = io_uring_peek_cqe(ring, &cqe);
155*25da2beaSAndroid Build Coastguard Worker 		if (!ret && cqe->res == -EOPNOTSUPP) {
156*25da2beaSAndroid Build Coastguard Worker 			no_iopoll = 1;
157*25da2beaSAndroid Build Coastguard Worker 			io_uring_cqe_seen(ring, cqe);
158*25da2beaSAndroid Build Coastguard Worker 			goto out;
159*25da2beaSAndroid Build Coastguard Worker 		}
160*25da2beaSAndroid Build Coastguard Worker 		fprintf(stderr, "submit got %d, wanted %d\n", ret, BUFFERS);
161*25da2beaSAndroid Build Coastguard Worker 		goto err;
162*25da2beaSAndroid Build Coastguard Worker 	}
163*25da2beaSAndroid Build Coastguard Worker 
164*25da2beaSAndroid Build Coastguard Worker 	for (i = 0; i < BUFFERS; i++) {
165*25da2beaSAndroid Build Coastguard Worker 		ret = io_uring_wait_cqe(ring, &cqe);
166*25da2beaSAndroid Build Coastguard Worker 		if (ret) {
167*25da2beaSAndroid Build Coastguard Worker 			fprintf(stderr, "wait_cqe=%d\n", ret);
168*25da2beaSAndroid Build Coastguard Worker 			goto err;
169*25da2beaSAndroid Build Coastguard Worker 		} else if (cqe->res == -EOPNOTSUPP) {
170*25da2beaSAndroid Build Coastguard Worker 			fprintf(stdout, "File/device/fs doesn't support polled IO\n");
171*25da2beaSAndroid Build Coastguard Worker 			no_iopoll = 1;
172*25da2beaSAndroid Build Coastguard Worker 			goto out;
173*25da2beaSAndroid Build Coastguard Worker 		} else if (cqe->res != BS) {
174*25da2beaSAndroid Build Coastguard Worker 			fprintf(stderr, "cqe res %d, wanted %d\n", cqe->res, BS);
175*25da2beaSAndroid Build Coastguard Worker 			goto err;
176*25da2beaSAndroid Build Coastguard Worker 		}
177*25da2beaSAndroid Build Coastguard Worker 		io_uring_cqe_seen(ring, cqe);
178*25da2beaSAndroid Build Coastguard Worker 	}
179*25da2beaSAndroid Build Coastguard Worker 
180*25da2beaSAndroid Build Coastguard Worker 	if (fixed) {
181*25da2beaSAndroid Build Coastguard Worker 		ret = io_uring_unregister_buffers(ring);
182*25da2beaSAndroid Build Coastguard Worker 		if (ret) {
183*25da2beaSAndroid Build Coastguard Worker 			fprintf(stderr, "buffer unreg failed: %d\n", ret);
184*25da2beaSAndroid Build Coastguard Worker 			goto err;
185*25da2beaSAndroid Build Coastguard Worker 		}
186*25da2beaSAndroid Build Coastguard Worker 	}
187*25da2beaSAndroid Build Coastguard Worker 	if (sqthread) {
188*25da2beaSAndroid Build Coastguard Worker 		ret = io_uring_unregister_files(ring);
189*25da2beaSAndroid Build Coastguard Worker 		if (ret) {
190*25da2beaSAndroid Build Coastguard Worker 			fprintf(stderr, "file unreg failed: %d\n", ret);
191*25da2beaSAndroid Build Coastguard Worker 			goto err;
192*25da2beaSAndroid Build Coastguard Worker 		}
193*25da2beaSAndroid Build Coastguard Worker 	}
194*25da2beaSAndroid Build Coastguard Worker 
195*25da2beaSAndroid Build Coastguard Worker out:
196*25da2beaSAndroid Build Coastguard Worker 	close(fd);
197*25da2beaSAndroid Build Coastguard Worker 	return 0;
198*25da2beaSAndroid Build Coastguard Worker err:
199*25da2beaSAndroid Build Coastguard Worker 	if (fd != -1)
200*25da2beaSAndroid Build Coastguard Worker 		close(fd);
201*25da2beaSAndroid Build Coastguard Worker 	return 1;
202*25da2beaSAndroid Build Coastguard Worker }
203*25da2beaSAndroid Build Coastguard Worker 
204*25da2beaSAndroid Build Coastguard Worker extern int __io_uring_flush_sq(struct io_uring *ring);
205*25da2beaSAndroid Build Coastguard Worker 
206*25da2beaSAndroid Build Coastguard Worker /*
207*25da2beaSAndroid Build Coastguard Worker  * if we are polling io_uring_submit needs to always enter the
208*25da2beaSAndroid Build Coastguard Worker  * kernel to fetch events
209*25da2beaSAndroid Build Coastguard Worker  */
test_io_uring_submit_enters(const char * file)210*25da2beaSAndroid Build Coastguard Worker static int test_io_uring_submit_enters(const char *file)
211*25da2beaSAndroid Build Coastguard Worker {
212*25da2beaSAndroid Build Coastguard Worker 	struct io_uring ring;
213*25da2beaSAndroid Build Coastguard Worker 	int fd, i, ret, ring_flags, open_flags;
214*25da2beaSAndroid Build Coastguard Worker 	unsigned head;
215*25da2beaSAndroid Build Coastguard Worker 	struct io_uring_cqe *cqe;
216*25da2beaSAndroid Build Coastguard Worker 
217*25da2beaSAndroid Build Coastguard Worker 	if (no_iopoll)
218*25da2beaSAndroid Build Coastguard Worker 		return 0;
219*25da2beaSAndroid Build Coastguard Worker 
220*25da2beaSAndroid Build Coastguard Worker 	ring_flags = IORING_SETUP_IOPOLL;
221*25da2beaSAndroid Build Coastguard Worker 	ret = io_uring_queue_init(64, &ring, ring_flags);
222*25da2beaSAndroid Build Coastguard Worker 	if (ret) {
223*25da2beaSAndroid Build Coastguard Worker 		fprintf(stderr, "ring create failed: %d\n", ret);
224*25da2beaSAndroid Build Coastguard Worker 		return 1;
225*25da2beaSAndroid Build Coastguard Worker 	}
226*25da2beaSAndroid Build Coastguard Worker 
227*25da2beaSAndroid Build Coastguard Worker 	open_flags = O_WRONLY | O_DIRECT;
228*25da2beaSAndroid Build Coastguard Worker 	fd = open(file, open_flags);
229*25da2beaSAndroid Build Coastguard Worker 	if (fd < 0) {
230*25da2beaSAndroid Build Coastguard Worker 		perror("file open");
231*25da2beaSAndroid Build Coastguard Worker 		goto err;
232*25da2beaSAndroid Build Coastguard Worker 	}
233*25da2beaSAndroid Build Coastguard Worker 
234*25da2beaSAndroid Build Coastguard Worker 	for (i = 0; i < BUFFERS; i++) {
235*25da2beaSAndroid Build Coastguard Worker 		struct io_uring_sqe *sqe;
236*25da2beaSAndroid Build Coastguard Worker 		off_t offset = BS * (rand() % BUFFERS);
237*25da2beaSAndroid Build Coastguard Worker 
238*25da2beaSAndroid Build Coastguard Worker 		sqe = io_uring_get_sqe(&ring);
239*25da2beaSAndroid Build Coastguard Worker 		io_uring_prep_writev(sqe, fd, &vecs[i], 1, offset);
240*25da2beaSAndroid Build Coastguard Worker 		sqe->user_data = 1;
241*25da2beaSAndroid Build Coastguard Worker 	}
242*25da2beaSAndroid Build Coastguard Worker 
243*25da2beaSAndroid Build Coastguard Worker 	/* submit manually to avoid adding IORING_ENTER_GETEVENTS */
244*25da2beaSAndroid Build Coastguard Worker 	ret = __sys_io_uring_enter(ring.ring_fd, __io_uring_flush_sq(&ring), 0,
245*25da2beaSAndroid Build Coastguard Worker 						0, NULL);
246*25da2beaSAndroid Build Coastguard Worker 	if (ret < 0)
247*25da2beaSAndroid Build Coastguard Worker 		goto err;
248*25da2beaSAndroid Build Coastguard Worker 
249*25da2beaSAndroid Build Coastguard Worker 	for (i = 0; i < 500; i++) {
250*25da2beaSAndroid Build Coastguard Worker 		ret = io_uring_submit(&ring);
251*25da2beaSAndroid Build Coastguard Worker 		if (ret != 0) {
252*25da2beaSAndroid Build Coastguard Worker 			fprintf(stderr, "still had %d sqes to submit, this is unexpected", ret);
253*25da2beaSAndroid Build Coastguard Worker 			goto err;
254*25da2beaSAndroid Build Coastguard Worker 		}
255*25da2beaSAndroid Build Coastguard Worker 
256*25da2beaSAndroid Build Coastguard Worker 		io_uring_for_each_cqe(&ring, head, cqe) {
257*25da2beaSAndroid Build Coastguard Worker 			/* runs after test_io so should not have happened */
258*25da2beaSAndroid Build Coastguard Worker 			if (cqe->res == -EOPNOTSUPP) {
259*25da2beaSAndroid Build Coastguard Worker 				fprintf(stdout, "File/device/fs doesn't support polled IO\n");
260*25da2beaSAndroid Build Coastguard Worker 				goto err;
261*25da2beaSAndroid Build Coastguard Worker 			}
262*25da2beaSAndroid Build Coastguard Worker 			goto ok;
263*25da2beaSAndroid Build Coastguard Worker 		}
264*25da2beaSAndroid Build Coastguard Worker 		usleep(10000);
265*25da2beaSAndroid Build Coastguard Worker 	}
266*25da2beaSAndroid Build Coastguard Worker err:
267*25da2beaSAndroid Build Coastguard Worker 	ret = 1;
268*25da2beaSAndroid Build Coastguard Worker 	if (fd != -1)
269*25da2beaSAndroid Build Coastguard Worker 		close(fd);
270*25da2beaSAndroid Build Coastguard Worker 
271*25da2beaSAndroid Build Coastguard Worker ok:
272*25da2beaSAndroid Build Coastguard Worker 	io_uring_queue_exit(&ring);
273*25da2beaSAndroid Build Coastguard Worker 	return ret;
274*25da2beaSAndroid Build Coastguard Worker }
275*25da2beaSAndroid Build Coastguard Worker 
test_io(const char * file,int write,int sqthread,int fixed,int buf_select)276*25da2beaSAndroid Build Coastguard Worker static int test_io(const char *file, int write, int sqthread, int fixed,
277*25da2beaSAndroid Build Coastguard Worker 		   int buf_select)
278*25da2beaSAndroid Build Coastguard Worker {
279*25da2beaSAndroid Build Coastguard Worker 	struct io_uring ring;
280*25da2beaSAndroid Build Coastguard Worker 	int ret, ring_flags = IORING_SETUP_IOPOLL;
281*25da2beaSAndroid Build Coastguard Worker 
282*25da2beaSAndroid Build Coastguard Worker 	if (no_iopoll)
283*25da2beaSAndroid Build Coastguard Worker 		return 0;
284*25da2beaSAndroid Build Coastguard Worker 
285*25da2beaSAndroid Build Coastguard Worker 	ret = t_create_ring(64, &ring, ring_flags);
286*25da2beaSAndroid Build Coastguard Worker 	if (ret == T_SETUP_SKIP)
287*25da2beaSAndroid Build Coastguard Worker 		return 0;
288*25da2beaSAndroid Build Coastguard Worker 	if (ret != T_SETUP_OK) {
289*25da2beaSAndroid Build Coastguard Worker 		fprintf(stderr, "ring create failed: %d\n", ret);
290*25da2beaSAndroid Build Coastguard Worker 		return 1;
291*25da2beaSAndroid Build Coastguard Worker 	}
292*25da2beaSAndroid Build Coastguard Worker 	ret = __test_io(file, &ring, write, sqthread, fixed, buf_select);
293*25da2beaSAndroid Build Coastguard Worker 	io_uring_queue_exit(&ring);
294*25da2beaSAndroid Build Coastguard Worker 	return ret;
295*25da2beaSAndroid Build Coastguard Worker }
296*25da2beaSAndroid Build Coastguard Worker 
probe_buf_select(void)297*25da2beaSAndroid Build Coastguard Worker static int probe_buf_select(void)
298*25da2beaSAndroid Build Coastguard Worker {
299*25da2beaSAndroid Build Coastguard Worker 	struct io_uring_probe *p;
300*25da2beaSAndroid Build Coastguard Worker 	struct io_uring ring;
301*25da2beaSAndroid Build Coastguard Worker 	int ret;
302*25da2beaSAndroid Build Coastguard Worker 
303*25da2beaSAndroid Build Coastguard Worker 	ret = io_uring_queue_init(1, &ring, 0);
304*25da2beaSAndroid Build Coastguard Worker 	if (ret) {
305*25da2beaSAndroid Build Coastguard Worker 		fprintf(stderr, "ring create failed: %d\n", ret);
306*25da2beaSAndroid Build Coastguard Worker 		return 1;
307*25da2beaSAndroid Build Coastguard Worker 	}
308*25da2beaSAndroid Build Coastguard Worker 
309*25da2beaSAndroid Build Coastguard Worker 	p = io_uring_get_probe_ring(&ring);
310*25da2beaSAndroid Build Coastguard Worker 	if (!p || !io_uring_opcode_supported(p, IORING_OP_PROVIDE_BUFFERS)) {
311*25da2beaSAndroid Build Coastguard Worker 		no_buf_select = 1;
312*25da2beaSAndroid Build Coastguard Worker 		fprintf(stdout, "Buffer select not supported, skipping\n");
313*25da2beaSAndroid Build Coastguard Worker 		return 0;
314*25da2beaSAndroid Build Coastguard Worker 	}
315*25da2beaSAndroid Build Coastguard Worker 	io_uring_free_probe(p);
316*25da2beaSAndroid Build Coastguard Worker 	return 0;
317*25da2beaSAndroid Build Coastguard Worker }
318*25da2beaSAndroid Build Coastguard Worker 
main(int argc,char * argv[])319*25da2beaSAndroid Build Coastguard Worker int main(int argc, char *argv[])
320*25da2beaSAndroid Build Coastguard Worker {
321*25da2beaSAndroid Build Coastguard Worker 	int i, ret, nr;
322*25da2beaSAndroid Build Coastguard Worker 	char buf[256];
323*25da2beaSAndroid Build Coastguard Worker 	char *fname;
324*25da2beaSAndroid Build Coastguard Worker 
325*25da2beaSAndroid Build Coastguard Worker 	if (probe_buf_select())
326*25da2beaSAndroid Build Coastguard Worker 		return 1;
327*25da2beaSAndroid Build Coastguard Worker 
328*25da2beaSAndroid Build Coastguard Worker 	if (argc > 1) {
329*25da2beaSAndroid Build Coastguard Worker 		fname = argv[1];
330*25da2beaSAndroid Build Coastguard Worker 	} else {
331*25da2beaSAndroid Build Coastguard Worker 		srand((unsigned)time(NULL));
332*25da2beaSAndroid Build Coastguard Worker 		snprintf(buf, sizeof(buf), ".basic-rw-%u-%u",
333*25da2beaSAndroid Build Coastguard Worker 			(unsigned)rand(), (unsigned)getpid());
334*25da2beaSAndroid Build Coastguard Worker 		fname = buf;
335*25da2beaSAndroid Build Coastguard Worker 		t_create_file(fname, FILE_SIZE);
336*25da2beaSAndroid Build Coastguard Worker 	}
337*25da2beaSAndroid Build Coastguard Worker 
338*25da2beaSAndroid Build Coastguard Worker 	vecs = t_create_buffers(BUFFERS, BS);
339*25da2beaSAndroid Build Coastguard Worker 
340*25da2beaSAndroid Build Coastguard Worker 	nr = 16;
341*25da2beaSAndroid Build Coastguard Worker 	if (no_buf_select)
342*25da2beaSAndroid Build Coastguard Worker 		nr = 8;
343*25da2beaSAndroid Build Coastguard Worker 	for (i = 0; i < nr; i++) {
344*25da2beaSAndroid Build Coastguard Worker 		int write = (i & 1) != 0;
345*25da2beaSAndroid Build Coastguard Worker 		int sqthread = (i & 2) != 0;
346*25da2beaSAndroid Build Coastguard Worker 		int fixed = (i & 4) != 0;
347*25da2beaSAndroid Build Coastguard Worker 		int buf_select = (i & 8) != 0;
348*25da2beaSAndroid Build Coastguard Worker 
349*25da2beaSAndroid Build Coastguard Worker 		ret = test_io(fname, write, sqthread, fixed, buf_select);
350*25da2beaSAndroid Build Coastguard Worker 		if (ret) {
351*25da2beaSAndroid Build Coastguard Worker 			fprintf(stderr, "test_io failed %d/%d/%d/%d\n",
352*25da2beaSAndroid Build Coastguard Worker 				write, sqthread, fixed, buf_select);
353*25da2beaSAndroid Build Coastguard Worker 			goto err;
354*25da2beaSAndroid Build Coastguard Worker 		}
355*25da2beaSAndroid Build Coastguard Worker 		if (no_iopoll)
356*25da2beaSAndroid Build Coastguard Worker 			break;
357*25da2beaSAndroid Build Coastguard Worker 	}
358*25da2beaSAndroid Build Coastguard Worker 
359*25da2beaSAndroid Build Coastguard Worker 	ret = test_io_uring_submit_enters(fname);
360*25da2beaSAndroid Build Coastguard Worker 	if (ret) {
361*25da2beaSAndroid Build Coastguard Worker 	    fprintf(stderr, "test_io_uring_submit_enters failed\n");
362*25da2beaSAndroid Build Coastguard Worker 	    goto err;
363*25da2beaSAndroid Build Coastguard Worker 	}
364*25da2beaSAndroid Build Coastguard Worker 
365*25da2beaSAndroid Build Coastguard Worker 	if (fname != argv[1])
366*25da2beaSAndroid Build Coastguard Worker 		unlink(fname);
367*25da2beaSAndroid Build Coastguard Worker 	return 0;
368*25da2beaSAndroid Build Coastguard Worker err:
369*25da2beaSAndroid Build Coastguard Worker 	if (fname != argv[1])
370*25da2beaSAndroid Build Coastguard Worker 		unlink(fname);
371*25da2beaSAndroid Build Coastguard Worker 	return 1;
372*25da2beaSAndroid Build Coastguard Worker }
373