xref: /aosp_15_r20/external/liburing/test/wakeup-hang.c (revision 25da2bea747f3a93b4c30fd9708b0618ef55a0e6)
1*25da2beaSAndroid Build Coastguard Worker /* SPDX-License-Identifier: MIT */
2*25da2beaSAndroid Build Coastguard Worker #include <sys/eventfd.h>
3*25da2beaSAndroid Build Coastguard Worker #include <unistd.h>
4*25da2beaSAndroid Build Coastguard Worker #include <stdio.h>
5*25da2beaSAndroid Build Coastguard Worker #include <stdlib.h>
6*25da2beaSAndroid Build Coastguard Worker #include <string.h>
7*25da2beaSAndroid Build Coastguard Worker #include <pthread.h>
8*25da2beaSAndroid Build Coastguard Worker #include <liburing.h>
9*25da2beaSAndroid Build Coastguard Worker #include <fcntl.h>
10*25da2beaSAndroid Build Coastguard Worker #include <poll.h>
11*25da2beaSAndroid Build Coastguard Worker #include <sys/time.h>
12*25da2beaSAndroid Build Coastguard Worker 
13*25da2beaSAndroid Build Coastguard Worker struct thread_data {
14*25da2beaSAndroid Build Coastguard Worker 	struct io_uring *ring;
15*25da2beaSAndroid Build Coastguard Worker 	int write_fd;
16*25da2beaSAndroid Build Coastguard Worker };
17*25da2beaSAndroid Build Coastguard Worker 
error_exit(char * message)18*25da2beaSAndroid Build Coastguard Worker static void error_exit(char *message)
19*25da2beaSAndroid Build Coastguard Worker {
20*25da2beaSAndroid Build Coastguard Worker 	perror(message);
21*25da2beaSAndroid Build Coastguard Worker 	exit(1);
22*25da2beaSAndroid Build Coastguard Worker }
23*25da2beaSAndroid Build Coastguard Worker 
listener_thread(void * data)24*25da2beaSAndroid Build Coastguard Worker static void *listener_thread(void *data)
25*25da2beaSAndroid Build Coastguard Worker {
26*25da2beaSAndroid Build Coastguard Worker 	struct thread_data *td = data;
27*25da2beaSAndroid Build Coastguard Worker 	struct io_uring_cqe *cqe;
28*25da2beaSAndroid Build Coastguard Worker 	int ret;
29*25da2beaSAndroid Build Coastguard Worker 
30*25da2beaSAndroid Build Coastguard Worker         ret = io_uring_wait_cqe(td->ring, &cqe);
31*25da2beaSAndroid Build Coastguard Worker         if (ret < 0) {
32*25da2beaSAndroid Build Coastguard Worker         	fprintf(stderr, "Error waiting for completion: %s\n",
33*25da2beaSAndroid Build Coastguard Worker                 	strerror(-ret));
34*25da2beaSAndroid Build Coastguard Worker 		goto err;
35*25da2beaSAndroid Build Coastguard Worker         }
36*25da2beaSAndroid Build Coastguard Worker 	if (cqe->res < 0) {
37*25da2beaSAndroid Build Coastguard Worker 		fprintf(stderr, "Error in async operation: %s\n", strerror(-cqe->res));
38*25da2beaSAndroid Build Coastguard Worker 		goto err;
39*25da2beaSAndroid Build Coastguard Worker         }
40*25da2beaSAndroid Build Coastguard Worker 	io_uring_cqe_seen(td->ring, cqe);
41*25da2beaSAndroid Build Coastguard Worker 	return NULL;
42*25da2beaSAndroid Build Coastguard Worker err:
43*25da2beaSAndroid Build Coastguard Worker 	return (void *) 1;
44*25da2beaSAndroid Build Coastguard Worker }
45*25da2beaSAndroid Build Coastguard Worker 
wakeup_io_uring(void * data)46*25da2beaSAndroid Build Coastguard Worker static void *wakeup_io_uring(void *data)
47*25da2beaSAndroid Build Coastguard Worker {
48*25da2beaSAndroid Build Coastguard Worker 	struct thread_data *td = data;
49*25da2beaSAndroid Build Coastguard Worker 	int res;
50*25da2beaSAndroid Build Coastguard Worker 
51*25da2beaSAndroid Build Coastguard Worker 	res = eventfd_write(td->write_fd, (eventfd_t) 1L);
52*25da2beaSAndroid Build Coastguard Worker 	if (res < 0) {
53*25da2beaSAndroid Build Coastguard Worker 		perror("eventfd_write");
54*25da2beaSAndroid Build Coastguard Worker 		return (void *) 1;
55*25da2beaSAndroid Build Coastguard Worker 	}
56*25da2beaSAndroid Build Coastguard Worker 	return NULL;
57*25da2beaSAndroid Build Coastguard Worker }
58*25da2beaSAndroid Build Coastguard Worker 
test_pipes(void)59*25da2beaSAndroid Build Coastguard Worker static int test_pipes(void)
60*25da2beaSAndroid Build Coastguard Worker {
61*25da2beaSAndroid Build Coastguard Worker 	struct io_uring_sqe *sqe;
62*25da2beaSAndroid Build Coastguard Worker 	struct thread_data td;
63*25da2beaSAndroid Build Coastguard Worker 	struct io_uring ring;
64*25da2beaSAndroid Build Coastguard Worker 	pthread_t t1, t2;
65*25da2beaSAndroid Build Coastguard Worker 	int ret, fds[2];
66*25da2beaSAndroid Build Coastguard Worker 	void *pret;
67*25da2beaSAndroid Build Coastguard Worker 
68*25da2beaSAndroid Build Coastguard Worker 	if (pipe(fds) < 0)
69*25da2beaSAndroid Build Coastguard Worker 		error_exit("eventfd");
70*25da2beaSAndroid Build Coastguard Worker 
71*25da2beaSAndroid Build Coastguard Worker 	ret = io_uring_queue_init(8, &ring, 0);
72*25da2beaSAndroid Build Coastguard Worker 	if (ret) {
73*25da2beaSAndroid Build Coastguard Worker 		fprintf(stderr, "Unable to setup io_uring: %s\n", strerror(-ret));
74*25da2beaSAndroid Build Coastguard Worker 		return 1;
75*25da2beaSAndroid Build Coastguard Worker 	}
76*25da2beaSAndroid Build Coastguard Worker 
77*25da2beaSAndroid Build Coastguard Worker 	td.write_fd = fds[1];
78*25da2beaSAndroid Build Coastguard Worker 	td.ring = &ring;
79*25da2beaSAndroid Build Coastguard Worker 
80*25da2beaSAndroid Build Coastguard Worker 	sqe = io_uring_get_sqe(&ring);
81*25da2beaSAndroid Build Coastguard Worker 	io_uring_prep_poll_add(sqe, fds[0], POLLIN);
82*25da2beaSAndroid Build Coastguard Worker 	sqe->user_data = 2;
83*25da2beaSAndroid Build Coastguard Worker 	ret = io_uring_submit(&ring);
84*25da2beaSAndroid Build Coastguard Worker 	if (ret != 1) {
85*25da2beaSAndroid Build Coastguard Worker 		fprintf(stderr, "ring_submit=%d\n", ret);
86*25da2beaSAndroid Build Coastguard Worker 		return 1;
87*25da2beaSAndroid Build Coastguard Worker 	}
88*25da2beaSAndroid Build Coastguard Worker 
89*25da2beaSAndroid Build Coastguard Worker 	pthread_create(&t1, NULL, listener_thread, &td);
90*25da2beaSAndroid Build Coastguard Worker 
91*25da2beaSAndroid Build Coastguard Worker 	sleep(1);
92*25da2beaSAndroid Build Coastguard Worker 
93*25da2beaSAndroid Build Coastguard Worker 	pthread_create(&t2, NULL, wakeup_io_uring, &td);
94*25da2beaSAndroid Build Coastguard Worker 	pthread_join(t1, &pret);
95*25da2beaSAndroid Build Coastguard Worker 
96*25da2beaSAndroid Build Coastguard Worker 	io_uring_queue_exit(&ring);
97*25da2beaSAndroid Build Coastguard Worker 	return pret != NULL;
98*25da2beaSAndroid Build Coastguard Worker }
99*25da2beaSAndroid Build Coastguard Worker 
test_eventfd(void)100*25da2beaSAndroid Build Coastguard Worker static int test_eventfd(void)
101*25da2beaSAndroid Build Coastguard Worker {
102*25da2beaSAndroid Build Coastguard Worker 	struct io_uring_sqe *sqe;
103*25da2beaSAndroid Build Coastguard Worker 	struct thread_data td;
104*25da2beaSAndroid Build Coastguard Worker 	struct io_uring ring;
105*25da2beaSAndroid Build Coastguard Worker 	pthread_t t1, t2;
106*25da2beaSAndroid Build Coastguard Worker 	int efd, ret;
107*25da2beaSAndroid Build Coastguard Worker 	void *pret;
108*25da2beaSAndroid Build Coastguard Worker 
109*25da2beaSAndroid Build Coastguard Worker 	efd = eventfd(0, 0);
110*25da2beaSAndroid Build Coastguard Worker 	if (efd < 0)
111*25da2beaSAndroid Build Coastguard Worker 		error_exit("eventfd");
112*25da2beaSAndroid Build Coastguard Worker 
113*25da2beaSAndroid Build Coastguard Worker 	ret = io_uring_queue_init(8, &ring, 0);
114*25da2beaSAndroid Build Coastguard Worker 	if (ret) {
115*25da2beaSAndroid Build Coastguard Worker 		fprintf(stderr, "Unable to setup io_uring: %s\n", strerror(-ret));
116*25da2beaSAndroid Build Coastguard Worker 		return 1;
117*25da2beaSAndroid Build Coastguard Worker 	}
118*25da2beaSAndroid Build Coastguard Worker 
119*25da2beaSAndroid Build Coastguard Worker 	td.write_fd = efd;
120*25da2beaSAndroid Build Coastguard Worker 	td.ring = &ring;
121*25da2beaSAndroid Build Coastguard Worker 
122*25da2beaSAndroid Build Coastguard Worker 	sqe = io_uring_get_sqe(&ring);
123*25da2beaSAndroid Build Coastguard Worker 	io_uring_prep_poll_add(sqe, efd, POLLIN);
124*25da2beaSAndroid Build Coastguard Worker 	sqe->user_data = 2;
125*25da2beaSAndroid Build Coastguard Worker 	ret = io_uring_submit(&ring);
126*25da2beaSAndroid Build Coastguard Worker 	if (ret != 1) {
127*25da2beaSAndroid Build Coastguard Worker 		fprintf(stderr, "ring_submit=%d\n", ret);
128*25da2beaSAndroid Build Coastguard Worker 		return 1;
129*25da2beaSAndroid Build Coastguard Worker 	}
130*25da2beaSAndroid Build Coastguard Worker 
131*25da2beaSAndroid Build Coastguard Worker 	pthread_create(&t1, NULL, listener_thread, &td);
132*25da2beaSAndroid Build Coastguard Worker 
133*25da2beaSAndroid Build Coastguard Worker 	sleep(1);
134*25da2beaSAndroid Build Coastguard Worker 
135*25da2beaSAndroid Build Coastguard Worker 	pthread_create(&t2, NULL, wakeup_io_uring, &td);
136*25da2beaSAndroid Build Coastguard Worker 	pthread_join(t1, &pret);
137*25da2beaSAndroid Build Coastguard Worker 
138*25da2beaSAndroid Build Coastguard Worker 	io_uring_queue_exit(&ring);
139*25da2beaSAndroid Build Coastguard Worker 	return pret != NULL;
140*25da2beaSAndroid Build Coastguard Worker }
141*25da2beaSAndroid Build Coastguard Worker 
main(int argc,char * argv[])142*25da2beaSAndroid Build Coastguard Worker int main(int argc, char *argv[])
143*25da2beaSAndroid Build Coastguard Worker {
144*25da2beaSAndroid Build Coastguard Worker 	int ret;
145*25da2beaSAndroid Build Coastguard Worker 
146*25da2beaSAndroid Build Coastguard Worker 	if (argc > 1)
147*25da2beaSAndroid Build Coastguard Worker 		return 0;
148*25da2beaSAndroid Build Coastguard Worker 
149*25da2beaSAndroid Build Coastguard Worker 	ret = test_pipes();
150*25da2beaSAndroid Build Coastguard Worker 	if (ret) {
151*25da2beaSAndroid Build Coastguard Worker 		fprintf(stderr, "test_pipe failed\n");
152*25da2beaSAndroid Build Coastguard Worker 		return ret;
153*25da2beaSAndroid Build Coastguard Worker 	}
154*25da2beaSAndroid Build Coastguard Worker 
155*25da2beaSAndroid Build Coastguard Worker 	ret = test_eventfd();
156*25da2beaSAndroid Build Coastguard Worker 	if (ret) {
157*25da2beaSAndroid Build Coastguard Worker 		fprintf(stderr, "test_eventfd failed\n");
158*25da2beaSAndroid Build Coastguard Worker 		return ret;
159*25da2beaSAndroid Build Coastguard Worker 	}
160*25da2beaSAndroid Build Coastguard Worker 
161*25da2beaSAndroid Build Coastguard Worker 	return 0;
162*25da2beaSAndroid Build Coastguard Worker }
163