xref: /aosp_15_r20/external/liburing/test/sq-poll-kthread.c (revision 25da2bea747f3a93b4c30fd9708b0618ef55a0e6)
1*25da2beaSAndroid Build Coastguard Worker /* SPDX-License-Identifier: MIT */
2*25da2beaSAndroid Build Coastguard Worker /*
3*25da2beaSAndroid Build Coastguard Worker  * Description: test if io_uring SQ poll kthread is stopped when the userspace
4*25da2beaSAndroid Build Coastguard Worker  *              process ended with or without closing the io_uring fd
5*25da2beaSAndroid Build Coastguard Worker  *
6*25da2beaSAndroid Build Coastguard Worker  */
7*25da2beaSAndroid Build Coastguard Worker #include <errno.h>
8*25da2beaSAndroid Build Coastguard Worker #include <fcntl.h>
9*25da2beaSAndroid Build Coastguard Worker #include <stdio.h>
10*25da2beaSAndroid Build Coastguard Worker #include <unistd.h>
11*25da2beaSAndroid Build Coastguard Worker #include <pthread.h>
12*25da2beaSAndroid Build Coastguard Worker #include <stdbool.h>
13*25da2beaSAndroid Build Coastguard Worker #include <stdlib.h>
14*25da2beaSAndroid Build Coastguard Worker #include <string.h>
15*25da2beaSAndroid Build Coastguard Worker #include <signal.h>
16*25da2beaSAndroid Build Coastguard Worker #include <poll.h>
17*25da2beaSAndroid Build Coastguard Worker #include <sys/wait.h>
18*25da2beaSAndroid Build Coastguard Worker #include <sys/epoll.h>
19*25da2beaSAndroid Build Coastguard Worker 
20*25da2beaSAndroid Build Coastguard Worker #include "liburing.h"
21*25da2beaSAndroid Build Coastguard Worker #include "helpers.h"
22*25da2beaSAndroid Build Coastguard Worker 
23*25da2beaSAndroid Build Coastguard Worker #define SQ_THREAD_IDLE  2000
24*25da2beaSAndroid Build Coastguard Worker #define BUF_SIZE        128
25*25da2beaSAndroid Build Coastguard Worker #define KTHREAD_NAME    "io_uring-sq"
26*25da2beaSAndroid Build Coastguard Worker 
27*25da2beaSAndroid Build Coastguard Worker enum {
28*25da2beaSAndroid Build Coastguard Worker 	TEST_OK = 0,
29*25da2beaSAndroid Build Coastguard Worker 	TEST_SKIPPED = 1,
30*25da2beaSAndroid Build Coastguard Worker 	TEST_FAILED = 2,
31*25da2beaSAndroid Build Coastguard Worker };
32*25da2beaSAndroid Build Coastguard Worker 
do_test_sq_poll_kthread_stopped(bool do_exit)33*25da2beaSAndroid Build Coastguard Worker static int do_test_sq_poll_kthread_stopped(bool do_exit)
34*25da2beaSAndroid Build Coastguard Worker {
35*25da2beaSAndroid Build Coastguard Worker 	int ret = 0, pipe1[2];
36*25da2beaSAndroid Build Coastguard Worker 	struct io_uring_params param;
37*25da2beaSAndroid Build Coastguard Worker 	struct io_uring ring;
38*25da2beaSAndroid Build Coastguard Worker 	struct io_uring_sqe *sqe;
39*25da2beaSAndroid Build Coastguard Worker 	struct io_uring_cqe *cqe;
40*25da2beaSAndroid Build Coastguard Worker 	uint8_t buf[BUF_SIZE];
41*25da2beaSAndroid Build Coastguard Worker 	struct iovec iov;
42*25da2beaSAndroid Build Coastguard Worker 
43*25da2beaSAndroid Build Coastguard Worker 	if (pipe(pipe1) != 0) {
44*25da2beaSAndroid Build Coastguard Worker 		perror("pipe");
45*25da2beaSAndroid Build Coastguard Worker 		return TEST_FAILED;
46*25da2beaSAndroid Build Coastguard Worker 	}
47*25da2beaSAndroid Build Coastguard Worker 
48*25da2beaSAndroid Build Coastguard Worker 	memset(&param, 0, sizeof(param));
49*25da2beaSAndroid Build Coastguard Worker 	param.flags |= IORING_SETUP_SQPOLL;
50*25da2beaSAndroid Build Coastguard Worker 	param.sq_thread_idle = SQ_THREAD_IDLE;
51*25da2beaSAndroid Build Coastguard Worker 
52*25da2beaSAndroid Build Coastguard Worker 	ret = t_create_ring_params(16, &ring, &param);
53*25da2beaSAndroid Build Coastguard Worker 	if (ret == T_SETUP_SKIP) {
54*25da2beaSAndroid Build Coastguard Worker 		ret = TEST_FAILED;
55*25da2beaSAndroid Build Coastguard Worker 		goto err_pipe;
56*25da2beaSAndroid Build Coastguard Worker 	} else if (ret != T_SETUP_OK) {
57*25da2beaSAndroid Build Coastguard Worker 		fprintf(stderr, "ring setup failed\n");
58*25da2beaSAndroid Build Coastguard Worker 		ret = TEST_FAILED;
59*25da2beaSAndroid Build Coastguard Worker 		goto err_pipe;
60*25da2beaSAndroid Build Coastguard Worker 	}
61*25da2beaSAndroid Build Coastguard Worker 
62*25da2beaSAndroid Build Coastguard Worker 	ret = io_uring_register_files(&ring, &pipe1[1], 1);
63*25da2beaSAndroid Build Coastguard Worker 	if (ret) {
64*25da2beaSAndroid Build Coastguard Worker 		fprintf(stderr, "file reg failed: %d\n", ret);
65*25da2beaSAndroid Build Coastguard Worker 		ret = TEST_FAILED;
66*25da2beaSAndroid Build Coastguard Worker 		goto err_uring;
67*25da2beaSAndroid Build Coastguard Worker 	}
68*25da2beaSAndroid Build Coastguard Worker 
69*25da2beaSAndroid Build Coastguard Worker 	iov.iov_base = buf;
70*25da2beaSAndroid Build Coastguard Worker 	iov.iov_len = BUF_SIZE;
71*25da2beaSAndroid Build Coastguard Worker 
72*25da2beaSAndroid Build Coastguard Worker 	sqe = io_uring_get_sqe(&ring);
73*25da2beaSAndroid Build Coastguard Worker 	if (!sqe) {
74*25da2beaSAndroid Build Coastguard Worker 		fprintf(stderr, "io_uring_get_sqe failed\n");
75*25da2beaSAndroid Build Coastguard Worker 		ret = TEST_FAILED;
76*25da2beaSAndroid Build Coastguard Worker 		goto err_uring;
77*25da2beaSAndroid Build Coastguard Worker 	}
78*25da2beaSAndroid Build Coastguard Worker 
79*25da2beaSAndroid Build Coastguard Worker 	io_uring_prep_writev(sqe, 0, &iov, 1, 0);
80*25da2beaSAndroid Build Coastguard Worker 	sqe->flags |= IOSQE_FIXED_FILE;
81*25da2beaSAndroid Build Coastguard Worker 
82*25da2beaSAndroid Build Coastguard Worker 	ret = io_uring_submit(&ring);
83*25da2beaSAndroid Build Coastguard Worker 	if (ret < 0) {
84*25da2beaSAndroid Build Coastguard Worker 		fprintf(stderr, "io_uring_submit failed - ret: %d\n",
85*25da2beaSAndroid Build Coastguard Worker 			ret);
86*25da2beaSAndroid Build Coastguard Worker 		ret = TEST_FAILED;
87*25da2beaSAndroid Build Coastguard Worker 		goto err_uring;
88*25da2beaSAndroid Build Coastguard Worker 	}
89*25da2beaSAndroid Build Coastguard Worker 
90*25da2beaSAndroid Build Coastguard Worker 	ret = io_uring_wait_cqe(&ring, &cqe);
91*25da2beaSAndroid Build Coastguard Worker 	if (ret < 0) {
92*25da2beaSAndroid Build Coastguard Worker 		fprintf(stderr, "io_uring_wait_cqe - ret: %d\n",
93*25da2beaSAndroid Build Coastguard Worker 			ret);
94*25da2beaSAndroid Build Coastguard Worker 		ret = TEST_FAILED;
95*25da2beaSAndroid Build Coastguard Worker 		goto err_uring;
96*25da2beaSAndroid Build Coastguard Worker 	}
97*25da2beaSAndroid Build Coastguard Worker 
98*25da2beaSAndroid Build Coastguard Worker 	if (cqe->res != BUF_SIZE) {
99*25da2beaSAndroid Build Coastguard Worker 		fprintf(stderr, "unexpected cqe->res %d [expected %d]\n",
100*25da2beaSAndroid Build Coastguard Worker 			cqe->res, BUF_SIZE);
101*25da2beaSAndroid Build Coastguard Worker 		ret = TEST_FAILED;
102*25da2beaSAndroid Build Coastguard Worker 		goto err_uring;
103*25da2beaSAndroid Build Coastguard Worker 
104*25da2beaSAndroid Build Coastguard Worker 	}
105*25da2beaSAndroid Build Coastguard Worker 
106*25da2beaSAndroid Build Coastguard Worker 	io_uring_cqe_seen(&ring, cqe);
107*25da2beaSAndroid Build Coastguard Worker 
108*25da2beaSAndroid Build Coastguard Worker 	ret = TEST_OK;
109*25da2beaSAndroid Build Coastguard Worker 
110*25da2beaSAndroid Build Coastguard Worker err_uring:
111*25da2beaSAndroid Build Coastguard Worker 	if (do_exit)
112*25da2beaSAndroid Build Coastguard Worker 		io_uring_queue_exit(&ring);
113*25da2beaSAndroid Build Coastguard Worker err_pipe:
114*25da2beaSAndroid Build Coastguard Worker 	close(pipe1[0]);
115*25da2beaSAndroid Build Coastguard Worker 	close(pipe1[1]);
116*25da2beaSAndroid Build Coastguard Worker 
117*25da2beaSAndroid Build Coastguard Worker 	return ret;
118*25da2beaSAndroid Build Coastguard Worker }
119*25da2beaSAndroid Build Coastguard Worker 
test_sq_poll_kthread_stopped(bool do_exit)120*25da2beaSAndroid Build Coastguard Worker int test_sq_poll_kthread_stopped(bool do_exit)
121*25da2beaSAndroid Build Coastguard Worker {
122*25da2beaSAndroid Build Coastguard Worker 	pid_t pid;
123*25da2beaSAndroid Build Coastguard Worker 	int status = 0;
124*25da2beaSAndroid Build Coastguard Worker 
125*25da2beaSAndroid Build Coastguard Worker 	pid = fork();
126*25da2beaSAndroid Build Coastguard Worker 
127*25da2beaSAndroid Build Coastguard Worker 	if (pid == 0) {
128*25da2beaSAndroid Build Coastguard Worker 		int ret = do_test_sq_poll_kthread_stopped(do_exit);
129*25da2beaSAndroid Build Coastguard Worker 		exit(ret);
130*25da2beaSAndroid Build Coastguard Worker 	}
131*25da2beaSAndroid Build Coastguard Worker 
132*25da2beaSAndroid Build Coastguard Worker 	pid = wait(&status);
133*25da2beaSAndroid Build Coastguard Worker 	if (status != 0)
134*25da2beaSAndroid Build Coastguard Worker 		return WEXITSTATUS(status);
135*25da2beaSAndroid Build Coastguard Worker 
136*25da2beaSAndroid Build Coastguard Worker 	sleep(1);
137*25da2beaSAndroid Build Coastguard Worker 	if (system("ps --ppid 2 | grep " KTHREAD_NAME) == 0) {
138*25da2beaSAndroid Build Coastguard Worker 		fprintf(stderr, "%s kthread still running!\n", KTHREAD_NAME);
139*25da2beaSAndroid Build Coastguard Worker 		return TEST_FAILED;
140*25da2beaSAndroid Build Coastguard Worker 	}
141*25da2beaSAndroid Build Coastguard Worker 
142*25da2beaSAndroid Build Coastguard Worker 	return 0;
143*25da2beaSAndroid Build Coastguard Worker }
144*25da2beaSAndroid Build Coastguard Worker 
main(int argc,char * argv[])145*25da2beaSAndroid Build Coastguard Worker int main(int argc, char *argv[])
146*25da2beaSAndroid Build Coastguard Worker {
147*25da2beaSAndroid Build Coastguard Worker 	int ret;
148*25da2beaSAndroid Build Coastguard Worker 
149*25da2beaSAndroid Build Coastguard Worker 	if (argc > 1)
150*25da2beaSAndroid Build Coastguard Worker 		return 0;
151*25da2beaSAndroid Build Coastguard Worker 
152*25da2beaSAndroid Build Coastguard Worker 	ret = test_sq_poll_kthread_stopped(true);
153*25da2beaSAndroid Build Coastguard Worker 	if (ret == TEST_SKIPPED) {
154*25da2beaSAndroid Build Coastguard Worker 		printf("test_sq_poll_kthread_stopped_exit: skipped\n");
155*25da2beaSAndroid Build Coastguard Worker 	} else if (ret == TEST_FAILED) {
156*25da2beaSAndroid Build Coastguard Worker 		fprintf(stderr, "test_sq_poll_kthread_stopped_exit failed\n");
157*25da2beaSAndroid Build Coastguard Worker 		return ret;
158*25da2beaSAndroid Build Coastguard Worker 	}
159*25da2beaSAndroid Build Coastguard Worker 
160*25da2beaSAndroid Build Coastguard Worker 	ret = test_sq_poll_kthread_stopped(false);
161*25da2beaSAndroid Build Coastguard Worker 	if (ret == TEST_SKIPPED) {
162*25da2beaSAndroid Build Coastguard Worker 		printf("test_sq_poll_kthread_stopped_noexit: skipped\n");
163*25da2beaSAndroid Build Coastguard Worker 	} else if (ret == TEST_FAILED) {
164*25da2beaSAndroid Build Coastguard Worker 		fprintf(stderr, "test_sq_poll_kthread_stopped_noexit failed\n");
165*25da2beaSAndroid Build Coastguard Worker 		return ret;
166*25da2beaSAndroid Build Coastguard Worker 	}
167*25da2beaSAndroid Build Coastguard Worker 
168*25da2beaSAndroid Build Coastguard Worker 	return 0;
169*25da2beaSAndroid Build Coastguard Worker }
170