1 /*
2 * Copyright (C) 2006 Apple Computer, Inc. All rights reserved.
3 * Copyright (C) 2008 Google Inc. All rights reserved.
4 * Copyright (C) 2007-2009 Torch Mobile, Inc.
5 *
6 * Redistribution and use in source and binary forms, with or without
7 * modification, are permitted provided that the following conditions are
8 * met:
9 *
10 * * Redistributions of source code must retain the above copyright
11 * notice, this list of conditions and the following disclaimer.
12 * * Redistributions in binary form must reproduce the above
13 * copyright notice, this list of conditions and the following disclaimer
14 * in the documentation and/or other materials provided with the
15 * distribution.
16 * * Neither the name of Google Inc. nor the names of its
17 * contributors may be used to endorse or promote products derived from
18 * this software without specific prior written permission.
19 *
20 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
21 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
22 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
23 * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
24 * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
25 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
26 * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
27 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
28 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
29 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
30 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
31 */
32
33 #include "config.h"
34 #include "CurrentTime.h"
35
36 #if PLATFORM(WIN_OS)
37
38 // Windows is first since we want to use hires timers, despite PLATFORM(CF)
39 // being defined.
40 // If defined, WIN32_LEAN_AND_MEAN disables timeBeginPeriod/timeEndPeriod.
41 #undef WIN32_LEAN_AND_MEAN
42 #include <windows.h>
43 #include <math.h>
44 #include <stdint.h>
45 #include <time.h>
46
47 #if USE(QUERY_PERFORMANCE_COUNTER)
48 #if PLATFORM(WINCE)
49 extern "C" time_t mktime(struct tm *t);
50 #else
51 #include <sys/timeb.h>
52 #include <sys/types.h>
53 #endif
54 #endif
55
56 #elif PLATFORM(CF)
57 #include <CoreFoundation/CFDate.h>
58 #elif PLATFORM(GTK)
59 #include <glib.h>
60 #elif PLATFORM(WX)
61 #include <wx/datetime.h>
62 #else // Posix systems relying on the gettimeofday()
63 #include <sys/time.h>
64 #endif
65
66 namespace WTF {
67
68 const double msPerSecond = 1000.0;
69
70 #if PLATFORM(WIN_OS)
71
72 #if USE(QUERY_PERFORMANCE_COUNTER)
73
74 static LARGE_INTEGER qpcFrequency;
75 static bool syncedTime;
76
highResUpTime()77 static double highResUpTime()
78 {
79 // We use QPC, but only after sanity checking its result, due to bugs:
80 // http://support.microsoft.com/kb/274323
81 // http://support.microsoft.com/kb/895980
82 // http://msdn.microsoft.com/en-us/library/ms644904.aspx ("...you can get different results on different processors due to bugs in the basic input/output system (BIOS) or the hardware abstraction layer (HAL)."
83
84 static LARGE_INTEGER qpcLast;
85 static DWORD tickCountLast;
86 static bool inited;
87
88 LARGE_INTEGER qpc;
89 QueryPerformanceCounter(&qpc);
90 DWORD tickCount = GetTickCount();
91
92 if (inited) {
93 __int64 qpcElapsed = ((qpc.QuadPart - qpcLast.QuadPart) * 1000) / qpcFrequency.QuadPart;
94 __int64 tickCountElapsed;
95 if (tickCount >= tickCountLast)
96 tickCountElapsed = (tickCount - tickCountLast);
97 else {
98 #if COMPILER(MINGW)
99 __int64 tickCountLarge = tickCount + 0x100000000ULL;
100 #else
101 __int64 tickCountLarge = tickCount + 0x100000000I64;
102 #endif
103 tickCountElapsed = tickCountLarge - tickCountLast;
104 }
105
106 // force a re-sync if QueryPerformanceCounter differs from GetTickCount by more than 500ms.
107 // (500ms value is from http://support.microsoft.com/kb/274323)
108 __int64 diff = tickCountElapsed - qpcElapsed;
109 if (diff > 500 || diff < -500)
110 syncedTime = false;
111 } else
112 inited = true;
113
114 qpcLast = qpc;
115 tickCountLast = tickCount;
116
117 return (1000.0 * qpc.QuadPart) / static_cast<double>(qpcFrequency.QuadPart);
118 }
119
lowResUTCTime()120 static double lowResUTCTime()
121 {
122 #if PLATFORM(WINCE)
123 SYSTEMTIME systemTime;
124 GetSystemTime(&systemTime);
125 struct tm tmtime;
126 tmtime.tm_year = systemTime.wYear - 1900;
127 tmtime.tm_mon = systemTime.wMonth - 1;
128 tmtime.tm_mday = systemTime.wDay;
129 tmtime.tm_wday = systemTime.wDayOfWeek;
130 tmtime.tm_hour = systemTime.wHour;
131 tmtime.tm_min = systemTime.wMinute;
132 tmtime.tm_sec = systemTime.wSecond;
133 time_t timet = mktime(&tmtime);
134 return timet * msPerSecond + systemTime.wMilliseconds;
135 #else
136 struct _timeb timebuffer;
137 _ftime(&timebuffer);
138 return timebuffer.time * msPerSecond + timebuffer.millitm;
139 #endif
140 }
141
qpcAvailable()142 static bool qpcAvailable()
143 {
144 static bool available;
145 static bool checked;
146
147 if (checked)
148 return available;
149
150 available = QueryPerformanceFrequency(&qpcFrequency);
151 checked = true;
152 return available;
153 }
154
currentTime()155 double currentTime()
156 {
157 // Use a combination of ftime and QueryPerformanceCounter.
158 // ftime returns the information we want, but doesn't have sufficient resolution.
159 // QueryPerformanceCounter has high resolution, but is only usable to measure time intervals.
160 // To combine them, we call ftime and QueryPerformanceCounter initially. Later calls will use QueryPerformanceCounter
161 // by itself, adding the delta to the saved ftime. We periodically re-sync to correct for drift.
162 static bool started;
163 static double syncLowResUTCTime;
164 static double syncHighResUpTime;
165 static double lastUTCTime;
166
167 double lowResTime = lowResUTCTime();
168
169 if (!qpcAvailable())
170 return lowResTime / 1000.0;
171
172 double highResTime = highResUpTime();
173
174 if (!syncedTime) {
175 timeBeginPeriod(1); // increase time resolution around low-res time getter
176 syncLowResUTCTime = lowResTime = lowResUTCTime();
177 timeEndPeriod(1); // restore time resolution
178 syncHighResUpTime = highResTime;
179 syncedTime = true;
180 }
181
182 double highResElapsed = highResTime - syncHighResUpTime;
183 double utc = syncLowResUTCTime + highResElapsed;
184
185 // force a clock re-sync if we've drifted
186 double lowResElapsed = lowResTime - syncLowResUTCTime;
187 const double maximumAllowedDriftMsec = 15.625 * 2.0; // 2x the typical low-res accuracy
188 if (fabs(highResElapsed - lowResElapsed) > maximumAllowedDriftMsec)
189 syncedTime = false;
190
191 // make sure time doesn't run backwards (only correct if difference is < 2 seconds, since DST or clock changes could occur)
192 const double backwardTimeLimit = 2000.0;
193 if (utc < lastUTCTime && (lastUTCTime - utc) < backwardTimeLimit)
194 return lastUTCTime / 1000.0;
195 lastUTCTime = utc;
196 return utc / 1000.0;
197 }
198
199 #else
200
currentSystemTime()201 static double currentSystemTime()
202 {
203 FILETIME ft;
204 GetCurrentFT(&ft);
205
206 // As per Windows documentation for FILETIME, copy the resulting FILETIME structure to a
207 // ULARGE_INTEGER structure using memcpy (using memcpy instead of direct assignment can
208 // prevent alignment faults on 64-bit Windows).
209
210 ULARGE_INTEGER t;
211 memcpy(&t, &ft, sizeof(t));
212
213 // Windows file times are in 100s of nanoseconds.
214 // To convert to seconds, we have to divide by 10,000,000, which is more quickly
215 // done by multiplying by 0.0000001.
216
217 // Between January 1, 1601 and January 1, 1970, there were 369 complete years,
218 // of which 89 were leap years (1700, 1800, and 1900 were not leap years).
219 // That is a total of 134774 days, which is 11644473600 seconds.
220
221 return t.QuadPart * 0.0000001 - 11644473600.0;
222 }
223
currentTime()224 double currentTime()
225 {
226 static bool init = false;
227 static double lastTime;
228 static DWORD lastTickCount;
229 if (!init) {
230 lastTime = currentSystemTime();
231 lastTickCount = GetTickCount();
232 init = true;
233 return lastTime;
234 }
235
236 DWORD tickCountNow = GetTickCount();
237 DWORD elapsed = tickCountNow - lastTickCount;
238 double timeNow = lastTime + (double)elapsed / 1000.;
239 if (elapsed >= 0x7FFFFFFF) {
240 lastTime = timeNow;
241 lastTickCount = tickCountNow;
242 }
243 return timeNow;
244 }
245
246 #endif // USE(QUERY_PERFORMANCE_COUNTER)
247
248 #elif PLATFORM(CF)
249
currentTime()250 double currentTime()
251 {
252 return CFAbsoluteTimeGetCurrent() + kCFAbsoluteTimeIntervalSince1970;
253 }
254
255 #elif PLATFORM(GTK)
256
257 // Note: GTK on Windows will pick up the PLATFORM(WIN) implementation above which provides
258 // better accuracy compared with Windows implementation of g_get_current_time:
259 // (http://www.google.com/codesearch/p?hl=en#HHnNRjks1t0/glib-2.5.2/glib/gmain.c&q=g_get_current_time).
260 // Non-Windows GTK builds could use gettimeofday() directly but for the sake of consistency lets use GTK function.
currentTime()261 double currentTime()
262 {
263 GTimeVal now;
264 g_get_current_time(&now);
265 return static_cast<double>(now.tv_sec) + static_cast<double>(now.tv_usec / 1000000.0);
266 }
267
268 #elif PLATFORM(WX)
269
currentTime()270 double currentTime()
271 {
272 wxDateTime now = wxDateTime::UNow();
273 return (double)now.GetTicks() + (double)(now.GetMillisecond() / 1000.0);
274 }
275
276 #else // Other Posix systems rely on the gettimeofday().
277
currentTime()278 double currentTime()
279 {
280 struct timeval now;
281 struct timezone zone;
282
283 gettimeofday(&now, &zone);
284 return static_cast<double>(now.tv_sec) + (double)(now.tv_usec / 1000000.0);
285 }
286
287 #endif
288
289 #if PLATFORM(ANDROID)
290
get_thread_msec()291 uint32_t get_thread_msec()
292 {
293 #if defined(HAVE_POSIX_CLOCKS)
294 struct timespec tm;
295
296 clock_gettime(CLOCK_THREAD_CPUTIME_ID, &tm);
297 return tm.tv_sec * 1000LL + tm.tv_nsec / 1000000;
298 #else
299 struct timeval now;
300 struct timezone zone;
301
302 gettimeofday(&now, &zone);
303 return now.tv_sec * 1000LL + now.tv_usec / 1000;
304 #endif
305 }
306
307 #endif
308
309 } // namespace WTF
310