xref: /xnu-8792.81.2/libsyscall/wrappers/utimensat.c (revision 19c3b8c28c31cb8130e034cfb5df6bf9ba342d90)
1*19c3b8c2SApple OSS Distributions /*
2*19c3b8c2SApple OSS Distributions  * Copyright (c) 2006, 2017 Apple Computer, Inc. All rights reserved.
3*19c3b8c2SApple OSS Distributions  *
4*19c3b8c2SApple OSS Distributions  * @APPLE_LICENSE_HEADER_START@
5*19c3b8c2SApple OSS Distributions  *
6*19c3b8c2SApple OSS Distributions  * This file contains Original Code and/or Modifications of Original Code
7*19c3b8c2SApple OSS Distributions  * as defined in and that are subject to the Apple Public Source License
8*19c3b8c2SApple OSS Distributions  * Version 2.0 (the 'License'). You may not use this file except in
9*19c3b8c2SApple OSS Distributions  * compliance with the License. Please obtain a copy of the License at
10*19c3b8c2SApple OSS Distributions  * http://www.opensource.apple.com/apsl/ and read it before using this
11*19c3b8c2SApple OSS Distributions  * file.
12*19c3b8c2SApple OSS Distributions  *
13*19c3b8c2SApple OSS Distributions  * The Original Code and all software distributed under the License are
14*19c3b8c2SApple OSS Distributions  * distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER
15*19c3b8c2SApple OSS Distributions  * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
16*19c3b8c2SApple OSS Distributions  * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
17*19c3b8c2SApple OSS Distributions  * FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR NON-INFRINGEMENT.
18*19c3b8c2SApple OSS Distributions  * Please see the License for the specific language governing rights and
19*19c3b8c2SApple OSS Distributions  * limitations under the License.
20*19c3b8c2SApple OSS Distributions  *
21*19c3b8c2SApple OSS Distributions  * @APPLE_LICENSE_HEADER_END@
22*19c3b8c2SApple OSS Distributions  */
23*19c3b8c2SApple OSS Distributions 
24*19c3b8c2SApple OSS Distributions #include <sys/types.h>
25*19c3b8c2SApple OSS Distributions #include <sys/stat.h>
26*19c3b8c2SApple OSS Distributions #include <sys/attr.h>
27*19c3b8c2SApple OSS Distributions #include <sys/time.h>
28*19c3b8c2SApple OSS Distributions #include <sys/fcntl.h>
29*19c3b8c2SApple OSS Distributions #include <unistd.h>
30*19c3b8c2SApple OSS Distributions #include <strings.h>
31*19c3b8c2SApple OSS Distributions 
32*19c3b8c2SApple OSS Distributions extern int __gettimeofday(struct timeval *, struct timezone *);
33*19c3b8c2SApple OSS Distributions extern int __commpage_gettimeofday(struct timeval *);
34*19c3b8c2SApple OSS Distributions 
35*19c3b8c2SApple OSS Distributions static struct timespec times_now[2] = {
36*19c3b8c2SApple OSS Distributions 	{ .tv_nsec = UTIME_NOW },
37*19c3b8c2SApple OSS Distributions 	{ .tv_nsec = UTIME_NOW }
38*19c3b8c2SApple OSS Distributions };
39*19c3b8c2SApple OSS Distributions 
40*19c3b8c2SApple OSS Distributions /*
41*19c3b8c2SApple OSS Distributions  * Resolve any UTIME_NOW or UTIME_OMIT and return the attributes buffer and
42*19c3b8c2SApple OSS Distributions  * attributes to pass.  Assumes times_in is writable.
43*19c3b8c2SApple OSS Distributions  */
44*19c3b8c2SApple OSS Distributions static int
prepare_times_array_and_attrs(struct timespec times_in[2],struct timespec times_out[2],size_t * times_out_size,int * flags)45*19c3b8c2SApple OSS Distributions prepare_times_array_and_attrs(struct timespec times_in[2],
46*19c3b8c2SApple OSS Distributions     struct timespec times_out[2], size_t *times_out_size, int *flags)
47*19c3b8c2SApple OSS Distributions {
48*19c3b8c2SApple OSS Distributions 	if (times_in[0].tv_nsec == UTIME_OMIT &&
49*19c3b8c2SApple OSS Distributions 	    times_in[1].tv_nsec == UTIME_OMIT) {
50*19c3b8c2SApple OSS Distributions 		return 0;
51*19c3b8c2SApple OSS Distributions 	}
52*19c3b8c2SApple OSS Distributions 
53*19c3b8c2SApple OSS Distributions 	if (times_in[0].tv_nsec == UTIME_NOW ||
54*19c3b8c2SApple OSS Distributions 	    times_in[1].tv_nsec == UTIME_NOW) {
55*19c3b8c2SApple OSS Distributions 		*flags |= FSOPT_UTIMES_NULL;
56*19c3b8c2SApple OSS Distributions 		struct timespec now = {};
57*19c3b8c2SApple OSS Distributions 		{
58*19c3b8c2SApple OSS Distributions 			/*
59*19c3b8c2SApple OSS Distributions 			 * TODO: Replace with nanosecond time when available
60*19c3b8c2SApple OSS Distributions 			 */
61*19c3b8c2SApple OSS Distributions 			struct timeval tv;
62*19c3b8c2SApple OSS Distributions 			if (__commpage_gettimeofday(&tv) != 0) {
63*19c3b8c2SApple OSS Distributions 				__gettimeofday(&tv, NULL);
64*19c3b8c2SApple OSS Distributions 			}
65*19c3b8c2SApple OSS Distributions 			TIMEVAL_TO_TIMESPEC(&tv, &now);
66*19c3b8c2SApple OSS Distributions 		}
67*19c3b8c2SApple OSS Distributions 
68*19c3b8c2SApple OSS Distributions 		if (times_in[0].tv_nsec == UTIME_NOW) {
69*19c3b8c2SApple OSS Distributions 			times_in[0] = now;
70*19c3b8c2SApple OSS Distributions 		}
71*19c3b8c2SApple OSS Distributions 		if (times_in[1].tv_nsec == UTIME_NOW) {
72*19c3b8c2SApple OSS Distributions 			times_in[1] = now;
73*19c3b8c2SApple OSS Distributions 		}
74*19c3b8c2SApple OSS Distributions 	}
75*19c3b8c2SApple OSS Distributions 
76*19c3b8c2SApple OSS Distributions 	int attrs = 0;
77*19c3b8c2SApple OSS Distributions 	*times_out_size = 0;
78*19c3b8c2SApple OSS Distributions 	struct timespec *times_cursor = times_out;
79*19c3b8c2SApple OSS Distributions 	if (times_in[1].tv_nsec != UTIME_OMIT) {
80*19c3b8c2SApple OSS Distributions 		attrs |= ATTR_CMN_MODTIME;
81*19c3b8c2SApple OSS Distributions 		*times_cursor++ = times_in[1];
82*19c3b8c2SApple OSS Distributions 		*times_out_size += sizeof(struct timespec);
83*19c3b8c2SApple OSS Distributions 	}
84*19c3b8c2SApple OSS Distributions 	if (times_in[0].tv_nsec != UTIME_OMIT) {
85*19c3b8c2SApple OSS Distributions 		attrs |= ATTR_CMN_ACCTIME;
86*19c3b8c2SApple OSS Distributions 		*times_cursor = times_in[0];
87*19c3b8c2SApple OSS Distributions 		*times_out_size += sizeof(struct timespec);
88*19c3b8c2SApple OSS Distributions 	}
89*19c3b8c2SApple OSS Distributions 	return attrs;
90*19c3b8c2SApple OSS Distributions }
91*19c3b8c2SApple OSS Distributions 
92*19c3b8c2SApple OSS Distributions int
futimens(int fd,const struct timespec _times_in[2])93*19c3b8c2SApple OSS Distributions futimens(int fd, const struct timespec _times_in[2])
94*19c3b8c2SApple OSS Distributions {
95*19c3b8c2SApple OSS Distributions 	struct timespec times_in[2];
96*19c3b8c2SApple OSS Distributions 	int flags_out;
97*19c3b8c2SApple OSS Distributions 
98*19c3b8c2SApple OSS Distributions 	if (_times_in != NULL) {
99*19c3b8c2SApple OSS Distributions 		memcpy(&times_in, _times_in, sizeof(times_in));
100*19c3b8c2SApple OSS Distributions 	} else {
101*19c3b8c2SApple OSS Distributions 		memcpy(&times_in, times_now, sizeof(times_in));
102*19c3b8c2SApple OSS Distributions 	}
103*19c3b8c2SApple OSS Distributions 
104*19c3b8c2SApple OSS Distributions 	size_t attrbuf_size = 0;
105*19c3b8c2SApple OSS Distributions 	struct timespec times_out[2] = {};
106*19c3b8c2SApple OSS Distributions 	struct attrlist a = {
107*19c3b8c2SApple OSS Distributions 		.bitmapcount = ATTR_BIT_MAP_COUNT
108*19c3b8c2SApple OSS Distributions 	};
109*19c3b8c2SApple OSS Distributions 	a.commonattr = prepare_times_array_and_attrs(times_in, times_out, &attrbuf_size, &flags_out);
110*19c3b8c2SApple OSS Distributions 
111*19c3b8c2SApple OSS Distributions 	return fsetattrlist(fd, &a, &times_out, attrbuf_size, 0);
112*19c3b8c2SApple OSS Distributions }
113*19c3b8c2SApple OSS Distributions 
114*19c3b8c2SApple OSS Distributions int
utimensat(int fd,const char * path,const struct timespec _times_in[2],int flags)115*19c3b8c2SApple OSS Distributions utimensat(int fd, const char *path, const struct timespec _times_in[2], int flags)
116*19c3b8c2SApple OSS Distributions {
117*19c3b8c2SApple OSS Distributions 	struct timespec times_in[2];
118*19c3b8c2SApple OSS Distributions 	int flags_out = 0;
119*19c3b8c2SApple OSS Distributions 
120*19c3b8c2SApple OSS Distributions 	if (_times_in != NULL) {
121*19c3b8c2SApple OSS Distributions 		memcpy(&times_in, _times_in, sizeof(times_in));
122*19c3b8c2SApple OSS Distributions 	} else {
123*19c3b8c2SApple OSS Distributions 		memcpy(&times_in, times_now, sizeof(times_in));
124*19c3b8c2SApple OSS Distributions 	}
125*19c3b8c2SApple OSS Distributions 
126*19c3b8c2SApple OSS Distributions 	size_t attrbuf_size = 0;
127*19c3b8c2SApple OSS Distributions 	struct timespec times_out[2] = {};
128*19c3b8c2SApple OSS Distributions 	struct attrlist a = {
129*19c3b8c2SApple OSS Distributions 		.bitmapcount = ATTR_BIT_MAP_COUNT
130*19c3b8c2SApple OSS Distributions 	};
131*19c3b8c2SApple OSS Distributions 	a.commonattr = prepare_times_array_and_attrs(times_in, times_out, &attrbuf_size, &flags_out);
132*19c3b8c2SApple OSS Distributions 
133*19c3b8c2SApple OSS Distributions 	if (flags & AT_SYMLINK_NOFOLLOW) {
134*19c3b8c2SApple OSS Distributions 		flags_out |= FSOPT_NOFOLLOW;
135*19c3b8c2SApple OSS Distributions 	}
136*19c3b8c2SApple OSS Distributions 	if (flags & AT_SYMLINK_NOFOLLOW_ANY) {
137*19c3b8c2SApple OSS Distributions 		flags_out |= FSOPT_NOFOLLOW_ANY;
138*19c3b8c2SApple OSS Distributions 	}
139*19c3b8c2SApple OSS Distributions 
140*19c3b8c2SApple OSS Distributions 	return setattrlistat(fd, path, &a, &times_out, attrbuf_size, flags_out);
141*19c3b8c2SApple OSS Distributions }
142