1*61046927SAndroid Build Coastguard Worker /*
2*61046927SAndroid Build Coastguard Worker * C11 <time.h> implementation
3*61046927SAndroid Build Coastguard Worker *
4*61046927SAndroid Build Coastguard Worker * (C) Copyright yohhoy 2012.
5*61046927SAndroid Build Coastguard Worker * Copyright 2022 Yonggang Luo
6*61046927SAndroid Build Coastguard Worker * Distributed under the Boost Software License, Version 1.0.
7*61046927SAndroid Build Coastguard Worker *
8*61046927SAndroid Build Coastguard Worker * Permission is hereby granted, free of charge, to any person or organization
9*61046927SAndroid Build Coastguard Worker * obtaining a copy of the software and accompanying documentation covered by
10*61046927SAndroid Build Coastguard Worker * this license (the "Software") to use, reproduce, display, distribute,
11*61046927SAndroid Build Coastguard Worker * execute, and transmit the Software, and to prepare [[derivative work]]s of the
12*61046927SAndroid Build Coastguard Worker * Software, and to permit third-parties to whom the Software is furnished to
13*61046927SAndroid Build Coastguard Worker * do so, all subject to the following:
14*61046927SAndroid Build Coastguard Worker *
15*61046927SAndroid Build Coastguard Worker * The copyright notices in the Software and this entire statement, including
16*61046927SAndroid Build Coastguard Worker * the above license grant, this restriction and the following disclaimer,
17*61046927SAndroid Build Coastguard Worker * must be included in all copies of the Software, in whole or in part, and
18*61046927SAndroid Build Coastguard Worker * all derivative works of the Software, unless such copies or derivative
19*61046927SAndroid Build Coastguard Worker * works are solely in the form of machine-executable object code generated by
20*61046927SAndroid Build Coastguard Worker * a source language processor.
21*61046927SAndroid Build Coastguard Worker *
22*61046927SAndroid Build Coastguard Worker * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
23*61046927SAndroid Build Coastguard Worker * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
24*61046927SAndroid Build Coastguard Worker * FITNESS FOR A PARTICULAR PURPOSE, TITLE AND NON-INFRINGEMENT. IN NO EVENT
25*61046927SAndroid Build Coastguard Worker * SHALL THE COPYRIGHT HOLDERS OR ANYONE DISTRIBUTING THE SOFTWARE BE LIABLE
26*61046927SAndroid Build Coastguard Worker * FOR ANY DAMAGES OR OTHER LIABILITY, WHETHER IN CONTRACT, TORT OR OTHERWISE,
27*61046927SAndroid Build Coastguard Worker * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
28*61046927SAndroid Build Coastguard Worker * DEALINGS IN THE SOFTWARE.
29*61046927SAndroid Build Coastguard Worker */
30*61046927SAndroid Build Coastguard Worker
31*61046927SAndroid Build Coastguard Worker #include "c11/time.h"
32*61046927SAndroid Build Coastguard Worker
33*61046927SAndroid Build Coastguard Worker #ifdef _TIMESPEC_GET_NEED_IMPL
34*61046927SAndroid Build Coastguard Worker
35*61046927SAndroid Build Coastguard Worker #if defined(_WIN32) && !defined(HAVE_PTHREAD)
36*61046927SAndroid Build Coastguard Worker
37*61046927SAndroid Build Coastguard Worker #include "c11/threads.h"
38*61046927SAndroid Build Coastguard Worker #include <windows.h>
39*61046927SAndroid Build Coastguard Worker
40*61046927SAndroid Build Coastguard Worker static LARGE_INTEGER frequency;
41*61046927SAndroid Build Coastguard Worker
42*61046927SAndroid Build Coastguard Worker static
43*61046927SAndroid Build Coastguard Worker void
c23_timespec_get_init(void)44*61046927SAndroid Build Coastguard Worker c23_timespec_get_init(void)
45*61046927SAndroid Build Coastguard Worker {
46*61046927SAndroid Build Coastguard Worker QueryPerformanceFrequency(&frequency);
47*61046927SAndroid Build Coastguard Worker }
48*61046927SAndroid Build Coastguard Worker
49*61046927SAndroid Build Coastguard Worker int
c23_timespec_get(struct timespec * ts,int base)50*61046927SAndroid Build Coastguard Worker c23_timespec_get(struct timespec *ts, int base)
51*61046927SAndroid Build Coastguard Worker {
52*61046927SAndroid Build Coastguard Worker /* difference between 1970 and 1601 */
53*61046927SAndroid Build Coastguard Worker #define _TIMESPEC_IMPL_UNIX_EPOCH_IN_TICKS 116444736000000000ull
54*61046927SAndroid Build Coastguard Worker /* 1 tick is 100 nanoseconds */
55*61046927SAndroid Build Coastguard Worker #define _TIMESPEC_IMPL_TICKS_PER_SECONDS 10000000ull
56*61046927SAndroid Build Coastguard Worker if (!ts)
57*61046927SAndroid Build Coastguard Worker return 0;
58*61046927SAndroid Build Coastguard Worker if (base == TIME_UTC) {
59*61046927SAndroid Build Coastguard Worker FILETIME ft;
60*61046927SAndroid Build Coastguard Worker ULARGE_INTEGER date;
61*61046927SAndroid Build Coastguard Worker LONGLONG ticks;
62*61046927SAndroid Build Coastguard Worker
63*61046927SAndroid Build Coastguard Worker GetSystemTimeAsFileTime(&ft);
64*61046927SAndroid Build Coastguard Worker date.HighPart = ft.dwHighDateTime;
65*61046927SAndroid Build Coastguard Worker date.LowPart = ft.dwLowDateTime;
66*61046927SAndroid Build Coastguard Worker ticks = (LONGLONG)(date.QuadPart - _TIMESPEC_IMPL_UNIX_EPOCH_IN_TICKS);
67*61046927SAndroid Build Coastguard Worker ts->tv_sec = ticks / _TIMESPEC_IMPL_TICKS_PER_SECONDS;
68*61046927SAndroid Build Coastguard Worker ts->tv_nsec = (ticks % _TIMESPEC_IMPL_TICKS_PER_SECONDS) * 100;
69*61046927SAndroid Build Coastguard Worker return base;
70*61046927SAndroid Build Coastguard Worker } else if (base == TIME_MONOTONIC || base == TIME_MONOTONIC_RAW) {
71*61046927SAndroid Build Coastguard Worker if (frequency.QuadPart == 0) {
72*61046927SAndroid Build Coastguard Worker static once_flag once = ONCE_FLAG_INIT;
73*61046927SAndroid Build Coastguard Worker call_once(&once, c23_timespec_get_init);
74*61046927SAndroid Build Coastguard Worker }
75*61046927SAndroid Build Coastguard Worker if (frequency.QuadPart != 0) {
76*61046927SAndroid Build Coastguard Worker LARGE_INTEGER now;
77*61046927SAndroid Build Coastguard Worker LONGLONG sec;
78*61046927SAndroid Build Coastguard Worker LONGLONG nsec;
79*61046927SAndroid Build Coastguard Worker QueryPerformanceCounter(&now);
80*61046927SAndroid Build Coastguard Worker sec = now.QuadPart / frequency.QuadPart;
81*61046927SAndroid Build Coastguard Worker nsec = (now.QuadPart - sec * frequency.QuadPart)
82*61046927SAndroid Build Coastguard Worker * 1000000000UL / frequency.QuadPart;
83*61046927SAndroid Build Coastguard Worker ts->tv_sec = (time_t)sec;
84*61046927SAndroid Build Coastguard Worker ts->tv_nsec = (long)nsec;
85*61046927SAndroid Build Coastguard Worker return base;
86*61046927SAndroid Build Coastguard Worker }
87*61046927SAndroid Build Coastguard Worker /* Otherwise timespec_get with TIME_MONOTONIC or TIME_MONOTONIC_RAW failed */
88*61046927SAndroid Build Coastguard Worker return 0;
89*61046927SAndroid Build Coastguard Worker }
90*61046927SAndroid Build Coastguard Worker return 0;
91*61046927SAndroid Build Coastguard Worker #undef _TIMESPEC_IMPL_UNIX_EPOCH_IN_TICKS
92*61046927SAndroid Build Coastguard Worker #undef _TIMESPEC_IMPL_TICKS_PER_SECONDS
93*61046927SAndroid Build Coastguard Worker }
94*61046927SAndroid Build Coastguard Worker
95*61046927SAndroid Build Coastguard Worker #else
96*61046927SAndroid Build Coastguard Worker
c23_timespec_get(struct timespec * ts,int base)97*61046927SAndroid Build Coastguard Worker int c23_timespec_get(struct timespec *ts, int base)
98*61046927SAndroid Build Coastguard Worker {
99*61046927SAndroid Build Coastguard Worker if (!ts)
100*61046927SAndroid Build Coastguard Worker return 0;
101*61046927SAndroid Build Coastguard Worker switch (base)
102*61046927SAndroid Build Coastguard Worker {
103*61046927SAndroid Build Coastguard Worker case TIME_UTC:
104*61046927SAndroid Build Coastguard Worker if (clock_gettime(CLOCK_REALTIME, ts) == 0)
105*61046927SAndroid Build Coastguard Worker return base;
106*61046927SAndroid Build Coastguard Worker break;
107*61046927SAndroid Build Coastguard Worker #ifdef CLOCK_MONOTONIC
108*61046927SAndroid Build Coastguard Worker case TIME_MONOTONIC:
109*61046927SAndroid Build Coastguard Worker if (clock_gettime(CLOCK_MONOTONIC, ts) == 0)
110*61046927SAndroid Build Coastguard Worker return base;
111*61046927SAndroid Build Coastguard Worker break;
112*61046927SAndroid Build Coastguard Worker #endif
113*61046927SAndroid Build Coastguard Worker #ifdef CLOCK_PROCESS_CPUTIME_ID
114*61046927SAndroid Build Coastguard Worker case TIME_ACTIVE:
115*61046927SAndroid Build Coastguard Worker if (clock_gettime(CLOCK_PROCESS_CPUTIME_ID, ts) == 0)
116*61046927SAndroid Build Coastguard Worker return base;
117*61046927SAndroid Build Coastguard Worker break;
118*61046927SAndroid Build Coastguard Worker #endif
119*61046927SAndroid Build Coastguard Worker #ifdef CLOCK_THREAD_CPUTIME_ID
120*61046927SAndroid Build Coastguard Worker case TIME_THREAD_ACTIVE:
121*61046927SAndroid Build Coastguard Worker if (clock_gettime(CLOCK_THREAD_CPUTIME_ID, ts) == 0)
122*61046927SAndroid Build Coastguard Worker return base;
123*61046927SAndroid Build Coastguard Worker break;
124*61046927SAndroid Build Coastguard Worker #endif
125*61046927SAndroid Build Coastguard Worker #ifdef CLOCK_MONOTONIC_RAW
126*61046927SAndroid Build Coastguard Worker case TIME_MONOTONIC_RAW:
127*61046927SAndroid Build Coastguard Worker if (clock_gettime(CLOCK_MONOTONIC_RAW, ts) == 0)
128*61046927SAndroid Build Coastguard Worker return base;
129*61046927SAndroid Build Coastguard Worker break;
130*61046927SAndroid Build Coastguard Worker #endif
131*61046927SAndroid Build Coastguard Worker default:
132*61046927SAndroid Build Coastguard Worker break;
133*61046927SAndroid Build Coastguard Worker }
134*61046927SAndroid Build Coastguard Worker return 0;
135*61046927SAndroid Build Coastguard Worker }
136*61046927SAndroid Build Coastguard Worker #endif
137*61046927SAndroid Build Coastguard Worker
138*61046927SAndroid Build Coastguard Worker #endif /* !HAVE_TIMESPEC_GET */
139