123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960 |
- #include <sys/stat.h>
- #include <sys/time.h>
- #include <fcntl.h>
- #include <errno.h>
- #include "syscall.h"
- #define IS32BIT(x) !((x)+0x80000000ULL>>32)
- #define NS_SPECIAL(ns) ((ns)==UTIME_NOW || (ns)==UTIME_OMIT)
- int utimensat(int fd, const char *path, const struct timespec times[2], int flags)
- {
- int r;
- if (times && times[0].tv_nsec==UTIME_NOW && times[1].tv_nsec==UTIME_NOW)
- times = 0;
- #ifdef SYS_utimensat_time64
- r = -ENOSYS;
- time_t s0=0, s1=0;
- long ns0=0, ns1=0;
- if (times) {
- ns0 = times[0].tv_nsec;
- ns1 = times[1].tv_nsec;
- if (!NS_SPECIAL(ns0)) s0 = times[0].tv_sec;
- if (!NS_SPECIAL(ns1)) s1 = times[1].tv_sec;
- }
- if (SYS_utimensat == SYS_utimensat_time64 || !IS32BIT(s0) || !IS32BIT(s1))
- r = __syscall(SYS_utimensat_time64, fd, path, times ?
- ((long long[]){s0, ns0, s1, ns1}) : 0, flags);
- if (SYS_utimensat == SYS_utimensat_time64 || r!=-ENOSYS)
- return __syscall_ret(r);
- if (!IS32BIT(s0) || !IS32BIT(s1))
- return __syscall_ret(-ENOTSUP);
- r = __syscall(SYS_utimensat, fd, path,
- times ? ((long[]){s0, ns0, s1, ns1}) : 0, flags);
- #else
- r = __syscall(SYS_utimensat, fd, path, times, flags);
- #endif
- #ifdef SYS_futimesat
- if (r != -ENOSYS || flags) return __syscall_ret(r);
- long *tv=0, tmp[4];
- if (times) {
- int i;
- tv = tmp;
- for (i=0; i<2; i++) {
- if (times[i].tv_nsec >= 1000000000ULL) {
- if (NS_SPECIAL(times[i].tv_nsec))
- return __syscall_ret(-ENOSYS);
- return __syscall_ret(-EINVAL);
- }
- tmp[2*i+0] = times[i].tv_sec;
- tmp[2*i+1] = times[i].tv_nsec / 1000;
- }
- }
- r = __syscall(SYS_futimesat, fd, path, tv);
- if (r != -ENOSYS || fd != AT_FDCWD) return __syscall_ret(r);
- r = __syscall(SYS_utimes, path, tv);
- #endif
- return __syscall_ret(r);
- }
|