VirtualBox

source: vbox/trunk/src/VBox/Additions/common/VBoxService/VBoxServiceTimeSync.cpp@ 20038

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

VBoxService/common: SVN props.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 20.7 KB
Line 
1/** $Id: VBoxServiceTimeSync.cpp 19374 2009-05-05 13:23:32Z vboxsync $ */
2/** @file
3 * VBoxService - Guest Additions TimeSync Service.
4 */
5
6/*
7 * Copyright (C) 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 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
18 * Clara, CA 95054 USA or visit http://www.sun.com if you need
19 * additional information or have any questions.
20 */
21
22
23/** @page pg_vboxservice_timesync The Time Sync Service
24 *
25 * The time sync service plays along with the Time Manager (TM) in the VMM
26 * to keep the guest time accurate using the host machine as reference.
27 * TM will try its best to make sure all timer ticks gets delivered so that
28 * there isn't normally any need to adjust the guest time.
29 *
30 * There are three normal (= acceptable) cases:
31 * -# When the service starts up. This is because ticks and such might
32 * be lost during VM and OS startup. (Need to figure out exactly why!)
33 * -# When the TM is unable to deliver all the ticks and swallows a
34 * backlog of ticks. The threshold for this is configurable with
35 * a default of 60 seconds.
36 * -# The time is adjusted on the host. This can be caused manually by
37 * the user or by some time sync daemon (NTP, LAN server, etc.).
38 *
39 * There are a number of very odd case where adjusting is needed. Here
40 * are some of them:
41 * -# Timer device emulation inaccurancies (like rounding).
42 * -# Inaccurancies in time source VirtualBox uses.
43 * -# The Guest and/or Host OS doesn't perform proper time keeping. This
44 * come about as a result of OS and/or hardware issues.
45 *
46 * The TM is our source for the host time and will make adjustments for
47 * current timer delivery lag. The simplistic approach taken by TM is to
48 * adjust the host time by the current guest timer delivery lag, meaning that
49 * if the guest is behind 1 second with PIT/RTC/++ ticks this should be reflected
50 * in the guest wall time as well.
51 *
52 * Now, there is any amount of trouble we can cause by changing the time.
53 * Most applications probably uses the wall time when they need to measure
54 * things. A walltime that is being juggled about every so often, even if just
55 * a little bit, could occationally upset these measurements by for instance
56 * yielding negative results.
57 *
58 * This bottom line here is that the time sync service isn't really supposed
59 * to do anything and will try avoid having to do anything when possible.
60 *
61 * The implementation uses the latency it takes to query host time as the
62 * absolute maximum precision to avoid messing up under timer tick catchup
63 * and/or heavy host/guest load. (Rational is that a *lot* of stuff may happen
64 * on our way back from ring-3 and TM/VMMDev since we're taking the route
65 * thru the inner EM loop with it's force action processing.)
66 *
67 * But this latency has to be measured from our perspective, which means it
68 * could just as easily come out as 0. (OS/2 and Windows guest only updates
69 * the current time when the timer ticks for instance.) The good thing is
70 * that this isn't really a problem since we won't ever do anything unless
71 * the drift is noticable.
72 *
73 * It now boils down to these three (configuration) factors:
74 * -# g_TimesyncMinAdjust - The minimum drift we will ever bother with.
75 * -# g_TimesyncLatencyFactor - The factor we multiply the latency by to
76 * calculate the dynamic minimum adjust factor.
77 * -# g_TimesyncMaxLatency - When to start discarding the data as utterly
78 * useless and take a rest (someone is too busy to give us good data).
79 */
80
81
82
83/*******************************************************************************
84* Header Files *
85*******************************************************************************/
86#ifdef RT_OS_WINDOWS
87# include <windows.h>
88# include <winbase.h>
89#else
90# include <unistd.h>
91# include <errno.h>
92# include <time.h>
93# include <sys/time.h>
94#endif
95
96#include <iprt/thread.h>
97#include <iprt/string.h>
98#include <iprt/semaphore.h>
99#include <iprt/time.h>
100#include <iprt/assert.h>
101#include <VBox/VBoxGuest.h>
102#include "VBoxServiceInternal.h"
103
104
105/*******************************************************************************
106* Global Variables *
107*******************************************************************************/
108/** The timesync interval (millseconds). */
109uint32_t g_TimeSyncInterval = 0;
110/**
111 * @see pg_vboxservice_timesync
112 *
113 * @remark OS/2: There is either a 1 second resolution on the DosSetDateTime
114 * API or a but in the settimeofday implementation. Thus, don't
115 * bother unless there is at least a 1 second drift.
116 */
117#ifdef RT_OS_OS2
118static uint32_t g_TimeSyncMinAdjust = 1000;
119#else
120static uint32_t g_TimeSyncMinAdjust = 100;
121#endif
122/** @see pg_vboxservice_timesync */
123static uint32_t g_TimeSyncLatencyFactor = 8;
124/** @see pg_vboxservice_timesync */
125static uint32_t g_TimeSyncMaxLatency = 250;
126
127/** The semaphore we're blocking on. */
128static RTSEMEVENTMULTI g_TimeSyncEvent = NIL_RTSEMEVENTMULTI;
129
130#ifdef RT_OS_WINDOWS
131/** Process token. */
132static HANDLE g_hTokenProcess = NULL;
133/** Old token privileges. */
134static TOKEN_PRIVILEGES g_TkOldPrivileges;
135/** Backup values for time adjustment. */
136static DWORD g_dwWinTimeAdjustment;
137static DWORD g_dwWinTimeIncrement;
138static BOOL g_bWinTimeAdjustmentDisabled;
139#endif
140
141
142/** @copydoc VBOXSERVICE::pfnPreInit */
143static DECLCALLBACK(int) VBoxServiceTimeSyncPreInit(void)
144{
145 return VINF_SUCCESS;
146}
147
148
149/** @copydoc VBOXSERVICE::pfnOption */
150static DECLCALLBACK(int) VBoxServiceTimeSyncOption(const char **ppszShort, int argc, char **argv, int *pi)
151{
152 int rc = -1;
153 if (ppszShort)
154 /* no short options */;
155 else if (!strcmp(argv[*pi], "--timesync-interval"))
156 rc = VBoxServiceArgUInt32(argc, argv, "", pi,
157 &g_TimeSyncInterval, 1, UINT32_MAX - 1);
158 else if (!strcmp(argv[*pi], "--timesync-min-adjust"))
159 rc = VBoxServiceArgUInt32(argc, argv, "", pi,
160 &g_TimeSyncMinAdjust, 0, 3600000);
161 else if (!strcmp(argv[*pi], "--timesync-latency-factor"))
162 rc = VBoxServiceArgUInt32(argc, argv, "", pi,
163 &g_TimeSyncLatencyFactor, 1, 1024);
164 else if (!strcmp(argv[*pi], "--timesync-max-latency"))
165 rc = VBoxServiceArgUInt32(argc, argv, "", pi,
166 &g_TimeSyncMaxLatency, 1, 3600000);
167 return rc;
168}
169
170
171/** @copydoc VBOXSERVICE::pfnInit */
172static DECLCALLBACK(int) VBoxServiceTimeSyncInit(void)
173{
174 /*
175 * If not specified, find the right interval default.
176 * Then create the event sem to block on.
177 */
178 if (!g_TimeSyncInterval)
179 g_TimeSyncInterval = g_DefaultInterval * 1000;
180 if (!g_TimeSyncInterval)
181 g_TimeSyncInterval = 10 * 1000;
182
183 int rc = RTSemEventMultiCreate(&g_TimeSyncEvent);
184 AssertRC(rc);
185#ifdef RT_OS_WINDOWS
186 if (RT_SUCCESS(rc))
187 {
188 /*
189 * Adjust priviledges of this process so we can make system time adjustments.
190 */
191 if (OpenProcessToken(GetCurrentProcess(), TOKEN_ADJUST_PRIVILEGES | TOKEN_QUERY, &g_hTokenProcess))
192 {
193 TOKEN_PRIVILEGES tkPriv;
194 RT_ZERO(tkPriv);
195 tkPriv.PrivilegeCount = 1;
196 tkPriv.Privileges[0].Attributes = SE_PRIVILEGE_ENABLED;
197 if (LookupPrivilegeValue(NULL, SE_SYSTEMTIME_NAME, &tkPriv.Privileges[0].Luid))
198 {
199 DWORD cbRet = sizeof(g_TkOldPrivileges);
200 if (!AdjustTokenPrivileges(g_hTokenProcess, FALSE, &tkPriv, sizeof(TOKEN_PRIVILEGES), &g_TkOldPrivileges, &cbRet))
201 {
202 DWORD dwErr = GetLastError();
203 rc = RTErrConvertFromWin32(dwErr);
204 VBoxServiceError("Adjusting token privileges (SE_SYSTEMTIME_NAME) failed with status code %u/%Rrc!\n", dwErr, rc);
205 }
206 }
207 else
208 {
209 DWORD dwErr = GetLastError();
210 rc = RTErrConvertFromWin32(dwErr);
211 VBoxServiceError("Looking up token privileges (SE_SYSTEMTIME_NAME) failed with status code %u/%Rrc!\n", dwErr, rc);
212 }
213
214 if (RT_FAILURE(rc))
215 {
216 CloseHandle(g_hTokenProcess);
217 g_hTokenProcess = NULL;
218 }
219 }
220 else
221 {
222 DWORD dwErr = GetLastError();
223 rc = RTErrConvertFromWin32(dwErr);
224 VBoxServiceError("Opening process token (SE_SYSTEMTIME_NAME) failed with status code %u/%Rrc!\n", dwErr, rc);
225 g_hTokenProcess = NULL;
226 }
227 }
228
229 if (!::GetSystemTimeAdjustment(&g_dwWinTimeAdjustment, &g_dwWinTimeIncrement, &g_bWinTimeAdjustmentDisabled))
230 {
231 DWORD dwErr = GetLastError();
232 rc = RTErrConvertFromWin32(dwErr);
233 VBoxServiceError("Could not get time adjustment values! Last error: %ld!\n", dwErr);
234 }
235 else VBoxServiceVerbose(3, "Windows time adjustment: Initially %ld (100ns) units per %ld (100 ns) units interval, disabled=%d\n",
236 g_dwWinTimeAdjustment, g_dwWinTimeIncrement, g_bWinTimeAdjustmentDisabled ? 1 : 0);
237#endif /* RT_OS_WINDOWS */
238
239 return rc;
240}
241
242
243/** @copydoc VBOXSERVICE::pfnWorker */
244DECLCALLBACK(int) VBoxServiceTimeSyncWorker(bool volatile *pfShutdown)
245{
246 RTTIME Time;
247 char sz[64];
248 int rc = VINF_SUCCESS;
249
250 /*
251 * Tell the control thread that it can continue
252 * spawning services.
253 */
254 RTThreadUserSignal(RTThreadSelf());
255
256 unsigned cErrors = 0;
257 for (;;)
258 {
259 /*
260 * Try get a reliable time reading.
261 */
262 int cTries = 3;
263 do
264 {
265 /* query it. */
266 RTTIMESPEC GuestNow0, GuestNow, HostNow;
267 RTTimeNow(&GuestNow0);
268 int rc2 = VbglR3GetHostTime(&HostNow);
269 if (RT_FAILURE(rc2))
270 {
271 if (cErrors++ < 10)
272 VBoxServiceError("VbglR3GetHostTime failed; rc2=%Rrc\n", rc2);
273 break;
274 }
275 RTTimeNow(&GuestNow);
276
277 /* calc latency and check if it's ok. */
278 RTTIMESPEC GuestElapsed = GuestNow;
279 RTTimeSpecSub(&GuestElapsed, &GuestNow0);
280 if ((uint32_t)RTTimeSpecGetMilli(&GuestElapsed) < g_TimeSyncMaxLatency)
281 {
282 /*
283 * Calculate the adjustment threshold and the current drift.
284 */
285 uint32_t MinAdjust = RTTimeSpecGetMilli(&GuestElapsed) * g_TimeSyncLatencyFactor;
286 if (MinAdjust < g_TimeSyncMinAdjust)
287 MinAdjust = g_TimeSyncMinAdjust;
288
289 RTTIMESPEC Drift = HostNow;
290 RTTimeSpecSub(&Drift, &GuestNow);
291 if (RTTimeSpecGetMilli(&Drift) < 0)
292 MinAdjust += g_TimeSyncMinAdjust; /* extra buffer against moving time backwards. */
293
294 RTTIMESPEC AbsDrift = Drift;
295 RTTimeSpecAbsolute(&AbsDrift);
296 if (g_cVerbosity >= 3)
297 {
298 VBoxServiceVerbose(3, "Host: %s (MinAdjust: %RU32 ms)\n",
299 RTTimeToString(RTTimeExplode(&Time, &HostNow), sz, sizeof(sz)), MinAdjust);
300 VBoxServiceVerbose(3, "Guest: - %s => %RDtimespec drift\n",
301 RTTimeToString(RTTimeExplode(&Time, &GuestNow), sz, sizeof(sz)),
302 &Drift);
303 }
304
305 uint32_t AbsDriftMilli = RTTimeSpecGetMilli(&AbsDrift);
306 if (AbsDriftMilli > MinAdjust)
307 {
308 /*
309 * The drift is too big, we have to make adjustments. :-/
310 * If we've got adjtime around, try that first - most
311 * *NIX systems have it. Fall back on settimeofday.
312 */
313#ifdef RT_OS_WINDOWS
314 DWORD dwWinTimeAdjustment, dwWinNewTimeAdjustment, dwWinTimeIncrement;
315 BOOL bWinTimeAdjustmentDisabled;
316 if (!::GetSystemTimeAdjustment(&dwWinTimeAdjustment, &dwWinTimeIncrement, &bWinTimeAdjustmentDisabled))
317 {
318 VBoxServiceError("GetSystemTimeAdjustment failed, error=%ld\n", GetLastError());
319 }
320 else
321 {
322 DWORD dwDiffMax = g_dwWinTimeAdjustment * 0.50;
323 DWORD dwDiffNew = dwWinTimeAdjustment * 0.10;
324
325 if (RTTimeSpecGetMilli(&Drift) > 0)
326 {
327 dwWinNewTimeAdjustment = dwWinTimeAdjustment + dwDiffNew;
328 if (dwWinNewTimeAdjustment > (g_dwWinTimeAdjustment + dwDiffMax))
329 {
330 dwWinNewTimeAdjustment = g_dwWinTimeAdjustment + dwDiffMax;
331 dwDiffNew = dwDiffMax;
332 }
333 }
334 else
335 {
336 dwWinNewTimeAdjustment = dwWinTimeAdjustment - dwDiffNew;
337 if (dwWinNewTimeAdjustment < (g_dwWinTimeAdjustment - dwDiffMax))
338 {
339 dwWinNewTimeAdjustment = g_dwWinTimeAdjustment - dwDiffMax;
340 dwDiffNew = dwDiffMax;
341 }
342 }
343
344 VBoxServiceVerbose(3, "Windows time adjustment: Drift=%ldms\n", RTTimeSpecGetMilli(&Drift));
345 VBoxServiceVerbose(3, "Windows time adjustment: OrgTA=%ld, CurTA=%ld, NewTA=%ld, DiffNew=%ld, DiffMax=%ld\n",
346 g_dwWinTimeAdjustment, dwWinTimeAdjustment, dwWinNewTimeAdjustment, dwDiffNew, dwDiffMax);
347
348 /* Is AbsDrift way too big? Then a minimum adjustment via SetSystemTimeAdjustment() would take ages.
349 So set the time in a hard manner. */
350 if (AbsDriftMilli > (60 * 1000 * 20)) /** @todo 20 minutes here hardcoded here. Needs configurable parameter later. */
351 {
352 SYSTEMTIME st = {0};
353 FILETIME ft = {0};
354
355 VBoxServiceVerbose(3, "Windows time adjustment: Setting system time directly.\n");
356
357 RTTimeSpecGetNtFileTime(&HostNow, &ft);
358 if (FALSE == FileTimeToSystemTime(&ft,&st))
359 VBoxServiceError("Cannot convert system times, error=%ld\n", GetLastError());
360
361 if (!::SetSystemTime(&st))
362 VBoxServiceError("SetSystemTime failed, error=%ld\n", GetLastError());
363 }
364 else
365 {
366 if (!::SetSystemTimeAdjustment(dwWinNewTimeAdjustment, FALSE /* Periodic adjustments enabled. */))
367 VBoxServiceError("SetSystemTimeAdjustment failed, error=%ld\n", GetLastError());
368 }
369 }
370
371#else /* !RT_OS_WINDOWS */
372 struct timeval tv;
373# if !defined(RT_OS_OS2) /* PORTME */
374 RTTimeSpecGetTimeval(&Drift, &tv);
375 if (adjtime(&tv, NULL) == 0)
376 {
377 if (g_cVerbosity >= 1)
378 VBoxServiceVerbose(1, "adjtime by %RDtimespec\n", &Drift);
379 cErrors = 0;
380 }
381 else
382# endif
383 {
384 errno = 0;
385 if (!gettimeofday(&tv, NULL))
386 {
387 RTTIMESPEC Tmp;
388 RTTimeSpecAdd(RTTimeSpecSetTimeval(&Tmp, &tv), &Drift);
389 if (!settimeofday(RTTimeSpecGetTimeval(&Tmp, &tv), NULL))
390 {
391 if (g_cVerbosity >= 1)
392 VBoxServiceVerbose(1, "settimeofday to %s\n",
393 RTTimeToString(RTTimeExplode(&Time, &Tmp), sz, sizeof(sz)));
394# ifdef DEBUG
395 if (g_cVerbosity >= 3)
396 VBoxServiceVerbose(2, " new time %s\n",
397 RTTimeToString(RTTimeExplode(&Time, RTTimeNow(&Tmp)), sz, sizeof(sz)));
398# endif
399 cErrors = 0;
400 }
401 else if (cErrors++ < 10)
402 VBoxServiceError("settimeofday failed; errno=%d: %s\n", errno, strerror(errno));
403 }
404 else if (cErrors++ < 10)
405 VBoxServiceError("gettimeofday failed; errno=%d: %s\n", errno, strerror(errno));
406 }
407#endif /* !RT_OS_WINDOWS */
408 }
409 else /* The time delta is <= MinAdjust, so don't do anything here (anymore). */
410 {
411#ifdef RT_OS_WINDOWS
412 if (::SetSystemTimeAdjustment(0, TRUE /* Periodic adjustments disabled. */))
413 VBoxServiceVerbose(3, "Windows Time Adjustment is now disabled.\n");
414#endif /* !RT_OS_WINDOWS */
415 }
416 break;
417 }
418 VBoxServiceVerbose(3, "%RDtimespec: latency too high (%RDtimespec) sleeping 1s\n", GuestElapsed);
419 RTThreadSleep(1000);
420 } while (--cTries > 0);
421
422 /*
423 * Block for a while.
424 *
425 * The event semaphore takes care of ignoring interruptions and it
426 * allows us to implement service wakeup later.
427 */
428 if (*pfShutdown)
429 break;
430 int rc2 = RTSemEventMultiWait(g_TimeSyncEvent, g_TimeSyncInterval);
431 if (*pfShutdown)
432 break;
433 if (rc2 != VERR_TIMEOUT && RT_FAILURE(rc2))
434 {
435 VBoxServiceError("RTSemEventMultiWait failed; rc2=%Rrc\n", rc2);
436 rc = rc2;
437 break;
438 }
439 }
440
441 RTSemEventMultiDestroy(g_TimeSyncEvent);
442 g_TimeSyncEvent = NIL_RTSEMEVENTMULTI;
443 return rc;
444}
445
446
447/** @copydoc VBOXSERVICE::pfnStop */
448static DECLCALLBACK(void) VBoxServiceTimeSyncStop(void)
449{
450 RTSemEventMultiSignal(g_TimeSyncEvent);
451}
452
453
454/** @copydoc VBOXSERVICE::pfnTerm */
455static DECLCALLBACK(void) VBoxServiceTimeSyncTerm(void)
456{
457#ifdef RT_OS_WINDOWS
458 /*
459 * Restore the SE_SYSTEMTIME_NAME token privileges (if init succeeded).
460 */
461 if (g_hTokenProcess)
462 {
463 if (!AdjustTokenPrivileges(g_hTokenProcess, FALSE, &g_TkOldPrivileges, sizeof(TOKEN_PRIVILEGES), NULL, NULL))
464 {
465 DWORD dwErr = GetLastError();
466 VBoxServiceError("Restoring token privileges (SE_SYSTEMTIME_NAME) failed with code %u!\n", dwErr);
467 }
468 CloseHandle(g_hTokenProcess);
469 g_hTokenProcess = NULL;
470 }
471#endif /* !RT_OS_WINDOWS */
472
473 if (g_TimeSyncEvent != NIL_RTSEMEVENTMULTI)
474 {
475 RTSemEventMultiDestroy(g_TimeSyncEvent);
476 g_TimeSyncEvent = NIL_RTSEMEVENTMULTI;
477 }
478}
479
480
481/**
482 * The 'timesync' service description.
483 */
484VBOXSERVICE g_TimeSync =
485{
486 /* pszName. */
487 "timesync",
488 /* pszDescription. */
489 "Time synchronization",
490 /* pszUsage. */
491 "[--timesync-interval <ms>] [--timesync-min-adjust <ms>] "
492 "[--timesync-latency-factor <x>] [--time-sync-max-latency <ms>]"
493 ,
494 /* pszOptions. */
495 " --timesync-interval Specifies the interval at which to synchronize the\n"
496 " time with the host. The default is 10000 ms.\n"
497 " --timesync-min-adjust The minimum absolute drift value measured\n"
498 " in milliseconds to make adjustments for.\n"
499 " The default is 1000 ms on OS/2 and 100 ms elsewhere.\n"
500 " --timesync-latency-factor The factor to multiply the time query latency\n"
501 " with to calculate the dynamic minimum adjust time.\n"
502 " The default is 8 times.\n"
503 " --timesync-max-latency The max host timer query latency to accept.\n"
504 " The default is 250 ms.\n"
505 ,
506 /* methods */
507 VBoxServiceTimeSyncPreInit,
508 VBoxServiceTimeSyncOption,
509 VBoxServiceTimeSyncInit,
510 VBoxServiceTimeSyncWorker,
511 VBoxServiceTimeSyncStop,
512 VBoxServiceTimeSyncTerm
513};
514
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