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