xref: /aosp_15_r20/external/liburing/test/hardlink.c (revision 25da2bea747f3a93b4c30fd9708b0618ef55a0e6)
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