1*25da2beaSAndroid Build Coastguard Worker /* SPDX-License-Identifier: MIT */
2*25da2beaSAndroid Build Coastguard Worker /*
3*25da2beaSAndroid Build Coastguard Worker * Description: test massive amounts of poll with cancel
4*25da2beaSAndroid Build Coastguard Worker *
5*25da2beaSAndroid Build Coastguard Worker */
6*25da2beaSAndroid Build Coastguard Worker #include <errno.h>
7*25da2beaSAndroid Build Coastguard Worker #include <stdio.h>
8*25da2beaSAndroid Build Coastguard Worker #include <unistd.h>
9*25da2beaSAndroid Build Coastguard Worker #include <stdlib.h>
10*25da2beaSAndroid Build Coastguard Worker #include <string.h>
11*25da2beaSAndroid Build Coastguard Worker #include <inttypes.h>
12*25da2beaSAndroid Build Coastguard Worker #include <poll.h>
13*25da2beaSAndroid Build Coastguard Worker #include <sys/wait.h>
14*25da2beaSAndroid Build Coastguard Worker #include <signal.h>
15*25da2beaSAndroid Build Coastguard Worker
16*25da2beaSAndroid Build Coastguard Worker #include "liburing.h"
17*25da2beaSAndroid Build Coastguard Worker
18*25da2beaSAndroid Build Coastguard Worker #define POLL_COUNT 30000
19*25da2beaSAndroid Build Coastguard Worker
20*25da2beaSAndroid Build Coastguard Worker static void *sqe_index[POLL_COUNT];
21*25da2beaSAndroid Build Coastguard Worker
reap_events(struct io_uring * ring,unsigned nr_events,int nowait)22*25da2beaSAndroid Build Coastguard Worker static int reap_events(struct io_uring *ring, unsigned nr_events, int nowait)
23*25da2beaSAndroid Build Coastguard Worker {
24*25da2beaSAndroid Build Coastguard Worker struct io_uring_cqe *cqe;
25*25da2beaSAndroid Build Coastguard Worker int i, ret = 0;
26*25da2beaSAndroid Build Coastguard Worker
27*25da2beaSAndroid Build Coastguard Worker for (i = 0; i < nr_events; i++) {
28*25da2beaSAndroid Build Coastguard Worker if (!i && !nowait)
29*25da2beaSAndroid Build Coastguard Worker ret = io_uring_wait_cqe(ring, &cqe);
30*25da2beaSAndroid Build Coastguard Worker else
31*25da2beaSAndroid Build Coastguard Worker ret = io_uring_peek_cqe(ring, &cqe);
32*25da2beaSAndroid Build Coastguard Worker if (ret) {
33*25da2beaSAndroid Build Coastguard Worker if (ret != -EAGAIN)
34*25da2beaSAndroid Build Coastguard Worker fprintf(stderr, "cqe peek failed: %d\n", ret);
35*25da2beaSAndroid Build Coastguard Worker break;
36*25da2beaSAndroid Build Coastguard Worker }
37*25da2beaSAndroid Build Coastguard Worker io_uring_cqe_seen(ring, cqe);
38*25da2beaSAndroid Build Coastguard Worker }
39*25da2beaSAndroid Build Coastguard Worker
40*25da2beaSAndroid Build Coastguard Worker return i ? i : ret;
41*25da2beaSAndroid Build Coastguard Worker }
42*25da2beaSAndroid Build Coastguard Worker
del_polls(struct io_uring * ring,int fd,int nr)43*25da2beaSAndroid Build Coastguard Worker static int del_polls(struct io_uring *ring, int fd, int nr)
44*25da2beaSAndroid Build Coastguard Worker {
45*25da2beaSAndroid Build Coastguard Worker int batch, i, ret;
46*25da2beaSAndroid Build Coastguard Worker struct io_uring_sqe *sqe;
47*25da2beaSAndroid Build Coastguard Worker
48*25da2beaSAndroid Build Coastguard Worker while (nr) {
49*25da2beaSAndroid Build Coastguard Worker batch = 1024;
50*25da2beaSAndroid Build Coastguard Worker if (batch > nr)
51*25da2beaSAndroid Build Coastguard Worker batch = nr;
52*25da2beaSAndroid Build Coastguard Worker
53*25da2beaSAndroid Build Coastguard Worker for (i = 0; i < batch; i++) {
54*25da2beaSAndroid Build Coastguard Worker void *data;
55*25da2beaSAndroid Build Coastguard Worker
56*25da2beaSAndroid Build Coastguard Worker sqe = io_uring_get_sqe(ring);
57*25da2beaSAndroid Build Coastguard Worker data = sqe_index[lrand48() % nr];
58*25da2beaSAndroid Build Coastguard Worker io_uring_prep_poll_remove(sqe, (__u64)(uintptr_t)data);
59*25da2beaSAndroid Build Coastguard Worker }
60*25da2beaSAndroid Build Coastguard Worker
61*25da2beaSAndroid Build Coastguard Worker ret = io_uring_submit(ring);
62*25da2beaSAndroid Build Coastguard Worker if (ret != batch) {
63*25da2beaSAndroid Build Coastguard Worker fprintf(stderr, "%s: failed submit, %d\n", __FUNCTION__, ret);
64*25da2beaSAndroid Build Coastguard Worker return 1;
65*25da2beaSAndroid Build Coastguard Worker }
66*25da2beaSAndroid Build Coastguard Worker nr -= batch;
67*25da2beaSAndroid Build Coastguard Worker ret = reap_events(ring, 2 * batch, 0);
68*25da2beaSAndroid Build Coastguard Worker }
69*25da2beaSAndroid Build Coastguard Worker return 0;
70*25da2beaSAndroid Build Coastguard Worker }
71*25da2beaSAndroid Build Coastguard Worker
add_polls(struct io_uring * ring,int fd,int nr)72*25da2beaSAndroid Build Coastguard Worker static int add_polls(struct io_uring *ring, int fd, int nr)
73*25da2beaSAndroid Build Coastguard Worker {
74*25da2beaSAndroid Build Coastguard Worker int batch, i, count, ret;
75*25da2beaSAndroid Build Coastguard Worker struct io_uring_sqe *sqe;
76*25da2beaSAndroid Build Coastguard Worker
77*25da2beaSAndroid Build Coastguard Worker count = 0;
78*25da2beaSAndroid Build Coastguard Worker while (nr) {
79*25da2beaSAndroid Build Coastguard Worker batch = 1024;
80*25da2beaSAndroid Build Coastguard Worker if (batch > nr)
81*25da2beaSAndroid Build Coastguard Worker batch = nr;
82*25da2beaSAndroid Build Coastguard Worker
83*25da2beaSAndroid Build Coastguard Worker for (i = 0; i < batch; i++) {
84*25da2beaSAndroid Build Coastguard Worker sqe = io_uring_get_sqe(ring);
85*25da2beaSAndroid Build Coastguard Worker io_uring_prep_poll_add(sqe, fd, POLLIN);
86*25da2beaSAndroid Build Coastguard Worker sqe_index[count++] = sqe;
87*25da2beaSAndroid Build Coastguard Worker sqe->user_data = (unsigned long) sqe;
88*25da2beaSAndroid Build Coastguard Worker }
89*25da2beaSAndroid Build Coastguard Worker
90*25da2beaSAndroid Build Coastguard Worker ret = io_uring_submit(ring);
91*25da2beaSAndroid Build Coastguard Worker if (ret != batch) {
92*25da2beaSAndroid Build Coastguard Worker fprintf(stderr, "%s: failed submit, %d\n", __FUNCTION__, ret);
93*25da2beaSAndroid Build Coastguard Worker return 1;
94*25da2beaSAndroid Build Coastguard Worker }
95*25da2beaSAndroid Build Coastguard Worker nr -= batch;
96*25da2beaSAndroid Build Coastguard Worker reap_events(ring, batch, 1);
97*25da2beaSAndroid Build Coastguard Worker }
98*25da2beaSAndroid Build Coastguard Worker return 0;
99*25da2beaSAndroid Build Coastguard Worker }
100*25da2beaSAndroid Build Coastguard Worker
main(int argc,char * argv[])101*25da2beaSAndroid Build Coastguard Worker int main(int argc, char *argv[])
102*25da2beaSAndroid Build Coastguard Worker {
103*25da2beaSAndroid Build Coastguard Worker struct io_uring ring;
104*25da2beaSAndroid Build Coastguard Worker struct io_uring_params p = { };
105*25da2beaSAndroid Build Coastguard Worker int pipe1[2];
106*25da2beaSAndroid Build Coastguard Worker int ret;
107*25da2beaSAndroid Build Coastguard Worker
108*25da2beaSAndroid Build Coastguard Worker if (argc > 1)
109*25da2beaSAndroid Build Coastguard Worker return 0;
110*25da2beaSAndroid Build Coastguard Worker
111*25da2beaSAndroid Build Coastguard Worker if (pipe(pipe1) != 0) {
112*25da2beaSAndroid Build Coastguard Worker perror("pipe");
113*25da2beaSAndroid Build Coastguard Worker return 1;
114*25da2beaSAndroid Build Coastguard Worker }
115*25da2beaSAndroid Build Coastguard Worker
116*25da2beaSAndroid Build Coastguard Worker p.flags = IORING_SETUP_CQSIZE;
117*25da2beaSAndroid Build Coastguard Worker p.cq_entries = 16384;
118*25da2beaSAndroid Build Coastguard Worker ret = io_uring_queue_init_params(1024, &ring, &p);
119*25da2beaSAndroid Build Coastguard Worker if (ret) {
120*25da2beaSAndroid Build Coastguard Worker if (ret == -EINVAL) {
121*25da2beaSAndroid Build Coastguard Worker fprintf(stdout, "No CQSIZE, trying without\n");
122*25da2beaSAndroid Build Coastguard Worker ret = io_uring_queue_init(1024, &ring, 0);
123*25da2beaSAndroid Build Coastguard Worker if (ret) {
124*25da2beaSAndroid Build Coastguard Worker fprintf(stderr, "ring setup failed: %d\n", ret);
125*25da2beaSAndroid Build Coastguard Worker return 1;
126*25da2beaSAndroid Build Coastguard Worker }
127*25da2beaSAndroid Build Coastguard Worker }
128*25da2beaSAndroid Build Coastguard Worker }
129*25da2beaSAndroid Build Coastguard Worker
130*25da2beaSAndroid Build Coastguard Worker add_polls(&ring, pipe1[0], 30000);
131*25da2beaSAndroid Build Coastguard Worker del_polls(&ring, pipe1[0], 30000);
132*25da2beaSAndroid Build Coastguard Worker
133*25da2beaSAndroid Build Coastguard Worker io_uring_queue_exit(&ring);
134*25da2beaSAndroid Build Coastguard Worker return 0;
135*25da2beaSAndroid Build Coastguard Worker }
136