1*25da2beaSAndroid Build Coastguard Worker /* SPDX-License-Identifier: MIT */
2*25da2beaSAndroid Build Coastguard Worker /*
3*25da2beaSAndroid Build Coastguard Worker * Test case testing exit without cleanup and io-wq work pending or queued.
4*25da2beaSAndroid Build Coastguard Worker *
5*25da2beaSAndroid Build Coastguard Worker * From Florian Fischer <[email protected]>
6*25da2beaSAndroid Build Coastguard Worker * Link: https://lore.kernel.org/io-uring/20211202165606.mqryio4yzubl7ms5@pasture/
7*25da2beaSAndroid Build Coastguard Worker *
8*25da2beaSAndroid Build Coastguard Worker */
9*25da2beaSAndroid Build Coastguard Worker #include <assert.h>
10*25da2beaSAndroid Build Coastguard Worker #include <err.h>
11*25da2beaSAndroid Build Coastguard Worker #include <errno.h>
12*25da2beaSAndroid Build Coastguard Worker #include <pthread.h>
13*25da2beaSAndroid Build Coastguard Worker #include <semaphore.h>
14*25da2beaSAndroid Build Coastguard Worker #include <stdio.h>
15*25da2beaSAndroid Build Coastguard Worker #include <stdlib.h>
16*25da2beaSAndroid Build Coastguard Worker #include <sys/sysinfo.h>
17*25da2beaSAndroid Build Coastguard Worker #include <unistd.h>
18*25da2beaSAndroid Build Coastguard Worker
19*25da2beaSAndroid Build Coastguard Worker #include "liburing.h"
20*25da2beaSAndroid Build Coastguard Worker #include "helpers.h"
21*25da2beaSAndroid Build Coastguard Worker
22*25da2beaSAndroid Build Coastguard Worker #define IORING_ENTRIES 8
23*25da2beaSAndroid Build Coastguard Worker
24*25da2beaSAndroid Build Coastguard Worker static pthread_t *threads;
25*25da2beaSAndroid Build Coastguard Worker static pthread_barrier_t init_barrier;
26*25da2beaSAndroid Build Coastguard Worker static int sleep_fd, notify_fd;
27*25da2beaSAndroid Build Coastguard Worker static sem_t sem;
28*25da2beaSAndroid Build Coastguard Worker
thread_func(void * arg)29*25da2beaSAndroid Build Coastguard Worker void *thread_func(void *arg)
30*25da2beaSAndroid Build Coastguard Worker {
31*25da2beaSAndroid Build Coastguard Worker struct io_uring ring;
32*25da2beaSAndroid Build Coastguard Worker int res;
33*25da2beaSAndroid Build Coastguard Worker
34*25da2beaSAndroid Build Coastguard Worker res = io_uring_queue_init(IORING_ENTRIES, &ring, 0);
35*25da2beaSAndroid Build Coastguard Worker if (res)
36*25da2beaSAndroid Build Coastguard Worker err(EXIT_FAILURE, "io_uring_queue_init failed");
37*25da2beaSAndroid Build Coastguard Worker
38*25da2beaSAndroid Build Coastguard Worker pthread_barrier_wait(&init_barrier);
39*25da2beaSAndroid Build Coastguard Worker
40*25da2beaSAndroid Build Coastguard Worker for(;;) {
41*25da2beaSAndroid Build Coastguard Worker struct io_uring_cqe *cqe;
42*25da2beaSAndroid Build Coastguard Worker struct io_uring_sqe *sqe;
43*25da2beaSAndroid Build Coastguard Worker uint64_t buf;
44*25da2beaSAndroid Build Coastguard Worker int res;
45*25da2beaSAndroid Build Coastguard Worker
46*25da2beaSAndroid Build Coastguard Worker sqe = io_uring_get_sqe(&ring);
47*25da2beaSAndroid Build Coastguard Worker assert(sqe);
48*25da2beaSAndroid Build Coastguard Worker
49*25da2beaSAndroid Build Coastguard Worker io_uring_prep_read(sqe, sleep_fd, &buf, sizeof(buf), 0);
50*25da2beaSAndroid Build Coastguard Worker
51*25da2beaSAndroid Build Coastguard Worker res = io_uring_submit_and_wait(&ring, 1);
52*25da2beaSAndroid Build Coastguard Worker if (res < 0)
53*25da2beaSAndroid Build Coastguard Worker err(EXIT_FAILURE, "io_uring_submit_and_wait failed");
54*25da2beaSAndroid Build Coastguard Worker
55*25da2beaSAndroid Build Coastguard Worker res = io_uring_peek_cqe(&ring, &cqe);
56*25da2beaSAndroid Build Coastguard Worker assert(!res);
57*25da2beaSAndroid Build Coastguard Worker if (cqe->res < 0) {
58*25da2beaSAndroid Build Coastguard Worker errno = -cqe->res;
59*25da2beaSAndroid Build Coastguard Worker err(EXIT_FAILURE, "read failed");
60*25da2beaSAndroid Build Coastguard Worker }
61*25da2beaSAndroid Build Coastguard Worker assert(cqe->res == sizeof(buf));
62*25da2beaSAndroid Build Coastguard Worker
63*25da2beaSAndroid Build Coastguard Worker sem_post(&sem);
64*25da2beaSAndroid Build Coastguard Worker
65*25da2beaSAndroid Build Coastguard Worker io_uring_cqe_seen(&ring, cqe);
66*25da2beaSAndroid Build Coastguard Worker }
67*25da2beaSAndroid Build Coastguard Worker
68*25da2beaSAndroid Build Coastguard Worker return NULL;
69*25da2beaSAndroid Build Coastguard Worker }
70*25da2beaSAndroid Build Coastguard Worker
main(int argc,char * argv[])71*25da2beaSAndroid Build Coastguard Worker int main(int argc, char *argv[])
72*25da2beaSAndroid Build Coastguard Worker {
73*25da2beaSAndroid Build Coastguard Worker int res, fds[2], i, cpus;
74*25da2beaSAndroid Build Coastguard Worker const uint64_t n = 0x42;
75*25da2beaSAndroid Build Coastguard Worker
76*25da2beaSAndroid Build Coastguard Worker if (argc > 1)
77*25da2beaSAndroid Build Coastguard Worker return 0;
78*25da2beaSAndroid Build Coastguard Worker
79*25da2beaSAndroid Build Coastguard Worker cpus = get_nprocs();
80*25da2beaSAndroid Build Coastguard Worker res = pthread_barrier_init(&init_barrier, NULL, cpus);
81*25da2beaSAndroid Build Coastguard Worker if (res)
82*25da2beaSAndroid Build Coastguard Worker err(EXIT_FAILURE, "pthread_barrier_init failed");
83*25da2beaSAndroid Build Coastguard Worker
84*25da2beaSAndroid Build Coastguard Worker res = sem_init(&sem, 0, 0);
85*25da2beaSAndroid Build Coastguard Worker if (res)
86*25da2beaSAndroid Build Coastguard Worker err(EXIT_FAILURE, "sem_init failed");
87*25da2beaSAndroid Build Coastguard Worker
88*25da2beaSAndroid Build Coastguard Worker threads = t_malloc(sizeof(pthread_t) * cpus);
89*25da2beaSAndroid Build Coastguard Worker
90*25da2beaSAndroid Build Coastguard Worker res = pipe(fds);
91*25da2beaSAndroid Build Coastguard Worker if (res)
92*25da2beaSAndroid Build Coastguard Worker err(EXIT_FAILURE, "pipe failed");
93*25da2beaSAndroid Build Coastguard Worker
94*25da2beaSAndroid Build Coastguard Worker sleep_fd = fds[0];
95*25da2beaSAndroid Build Coastguard Worker notify_fd = fds[1];
96*25da2beaSAndroid Build Coastguard Worker
97*25da2beaSAndroid Build Coastguard Worker for (i = 0; i < cpus; i++) {
98*25da2beaSAndroid Build Coastguard Worker errno = pthread_create(&threads[i], NULL, thread_func, NULL);
99*25da2beaSAndroid Build Coastguard Worker if (errno)
100*25da2beaSAndroid Build Coastguard Worker err(EXIT_FAILURE, "pthread_create failed");
101*25da2beaSAndroid Build Coastguard Worker }
102*25da2beaSAndroid Build Coastguard Worker
103*25da2beaSAndroid Build Coastguard Worker // Write #cpus notifications
104*25da2beaSAndroid Build Coastguard Worker for (i = 0; i < cpus; i++) {
105*25da2beaSAndroid Build Coastguard Worker res = write(notify_fd, &n, sizeof(n));
106*25da2beaSAndroid Build Coastguard Worker if (res < 0)
107*25da2beaSAndroid Build Coastguard Worker err(EXIT_FAILURE, "write failed");
108*25da2beaSAndroid Build Coastguard Worker assert(res == sizeof(n));
109*25da2beaSAndroid Build Coastguard Worker }
110*25da2beaSAndroid Build Coastguard Worker
111*25da2beaSAndroid Build Coastguard Worker // Await that all notifications were received
112*25da2beaSAndroid Build Coastguard Worker for (i = 0; i < cpus; i++)
113*25da2beaSAndroid Build Coastguard Worker sem_wait(&sem);
114*25da2beaSAndroid Build Coastguard Worker
115*25da2beaSAndroid Build Coastguard Worker // Exit without resource cleanup
116*25da2beaSAndroid Build Coastguard Worker exit(EXIT_SUCCESS);
117*25da2beaSAndroid Build Coastguard Worker }
118