1*25da2beaSAndroid Build Coastguard Worker /* SPDX-License-Identifier: MIT */
2*25da2beaSAndroid Build Coastguard Worker /*
3*25da2beaSAndroid Build Coastguard Worker * Description: test that pathname resolution works from async context when
4*25da2beaSAndroid Build Coastguard Worker * using /proc/self/ which should be the original submitting task, not the
5*25da2beaSAndroid Build Coastguard Worker * async worker.
6*25da2beaSAndroid Build Coastguard Worker *
7*25da2beaSAndroid Build Coastguard Worker */
8*25da2beaSAndroid Build Coastguard Worker #include <errno.h>
9*25da2beaSAndroid Build Coastguard Worker #include <stdio.h>
10*25da2beaSAndroid Build Coastguard Worker #include <unistd.h>
11*25da2beaSAndroid Build Coastguard Worker #include <stdlib.h>
12*25da2beaSAndroid Build Coastguard Worker #include <string.h>
13*25da2beaSAndroid Build Coastguard Worker #include <fcntl.h>
14*25da2beaSAndroid Build Coastguard Worker
15*25da2beaSAndroid Build Coastguard Worker #include "liburing.h"
16*25da2beaSAndroid Build Coastguard Worker
io_openat2(struct io_uring * ring,const char * path,int dfd)17*25da2beaSAndroid Build Coastguard Worker static int io_openat2(struct io_uring *ring, const char *path, int dfd)
18*25da2beaSAndroid Build Coastguard Worker {
19*25da2beaSAndroid Build Coastguard Worker struct io_uring_cqe *cqe;
20*25da2beaSAndroid Build Coastguard Worker struct io_uring_sqe *sqe;
21*25da2beaSAndroid Build Coastguard Worker struct open_how how;
22*25da2beaSAndroid Build Coastguard Worker int ret;
23*25da2beaSAndroid Build Coastguard Worker
24*25da2beaSAndroid Build Coastguard Worker sqe = io_uring_get_sqe(ring);
25*25da2beaSAndroid Build Coastguard Worker if (!sqe) {
26*25da2beaSAndroid Build Coastguard Worker fprintf(stderr, "get sqe failed\n");
27*25da2beaSAndroid Build Coastguard Worker goto err;
28*25da2beaSAndroid Build Coastguard Worker }
29*25da2beaSAndroid Build Coastguard Worker memset(&how, 0, sizeof(how));
30*25da2beaSAndroid Build Coastguard Worker how.flags = O_RDONLY;
31*25da2beaSAndroid Build Coastguard Worker io_uring_prep_openat2(sqe, dfd, path, &how);
32*25da2beaSAndroid Build Coastguard Worker
33*25da2beaSAndroid Build Coastguard Worker ret = io_uring_submit(ring);
34*25da2beaSAndroid Build Coastguard Worker if (ret <= 0) {
35*25da2beaSAndroid Build Coastguard Worker fprintf(stderr, "sqe submit failed: %d\n", ret);
36*25da2beaSAndroid Build Coastguard Worker goto err;
37*25da2beaSAndroid Build Coastguard Worker }
38*25da2beaSAndroid Build Coastguard Worker
39*25da2beaSAndroid Build Coastguard Worker ret = io_uring_wait_cqe(ring, &cqe);
40*25da2beaSAndroid Build Coastguard Worker if (ret < 0) {
41*25da2beaSAndroid Build Coastguard Worker fprintf(stderr, "wait completion %d\n", ret);
42*25da2beaSAndroid Build Coastguard Worker goto err;
43*25da2beaSAndroid Build Coastguard Worker }
44*25da2beaSAndroid Build Coastguard Worker ret = cqe->res;
45*25da2beaSAndroid Build Coastguard Worker io_uring_cqe_seen(ring, cqe);
46*25da2beaSAndroid Build Coastguard Worker return ret;
47*25da2beaSAndroid Build Coastguard Worker err:
48*25da2beaSAndroid Build Coastguard Worker return -1;
49*25da2beaSAndroid Build Coastguard Worker }
50*25da2beaSAndroid Build Coastguard Worker
main(int argc,char * argv[])51*25da2beaSAndroid Build Coastguard Worker int main(int argc, char *argv[])
52*25da2beaSAndroid Build Coastguard Worker {
53*25da2beaSAndroid Build Coastguard Worker struct io_uring ring;
54*25da2beaSAndroid Build Coastguard Worker char buf[64];
55*25da2beaSAndroid Build Coastguard Worker int ret;
56*25da2beaSAndroid Build Coastguard Worker
57*25da2beaSAndroid Build Coastguard Worker if (argc > 1)
58*25da2beaSAndroid Build Coastguard Worker return 0;
59*25da2beaSAndroid Build Coastguard Worker
60*25da2beaSAndroid Build Coastguard Worker ret = io_uring_queue_init(1, &ring, 0);
61*25da2beaSAndroid Build Coastguard Worker if (ret) {
62*25da2beaSAndroid Build Coastguard Worker fprintf(stderr, "ring setup failed\n");
63*25da2beaSAndroid Build Coastguard Worker return 1;
64*25da2beaSAndroid Build Coastguard Worker }
65*25da2beaSAndroid Build Coastguard Worker
66*25da2beaSAndroid Build Coastguard Worker ret = io_openat2(&ring, "/proc/self/comm", -1);
67*25da2beaSAndroid Build Coastguard Worker if (ret < 0) {
68*25da2beaSAndroid Build Coastguard Worker if (ret == -EOPNOTSUPP)
69*25da2beaSAndroid Build Coastguard Worker return 0;
70*25da2beaSAndroid Build Coastguard Worker if (ret == -EINVAL) {
71*25da2beaSAndroid Build Coastguard Worker fprintf(stdout, "openat2 not supported, skipping\n");
72*25da2beaSAndroid Build Coastguard Worker return 0;
73*25da2beaSAndroid Build Coastguard Worker }
74*25da2beaSAndroid Build Coastguard Worker fprintf(stderr, "openat2 failed: %s\n", strerror(-ret));
75*25da2beaSAndroid Build Coastguard Worker return 1;
76*25da2beaSAndroid Build Coastguard Worker }
77*25da2beaSAndroid Build Coastguard Worker
78*25da2beaSAndroid Build Coastguard Worker memset(buf, 0, sizeof(buf));
79*25da2beaSAndroid Build Coastguard Worker ret = read(ret, buf, sizeof(buf));
80*25da2beaSAndroid Build Coastguard Worker if (ret < 0) {
81*25da2beaSAndroid Build Coastguard Worker perror("read");
82*25da2beaSAndroid Build Coastguard Worker return 1;
83*25da2beaSAndroid Build Coastguard Worker }
84*25da2beaSAndroid Build Coastguard Worker
85*25da2beaSAndroid Build Coastguard Worker if (strncmp(buf, "self", 4)) {
86*25da2beaSAndroid Build Coastguard Worker fprintf(stderr, "got comm=<%s>, wanted <self>\n", buf);
87*25da2beaSAndroid Build Coastguard Worker return 1;
88*25da2beaSAndroid Build Coastguard Worker }
89*25da2beaSAndroid Build Coastguard Worker
90*25da2beaSAndroid Build Coastguard Worker return 0;
91*25da2beaSAndroid Build Coastguard Worker }
92