1*25da2beaSAndroid Build Coastguard Worker /* SPDX-License-Identifier: MIT */
2*25da2beaSAndroid Build Coastguard Worker /*
3*25da2beaSAndroid Build Coastguard Worker * Description: test if issuing IO from thread and immediately exiting will
4*25da2beaSAndroid Build Coastguard Worker * proceed correctly.
5*25da2beaSAndroid Build Coastguard Worker *
6*25da2beaSAndroid Build Coastguard Worker * Original test case from: https://github.com/axboe/liburing/issues/582
7*25da2beaSAndroid Build Coastguard Worker */
8*25da2beaSAndroid Build Coastguard Worker #include <unistd.h>
9*25da2beaSAndroid Build Coastguard Worker #include <pthread.h>
10*25da2beaSAndroid Build Coastguard Worker #include <sys/timerfd.h>
11*25da2beaSAndroid Build Coastguard Worker #include <string.h>
12*25da2beaSAndroid Build Coastguard Worker #include <stdio.h>
13*25da2beaSAndroid Build Coastguard Worker
14*25da2beaSAndroid Build Coastguard Worker #include "liburing.h"
15*25da2beaSAndroid Build Coastguard Worker #include "helpers.h"
16*25da2beaSAndroid Build Coastguard Worker
17*25da2beaSAndroid Build Coastguard Worker struct data {
18*25da2beaSAndroid Build Coastguard Worker struct io_uring *ring;
19*25da2beaSAndroid Build Coastguard Worker int timer_fd1;
20*25da2beaSAndroid Build Coastguard Worker int timer_fd2;
21*25da2beaSAndroid Build Coastguard Worker uint64_t buf1;
22*25da2beaSAndroid Build Coastguard Worker uint64_t buf2;
23*25da2beaSAndroid Build Coastguard Worker };
24*25da2beaSAndroid Build Coastguard Worker
submit(void * data)25*25da2beaSAndroid Build Coastguard Worker void *submit(void *data)
26*25da2beaSAndroid Build Coastguard Worker {
27*25da2beaSAndroid Build Coastguard Worker struct io_uring_sqe *sqe;
28*25da2beaSAndroid Build Coastguard Worker struct data *d = data;
29*25da2beaSAndroid Build Coastguard Worker int ret;
30*25da2beaSAndroid Build Coastguard Worker
31*25da2beaSAndroid Build Coastguard Worker sqe = io_uring_get_sqe(d->ring);
32*25da2beaSAndroid Build Coastguard Worker io_uring_prep_read(sqe, d->timer_fd1, &d->buf1, sizeof(d->buf1), 0);
33*25da2beaSAndroid Build Coastguard Worker
34*25da2beaSAndroid Build Coastguard Worker sqe = io_uring_get_sqe(d->ring);
35*25da2beaSAndroid Build Coastguard Worker io_uring_prep_read(sqe, d->timer_fd2, &d->buf2, sizeof(d->buf2), 0);
36*25da2beaSAndroid Build Coastguard Worker
37*25da2beaSAndroid Build Coastguard Worker ret = io_uring_submit(d->ring);
38*25da2beaSAndroid Build Coastguard Worker if (ret != 2)
39*25da2beaSAndroid Build Coastguard Worker return (void *) (uintptr_t) 1;
40*25da2beaSAndroid Build Coastguard Worker
41*25da2beaSAndroid Build Coastguard Worker /* Exit suddenly. */
42*25da2beaSAndroid Build Coastguard Worker return NULL;
43*25da2beaSAndroid Build Coastguard Worker }
44*25da2beaSAndroid Build Coastguard Worker
test(int flags)45*25da2beaSAndroid Build Coastguard Worker static int test(int flags)
46*25da2beaSAndroid Build Coastguard Worker {
47*25da2beaSAndroid Build Coastguard Worker struct io_uring_params params = { .flags = flags, };
48*25da2beaSAndroid Build Coastguard Worker struct io_uring ring;
49*25da2beaSAndroid Build Coastguard Worker struct data d = { .ring = &ring, };
50*25da2beaSAndroid Build Coastguard Worker pthread_t thread;
51*25da2beaSAndroid Build Coastguard Worker void *res;
52*25da2beaSAndroid Build Coastguard Worker int ret;
53*25da2beaSAndroid Build Coastguard Worker
54*25da2beaSAndroid Build Coastguard Worker ret = t_create_ring_params(8, &ring, ¶ms);
55*25da2beaSAndroid Build Coastguard Worker if (ret == T_SETUP_SKIP)
56*25da2beaSAndroid Build Coastguard Worker return 0;
57*25da2beaSAndroid Build Coastguard Worker else if (ret != T_SETUP_OK)
58*25da2beaSAndroid Build Coastguard Worker return 1;
59*25da2beaSAndroid Build Coastguard Worker
60*25da2beaSAndroid Build Coastguard Worker d.timer_fd1 = timerfd_create(CLOCK_MONOTONIC, TFD_CLOEXEC);
61*25da2beaSAndroid Build Coastguard Worker if (d.timer_fd1 < 0) {
62*25da2beaSAndroid Build Coastguard Worker perror("timerfd_create");
63*25da2beaSAndroid Build Coastguard Worker return 1;
64*25da2beaSAndroid Build Coastguard Worker }
65*25da2beaSAndroid Build Coastguard Worker d.timer_fd2 = timerfd_create(CLOCK_MONOTONIC, TFD_CLOEXEC);
66*25da2beaSAndroid Build Coastguard Worker if (d.timer_fd2 < 0) {
67*25da2beaSAndroid Build Coastguard Worker perror("timerfd_create");
68*25da2beaSAndroid Build Coastguard Worker return 1;
69*25da2beaSAndroid Build Coastguard Worker }
70*25da2beaSAndroid Build Coastguard Worker
71*25da2beaSAndroid Build Coastguard Worker pthread_create(&thread, NULL, submit, &d);
72*25da2beaSAndroid Build Coastguard Worker pthread_join(thread, &res);
73*25da2beaSAndroid Build Coastguard Worker
74*25da2beaSAndroid Build Coastguard Worker /** Wait for completions and do stuff ... **/
75*25da2beaSAndroid Build Coastguard Worker
76*25da2beaSAndroid Build Coastguard Worker io_uring_queue_exit(&ring);
77*25da2beaSAndroid Build Coastguard Worker
78*25da2beaSAndroid Build Coastguard Worker close(d.timer_fd1);
79*25da2beaSAndroid Build Coastguard Worker close(d.timer_fd2);
80*25da2beaSAndroid Build Coastguard Worker return !!res;
81*25da2beaSAndroid Build Coastguard Worker }
82*25da2beaSAndroid Build Coastguard Worker
main(int argc,char * argv[])83*25da2beaSAndroid Build Coastguard Worker int main(int argc, char *argv[])
84*25da2beaSAndroid Build Coastguard Worker {
85*25da2beaSAndroid Build Coastguard Worker int ret, i;
86*25da2beaSAndroid Build Coastguard Worker
87*25da2beaSAndroid Build Coastguard Worker for (i = 0; i < 1000; i++) {
88*25da2beaSAndroid Build Coastguard Worker ret = test(0);
89*25da2beaSAndroid Build Coastguard Worker if (ret) {
90*25da2beaSAndroid Build Coastguard Worker fprintf(stderr, "Test failed\n");
91*25da2beaSAndroid Build Coastguard Worker return ret;
92*25da2beaSAndroid Build Coastguard Worker }
93*25da2beaSAndroid Build Coastguard Worker }
94*25da2beaSAndroid Build Coastguard Worker
95*25da2beaSAndroid Build Coastguard Worker for (i = 0; i < 1000; i++) {
96*25da2beaSAndroid Build Coastguard Worker ret = test(IORING_SETUP_IOPOLL);
97*25da2beaSAndroid Build Coastguard Worker if (ret) {
98*25da2beaSAndroid Build Coastguard Worker fprintf(stderr, "Test IOPOLL failed\n");
99*25da2beaSAndroid Build Coastguard Worker return ret;
100*25da2beaSAndroid Build Coastguard Worker }
101*25da2beaSAndroid Build Coastguard Worker }
102*25da2beaSAndroid Build Coastguard Worker
103*25da2beaSAndroid Build Coastguard Worker for (i = 0; i < 100; i++) {
104*25da2beaSAndroid Build Coastguard Worker ret = test(IORING_SETUP_SQPOLL);
105*25da2beaSAndroid Build Coastguard Worker if (ret) {
106*25da2beaSAndroid Build Coastguard Worker fprintf(stderr, "Test SQPOLL failed\n");
107*25da2beaSAndroid Build Coastguard Worker return ret;
108*25da2beaSAndroid Build Coastguard Worker }
109*25da2beaSAndroid Build Coastguard Worker }
110*25da2beaSAndroid Build Coastguard Worker
111*25da2beaSAndroid Build Coastguard Worker return 0;
112*25da2beaSAndroid Build Coastguard Worker }
113