xref: /xnu-12377.81.4/tests/utimensat.c (revision 043036a2b3718f7f0be807e2870f8f47d3fa0796)
1*043036a2SApple OSS Distributions #include <sys/cdefs.h>
2*043036a2SApple OSS Distributions #include <sys/param.h>
3*043036a2SApple OSS Distributions #include <sys/stat.h>
4*043036a2SApple OSS Distributions #include <sys/time.h>
5*043036a2SApple OSS Distributions #include <errno.h>
6*043036a2SApple OSS Distributions #include <fcntl.h>
7*043036a2SApple OSS Distributions #include <limits.h>
8*043036a2SApple OSS Distributions #include <paths.h>
9*043036a2SApple OSS Distributions #include <stdio.h>
10*043036a2SApple OSS Distributions #include <string.h>
11*043036a2SApple OSS Distributions #include <unistd.h>
12*043036a2SApple OSS Distributions 
13*043036a2SApple OSS Distributions #include <darwintest.h>
14*043036a2SApple OSS Distributions #include <darwintest_utils.h>
15*043036a2SApple OSS Distributions 
16*043036a2SApple OSS Distributions T_GLOBAL_META(T_META_RUN_CONCURRENTLY(true));
17*043036a2SApple OSS Distributions 
18*043036a2SApple OSS Distributions #define FILENAME "utimensat"
19*043036a2SApple OSS Distributions 
20*043036a2SApple OSS Distributions static const struct timespec tptr[][2] = {
21*043036a2SApple OSS Distributions 	{ { 0x12345678, 987654321 }, { 0x15263748, 123456789 }, },
22*043036a2SApple OSS Distributions 
23*043036a2SApple OSS Distributions 	{ { 0, UTIME_NOW }, { 0x15263748, 123456789 }, },
24*043036a2SApple OSS Distributions 	{ { 0x12345678, 987654321 }, { 0, UTIME_NOW }, },
25*043036a2SApple OSS Distributions 	{ { 0, UTIME_NOW }, { 0, UTIME_NOW }, },
26*043036a2SApple OSS Distributions 
27*043036a2SApple OSS Distributions 	{ { 0, UTIME_OMIT }, { 0x15263748, 123456789 }, },
28*043036a2SApple OSS Distributions 	{ { 0x12345678, 987654321 }, { 0, UTIME_OMIT }, },
29*043036a2SApple OSS Distributions 	{ { 0, UTIME_OMIT }, { 0, UTIME_OMIT }, },
30*043036a2SApple OSS Distributions 
31*043036a2SApple OSS Distributions 	{ { 0, UTIME_NOW }, { 0, UTIME_OMIT }, },
32*043036a2SApple OSS Distributions 	{ { 0, UTIME_OMIT }, { 0, UTIME_NOW }, },
33*043036a2SApple OSS Distributions };
34*043036a2SApple OSS Distributions 
35*043036a2SApple OSS Distributions T_DECL(utimensat, "Try various versions of utimensat")
36*043036a2SApple OSS Distributions {
37*043036a2SApple OSS Distributions 	T_SETUPBEGIN;
38*043036a2SApple OSS Distributions 	T_ASSERT_POSIX_SUCCESS(chdir(dt_tmpdir()), NULL);
39*043036a2SApple OSS Distributions 	// Skip the test if the current working directory is not on APFS.
40*043036a2SApple OSS Distributions 	struct statfs sfs = { 0 };
41*043036a2SApple OSS Distributions 	T_QUIET; T_ASSERT_POSIX_SUCCESS(statfs(".", &sfs), NULL);
42*043036a2SApple OSS Distributions 	if (memcmp(&sfs.f_fstypename[0], "apfs", strlen("apfs")) != 0) {
43*043036a2SApple OSS Distributions 		T_SKIP("utimensat is APFS-only, but working directory is non-APFS");
44*043036a2SApple OSS Distributions 	}
45*043036a2SApple OSS Distributions 	T_SETUPEND;
46*043036a2SApple OSS Distributions 
47*043036a2SApple OSS Distributions 	struct stat pre_st, post_st;
48*043036a2SApple OSS Distributions 	int fd;
49*043036a2SApple OSS Distributions 
50*043036a2SApple OSS Distributions 	T_ASSERT_POSIX_SUCCESS((fd = open(FILENAME, O_CREAT | O_RDWR, 0644)), NULL);
51*043036a2SApple OSS Distributions 	T_ASSERT_POSIX_SUCCESS(close(fd), NULL);
52*043036a2SApple OSS Distributions 
53*043036a2SApple OSS Distributions 	for (size_t i = 0; i < sizeof(tptr) / sizeof(tptr[0]); i++) {
54*043036a2SApple OSS Distributions 		T_LOG("=== {%ld, %ld} {%ld, %ld} ===",
55*043036a2SApple OSS Distributions 		    tptr[i][0].tv_sec, tptr[i][0].tv_nsec,
56*043036a2SApple OSS Distributions 		    tptr[i][1].tv_sec, tptr[i][1].tv_nsec);
57*043036a2SApple OSS Distributions 
58*043036a2SApple OSS Distributions 		struct timespec now;
59*043036a2SApple OSS Distributions 		clock_gettime(CLOCK_REALTIME, &now);
60*043036a2SApple OSS Distributions 
61*043036a2SApple OSS Distributions 		T_ASSERT_POSIX_SUCCESS(stat(FILENAME, &pre_st), NULL);
62*043036a2SApple OSS Distributions 		T_ASSERT_POSIX_SUCCESS(utimensat(AT_FDCWD, FILENAME, tptr[i], 0), NULL);
63*043036a2SApple OSS Distributions 		T_ASSERT_POSIX_SUCCESS(stat(FILENAME, &post_st), NULL);
64*043036a2SApple OSS Distributions 
65*043036a2SApple OSS Distributions 		if (tptr[i][0].tv_nsec == UTIME_NOW) {
66*043036a2SApple OSS Distributions 			T_ASSERT_GE(post_st.st_atimespec.tv_sec, now.tv_sec, NULL);
67*043036a2SApple OSS Distributions 		} else if (tptr[i][0].tv_nsec == UTIME_OMIT) {
68*043036a2SApple OSS Distributions 			T_ASSERT_EQ(post_st.st_atimespec.tv_sec, pre_st.st_atimespec.tv_sec, NULL);
69*043036a2SApple OSS Distributions 			T_ASSERT_EQ(post_st.st_atimespec.tv_nsec, pre_st.st_atimespec.tv_nsec, NULL);
70*043036a2SApple OSS Distributions 		} else {
71*043036a2SApple OSS Distributions 			T_ASSERT_EQ(post_st.st_atimespec.tv_sec, tptr[i][0].tv_sec, NULL);
72*043036a2SApple OSS Distributions 			T_ASSERT_EQ(post_st.st_atimespec.tv_nsec, tptr[i][0].tv_nsec, NULL);
73*043036a2SApple OSS Distributions 		}
74*043036a2SApple OSS Distributions 
75*043036a2SApple OSS Distributions 		if (tptr[i][1].tv_nsec == UTIME_NOW) {
76*043036a2SApple OSS Distributions 			T_ASSERT_GE(post_st.st_mtimespec.tv_sec, now.tv_sec, NULL);
77*043036a2SApple OSS Distributions 		} else if (tptr[i][1].tv_nsec == UTIME_OMIT) {
78*043036a2SApple OSS Distributions 			T_ASSERT_EQ(post_st.st_mtimespec.tv_sec, pre_st.st_mtimespec.tv_sec, NULL);
79*043036a2SApple OSS Distributions 			T_ASSERT_EQ(post_st.st_mtimespec.tv_nsec, pre_st.st_mtimespec.tv_nsec, NULL);
80*043036a2SApple OSS Distributions 		} else {
81*043036a2SApple OSS Distributions 			T_ASSERT_EQ(post_st.st_mtimespec.tv_sec, tptr[i][1].tv_sec, NULL);
82*043036a2SApple OSS Distributions 			T_ASSERT_EQ(post_st.st_mtimespec.tv_nsec, tptr[i][1].tv_nsec, NULL);
83*043036a2SApple OSS Distributions 		}
84*043036a2SApple OSS Distributions 	}
85*043036a2SApple OSS Distributions }
86