VirtualBox

source: vbox/trunk/include/iprt/time.h@ 69411

Last change on this file since 69411 was 69105, checked in by vboxsync, 7 years ago

include/iprt/: (C) year

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 33.2 KB
Line 
1/** @file
2 * IPRT - Time.
3 */
4
5/*
6 * Copyright (C) 2006-2017 Oracle Corporation
7 *
8 * This file is part of VirtualBox Open Source Edition (OSE), as
9 * available from http://www.virtualbox.org. This file is free software;
10 * you can redistribute it and/or modify it under the terms of the GNU
11 * General Public License (GPL) as published by the Free Software
12 * Foundation, in version 2 as it comes in the "COPYING" file of the
13 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
14 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
15 *
16 * The contents of this file may alternatively be used under the terms
17 * of the Common Development and Distribution License Version 1.0
18 * (CDDL) only, as it comes in the "COPYING.CDDL" file of the
19 * VirtualBox OSE distribution, in which case the provisions of the
20 * CDDL are applicable instead of those of the GPL.
21 *
22 * You may elect to license modified versions of this file under the
23 * terms and conditions of either the GPL or the CDDL or both.
24 */
25
26#ifndef ___iprt_time_h
27#define ___iprt_time_h
28
29#include <iprt/cdefs.h>
30#include <iprt/types.h>
31
32RT_C_DECLS_BEGIN
33
34/** @defgroup grp_rt_time RTTime - Time
35 * @ingroup grp_rt
36 * @{
37 */
38
39/** Time Specification.
40 *
41 * Use the inline RTTimeSpecGet/Set to operate on structure this so we
42 * can easily change the representation if required later.
43 *
44 * The current representation is in nanoseconds relative to the unix epoch
45 * (1970-01-01 00:00:00 UTC). This gives us an approximate span from
46 * 1678 to 2262 without sacrificing the resolution offered by the various
47 * host OSes (BSD & LINUX 1ns, NT 100ns).
48 */
49typedef struct RTTIMESPEC
50{
51 /** Nanoseconds since epoch.
52 * The name is intentially too long to be comfortable to use because you should be
53 * using inline helpers! */
54 int64_t i64NanosecondsRelativeToUnixEpoch;
55} RTTIMESPEC;
56
57
58/** @name RTTIMESPEC methods
59 * @{ */
60
61/**
62 * Gets the time as nanoseconds relative to the unix epoch.
63 *
64 * @returns Nanoseconds relative to unix epoch.
65 * @param pTime The time spec to interpret.
66 */
67DECLINLINE(int64_t) RTTimeSpecGetNano(PCRTTIMESPEC pTime)
68{
69 return pTime->i64NanosecondsRelativeToUnixEpoch;
70}
71
72
73/**
74 * Sets the time give by nanoseconds relative to the unix epoch.
75 *
76 * @returns pTime.
77 * @param pTime The time spec to modify.
78 * @param i64Nano The new time in nanoseconds.
79 */
80DECLINLINE(PRTTIMESPEC) RTTimeSpecSetNano(PRTTIMESPEC pTime, int64_t i64Nano)
81{
82 pTime->i64NanosecondsRelativeToUnixEpoch = i64Nano;
83 return pTime;
84}
85
86
87/**
88 * Gets the time as microseconds relative to the unix epoch.
89 *
90 * @returns microseconds relative to unix epoch.
91 * @param pTime The time spec to interpret.
92 */
93DECLINLINE(int64_t) RTTimeSpecGetMicro(PCRTTIMESPEC pTime)
94{
95 return pTime->i64NanosecondsRelativeToUnixEpoch / RT_NS_1US;
96}
97
98
99/**
100 * Sets the time given by microseconds relative to the unix epoch.
101 *
102 * @returns pTime.
103 * @param pTime The time spec to modify.
104 * @param i64Micro The new time in microsecond.
105 */
106DECLINLINE(PRTTIMESPEC) RTTimeSpecSetMicro(PRTTIMESPEC pTime, int64_t i64Micro)
107{
108 pTime->i64NanosecondsRelativeToUnixEpoch = i64Micro * RT_NS_1US;
109 return pTime;
110}
111
112
113/**
114 * Gets the time as milliseconds relative to the unix epoch.
115 *
116 * @returns milliseconds relative to unix epoch.
117 * @param pTime The time spec to interpret.
118 */
119DECLINLINE(int64_t) RTTimeSpecGetMilli(PCRTTIMESPEC pTime)
120{
121 return pTime->i64NanosecondsRelativeToUnixEpoch / RT_NS_1MS;
122}
123
124
125/**
126 * Sets the time given by milliseconds relative to the unix epoch.
127 *
128 * @returns pTime.
129 * @param pTime The time spec to modify.
130 * @param i64Milli The new time in milliseconds.
131 */
132DECLINLINE(PRTTIMESPEC) RTTimeSpecSetMilli(PRTTIMESPEC pTime, int64_t i64Milli)
133{
134 pTime->i64NanosecondsRelativeToUnixEpoch = i64Milli * RT_NS_1MS;
135 return pTime;
136}
137
138
139/**
140 * Gets the time as seconds relative to the unix epoch.
141 *
142 * @returns seconds relative to unix epoch.
143 * @param pTime The time spec to interpret.
144 */
145DECLINLINE(int64_t) RTTimeSpecGetSeconds(PCRTTIMESPEC pTime)
146{
147 return pTime->i64NanosecondsRelativeToUnixEpoch / RT_NS_1SEC;
148}
149
150
151/**
152 * Sets the time given by seconds relative to the unix epoch.
153 *
154 * @returns pTime.
155 * @param pTime The time spec to modify.
156 * @param i64Seconds The new time in seconds.
157 */
158DECLINLINE(PRTTIMESPEC) RTTimeSpecSetSeconds(PRTTIMESPEC pTime, int64_t i64Seconds)
159{
160 pTime->i64NanosecondsRelativeToUnixEpoch = i64Seconds * RT_NS_1SEC;
161 return pTime;
162}
163
164
165/**
166 * Makes the time spec absolute like abs() does (i.e. a positive value).
167 *
168 * @returns pTime.
169 * @param pTime The time spec to modify.
170 */
171DECLINLINE(PRTTIMESPEC) RTTimeSpecAbsolute(PRTTIMESPEC pTime)
172{
173 if (pTime->i64NanosecondsRelativeToUnixEpoch < 0)
174 pTime->i64NanosecondsRelativeToUnixEpoch = -pTime->i64NanosecondsRelativeToUnixEpoch;
175 return pTime;
176}
177
178
179/**
180 * Negates the time.
181 *
182 * @returns pTime.
183 * @param pTime The time spec to modify.
184 */
185DECLINLINE(PRTTIMESPEC) RTTimeSpecNegate(PRTTIMESPEC pTime)
186{
187 pTime->i64NanosecondsRelativeToUnixEpoch = -pTime->i64NanosecondsRelativeToUnixEpoch;
188 return pTime;
189}
190
191
192/**
193 * Adds a time period to the time.
194 *
195 * @returns pTime.
196 * @param pTime The time spec to modify.
197 * @param pTimeAdd The time spec to add to pTime.
198 */
199DECLINLINE(PRTTIMESPEC) RTTimeSpecAdd(PRTTIMESPEC pTime, PCRTTIMESPEC pTimeAdd)
200{
201 pTime->i64NanosecondsRelativeToUnixEpoch += pTimeAdd->i64NanosecondsRelativeToUnixEpoch;
202 return pTime;
203}
204
205
206/**
207 * Adds a time period give as nanoseconds from the time.
208 *
209 * @returns pTime.
210 * @param pTime The time spec to modify.
211 * @param i64Nano The time period in nanoseconds.
212 */
213DECLINLINE(PRTTIMESPEC) RTTimeSpecAddNano(PRTTIMESPEC pTime, int64_t i64Nano)
214{
215 pTime->i64NanosecondsRelativeToUnixEpoch += i64Nano;
216 return pTime;
217}
218
219
220/**
221 * Adds a time period give as microseconds from the time.
222 *
223 * @returns pTime.
224 * @param pTime The time spec to modify.
225 * @param i64Micro The time period in microseconds.
226 */
227DECLINLINE(PRTTIMESPEC) RTTimeSpecAddMicro(PRTTIMESPEC pTime, int64_t i64Micro)
228{
229 pTime->i64NanosecondsRelativeToUnixEpoch += i64Micro * RT_NS_1US;
230 return pTime;
231}
232
233
234/**
235 * Adds a time period give as milliseconds from the time.
236 *
237 * @returns pTime.
238 * @param pTime The time spec to modify.
239 * @param i64Milli The time period in milliseconds.
240 */
241DECLINLINE(PRTTIMESPEC) RTTimeSpecAddMilli(PRTTIMESPEC pTime, int64_t i64Milli)
242{
243 pTime->i64NanosecondsRelativeToUnixEpoch += i64Milli * RT_NS_1MS;
244 return pTime;
245}
246
247
248/**
249 * Adds a time period give as seconds from the time.
250 *
251 * @returns pTime.
252 * @param pTime The time spec to modify.
253 * @param i64Seconds The time period in seconds.
254 */
255DECLINLINE(PRTTIMESPEC) RTTimeSpecAddSeconds(PRTTIMESPEC pTime, int64_t i64Seconds)
256{
257 pTime->i64NanosecondsRelativeToUnixEpoch += i64Seconds * RT_NS_1SEC;
258 return pTime;
259}
260
261
262/**
263 * Subtracts a time period from the time.
264 *
265 * @returns pTime.
266 * @param pTime The time spec to modify.
267 * @param pTimeSub The time spec to subtract from pTime.
268 */
269DECLINLINE(PRTTIMESPEC) RTTimeSpecSub(PRTTIMESPEC pTime, PCRTTIMESPEC pTimeSub)
270{
271 pTime->i64NanosecondsRelativeToUnixEpoch -= pTimeSub->i64NanosecondsRelativeToUnixEpoch;
272 return pTime;
273}
274
275
276/**
277 * Subtracts a time period give as nanoseconds from the time.
278 *
279 * @returns pTime.
280 * @param pTime The time spec to modify.
281 * @param i64Nano The time period in nanoseconds.
282 */
283DECLINLINE(PRTTIMESPEC) RTTimeSpecSubNano(PRTTIMESPEC pTime, int64_t i64Nano)
284{
285 pTime->i64NanosecondsRelativeToUnixEpoch -= i64Nano;
286 return pTime;
287}
288
289
290/**
291 * Subtracts a time period give as microseconds from the time.
292 *
293 * @returns pTime.
294 * @param pTime The time spec to modify.
295 * @param i64Micro The time period in microseconds.
296 */
297DECLINLINE(PRTTIMESPEC) RTTimeSpecSubMicro(PRTTIMESPEC pTime, int64_t i64Micro)
298{
299 pTime->i64NanosecondsRelativeToUnixEpoch -= i64Micro * RT_NS_1US;
300 return pTime;
301}
302
303
304/**
305 * Subtracts a time period give as milliseconds from the time.
306 *
307 * @returns pTime.
308 * @param pTime The time spec to modify.
309 * @param i64Milli The time period in milliseconds.
310 */
311DECLINLINE(PRTTIMESPEC) RTTimeSpecSubMilli(PRTTIMESPEC pTime, int64_t i64Milli)
312{
313 pTime->i64NanosecondsRelativeToUnixEpoch -= i64Milli * RT_NS_1MS;
314 return pTime;
315}
316
317
318/**
319 * Subtracts a time period give as seconds from the time.
320 *
321 * @returns pTime.
322 * @param pTime The time spec to modify.
323 * @param i64Seconds The time period in seconds.
324 */
325DECLINLINE(PRTTIMESPEC) RTTimeSpecSubSeconds(PRTTIMESPEC pTime, int64_t i64Seconds)
326{
327 pTime->i64NanosecondsRelativeToUnixEpoch -= i64Seconds * RT_NS_1SEC;
328 return pTime;
329}
330
331
332/**
333 * Gives the time in seconds and nanoseconds.
334 *
335 * @returns pTime.
336 * @param pTime The time spec to interpret.
337 * @param *pi32Seconds Where to store the time period in seconds.
338 * @param *pi32Nano Where to store the time period in nanoseconds.
339 */
340DECLINLINE(void) RTTimeSpecGetSecondsAndNano(PRTTIMESPEC pTime, int32_t *pi32Seconds, int32_t *pi32Nano)
341{
342 int64_t i64 = RTTimeSpecGetNano(pTime);
343 int32_t i32Nano = (int32_t)(i64 % RT_NS_1SEC);
344 i64 /= RT_NS_1SEC;
345 if (i32Nano < 0)
346 {
347 i32Nano += RT_NS_1SEC;
348 i64--;
349 }
350 *pi32Seconds = (int32_t)i64;
351 *pi32Nano = i32Nano;
352}
353
354
355/* PORTME: Add struct timeval guard macro here. */
356#if defined(RTTIME_INCL_TIMEVAL) || defined(_STRUCT_TIMEVAL) || defined(_SYS__TIMEVAL_H_) || defined(_SYS_TIME_H) || defined(_TIMEVAL) || defined(_LINUX_TIME_H) \
357 || (defined(RT_OS_NETBSD) && defined(_SYS_TIME_H_))
358/**
359 * Gets the time as POSIX timeval.
360 *
361 * @returns pTime.
362 * @param pTime The time spec to interpret.
363 * @param pTimeval Where to store the time as POSIX timeval.
364 */
365DECLINLINE(struct timeval *) RTTimeSpecGetTimeval(PCRTTIMESPEC pTime, struct timeval *pTimeval)
366{
367 int64_t i64 = RTTimeSpecGetMicro(pTime);
368 int32_t i32Micro = (int32_t)(i64 % RT_US_1SEC);
369 i64 /= RT_US_1SEC;
370 if (i32Micro < 0)
371 {
372 i32Micro += RT_US_1SEC;
373 i64--;
374 }
375 pTimeval->tv_sec = (time_t)i64;
376 pTimeval->tv_usec = i32Micro;
377 return pTimeval;
378}
379
380/**
381 * Sets the time as POSIX timeval.
382 *
383 * @returns pTime.
384 * @param pTime The time spec to modify.
385 * @param pTimeval Pointer to the POSIX timeval struct with the new time.
386 */
387DECLINLINE(PRTTIMESPEC) RTTimeSpecSetTimeval(PRTTIMESPEC pTime, const struct timeval *pTimeval)
388{
389 return RTTimeSpecAddMicro(RTTimeSpecSetSeconds(pTime, pTimeval->tv_sec), pTimeval->tv_usec);
390}
391#endif /* various ways of detecting struct timeval */
392
393
394/* PORTME: Add struct timespec guard macro here. */
395#if defined(RTTIME_INCL_TIMESPEC) || defined(_STRUCT_TIMESPEC) || defined(_SYS__TIMESPEC_H_) || defined(TIMEVAL_TO_TIMESPEC) || defined(_TIMESPEC) \
396 || (defined(RT_OS_NETBSD) && defined(_SYS_TIME_H_))
397/**
398 * Gets the time as POSIX timespec.
399 *
400 * @returns pTime.
401 * @param pTime The time spec to interpret.
402 * @param pTimespec Where to store the time as POSIX timespec.
403 */
404DECLINLINE(struct timespec *) RTTimeSpecGetTimespec(PCRTTIMESPEC pTime, struct timespec *pTimespec)
405{
406 int64_t i64 = RTTimeSpecGetNano(pTime);
407 int32_t i32Nano = (int32_t)(i64 % RT_NS_1SEC);
408 i64 /= RT_NS_1SEC;
409 if (i32Nano < 0)
410 {
411 i32Nano += RT_NS_1SEC;
412 i64--;
413 }
414 pTimespec->tv_sec = (time_t)i64;
415 pTimespec->tv_nsec = i32Nano;
416 return pTimespec;
417}
418
419/**
420 * Sets the time as POSIX timespec.
421 *
422 * @returns pTime.
423 * @param pTime The time spec to modify.
424 * @param pTimespec Pointer to the POSIX timespec struct with the new time.
425 */
426DECLINLINE(PRTTIMESPEC) RTTimeSpecSetTimespec(PRTTIMESPEC pTime, const struct timespec *pTimespec)
427{
428 return RTTimeSpecAddNano(RTTimeSpecSetSeconds(pTime, pTimespec->tv_sec), pTimespec->tv_nsec);
429}
430#endif /* various ways of detecting struct timespec */
431
432
433
434/** The offset of the unix epoch and the base for NT time (in 100ns units).
435 * Nt time starts at 1601-01-01 00:00:00. */
436#define RTTIME_NT_TIME_OFFSET_UNIX (116444736000000000LL)
437
438
439/**
440 * Gets the time as NT time.
441 *
442 * @returns Nt time.
443 * @param pTime The time spec to interpret.
444 */
445DECLINLINE(uint64_t) RTTimeSpecGetNtTime(PCRTTIMESPEC pTime)
446{
447 return pTime->i64NanosecondsRelativeToUnixEpoch / 100
448 + RTTIME_NT_TIME_OFFSET_UNIX;
449}
450
451
452/**
453 * Sets the time given by Nt time.
454 *
455 * @returns pTime.
456 * @param pTime The time spec to modify.
457 * @param u64NtTime The new time in Nt time.
458 */
459DECLINLINE(PRTTIMESPEC) RTTimeSpecSetNtTime(PRTTIMESPEC pTime, uint64_t u64NtTime)
460{
461 pTime->i64NanosecondsRelativeToUnixEpoch =
462 ((int64_t)u64NtTime - RTTIME_NT_TIME_OFFSET_UNIX) * 100;
463 return pTime;
464}
465
466
467#ifdef _FILETIME_
468/**
469 * Gets the time as NT file time.
470 *
471 * @returns pFileTime.
472 * @param pTime The time spec to interpret.
473 * @param pFileTime Pointer to NT filetime structure.
474 */
475DECLINLINE(PFILETIME) RTTimeSpecGetNtFileTime(PCRTTIMESPEC pTime, PFILETIME pFileTime)
476{
477 *((uint64_t *)pFileTime) = RTTimeSpecGetNtTime(pTime);
478 return pFileTime;
479}
480
481/**
482 * Sets the time as NT file time.
483 *
484 * @returns pTime.
485 * @param pTime The time spec to modify.
486 * @param pFileTime Where to store the time as Nt file time.
487 */
488DECLINLINE(PRTTIMESPEC) RTTimeSpecSetNtFileTime(PRTTIMESPEC pTime, const FILETIME *pFileTime)
489{
490 return RTTimeSpecSetNtTime(pTime, *(const uint64_t *)pFileTime);
491}
492#endif
493
494
495/** The offset to the start of DOS time.
496 * DOS time starts 1980-01-01 00:00:00. */
497#define RTTIME_OFFSET_DOS_TIME (315532800000000000LL)
498
499
500/**
501 * Gets the time as seconds relative to the start of dos time.
502 *
503 * @returns seconds relative to the start of dos time.
504 * @param pTime The time spec to interpret.
505 */
506DECLINLINE(int64_t) RTTimeSpecGetDosSeconds(PCRTTIMESPEC pTime)
507{
508 return (pTime->i64NanosecondsRelativeToUnixEpoch - RTTIME_OFFSET_DOS_TIME)
509 / RT_NS_1SEC;
510}
511
512
513/**
514 * Sets the time given by seconds relative to the start of dos time.
515 *
516 * @returns pTime.
517 * @param pTime The time spec to modify.
518 * @param i64Seconds The new time in seconds relative to the start of dos time.
519 */
520DECLINLINE(PRTTIMESPEC) RTTimeSpecSetDosSeconds(PRTTIMESPEC pTime, int64_t i64Seconds)
521{
522 pTime->i64NanosecondsRelativeToUnixEpoch = i64Seconds * RT_NS_1SEC
523 + RTTIME_OFFSET_DOS_TIME;
524 return pTime;
525}
526
527
528/**
529 * Compare two time specs.
530 *
531 * @returns true they are equal.
532 * @returns false they are not equal.
533 * @param pTime1 The 1st time spec.
534 * @param pTime2 The 2nd time spec.
535 */
536DECLINLINE(bool) RTTimeSpecIsEqual(PCRTTIMESPEC pTime1, PCRTTIMESPEC pTime2)
537{
538 return pTime1->i64NanosecondsRelativeToUnixEpoch == pTime2->i64NanosecondsRelativeToUnixEpoch;
539}
540
541
542/**
543 * Compare two time specs.
544 *
545 * @returns 0 if equal, -1 if @a pLeft is smaller, 1 if @a pLeft is larger.
546 * @returns false they are not equal.
547 * @param pLeft The 1st time spec.
548 * @param pRight The 2nd time spec.
549 */
550DECLINLINE(int) RTTimeSpecCompare(PCRTTIMESPEC pLeft, PCRTTIMESPEC pRight)
551{
552 if (pLeft->i64NanosecondsRelativeToUnixEpoch == pRight->i64NanosecondsRelativeToUnixEpoch)
553 return 0;
554 return pLeft->i64NanosecondsRelativeToUnixEpoch < pRight->i64NanosecondsRelativeToUnixEpoch ? -1 : 1;
555}
556
557
558/**
559 * Converts a time spec to a ISO date string.
560 *
561 * @returns psz on success.
562 * @returns NULL on buffer underflow.
563 * @param pTime The time spec.
564 * @param psz Where to store the string.
565 * @param cb The size of the buffer.
566 */
567RTDECL(char *) RTTimeSpecToString(PCRTTIMESPEC pTime, char *psz, size_t cb);
568
569/**
570 * Attempts to convert an ISO date string to a time structure.
571 *
572 * We're a little forgiving with zero padding, unspecified parts, and leading
573 * and trailing spaces.
574 *
575 * @retval pTime on success,
576 * @retval NULL on failure.
577 * @param pTime The time spec.
578 * @param pszString The ISO date string to convert.
579 */
580RTDECL(PRTTIMESPEC) RTTimeSpecFromString(PRTTIMESPEC pTime, const char *pszString);
581
582/** @} */
583
584
585/**
586 * Exploded time.
587 */
588#pragma pack(1)
589typedef struct RTTIME
590{
591 /** The year number. */
592 int32_t i32Year;
593 /** The month of the year (1-12). January is 1. */
594 uint8_t u8Month;
595 /** The day of the week (0-6). Monday is 0. */
596 uint8_t u8WeekDay;
597 /** The day of the year (1-366). January the 1st is 1. */
598 uint16_t u16YearDay;
599 /** The day of the month (1-31). */
600 uint8_t u8MonthDay;
601 /** Hour of the day (0-23). */
602 uint8_t u8Hour;
603 /** The minute of the hour (0-59). */
604 uint8_t u8Minute;
605 /** The second of the minute (0-60).
606 * (u32Nanosecond / 1000000) */
607 uint8_t u8Second;
608 /** The nanoseconds of the second (0-999999999). */
609 uint32_t u32Nanosecond;
610 /** Flags, of the RTTIME_FLAGS_* \#defines. */
611 uint32_t fFlags;
612 /** UCT time offset in minutes (-840-840).
613 * @remarks The implementation of RTTimeLocal* isn't quite there yet, so this might not be 100% correct. */
614 int32_t offUTC;
615} RTTIME;
616#pragma pack()
617/** Pointer to a exploded time structure. */
618typedef RTTIME *PRTTIME;
619/** Pointer to a const exploded time structure. */
620typedef const RTTIME *PCRTTIME;
621
622/** @name RTTIME::fFlags values.
623 * @{ */
624/** Set if the time is UTC. If clear the time local time. */
625#define RTTIME_FLAGS_TYPE_MASK 3
626/** the time is UTC time. */
627#define RTTIME_FLAGS_TYPE_UTC 2
628/** The time is local time. */
629#define RTTIME_FLAGS_TYPE_LOCAL 3
630
631/** Set if the time is local and daylight saving time is in effect.
632 * Not bit is not valid if RTTIME_FLAGS_NO_DST_DATA is set. */
633#define RTTIME_FLAGS_DST RT_BIT(4)
634/** Set if the time is local and there is no data available on daylight saving time. */
635#define RTTIME_FLAGS_NO_DST_DATA RT_BIT(5)
636/** Set if the year is a leap year.
637 * This is mutual exclusiv with RTTIME_FLAGS_COMMON_YEAR. */
638#define RTTIME_FLAGS_LEAP_YEAR RT_BIT(6)
639/** Set if the year is a common year.
640 * This is mutual exclusiv with RTTIME_FLAGS_LEAP_YEAR. */
641#define RTTIME_FLAGS_COMMON_YEAR RT_BIT(7)
642/** The mask of valid flags. */
643#define RTTIME_FLAGS_MASK UINT32_C(0xff)
644/** @} */
645
646
647/**
648 * Gets the current system time (UTC).
649 *
650 * @returns pTime.
651 * @param pTime Where to store the time.
652 */
653RTDECL(PRTTIMESPEC) RTTimeNow(PRTTIMESPEC pTime);
654
655/**
656 * Sets the system time.
657 *
658 * @returns IPRT status code
659 * @param pTime The new system time (UTC).
660 *
661 * @remarks This will usually fail because changing the wall time is usually
662 * requires extra privileges.
663 */
664RTDECL(int) RTTimeSet(PCRTTIMESPEC pTime);
665
666/**
667 * Explodes a time spec (UTC).
668 *
669 * @returns pTime.
670 * @param pTime Where to store the exploded time.
671 * @param pTimeSpec The time spec to exploded.
672 */
673RTDECL(PRTTIME) RTTimeExplode(PRTTIME pTime, PCRTTIMESPEC pTimeSpec);
674
675/**
676 * Implodes exploded time to a time spec (UTC).
677 *
678 * @returns pTime on success.
679 * @returns NULL if the pTime data is invalid.
680 * @param pTimeSpec Where to store the imploded UTC time.
681 * If pTime specifies a time which outside the range, maximum or
682 * minimum values will be returned.
683 * @param pTime Pointer to the exploded time to implode.
684 * The fields u8Month, u8WeekDay and u8MonthDay are not used,
685 * and all the other fields are expected to be within their
686 * bounds. Use RTTimeNormalize() to calculate u16YearDay and
687 * normalize the ranges of the fields.
688 */
689RTDECL(PRTTIMESPEC) RTTimeImplode(PRTTIMESPEC pTimeSpec, PCRTTIME pTime);
690
691/**
692 * Normalizes the fields of a time structure.
693 *
694 * It is possible to calculate year-day from month/day and vice
695 * versa. If you adjust any of of these, make sure to zero the
696 * other so you make it clear which of the fields to use. If
697 * it's ambiguous, the year-day field is used (and you get
698 * assertions in debug builds).
699 *
700 * All the time fields and the year-day or month/day fields will
701 * be adjusted for overflows. (Since all fields are unsigned, there
702 * is no underflows.) It is possible to exploit this for simple
703 * date math, though the recommended way of doing that to implode
704 * the time into a timespec and do the math on that.
705 *
706 * @returns pTime on success.
707 * @returns NULL if the data is invalid.
708 *
709 * @param pTime The time structure to normalize.
710 *
711 * @remarks This function doesn't work with local time, only with UTC time.
712 */
713RTDECL(PRTTIME) RTTimeNormalize(PRTTIME pTime);
714
715/**
716 * Gets the current local system time.
717 *
718 * @returns pTime.
719 * @param pTime Where to store the local time.
720 */
721RTDECL(PRTTIMESPEC) RTTimeLocalNow(PRTTIMESPEC pTime);
722
723/**
724 * Gets the delta between UTC and local time.
725 *
726 * @code
727 * RTTIMESPEC LocalTime;
728 * RTTimeSpecAddNano(RTTimeNow(&LocalTime), RTTimeLocalDeltaNano());
729 * @endcode
730 *
731 * @returns Returns the nanosecond delta between UTC and local time.
732 */
733RTDECL(int64_t) RTTimeLocalDeltaNano(void);
734
735/**
736 * Explodes a time spec to the localized timezone.
737 *
738 * @returns pTime.
739 * @param pTime Where to store the exploded time.
740 * @param pTimeSpec The time spec to exploded (UTC).
741 */
742RTDECL(PRTTIME) RTTimeLocalExplode(PRTTIME pTime, PCRTTIMESPEC pTimeSpec);
743
744/**
745 * Normalizes the fields of a time structure containing local time.
746 *
747 * See RTTimeNormalize for details.
748 *
749 * @returns pTime on success.
750 * @returns NULL if the data is invalid.
751 * @param pTime The time structure to normalize.
752 */
753RTDECL(PRTTIME) RTTimeLocalNormalize(PRTTIME pTime);
754
755/**
756 * Converts a time spec to a ISO date string.
757 *
758 * @returns psz on success.
759 * @returns NULL on buffer underflow.
760 * @param pTime The time. Caller should've normalized this.
761 * @param psz Where to store the string.
762 * @param cb The size of the buffer.
763 */
764RTDECL(char *) RTTimeToString(PCRTTIME pTime, char *psz, size_t cb);
765
766/**
767 * Attempts to convert an ISO date string to a time structure.
768 *
769 * We're a little forgiving with zero padding, unspecified parts, and leading
770 * and trailing spaces.
771 *
772 * @retval pTime on success,
773 * @retval NULL on failure.
774 * @param pTime Where to store the time on success.
775 * @param pszString The ISO date string to convert.
776 */
777RTDECL(PRTTIME) RTTimeFromString(PRTTIME pTime, const char *pszString);
778
779/**
780 * Checks if a year is a leap year or not.
781 *
782 * @returns true if it's a leap year.
783 * @returns false if it's a common year.
784 * @param i32Year The year in question.
785 */
786RTDECL(bool) RTTimeIsLeapYear(int32_t i32Year);
787
788/**
789 * Gets the current nanosecond timestamp.
790 *
791 * @returns nanosecond timestamp.
792 */
793RTDECL(uint64_t) RTTimeNanoTS(void);
794
795/**
796 * Gets the current millisecond timestamp.
797 *
798 * @returns millisecond timestamp.
799 */
800RTDECL(uint64_t) RTTimeMilliTS(void);
801
802/**
803 * Debugging the time api.
804 *
805 * @returns the number of 1ns steps which has been applied by RTTimeNanoTS().
806 */
807RTDECL(uint32_t) RTTimeDbgSteps(void);
808
809/**
810 * Debugging the time api.
811 *
812 * @returns the number of times the TSC interval expired RTTimeNanoTS().
813 */
814RTDECL(uint32_t) RTTimeDbgExpired(void);
815
816/**
817 * Debugging the time api.
818 *
819 * @returns the number of bad previous values encountered by RTTimeNanoTS().
820 */
821RTDECL(uint32_t) RTTimeDbgBad(void);
822
823/**
824 * Debugging the time api.
825 *
826 * @returns the number of update races in RTTimeNanoTS().
827 */
828RTDECL(uint32_t) RTTimeDbgRaces(void);
829
830/** @name RTTimeNanoTS GIP worker functions, for TM.
831 * @{ */
832/** Pointer to a RTTIMENANOTSDATA structure. */
833typedef struct RTTIMENANOTSDATA *PRTTIMENANOTSDATA;
834
835/**
836 * Nanosecond timestamp data.
837 *
838 * This is used to keep track of statistics and callback so IPRT
839 * and TM (VirtualBox) can share code.
840 *
841 * @remark Keep this in sync with the assembly version in timesupA.asm.
842 */
843typedef struct RTTIMENANOTSDATA
844{
845 /** Where the previous timestamp is stored.
846 * This is maintained to ensure that time doesn't go backwards or anything. */
847 uint64_t volatile *pu64Prev;
848
849 /**
850 * Helper function that's used by the assembly routines when something goes bust.
851 *
852 * @param pData Pointer to this structure.
853 * @param u64NanoTS The calculated nano ts.
854 * @param u64DeltaPrev The delta relative to the previously returned timestamp.
855 * @param u64PrevNanoTS The previously returned timestamp (as it was read it).
856 */
857 DECLCALLBACKMEMBER(void, pfnBad)(PRTTIMENANOTSDATA pData, uint64_t u64NanoTS, uint64_t u64DeltaPrev, uint64_t u64PrevNanoTS);
858
859 /**
860 * Callback for when rediscovery is required.
861 *
862 * @returns Nanosecond timestamp.
863 * @param pData Pointer to this structure.
864 */
865 DECLCALLBACKMEMBER(uint64_t, pfnRediscover)(PRTTIMENANOTSDATA pData);
866
867 /**
868 * Callback for when some CPU index related stuff goes wrong.
869 *
870 * @returns Nanosecond timestamp.
871 * @param pData Pointer to this structure.
872 * @param idApic The APIC ID if available, otherwise (UINT16_MAX-1).
873 * @param iCpuSet The CPU set index if available, otherwise
874 * (UINT16_MAX-1).
875 * @param iGipCpu The GIP CPU array index if available, otherwise
876 * (UINT16_MAX-1).
877 */
878 DECLCALLBACKMEMBER(uint64_t, pfnBadCpuIndex)(PRTTIMENANOTSDATA pData, uint16_t idApic, uint16_t iCpuSet, uint16_t iGipCpu);
879
880 /** Number of 1ns steps because of overshooting the period. */
881 uint32_t c1nsSteps;
882 /** The number of times the interval expired (overflow). */
883 uint32_t cExpired;
884 /** Number of "bad" previous values. */
885 uint32_t cBadPrev;
886 /** The number of update races. */
887 uint32_t cUpdateRaces;
888} RTTIMENANOTSDATA;
889
890#ifndef IN_RING3
891/**
892 * The Ring-3 layout of the RTTIMENANOTSDATA structure.
893 */
894typedef struct RTTIMENANOTSDATAR3
895{
896 R3PTRTYPE(uint64_t volatile *) pu64Prev;
897 DECLR3CALLBACKMEMBER(void, pfnBad,(PRTTIMENANOTSDATA pData, uint64_t u64NanoTS, uint64_t u64DeltaPrev, uint64_t u64PrevNanoTS));
898 DECLR3CALLBACKMEMBER(uint64_t, pfnRediscover,(PRTTIMENANOTSDATA pData));
899 DECLR3CALLBACKMEMBER(uint64_t, pfnBadCpuIndex,(PRTTIMENANOTSDATA pData, uint16_t idApic, uint16_t iCpuSet, uint16_t iGipCpu));
900 uint32_t c1nsSteps;
901 uint32_t cExpired;
902 uint32_t cBadPrev;
903 uint32_t cUpdateRaces;
904} RTTIMENANOTSDATAR3;
905#else
906typedef RTTIMENANOTSDATA RTTIMENANOTSDATAR3;
907#endif
908
909#ifndef IN_RING0
910/**
911 * The Ring-3 layout of the RTTIMENANOTSDATA structure.
912 */
913typedef struct RTTIMENANOTSDATAR0
914{
915 R0PTRTYPE(uint64_t volatile *) pu64Prev;
916 DECLR0CALLBACKMEMBER(void, pfnBad,(PRTTIMENANOTSDATA pData, uint64_t u64NanoTS, uint64_t u64DeltaPrev, uint64_t u64PrevNanoTS));
917 DECLR0CALLBACKMEMBER(uint64_t, pfnRediscover,(PRTTIMENANOTSDATA pData));
918 DECLR0CALLBACKMEMBER(uint64_t, pfnBadCpuIndex,(PRTTIMENANOTSDATA pData, uint16_t idApic, uint16_t iCpuSet, uint16_t iGipCpu));
919 uint32_t c1nsSteps;
920 uint32_t cExpired;
921 uint32_t cBadPrev;
922 uint32_t cUpdateRaces;
923} RTTIMENANOTSDATAR0;
924#else
925typedef RTTIMENANOTSDATA RTTIMENANOTSDATAR0;
926#endif
927
928#ifndef IN_RC
929/**
930 * The RC layout of the RTTIMENANOTSDATA structure.
931 */
932typedef struct RTTIMENANOTSDATARC
933{
934 RCPTRTYPE(uint64_t volatile *) pu64Prev;
935 DECLRCCALLBACKMEMBER(void, pfnBad,(PRTTIMENANOTSDATA pData, uint64_t u64NanoTS, uint64_t u64DeltaPrev, uint64_t u64PrevNanoTS));
936 DECLRCCALLBACKMEMBER(uint64_t, pfnRediscover,(PRTTIMENANOTSDATA pData));
937 DECLRCCALLBACKMEMBER(uint64_t, pfnBadCpuIndex,(PRTTIMENANOTSDATA pData, uint16_t idApic, uint16_t iCpuSet, uint16_t iGipCpu));
938 uint32_t c1nsSteps;
939 uint32_t cExpired;
940 uint32_t cBadPrev;
941 uint32_t cUpdateRaces;
942} RTTIMENANOTSDATARC;
943#else
944typedef RTTIMENANOTSDATA RTTIMENANOTSDATARC;
945#endif
946
947/** Internal RTTimeNanoTS worker (assembly). */
948typedef DECLCALLBACK(uint64_t) FNTIMENANOTSINTERNAL(PRTTIMENANOTSDATA pData);
949/** Pointer to an internal RTTimeNanoTS worker (assembly). */
950typedef FNTIMENANOTSINTERNAL *PFNTIMENANOTSINTERNAL;
951RTDECL(uint64_t) RTTimeNanoTSLegacySyncInvarNoDelta(PRTTIMENANOTSDATA pData);
952RTDECL(uint64_t) RTTimeNanoTSLFenceSyncInvarNoDelta(PRTTIMENANOTSDATA pData);
953#ifdef IN_RING3
954RTDECL(uint64_t) RTTimeNanoTSLegacyAsyncUseApicId(PRTTIMENANOTSDATA pData);
955RTDECL(uint64_t) RTTimeNanoTSLegacyAsyncUseRdtscp(PRTTIMENANOTSDATA pData);
956RTDECL(uint64_t) RTTimeNanoTSLegacyAsyncUseRdtscpGroupChNumCl(PRTTIMENANOTSDATA pData);
957RTDECL(uint64_t) RTTimeNanoTSLegacyAsyncUseIdtrLim(PRTTIMENANOTSDATA pData);
958RTDECL(uint64_t) RTTimeNanoTSLegacySyncInvarWithDeltaUseApicId(PRTTIMENANOTSDATA pData);
959RTDECL(uint64_t) RTTimeNanoTSLegacySyncInvarWithDeltaUseRdtscp(PRTTIMENANOTSDATA pData);
960RTDECL(uint64_t) RTTimeNanoTSLegacySyncInvarWithDeltaUseIdtrLim(PRTTIMENANOTSDATA pData);
961RTDECL(uint64_t) RTTimeNanoTSLFenceAsyncUseApicId(PRTTIMENANOTSDATA pData);
962RTDECL(uint64_t) RTTimeNanoTSLFenceAsyncUseRdtscp(PRTTIMENANOTSDATA pData);
963RTDECL(uint64_t) RTTimeNanoTSLFenceAsyncUseRdtscpGroupChNumCl(PRTTIMENANOTSDATA pData);
964RTDECL(uint64_t) RTTimeNanoTSLFenceAsyncUseIdtrLim(PRTTIMENANOTSDATA pData);
965RTDECL(uint64_t) RTTimeNanoTSLFenceSyncInvarWithDeltaUseApicId(PRTTIMENANOTSDATA pData);
966RTDECL(uint64_t) RTTimeNanoTSLFenceSyncInvarWithDeltaUseRdtscp(PRTTIMENANOTSDATA pData);
967RTDECL(uint64_t) RTTimeNanoTSLFenceSyncInvarWithDeltaUseIdtrLim(PRTTIMENANOTSDATA pData);
968#else
969RTDECL(uint64_t) RTTimeNanoTSLegacyAsync(PRTTIMENANOTSDATA pData);
970RTDECL(uint64_t) RTTimeNanoTSLegacySyncInvarWithDelta(PRTTIMENANOTSDATA pData);
971RTDECL(uint64_t) RTTimeNanoTSLFenceAsync(PRTTIMENANOTSDATA pData);
972RTDECL(uint64_t) RTTimeNanoTSLFenceSyncInvarWithDelta(PRTTIMENANOTSDATA pData);
973#endif
974
975/** @} */
976
977
978/**
979 * Gets the current nanosecond timestamp.
980 *
981 * This differs from RTTimeNanoTS in that it will use system APIs and not do any
982 * resolution or performance optimizations.
983 *
984 * @returns nanosecond timestamp.
985 */
986RTDECL(uint64_t) RTTimeSystemNanoTS(void);
987
988/**
989 * Gets the current millisecond timestamp.
990 *
991 * This differs from RTTimeNanoTS in that it will use system APIs and not do any
992 * resolution or performance optimizations.
993 *
994 * @returns millisecond timestamp.
995 */
996RTDECL(uint64_t) RTTimeSystemMilliTS(void);
997
998/**
999 * Get the nanosecond timestamp relative to program startup.
1000 *
1001 * @returns Timestamp relative to program startup.
1002 */
1003RTDECL(uint64_t) RTTimeProgramNanoTS(void);
1004
1005/**
1006 * Get the microsecond timestamp relative to program startup.
1007 *
1008 * @returns Timestamp relative to program startup.
1009 */
1010RTDECL(uint64_t) RTTimeProgramMicroTS(void);
1011
1012/**
1013 * Get the millisecond timestamp relative to program startup.
1014 *
1015 * @returns Timestamp relative to program startup.
1016 */
1017RTDECL(uint64_t) RTTimeProgramMilliTS(void);
1018
1019/**
1020 * Get the second timestamp relative to program startup.
1021 *
1022 * @returns Timestamp relative to program startup.
1023 */
1024RTDECL(uint32_t) RTTimeProgramSecTS(void);
1025
1026/**
1027 * Get the RTTimeNanoTS() of when the program started.
1028 *
1029 * @returns Program startup timestamp.
1030 */
1031RTDECL(uint64_t) RTTimeProgramStartNanoTS(void);
1032
1033
1034/**
1035 * Time zone information.
1036 */
1037typedef struct RTTIMEZONEINFO
1038{
1039 /** Unix time zone name (continent/country[/city]|). */
1040 const char *pszUnixName;
1041 /** Windows time zone name. */
1042 const char *pszWindowsName;
1043 /** The length of the unix time zone name. */
1044 uint8_t cchUnixName;
1045 /** The length of the windows time zone name. */
1046 uint8_t cchWindowsName;
1047 /** Two letter country/territory code if applicable, otherwise 'ZZ'. */
1048 char szCountry[3];
1049 /** Two letter windows country/territory code if applicable.
1050 * Empty string if no windows mapping. */
1051 char szWindowsCountry[3];
1052#if 0 /* Add when needed and it's been extracted. */
1053 /** The standard delta in minutes (add to UTC). */
1054 int16_t cMinStdDelta;
1055 /** The daylight saving time delta in minutes (add to UTC). */
1056 int16_t cMinDstDelta;
1057#endif
1058 /** closest matching windows time zone index. */
1059 uint32_t idxWindows;
1060 /** Flags, RTTIMEZONEINFO_F_XXX. */
1061 uint32_t fFlags;
1062} RTTIMEZONEINFO;
1063/** Pointer to time zone info. */
1064typedef RTTIMEZONEINFO const *PCRTTIMEZONEINFO;
1065
1066/** @name RTTIMEZONEINFO_F_XXX - time zone info flags.
1067 * @{ */
1068/** Indicates golden mapping entry for a windows time zone name. */
1069#define RTTIMEZONEINFO_F_GOLDEN RT_BIT_32(0)
1070/** @} */
1071
1072/**
1073 * Looks up static time zone information by unix name.
1074 *
1075 * @returns Pointer to info entry if found, NULL if not.
1076 * @param pszName The unix zone name (TZ).
1077 */
1078RTDECL(PCRTTIMEZONEINFO) RTTimeZoneGetInfoByUnixName(const char *pszName);
1079
1080/**
1081 * Looks up static time zone information by window name.
1082 *
1083 * @returns Pointer to info entry if found, NULL if not.
1084 * @param pszName The windows zone name (reg key).
1085 */
1086RTDECL(PCRTTIMEZONEINFO) RTTimeZoneGetInfoByWindowsName(const char *pszName);
1087
1088/**
1089 * Looks up static time zone information by windows index.
1090 *
1091 * @returns Pointer to info entry if found, NULL if not.
1092 * @param idxZone The windows timezone index.
1093 */
1094RTDECL(PCRTTIMEZONEINFO) RTTimeZoneGetInfoByWindowsIndex(uint32_t idxZone);
1095
1096/**
1097 * Get the current time zone (TZ).
1098 *
1099 * @returns IPRT status code.
1100 * @param pszName Where to return the time zone name.
1101 * @param cbName The size of the name buffer.
1102 */
1103RTDECL(int) RTTimeZoneGetCurrent(char *pszName, size_t cbName);
1104
1105/** @} */
1106
1107RT_C_DECLS_END
1108
1109#endif
1110
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette