xref: /xnu-10002.1.13/tests/rename_excl.c (revision 1031c584a5e37aff177559b9f69dbd3c8c3fd30a)
1*1031c584SApple OSS Distributions #include <darwintest.h>
2*1031c584SApple OSS Distributions #include <darwintest_utils.h>
3*1031c584SApple OSS Distributions #include <dirent.h>
4*1031c584SApple OSS Distributions #include <errno.h>
5*1031c584SApple OSS Distributions #include <fcntl.h>
6*1031c584SApple OSS Distributions #include <stdio.h>
7*1031c584SApple OSS Distributions #include <stdlib.h>
8*1031c584SApple OSS Distributions #include <unistd.h>
9*1031c584SApple OSS Distributions #include <sys/stat.h>
10*1031c584SApple OSS Distributions 
11*1031c584SApple OSS Distributions 
12*1031c584SApple OSS Distributions T_GLOBAL_META(
13*1031c584SApple OSS Distributions 	T_META_NAMESPACE("xnu.vfs"),
14*1031c584SApple OSS Distributions 	T_META_CHECK_LEAKS(false)
15*1031c584SApple OSS Distributions 	);
16*1031c584SApple OSS Distributions 
17*1031c584SApple OSS Distributions #define TEST_DIR         "rename_dir"
18*1031c584SApple OSS Distributions #define TEST_FILE1       TEST_DIR "/file1"
19*1031c584SApple OSS Distributions #define TEST_FILE1_UC    TEST_DIR "/FILE1"
20*1031c584SApple OSS Distributions #define TEST_FILE2       TEST_DIR "/file2"
21*1031c584SApple OSS Distributions #define TEST_FILE3_HL    TEST_DIR "/file3"
22*1031c584SApple OSS Distributions 
23*1031c584SApple OSS Distributions static void
cleanup(void)24*1031c584SApple OSS Distributions cleanup(void)
25*1031c584SApple OSS Distributions {
26*1031c584SApple OSS Distributions 	(void) remove(TEST_FILE1);
27*1031c584SApple OSS Distributions 	(void) remove(TEST_FILE1_UC);
28*1031c584SApple OSS Distributions 	(void) remove(TEST_FILE2);
29*1031c584SApple OSS Distributions 	(void) remove(TEST_FILE3_HL);
30*1031c584SApple OSS Distributions 	(void) rmdir(TEST_DIR);
31*1031c584SApple OSS Distributions }
32*1031c584SApple OSS Distributions 
33*1031c584SApple OSS Distributions /*
34*1031c584SApple OSS Distributions  * This unit-test validates the behavior of renamex_np() with RENAME_EXCL flag.
35*1031c584SApple OSS Distributions  * On either a case-insensitve/case-sensitive volume:
36*1031c584SApple OSS Distributions  * 1. rename from source to existing target should succeed when the change is
37*1031c584SApple OSS Distributions  *    only case-variant (for e.g rename_dir/file1 -> rename_dir/FILE1)
38*1031c584SApple OSS Distributions  * 2. rename from source to existing target should fail with EEXIST
39*1031c584SApple OSS Distributions  * 3. rename from source to existing target which is a hardlink of the source
40*1031c584SApple OSS Distributions  *    should fail with EEXIST
41*1031c584SApple OSS Distributions  *
42*1031c584SApple OSS Distributions  * On case-insensitive volume:
43*1031c584SApple OSS Distributions  * 1. rename from source to itself should succeed
44*1031c584SApple OSS Distributions  *    (rename_dir/file1 -> rename_dir/file1)
45*1031c584SApple OSS Distributions  *
46*1031c584SApple OSS Distributions  * On case-sensitive volume:
47*1031c584SApple OSS Distributions  * 1. rename from source to itself should fail with EEXIST
48*1031c584SApple OSS Distributions  *    (rename_dir/file1 -> rename_dir/file1)
49*1031c584SApple OSS Distributions  */
50*1031c584SApple OSS Distributions 
51*1031c584SApple OSS Distributions T_DECL(rename_excl_with_case_variant,
52*1031c584SApple OSS Distributions     "test renamex_np() with RENAME_EXCL flag for files with case variants")
53*1031c584SApple OSS Distributions {
54*1031c584SApple OSS Distributions 	const char *tmpdir = dt_tmpdir();
55*1031c584SApple OSS Distributions 	long case_sensitive_vol;
56*1031c584SApple OSS Distributions 	int err, saved_errno;
57*1031c584SApple OSS Distributions 	int fd;
58*1031c584SApple OSS Distributions 
59*1031c584SApple OSS Distributions 	T_SETUPBEGIN;
60*1031c584SApple OSS Distributions 
61*1031c584SApple OSS Distributions 	atexit(cleanup);
62*1031c584SApple OSS Distributions 
63*1031c584SApple OSS Distributions 	T_ASSERT_POSIX_ZERO(chdir(tmpdir),
64*1031c584SApple OSS Distributions 	    "Setup: changing to tmpdir: %s", tmpdir);
65*1031c584SApple OSS Distributions 
66*1031c584SApple OSS Distributions 	T_ASSERT_POSIX_SUCCESS(mkdir(TEST_DIR, 0777),
67*1031c584SApple OSS Distributions 	    "Setup: creating test dir: %s", TEST_DIR);
68*1031c584SApple OSS Distributions 
69*1031c584SApple OSS Distributions 	T_WITH_ERRNO;
70*1031c584SApple OSS Distributions 	fd = open(TEST_FILE1, O_CREAT | O_RDWR, 0666);
71*1031c584SApple OSS Distributions 	T_ASSERT_TRUE(fd != -1, "Creating test file1: %s", TEST_FILE1);
72*1031c584SApple OSS Distributions 
73*1031c584SApple OSS Distributions 	T_ASSERT_POSIX_SUCCESS(close(fd), "Closing test file1: %s",
74*1031c584SApple OSS Distributions 	    TEST_FILE1);
75*1031c584SApple OSS Distributions 
76*1031c584SApple OSS Distributions 	T_WITH_ERRNO;
77*1031c584SApple OSS Distributions 	fd = open(TEST_FILE2, O_CREAT | O_RDWR, 0666);
78*1031c584SApple OSS Distributions 	T_ASSERT_TRUE(fd != -1, "Creating test file2: %s", TEST_FILE2);
79*1031c584SApple OSS Distributions 
80*1031c584SApple OSS Distributions 	T_ASSERT_POSIX_SUCCESS(close(fd), "Closing test file2: %s",
81*1031c584SApple OSS Distributions 	    TEST_FILE2);
82*1031c584SApple OSS Distributions 
83*1031c584SApple OSS Distributions 	T_ASSERT_POSIX_SUCCESS(link(TEST_FILE1, TEST_FILE3_HL),
84*1031c584SApple OSS Distributions 	    "Creating hardlink for %s from source: %s",
85*1031c584SApple OSS Distributions 	    TEST_FILE3_HL, TEST_FILE1);
86*1031c584SApple OSS Distributions 
87*1031c584SApple OSS Distributions 	case_sensitive_vol = pathconf(TEST_DIR, _PC_CASE_SENSITIVE);
88*1031c584SApple OSS Distributions 	T_ASSERT_TRUE(case_sensitive_vol != -1,
89*1031c584SApple OSS Distributions 	    "Checking if target volume is case-sensitive, is_case_sensitive: %ld",
90*1031c584SApple OSS Distributions 	    case_sensitive_vol);
91*1031c584SApple OSS Distributions 
92*1031c584SApple OSS Distributions 	T_SETUPEND;
93*1031c584SApple OSS Distributions 
94*1031c584SApple OSS Distributions 	err = renamex_np(TEST_FILE1, TEST_FILE2, RENAME_EXCL);
95*1031c584SApple OSS Distributions 	saved_errno = errno;
96*1031c584SApple OSS Distributions 	T_ASSERT_TRUE((err == -1 && saved_errno == EEXIST),
97*1031c584SApple OSS Distributions 	    "Renaming with RENAME_EXCL from source: %s to target: %s",
98*1031c584SApple OSS Distributions 	    TEST_FILE1, TEST_FILE2);
99*1031c584SApple OSS Distributions 
100*1031c584SApple OSS Distributions 	err = renamex_np(TEST_FILE1, TEST_FILE3_HL, RENAME_EXCL);
101*1031c584SApple OSS Distributions 	saved_errno = errno;
102*1031c584SApple OSS Distributions 	T_ASSERT_TRUE((err == -1 && saved_errno == EEXIST),
103*1031c584SApple OSS Distributions 	    "Renaming with RENAME_EXCL from source: %s to hardlink target: %s",
104*1031c584SApple OSS Distributions 	    TEST_FILE1, TEST_FILE3_HL);
105*1031c584SApple OSS Distributions 
106*1031c584SApple OSS Distributions 	if (case_sensitive_vol) {
107*1031c584SApple OSS Distributions 		err = renamex_np(TEST_FILE1, TEST_FILE1, RENAME_EXCL);
108*1031c584SApple OSS Distributions 		saved_errno = errno;
109*1031c584SApple OSS Distributions 		T_ASSERT_TRUE((err == -1 && saved_errno == EEXIST),
110*1031c584SApple OSS Distributions 		    "Renaming with RENAME_EXCL from source: %s to target: %s",
111*1031c584SApple OSS Distributions 		    TEST_FILE1, TEST_FILE1);
112*1031c584SApple OSS Distributions 	} else {
113*1031c584SApple OSS Distributions 		T_ASSERT_POSIX_SUCCESS(renamex_np(TEST_FILE1, TEST_FILE1, RENAME_EXCL),
114*1031c584SApple OSS Distributions 		    "Renaming with RENAME_EXCL from source: %s to target: %s",
115*1031c584SApple OSS Distributions 		    TEST_FILE1, TEST_FILE1);
116*1031c584SApple OSS Distributions 	}
117*1031c584SApple OSS Distributions 
118*1031c584SApple OSS Distributions 	T_ASSERT_POSIX_SUCCESS(renamex_np(TEST_FILE1, TEST_FILE1_UC, RENAME_EXCL),
119*1031c584SApple OSS Distributions 	    "Renaming with RENAME_EXCL from source: %s to target: %s",
120*1031c584SApple OSS Distributions 	    TEST_FILE1, TEST_FILE1_UC);
121*1031c584SApple OSS Distributions }
122