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(×_in, _times_in, sizeof(times_in));
100*19c3b8c2SApple OSS Distributions } else {
101*19c3b8c2SApple OSS Distributions memcpy(×_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, ×_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(×_in, _times_in, sizeof(times_in));
122*19c3b8c2SApple OSS Distributions } else {
123*19c3b8c2SApple OSS Distributions memcpy(×_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, ×_out, attrbuf_size, flags_out);
141*19c3b8c2SApple OSS Distributions }
142