1*25da2beaSAndroid Build Coastguard Worker /* SPDX-License-Identifier: MIT */
2*25da2beaSAndroid Build Coastguard Worker /*
3*25da2beaSAndroid Build Coastguard Worker * Description: test io_uring linkat handling
4*25da2beaSAndroid Build Coastguard Worker */
5*25da2beaSAndroid Build Coastguard Worker #include <fcntl.h>
6*25da2beaSAndroid Build Coastguard Worker #include <stdio.h>
7*25da2beaSAndroid Build Coastguard Worker #include <string.h>
8*25da2beaSAndroid Build Coastguard Worker #include <sys/stat.h>
9*25da2beaSAndroid Build Coastguard Worker #include <sys/types.h>
10*25da2beaSAndroid Build Coastguard Worker #include <unistd.h>
11*25da2beaSAndroid Build Coastguard Worker
12*25da2beaSAndroid Build Coastguard Worker #include "liburing.h"
13*25da2beaSAndroid Build Coastguard Worker
14*25da2beaSAndroid Build Coastguard Worker
do_linkat(struct io_uring * ring,const char * oldname,const char * newname)15*25da2beaSAndroid Build Coastguard Worker static int do_linkat(struct io_uring *ring, const char *oldname, const char *newname)
16*25da2beaSAndroid Build Coastguard Worker {
17*25da2beaSAndroid Build Coastguard Worker int ret;
18*25da2beaSAndroid Build Coastguard Worker struct io_uring_sqe *sqe;
19*25da2beaSAndroid Build Coastguard Worker struct io_uring_cqe *cqe;
20*25da2beaSAndroid Build Coastguard Worker
21*25da2beaSAndroid Build Coastguard Worker sqe = io_uring_get_sqe(ring);
22*25da2beaSAndroid Build Coastguard Worker if (!sqe) {
23*25da2beaSAndroid Build Coastguard Worker fprintf(stderr, "sqe get failed\n");
24*25da2beaSAndroid Build Coastguard Worker goto err;
25*25da2beaSAndroid Build Coastguard Worker }
26*25da2beaSAndroid Build Coastguard Worker io_uring_prep_linkat(sqe, AT_FDCWD, oldname, AT_FDCWD, newname, 0);
27*25da2beaSAndroid Build Coastguard Worker
28*25da2beaSAndroid Build Coastguard Worker ret = io_uring_submit(ring);
29*25da2beaSAndroid Build Coastguard Worker if (ret != 1) {
30*25da2beaSAndroid Build Coastguard Worker fprintf(stderr, "submit failed: %d\n", ret);
31*25da2beaSAndroid Build Coastguard Worker goto err;
32*25da2beaSAndroid Build Coastguard Worker }
33*25da2beaSAndroid Build Coastguard Worker
34*25da2beaSAndroid Build Coastguard Worker ret = io_uring_wait_cqes(ring, &cqe, 1, 0, 0);
35*25da2beaSAndroid Build Coastguard Worker if (ret) {
36*25da2beaSAndroid Build Coastguard Worker fprintf(stderr, "wait_cqe failed: %d\n", ret);
37*25da2beaSAndroid Build Coastguard Worker goto err;
38*25da2beaSAndroid Build Coastguard Worker }
39*25da2beaSAndroid Build Coastguard Worker ret = cqe->res;
40*25da2beaSAndroid Build Coastguard Worker io_uring_cqe_seen(ring, cqe);
41*25da2beaSAndroid Build Coastguard Worker return ret;
42*25da2beaSAndroid Build Coastguard Worker err:
43*25da2beaSAndroid Build Coastguard Worker return 1;
44*25da2beaSAndroid Build Coastguard Worker }
45*25da2beaSAndroid Build Coastguard Worker
files_linked_ok(const char * fn1,const char * fn2)46*25da2beaSAndroid Build Coastguard Worker int files_linked_ok(const char* fn1, const char *fn2)
47*25da2beaSAndroid Build Coastguard Worker {
48*25da2beaSAndroid Build Coastguard Worker struct stat s1, s2;
49*25da2beaSAndroid Build Coastguard Worker
50*25da2beaSAndroid Build Coastguard Worker if (stat(fn1, &s1)) {
51*25da2beaSAndroid Build Coastguard Worker fprintf(stderr, "stat(%s): %s\n", fn1, strerror(errno));
52*25da2beaSAndroid Build Coastguard Worker return 0;
53*25da2beaSAndroid Build Coastguard Worker }
54*25da2beaSAndroid Build Coastguard Worker if (stat(fn2, &s2)) {
55*25da2beaSAndroid Build Coastguard Worker fprintf(stderr, "stat(%s): %s\n", fn2, strerror(errno));
56*25da2beaSAndroid Build Coastguard Worker return 0;
57*25da2beaSAndroid Build Coastguard Worker }
58*25da2beaSAndroid Build Coastguard Worker if (s1.st_dev != s2.st_dev || s1.st_ino != s2.st_ino) {
59*25da2beaSAndroid Build Coastguard Worker fprintf(stderr, "linked files have different device / inode numbers\n");
60*25da2beaSAndroid Build Coastguard Worker return 0;
61*25da2beaSAndroid Build Coastguard Worker }
62*25da2beaSAndroid Build Coastguard Worker if (s1.st_nlink != 2 || s2.st_nlink != 2) {
63*25da2beaSAndroid Build Coastguard Worker fprintf(stderr, "linked files have unexpected links count\n");
64*25da2beaSAndroid Build Coastguard Worker return 0;
65*25da2beaSAndroid Build Coastguard Worker }
66*25da2beaSAndroid Build Coastguard Worker return 1;
67*25da2beaSAndroid Build Coastguard Worker }
68*25da2beaSAndroid Build Coastguard Worker
main(int argc,char * argv[])69*25da2beaSAndroid Build Coastguard Worker int main(int argc, char *argv[])
70*25da2beaSAndroid Build Coastguard Worker {
71*25da2beaSAndroid Build Coastguard Worker static const char target[] = "io_uring-linkat-test-target";
72*25da2beaSAndroid Build Coastguard Worker static const char linkname[] = "io_uring-linkat-test-link";
73*25da2beaSAndroid Build Coastguard Worker int ret;
74*25da2beaSAndroid Build Coastguard Worker struct io_uring ring;
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 ret = io_uring_queue_init(8, &ring, 0);
80*25da2beaSAndroid Build Coastguard Worker if (ret) {
81*25da2beaSAndroid Build Coastguard Worker fprintf(stderr, "queue init failed: %d\n", ret);
82*25da2beaSAndroid Build Coastguard Worker return ret;
83*25da2beaSAndroid Build Coastguard Worker }
84*25da2beaSAndroid Build Coastguard Worker
85*25da2beaSAndroid Build Coastguard Worker ret = open(target, O_CREAT | O_RDWR | O_EXCL, 0600);
86*25da2beaSAndroid Build Coastguard Worker if (ret < 0) {
87*25da2beaSAndroid Build Coastguard Worker perror("open");
88*25da2beaSAndroid Build Coastguard Worker goto err;
89*25da2beaSAndroid Build Coastguard Worker }
90*25da2beaSAndroid Build Coastguard Worker if (write(ret, "linktest", 8) != 8) {
91*25da2beaSAndroid Build Coastguard Worker close(ret);
92*25da2beaSAndroid Build Coastguard Worker goto err1;
93*25da2beaSAndroid Build Coastguard Worker }
94*25da2beaSAndroid Build Coastguard Worker close(ret);
95*25da2beaSAndroid Build Coastguard Worker
96*25da2beaSAndroid Build Coastguard Worker ret = do_linkat(&ring, target, linkname);
97*25da2beaSAndroid Build Coastguard Worker if (ret < 0) {
98*25da2beaSAndroid Build Coastguard Worker if (ret == -EBADF || ret == -EINVAL) {
99*25da2beaSAndroid Build Coastguard Worker fprintf(stdout, "linkat not supported, skipping\n");
100*25da2beaSAndroid Build Coastguard Worker goto out;
101*25da2beaSAndroid Build Coastguard Worker }
102*25da2beaSAndroid Build Coastguard Worker fprintf(stderr, "linkat: %s\n", strerror(-ret));
103*25da2beaSAndroid Build Coastguard Worker goto err1;
104*25da2beaSAndroid Build Coastguard Worker } else if (ret) {
105*25da2beaSAndroid Build Coastguard Worker goto err1;
106*25da2beaSAndroid Build Coastguard Worker }
107*25da2beaSAndroid Build Coastguard Worker
108*25da2beaSAndroid Build Coastguard Worker if (!files_linked_ok(linkname, target))
109*25da2beaSAndroid Build Coastguard Worker goto err2;
110*25da2beaSAndroid Build Coastguard Worker
111*25da2beaSAndroid Build Coastguard Worker ret = do_linkat(&ring, target, linkname);
112*25da2beaSAndroid Build Coastguard Worker if (ret != -EEXIST) {
113*25da2beaSAndroid Build Coastguard Worker fprintf(stderr, "test_linkat linkname already exists failed: %d\n", ret);
114*25da2beaSAndroid Build Coastguard Worker goto err2;
115*25da2beaSAndroid Build Coastguard Worker }
116*25da2beaSAndroid Build Coastguard Worker
117*25da2beaSAndroid Build Coastguard Worker ret = do_linkat(&ring, target, "surely/this/does/not/exist");
118*25da2beaSAndroid Build Coastguard Worker if (ret != -ENOENT) {
119*25da2beaSAndroid Build Coastguard Worker fprintf(stderr, "test_linkat no parent failed: %d\n", ret);
120*25da2beaSAndroid Build Coastguard Worker goto err2;
121*25da2beaSAndroid Build Coastguard Worker }
122*25da2beaSAndroid Build Coastguard Worker
123*25da2beaSAndroid Build Coastguard Worker out:
124*25da2beaSAndroid Build Coastguard Worker unlinkat(AT_FDCWD, linkname, 0);
125*25da2beaSAndroid Build Coastguard Worker unlinkat(AT_FDCWD, target, 0);
126*25da2beaSAndroid Build Coastguard Worker io_uring_queue_exit(&ring);
127*25da2beaSAndroid Build Coastguard Worker return 0;
128*25da2beaSAndroid Build Coastguard Worker err2:
129*25da2beaSAndroid Build Coastguard Worker unlinkat(AT_FDCWD, linkname, 0);
130*25da2beaSAndroid Build Coastguard Worker err1:
131*25da2beaSAndroid Build Coastguard Worker unlinkat(AT_FDCWD, target, 0);
132*25da2beaSAndroid Build Coastguard Worker err:
133*25da2beaSAndroid Build Coastguard Worker io_uring_queue_exit(&ring);
134*25da2beaSAndroid Build Coastguard Worker return 1;
135*25da2beaSAndroid Build Coastguard Worker }
136*25da2beaSAndroid Build Coastguard Worker
137