xref: /freebsd/contrib/ntp/libntp/lib/isc/win32/time.c (revision 5ca8e32633c4ffbbcd6762e5888b6a4ba0708c6c)
1 /*
2  * Copyright (C) 2004, 2006-2009  Internet Systems Consortium, Inc. ("ISC")
3  * Copyright (C) 1998-2001, 2003  Internet Software Consortium.
4  *
5  * Permission to use, copy, modify, and/or distribute this software for any
6  * purpose with or without fee is hereby granted, provided that the above
7  * copyright notice and this permission notice appear in all copies.
8  *
9  * THE SOFTWARE IS PROVIDED "AS IS" AND ISC DISCLAIMS ALL WARRANTIES WITH
10  * REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
11  * AND FITNESS.  IN NO EVENT SHALL ISC BE LIABLE FOR ANY SPECIAL, DIRECT,
12  * INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
13  * LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE
14  * OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
15  * PERFORMANCE OF THIS SOFTWARE.
16  */
17 
18 /* $Id: time.c,v 1.52 2009/08/14 07:51:08 marka Exp $ */
19 
20 #include <config.h>
21 
22 #include <errno.h>
23 #include <limits.h>
24 #include <stddef.h>
25 #include <stdlib.h>
26 #include <string.h>
27 #include <time.h>
28 
29 #include <windows.h>
30 
31 #include <isc/assertions.h>
32 #include <isc/time.h>
33 #include <isc/util.h>
34 
35 /*
36  * struct FILETIME uses "100-nanoseconds intervals".
37  * NS / S = 1000000000 (10^9).
38  * While it is reasonably obvious that this makes the needed
39  * conversion factor 10^7, it is coded this way for additional clarity.
40  */
41 #define NS_PER_S 	1000000000
42 #define NS_INTERVAL	100
43 #define INTERVALS_PER_S (NS_PER_S / NS_INTERVAL)
44 #define UINT64_MAX	_UI64_MAX
45 
46 /***
47  *** Absolute Times
48  ***/
49 
50 static isc_time_t epoch = { { 0, 0 } };
51 LIBISC_EXTERNAL_DATA isc_time_t *isc_time_epoch = &epoch;
52 
53 /***
54  *** Intervals
55  ***/
56 
57 static isc_interval_t zero_interval = { 0 };
58 LIBISC_EXTERNAL_DATA isc_interval_t *isc_interval_zero = &zero_interval;
59 
60 void
61 isc_interval_set(isc_interval_t *i, unsigned int seconds,
62 		 unsigned int nanoseconds)
63 {
64 	REQUIRE(i != NULL);
65 	REQUIRE(nanoseconds < NS_PER_S);
66 
67 	/*
68 	 * This rounds nanoseconds up not down.
69 	 */
70 	i->interval = (LONGLONG)seconds * INTERVALS_PER_S
71 		+ (nanoseconds + NS_INTERVAL - 1) / NS_INTERVAL;
72 }
73 
74 isc_boolean_t
75 isc_interval_iszero(const isc_interval_t *i) {
76 	REQUIRE(i != NULL);
77 	if (i->interval == 0)
78 		return (ISC_TRUE);
79 
80 	return (ISC_FALSE);
81 }
82 
83 void
84 isc_time_set(isc_time_t *t, unsigned int seconds, unsigned int nanoseconds) {
85 	SYSTEMTIME epoch = { 1970, 1, 4, 1, 0, 0, 0, 0 };
86 	FILETIME temp;
87 	ULARGE_INTEGER i1;
88 
89 	REQUIRE(t != NULL);
90 	REQUIRE(nanoseconds < NS_PER_S);
91 
92 	SystemTimeToFileTime(&epoch, &temp);
93 
94 	i1.LowPart = t->absolute.dwLowDateTime;
95 	i1.HighPart = t->absolute.dwHighDateTime;
96 
97 	i1.QuadPart += (unsigned __int64)nanoseconds/100;
98 	i1.QuadPart += (unsigned __int64)seconds*10000000;
99 
100 	t->absolute.dwLowDateTime = i1.LowPart;
101 	t->absolute.dwHighDateTime = i1.HighPart;
102 }
103 
104 void
105 isc_time_settoepoch(isc_time_t *t) {
106 	REQUIRE(t != NULL);
107 
108 	t->absolute.dwLowDateTime = 0;
109 	t->absolute.dwHighDateTime = 0;
110 }
111 
112 isc_boolean_t
113 isc_time_isepoch(const isc_time_t *t) {
114 	REQUIRE(t != NULL);
115 
116 	if (t->absolute.dwLowDateTime == 0 &&
117 	    t->absolute.dwHighDateTime == 0)
118 		return (ISC_TRUE);
119 
120 	return (ISC_FALSE);
121 }
122 
123 isc_result_t
124 isc_time_now(isc_time_t *t) {
125 	REQUIRE(t != NULL);
126 
127 	GetSystemTimeAsFileTime(&t->absolute);
128 
129 	return (ISC_R_SUCCESS);
130 }
131 
132 isc_result_t
133 isc_time_nowplusinterval(isc_time_t *t, const isc_interval_t *i) {
134 	ULARGE_INTEGER i1;
135 
136 	REQUIRE(t != NULL);
137 	REQUIRE(i != NULL);
138 
139 	GetSystemTimeAsFileTime(&t->absolute);
140 
141 	i1.LowPart = t->absolute.dwLowDateTime;
142 	i1.HighPart = t->absolute.dwHighDateTime;
143 
144 	if (UINT64_MAX - i1.QuadPart < (unsigned __int64)i->interval)
145 		return (ISC_R_RANGE);
146 
147 	i1.QuadPart += i->interval;
148 
149 	t->absolute.dwLowDateTime  = i1.LowPart;
150 	t->absolute.dwHighDateTime = i1.HighPart;
151 
152 	return (ISC_R_SUCCESS);
153 }
154 
155 int
156 isc_time_compare(const isc_time_t *t1, const isc_time_t *t2) {
157 	REQUIRE(t1 != NULL && t2 != NULL);
158 
159 	return ((int)CompareFileTime(&t1->absolute, &t2->absolute));
160 }
161 
162 isc_result_t
163 isc_time_add(const isc_time_t *t, const isc_interval_t *i, isc_time_t *result)
164 {
165 	ULARGE_INTEGER i1;
166 
167 	REQUIRE(t != NULL && i != NULL && result != NULL);
168 
169 	i1.LowPart = t->absolute.dwLowDateTime;
170 	i1.HighPart = t->absolute.dwHighDateTime;
171 
172 	if (UINT64_MAX - i1.QuadPart < (unsigned __int64)i->interval)
173 		return (ISC_R_RANGE);
174 
175 	i1.QuadPart += i->interval;
176 
177 	result->absolute.dwLowDateTime = i1.LowPart;
178 	result->absolute.dwHighDateTime = i1.HighPart;
179 
180 	return (ISC_R_SUCCESS);
181 }
182 
183 isc_result_t
184 isc_time_subtract(const isc_time_t *t, const isc_interval_t *i,
185 		  isc_time_t *result) {
186 	ULARGE_INTEGER i1;
187 
188 	REQUIRE(t != NULL && i != NULL && result != NULL);
189 
190 	i1.LowPart = t->absolute.dwLowDateTime;
191 	i1.HighPart = t->absolute.dwHighDateTime;
192 
193 	if (i1.QuadPart < (unsigned __int64) i->interval)
194 		return (ISC_R_RANGE);
195 
196 	i1.QuadPart -= i->interval;
197 
198 	result->absolute.dwLowDateTime = i1.LowPart;
199 	result->absolute.dwHighDateTime = i1.HighPart;
200 
201 	return (ISC_R_SUCCESS);
202 }
203 
204 isc_uint64_t
205 isc_time_microdiff(const isc_time_t *t1, const isc_time_t *t2) {
206 	ULARGE_INTEGER i1, i2;
207 	LONGLONG i3;
208 
209 	REQUIRE(t1 != NULL && t2 != NULL);
210 
211 	i1.LowPart  = t1->absolute.dwLowDateTime;
212 	i1.HighPart = t1->absolute.dwHighDateTime;
213 	i2.LowPart  = t2->absolute.dwLowDateTime;
214 	i2.HighPart = t2->absolute.dwHighDateTime;
215 
216 	if (i1.QuadPart <= i2.QuadPart)
217 		return (0);
218 
219 	/*
220 	 * Convert to microseconds.
221 	 */
222 	i3 = (i1.QuadPart - i2.QuadPart) / 10;
223 
224 	return (i3);
225 }
226 
227 isc_uint32_t
228 isc_time_seconds(const isc_time_t *t) {
229 	SYSTEMTIME epoch = { 1970, 1, 4, 1, 0, 0, 0, 0 };
230 	FILETIME temp;
231 	ULARGE_INTEGER i1, i2;
232 	LONGLONG i3;
233 
234 	SystemTimeToFileTime(&epoch, &temp);
235 
236 	i1.LowPart  = t->absolute.dwLowDateTime;
237 	i1.HighPart = t->absolute.dwHighDateTime;
238 	i2.LowPart  = temp.dwLowDateTime;
239 	i2.HighPart = temp.dwHighDateTime;
240 
241 	i3 = (i1.QuadPart - i2.QuadPart) / 10000000;
242 
243 	return ((isc_uint32_t)i3);
244 }
245 
246 isc_uint32_t
247 isc_time_nanoseconds(const isc_time_t *t) {
248 	ULARGE_INTEGER i;
249 
250 	i.LowPart  = t->absolute.dwLowDateTime;
251 	i.HighPart = t->absolute.dwHighDateTime;
252 	return ((isc_uint32_t)(i.QuadPart % 10000000) * 100);
253 }
254 
255 void
256 isc_time_formattimestamp(const isc_time_t *t, char *buf, unsigned int len) {
257 	FILETIME localft;
258 	SYSTEMTIME st;
259 	char DateBuf[50];
260 	char TimeBuf[50];
261 
262 	static const char badtime[] = "99-Bad-9999 99:99:99.999";
263 
264 	REQUIRE(len > 0);
265 	if (FileTimeToLocalFileTime(&t->absolute, &localft) &&
266 	    FileTimeToSystemTime(&localft, &st)) {
267 		GetDateFormat(LOCALE_USER_DEFAULT, 0, &st, "dd-MMM-yyyy",
268 			      DateBuf, 50);
269 		GetTimeFormat(LOCALE_USER_DEFAULT, TIME_NOTIMEMARKER|
270 			      TIME_FORCE24HOURFORMAT, &st, NULL, TimeBuf, 50);
271 
272 		snprintf(buf, len, "%s %s.%03u", DateBuf, TimeBuf,
273 			 st.wMilliseconds);
274 
275 	} else
276 		snprintf(buf, len, badtime);
277 }
278 
279 void
280 isc_time_formathttptimestamp(const isc_time_t *t, char *buf, unsigned int len) {
281 	SYSTEMTIME st;
282 	char DateBuf[50];
283 	char TimeBuf[50];
284 
285 /* strftime() format: "%a, %d %b %Y %H:%M:%S GMT" */
286 
287 	REQUIRE(len > 0);
288 	if (FileTimeToSystemTime(&t->absolute, &st)) {
289 		GetDateFormat(LOCALE_USER_DEFAULT, 0, &st,
290 			      "ddd',', dd-MMM-yyyy", DateBuf, 50);
291 		GetTimeFormat(LOCALE_USER_DEFAULT,
292 			      TIME_NOTIMEMARKER | TIME_FORCE24HOURFORMAT,
293 			      &st, "hh':'mm':'ss", TimeBuf, 50);
294 
295 		snprintf(buf, len, "%s %s GMT", DateBuf, TimeBuf);
296 	} else {
297 		buf[0] = 0;
298 	}
299 }
300 
301 void
302 isc_time_formatISO8601(const isc_time_t *t, char *buf, unsigned int len) {
303 	SYSTEMTIME st;
304 	char DateBuf[50];
305 	char TimeBuf[50];
306 
307 /* strtime() format: "%Y-%m-%dT%H:%M:%SZ" */
308 
309 	REQUIRE(len > 0);
310 	if (FileTimeToSystemTime(&t->absolute, &st)) {
311 		GetDateFormat(LOCALE_NEUTRAL, 0, &st, "yyyy-MM-dd",
312 			      DateBuf, 50);
313 		GetTimeFormat(LOCALE_NEUTRAL,
314 			      TIME_NOTIMEMARKER | TIME_FORCE24HOURFORMAT,
315 			      &st, "hh':'mm':'ss", TimeBuf, 50);
316 		snprintf(buf, len, "%s%sZ", DateBuf, TimeBuf);
317 	} else {
318 		buf[0] = 0;
319 	}
320 }
321