VirtualBox

source: vbox/trunk/src/VBox/Runtime/r3/win/timer-win.cpp@ 20364

Last change on this file since 20364 was 20364, checked in by vboxsync, 16 years ago

IPRT: BEGIN_DECLS -> RT_BEGIN_DECLS; END_DECLS -> RT_END_DECLS.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 14.8 KB
Line 
1/* $Id: timer-win.cpp 20364 2009-06-08 00:17:43Z vboxsync $ */
2/** @file
3 * IPRT - Timer.
4 */
5
6/*
7 * Copyright (C) 2006-2007 Sun Microsystems, Inc.
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 *
17 * The contents of this file may alternatively be used under the terms
18 * of the Common Development and Distribution License Version 1.0
19 * (CDDL) only, as it comes in the "COPYING.CDDL" file of the
20 * VirtualBox OSE distribution, in which case the provisions of the
21 * CDDL are applicable instead of those of the GPL.
22 *
23 * You may elect to license modified versions of this file under the
24 * terms and conditions of either the GPL or the CDDL or both.
25 *
26 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
27 * Clara, CA 95054 USA or visit http://www.sun.com if you need
28 * additional information or have any questions.
29 */
30
31
32/* Which code to use is determined here...
33 *
34 * The default is to use wait on NT timers directly with no APC since this
35 * is supposed to give the shortest kernel code paths.
36 *
37 * The USE_APC variation will do as above except that an APC routine is
38 * handling the callback action.
39 *
40 * The USE_WINMM version will use the NT timer wrappers in WinMM which may
41 * result in some 0.1% better correctness in number of delivered ticks. However,
42 * this codepath have more overhead (it uses APC among other things), and I'm not
43 * quite sure if it's actually any more correct.
44 *
45 * The USE_CATCH_UP will play catch up when the timer lags behind. However this
46 * requires a monotonous time source.
47 *
48 * The default mode which we are using is using relative periods of time and thus
49 * will never suffer from errors in the time source. Neither will it try catch up
50 * missed ticks. This suits our current purposes best I'd say.
51 */
52#undef USE_APC
53#undef USE_WINMM
54#undef USE_CATCH_UP
55
56
57/*******************************************************************************
58* Header Files *
59*******************************************************************************/
60#define LOG_GROUP RTLOGGROUP_TIMER
61#define _WIN32_WINNT 0x0500
62#include <Windows.h>
63
64#include <iprt/timer.h>
65#ifdef USE_CATCH_UP
66# include <iprt/time.h>
67#endif
68#include <iprt/alloc.h>
69#include <iprt/assert.h>
70#include <iprt/thread.h>
71#include <iprt/log.h>
72#include <iprt/asm.h>
73#include <iprt/semaphore.h>
74#include <iprt/err.h>
75#include "internal/magics.h"
76
77RT_BEGIN_DECLS
78/* from sysinternals. */
79NTSYSAPI LONG NTAPI NtSetTimerResolution(IN ULONG DesiredResolution, IN BOOLEAN SetResolution, OUT PULONG CurrentResolution);
80NTSYSAPI LONG NTAPI NtQueryTimerResolution(OUT PULONG MaximumResolution, OUT PULONG MinimumResolution, OUT PULONG CurrentResolution);
81RT_END_DECLS
82
83
84/*******************************************************************************
85* Structures and Typedefs *
86*******************************************************************************/
87/**
88 * The internal representation of a timer handle.
89 */
90typedef struct RTTIMER
91{
92 /** Magic.
93 * This is RTTIMER_MAGIC, but changes to something else before the timer
94 * is destroyed to indicate clearly that thread should exit. */
95 volatile uint32_t u32Magic;
96 /** User argument. */
97 void *pvUser;
98 /** Callback. */
99 PFNRTTIMER pfnTimer;
100 /** The current tick. */
101 uint64_t iTick;
102 /** The interval. */
103 unsigned uMilliesInterval;
104#ifdef USE_WINMM
105 /** Win32 timer id. */
106 UINT TimerId;
107#else
108 /** Time handle. */
109 HANDLE hTimer;
110# ifdef USE_APC
111 /** Handle to wait on. */
112 HANDLE hevWait;
113# endif
114 /** USE_CATCH_UP: ns time of the next tick.
115 * !USE_CATCH_UP: -uMilliesInterval * 10000 */
116 LARGE_INTEGER llNext;
117 /** The thread handle of the timer thread. */
118 RTTHREAD Thread;
119 /** The error/status of the timer.
120 * Initially -1, set to 0 when the timer have been successfully started, and
121 * to errno on failure in starting the timer. */
122 volatile int iError;
123#endif
124} RTTIMER;
125
126
127
128#ifdef USE_WINMM
129/**
130 * Win32 callback wrapper.
131 */
132static void CALLBACK rttimerCallback(UINT uTimerID, UINT uMsg, DWORD_PTR dwUser, DWORD_PTR dw1, DWORD_PTR dw2)
133{
134 PRTTIMER pTimer = (PRTTIMER)(void *)dwUser;
135 Assert(pTimer->TimerId == uTimerID);
136 pTimer->pfnTimer(pTimer, pTimer->pvUser, ++pTimer->iTick);
137 NOREF(uMsg); NOREF(dw1); NOREF(dw2); NOREF(uTimerID);
138}
139#else /* !USE_WINMM */
140
141#ifdef USE_APC
142/**
143 * Async callback.
144 *
145 * @param lpArgToCompletionRoutine Pointer to our timer structure.
146 */
147VOID CALLBACK rttimerAPCProc(LPVOID lpArgToCompletionRoutine, DWORD dwTimerLowValue, DWORD dwTimerHighValue)
148{
149 PRTTIMER pTimer = (PRTTIMER)lpArgToCompletionRoutine;
150
151 /*
152 * Check if we're begin destroyed.
153 */
154 if (pTimer->u32Magic != RTTIMER_MAGIC)
155 return;
156
157 /*
158 * Callback the handler.
159 */
160 pTimer->pfnTimer(pTimer, pTimer->pvUser, ++pTimer->iTick);
161
162 /*
163 * Rearm the timer handler.
164 */
165#ifdef USE_CATCH_UP
166 pTimer->llNext.QuadPart += (int64_t)pTimer->uMilliesInterval * 10000;
167 LARGE_INTEGER ll;
168 ll.QuadPart = RTTimeNanoTS() - pTimer->llNext.QuadPart;
169 if (ll.QuadPart < -500000)
170 ll.QuadPart = ll.QuadPart / 100;
171 else
172 ll.QuadPart = -500000 / 100; /* need to catch up, do a minimum wait of 0.5ms. */
173#else
174 LARGE_INTEGER ll = pTimer->llNext;
175#endif
176 BOOL frc = SetWaitableTimer(pTimer->hTimer, &ll, 0, rttimerAPCProc, pTimer, FALSE);
177 AssertMsg(frc || pTimer->u32Magic != RTTIMER_MAGIC, ("last error %d\n", GetLastError()));
178}
179#endif /* USE_APC */
180
181/**
182 * Timer thread.
183 */
184static DECLCALLBACK(int) rttimerCallback(RTTHREAD Thread, void *pvArg)
185{
186 PRTTIMER pTimer = (PRTTIMER)(void *)pvArg;
187 Assert(pTimer->u32Magic == RTTIMER_MAGIC);
188
189 /*
190 * Bounce our priority up quite a bit.
191 */
192 if ( !SetThreadPriority(GetCurrentThread(), THREAD_PRIORITY_TIME_CRITICAL)
193 /*&& !SetThreadPriority(GetCurrentThread(), THREAD_PRIORITY_HIGHEST)*/)
194 {
195 int rc = GetLastError();
196 AssertMsgFailed(("Failed to set priority class lasterror %d.\n", rc));
197 pTimer->iError = RTErrConvertFromWin32(rc);
198 return rc;
199 }
200
201 /*
202 * Start the waitable timer.
203 */
204
205#ifdef USE_CATCH_UP
206 const int64_t NSInterval = (int64_t)pTimer->uMilliesInterval * 1000000;
207 pTimer->llNext.QuadPart = RTTimeNanoTS() + NSInterval;
208#else
209 pTimer->llNext.QuadPart = -(int64_t)pTimer->uMilliesInterval * 10000;
210#endif
211 LARGE_INTEGER ll;
212 ll.QuadPart = -(int64_t)pTimer->uMilliesInterval * 10000;
213#ifdef USE_APC
214 if (!SetWaitableTimer(pTimer->hTimer, &ll, 0, rttimerAPCProc, pTimer, FALSE))
215#else
216 if (!SetWaitableTimer(pTimer->hTimer, &ll, 0, NULL, NULL, FALSE))
217#endif
218 {
219 int rc = GetLastError();
220 AssertMsgFailed(("Failed to set timer, lasterr %d.\n", rc));
221 pTimer->iError = RTErrConvertFromWin32(rc);
222 RTThreadUserSignal(Thread);
223 return rc;
224 }
225
226 /*
227 * Wait for the semaphore to be posted.
228 */
229 RTThreadUserSignal(Thread);
230 for (;pTimer->u32Magic == RTTIMER_MAGIC;)
231 {
232#ifdef USE_APC
233 int rc = WaitForSingleObjectEx(pTimer->hevWait, INFINITE, TRUE);
234 if (rc != WAIT_OBJECT_0 && rc != WAIT_IO_COMPLETION)
235#else
236 int rc = WaitForSingleObjectEx(pTimer->hTimer, INFINITE, FALSE);
237 if (pTimer->u32Magic != RTTIMER_MAGIC)
238 break;
239 if (rc == WAIT_OBJECT_0)
240 {
241 /*
242 * Callback the handler.
243 */
244 pTimer->pfnTimer(pTimer, pTimer->pvUser, ++pTimer->iTick);
245
246 /*
247 * Rearm the timer handler.
248 */
249#ifdef USE_CATCH_UP
250 pTimer->llNext.QuadPart += NSInterval;
251 LARGE_INTEGER ll;
252 ll.QuadPart = RTTimeNanoTS() - pTimer->llNext.QuadPart;
253 if (ll.QuadPart < -500000)
254 ll.QuadPart = ll.QuadPart / 100;
255 else
256 ll.QuadPart = -500000 / 100; /* need to catch up, do a minimum wait of 0.5ms. */
257#else
258 LARGE_INTEGER ll = pTimer->llNext;
259#endif
260 BOOL frc = SetWaitableTimer(pTimer->hTimer, &ll, 0, NULL, NULL, FALSE);
261 AssertMsg(frc || pTimer->u32Magic != RTTIMER_MAGIC, ("last error %d\n", GetLastError()));
262 }
263 else
264#endif
265 {
266 /*
267 * We failed during wait, so just signal the destructor and exit.
268 */
269 int rc2 = GetLastError();
270 RTThreadUserSignal(Thread);
271 AssertMsgFailed(("Wait on hTimer failed, rc=%d lasterr=%d\n", rc, rc2));
272 return -1;
273 }
274 }
275
276 /*
277 * Exit.
278 */
279 RTThreadUserSignal(Thread);
280 return 0;
281}
282#endif /* !USE_WINMM */
283
284
285RTDECL(int) RTTimerCreate(PRTTIMER *ppTimer, unsigned uMilliesInterval, PFNRTTIMER pfnTimer, void *pvUser)
286{
287#ifndef USE_WINMM
288 /*
289 * On windows we'll have to set the timer resolution before
290 * we start the timer.
291 */
292 ULONG ulMax = ~0;
293 ULONG ulMin = ~0;
294 ULONG ulCur = ~0;
295 NtQueryTimerResolution(&ulMax, &ulMin, &ulCur);
296 Log(("NtQueryTimerResolution -> ulMax=%lu00ns ulMin=%lu00ns ulCur=%lu00ns\n", ulMax, ulMin, ulCur));
297 if (ulCur > ulMin && ulCur > 10000 /* = 1ms */)
298 {
299 if (NtSetTimerResolution(10000, TRUE, &ulCur) >= 0)
300 Log(("Changed timer resolution to 1ms.\n"));
301 else if (NtSetTimerResolution(20000, TRUE, &ulCur) >= 0)
302 Log(("Changed timer resolution to 2ms.\n"));
303 else if (NtSetTimerResolution(40000, TRUE, &ulCur) >= 0)
304 Log(("Changed timer resolution to 4ms.\n"));
305 else if (ulMin <= 50000 && NtSetTimerResolution(ulMin, TRUE, &ulCur) >= 0)
306 Log(("Changed timer resolution to %lu *100ns.\n", ulMin));
307 else
308 {
309 AssertMsgFailed(("Failed to configure timer resolution!\n"));
310 return VERR_INTERNAL_ERROR;
311 }
312 }
313#endif /* !USE_WINN */
314
315 /*
316 * Create new timer.
317 */
318 int rc;
319 PRTTIMER pTimer = (PRTTIMER)RTMemAlloc(sizeof(*pTimer));
320 if (pTimer)
321 {
322 pTimer->u32Magic = RTTIMER_MAGIC;
323 pTimer->pvUser = pvUser;
324 pTimer->pfnTimer = pfnTimer;
325 pTimer->iTick = 0;
326 pTimer->uMilliesInterval = uMilliesInterval;
327#ifdef USE_WINMM
328 /* sync kill doesn't work. */
329 pTimer->TimerId = timeSetEvent(uMilliesInterval, 0, rttimerCallback, (DWORD_PTR)pTimer, TIME_PERIODIC | TIME_CALLBACK_FUNCTION);
330 if (pTimer->TimerId)
331 {
332 ULONG ulMax = ~0;
333 ULONG ulMin = ~0;
334 ULONG ulCur = ~0;
335 NtQueryTimerResolution(&ulMax, &ulMin, &ulCur);
336 Log(("NtQueryTimerResolution -> ulMax=%lu00ns ulMin=%lu00ns ulCur=%lu00ns\n", ulMax, ulMin, ulCur));
337
338 *ppTimer = pTimer;
339 return VINF_SUCCESS;
340 }
341 rc = VERR_INVALID_PARAMETER;
342
343#else /* !USE_WINMM */
344
345 /*
346 * Create Win32 event semaphore.
347 */
348 pTimer->iError = 0;
349 pTimer->hTimer = CreateWaitableTimer(NULL, TRUE, NULL);
350 if (pTimer->hTimer)
351 {
352#ifdef USE_APC
353 /*
354 * Create wait semaphore.
355 */
356 pTimer->hevWait = CreateEvent(NULL, FALSE, FALSE, NULL);
357 if (pTimer->hevWait)
358#endif
359 {
360 /*
361 * Kick off the timer thread.
362 */
363 rc = RTThreadCreate(&pTimer->Thread, rttimerCallback, pTimer, 0, RTTHREADTYPE_TIMER, RTTHREADFLAGS_WAITABLE, "Timer");
364 if (RT_SUCCESS(rc))
365 {
366 /*
367 * Wait for the timer to successfully create the timer
368 * If we don't get a response in 10 secs, then we assume we're screwed.
369 */
370 rc = RTThreadUserWait(pTimer->Thread, 10000);
371 if (RT_SUCCESS(rc))
372 {
373 rc = pTimer->iError;
374 if (RT_SUCCESS(rc))
375 {
376 *ppTimer = pTimer;
377 return VINF_SUCCESS;
378 }
379 }
380 ASMAtomicXchgU32(&pTimer->u32Magic, RTTIMER_MAGIC + 1);
381 RTThreadWait(pTimer->Thread, 250, NULL);
382 CancelWaitableTimer(pTimer->hTimer);
383 }
384#ifdef USE_APC
385 CloseHandle(pTimer->hevWait);
386#endif
387 }
388 CloseHandle(pTimer->hTimer);
389 }
390#endif /* !USE_WINMM */
391
392 AssertMsgFailed(("Failed to create timer uMilliesInterval=%d. rc=%d\n", uMilliesInterval, rc));
393 RTMemFree(pTimer);
394 }
395 else
396 rc = VERR_NO_MEMORY;
397 return rc;
398}
399
400
401RTR3DECL(int) RTTimerDestroy(PRTTIMER pTimer)
402{
403 /* NULL is ok. */
404 if (!pTimer)
405 return VINF_SUCCESS;
406
407 /*
408 * Validate handle first.
409 */
410 int rc;
411 if ( VALID_PTR(pTimer)
412 && pTimer->u32Magic == RTTIMER_MAGIC)
413 {
414#ifdef USE_WINMM
415 /*
416 * Kill the timer and exit.
417 */
418 rc = timeKillEvent(pTimer->TimerId);
419 AssertMsg(rc == TIMERR_NOERROR, ("timeKillEvent -> %d\n", rc));
420 ASMAtomicXchgU32(&pTimer->u32Magic, RTTIMER_MAGIC + 1);
421 RTThreadSleep(1);
422
423#else /* !USE_WINMM */
424
425 /*
426 * Signal that we want the thread to exit.
427 */
428 ASMAtomicXchgU32(&pTimer->u32Magic, RTTIMER_MAGIC + 1);
429#ifdef USE_APC
430 SetEvent(pTimer->hevWait);
431 CloseHandle(pTimer->hevWait);
432 rc = CancelWaitableTimer(pTimer->hTimer);
433 AssertMsg(rc, ("CancelWaitableTimer lasterr=%d\n", GetLastError()));
434#else
435 LARGE_INTEGER ll = {0};
436 ll.LowPart = 100;
437 rc = SetWaitableTimer(pTimer->hTimer, &ll, 0, NULL, NULL, FALSE);
438 AssertMsg(rc, ("CancelWaitableTimer lasterr=%d\n", GetLastError()));
439#endif
440
441 /*
442 * Wait for the thread to exit.
443 * And if it don't wanna exit, we'll get kill it.
444 */
445 rc = RTThreadWait(pTimer->Thread, 1000, NULL);
446 if (RT_FAILURE(rc))
447 TerminateThread((HANDLE)RTThreadGetNative(pTimer->Thread), -1);
448
449 /*
450 * Free resource.
451 */
452 rc = CloseHandle(pTimer->hTimer);
453 AssertMsg(rc, ("CloseHandle lasterr=%d\n", GetLastError()));
454
455#endif /* !USE_WINMM */
456 RTMemFree(pTimer);
457 return rc;
458 }
459
460 rc = VERR_INVALID_HANDLE;
461 AssertMsgFailed(("Failed to destroy timer %p. rc=%d\n", pTimer, rc));
462 return rc;
463}
464
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