summaryrefslogtreecommitdiff
path: root/src/stat/utimensat.c
blob: 730723a9ea70d4f88e20155f1f7e33b366a2281e (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
#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);
}