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