/* $Id: timesup.cpp 6677 2008-01-31 19:27:56Z vboxsync $ */ /** @file * innotek Portable Runtime - Time using SUPLib. */ /* * Copyright (C) 2006-2007 innotek GmbH * * This file is part of VirtualBox Open Source Edition (OSE), as * available from http://www.virtualbox.org. This file is free software; * you can redistribute it and/or modify it under the terms of the GNU * General Public License (GPL) as published by the Free Software * Foundation, in version 2 as it comes in the "COPYING" file of the * VirtualBox OSE distribution. VirtualBox OSE is distributed in the * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind. * * The contents of this file may alternatively be used under the terms * of the Common Development and Distribution License Version 1.0 * (CDDL) only, as it comes in the "COPYING.CDDL" file of the * VirtualBox OSE distribution, in which case the provisions of the * CDDL are applicable instead of those of the GPL. * * You may elect to license modified versions of this file under the * terms and conditions of either the GPL or the CDDL or both. */ /******************************************************************************* * Header Files * *******************************************************************************/ #define LOG_GROUP RTLOGGROUP_TIME #include #include #include #include #include #ifndef IN_GUEST # include # include #endif #include "internal/time.h" /******************************************************************************* * Internal Functions * *******************************************************************************/ #if !defined(IN_GUEST) && !defined(RT_NO_GIP) static DECLCALLBACK(void) rtTimeNanoTSInternalBitch(PRTTIMENANOTSDATA pData, uint64_t u64NanoTS, uint64_t u64DeltaPrev, uint64_t u64PrevNanoTS); static DECLCALLBACK(uint64_t) rtTimeNanoTSInternalFallback(PRTTIMENANOTSDATA pData); static DECLCALLBACK(uint64_t) rtTimeNanoTSInternalRediscover(PRTTIMENANOTSDATA pData); #endif /******************************************************************************* * Global Variables * *******************************************************************************/ #if !defined(IN_GUEST) && !defined(RT_NO_GIP) /** The previous timestamp value returned by RTTimeNanoTS. */ static uint64_t g_TimeNanoTSPrev = 0; /** The RTTimeNanoTS data structure that's passed down to the worker functions. */ static RTTIMENANOTSDATA g_TimeNanoTSData = { /* .pu64Prev = */ &g_TimeNanoTSPrev, /* .pfnBad = */ rtTimeNanoTSInternalBitch, /* .pfnRediscover = */ rtTimeNanoTSInternalRediscover, /* .pvDummy = */ NULL, /* .c1nsSteps = */ 0, /* .cExpired = */ 0, /* .cBadPrev = */ 0, /* .cUpdateRaces = */ 0 }; /** The index into g_apfnWorkers for the function to use. * This cannot be a pointer because that'll break down in GC due to code relocation. */ static uint32_t g_iWorker = 0; /** Array of rtTimeNanoTSInternal worker functions. * This array is indexed by g_iWorker. */ static const PFNTIMENANOTSINTERNAL g_apfnWorkers[] = { #define RTTIMENANO_WORKER_DETECT 0 rtTimeNanoTSInternalRediscover, #define RTTIMENANO_WORKER_SYNC_CPUID 1 RTTimeNanoTSLegacySync, #define RTTIMENANO_WORKER_ASYNC_CPUID 2 RTTimeNanoTSLegacyAsync, #define RTTIMENANO_WORKER_SYNC_LFENCE 3 RTTimeNanoTSLFenceSync, #define RTTIMENANO_WORKER_ASYNC_LFENCE 4 RTTimeNanoTSLFenceAsync, #define RTTIMENANO_WORKER_FALLBACK 5 rtTimeNanoTSInternalFallback, }; /** * Helper function that's used by the assembly routines when something goes bust. * * @param pData Pointer to the data structure. * @param u64NanoTS The calculated nano ts. * @param u64DeltaPrev The delta relative to the previously returned timestamp. * @param u64PrevNanoTS The previously returned timestamp (as it was read it). */ static DECLCALLBACK(void) rtTimeNanoTSInternalBitch(PRTTIMENANOTSDATA pData, uint64_t u64NanoTS, uint64_t u64DeltaPrev, uint64_t u64PrevNanoTS) { pData->cBadPrev++; if ((int64_t)u64DeltaPrev < 0) LogRel(("TM: u64DeltaPrev=%RI64 u64PrevNanoTS=0x%016RX64 u64NanoTS=0x%016RX64\n", u64DeltaPrev, u64PrevNanoTS, u64NanoTS)); else Log(("TM: u64DeltaPrev=%RI64 u64PrevNanoTS=0x%016RX64 u64NanoTS=0x%016RX64 (debugging?)\n", u64DeltaPrev, u64PrevNanoTS, u64NanoTS)); } /** * Fallback function. */ static DECLCALLBACK(uint64_t) rtTimeNanoTSInternalFallback(PRTTIMENANOTSDATA pData) { PSUPGLOBALINFOPAGE pGip = g_pSUPGlobalInfoPage; if ( pGip && pGip->u32Magic == SUPGLOBALINFOPAGE_MAGIC && ( pGip->u32Mode == SUPGIPMODE_SYNC_TSC || pGip->u32Mode == SUPGIPMODE_ASYNC_TSC)) return rtTimeNanoTSInternalRediscover(pData); NOREF(pData); #if defined(IN_RING3) /** @todo Add ring-0 RTTimeSystemNanoTS to all hosts. */ return RTTimeSystemNanoTS(); #else return 0; #endif } /** * Called the first time somebody asks for the time or when the GIP * is mapped/unmapped. */ static DECLCALLBACK(uint64_t) rtTimeNanoTSInternalRediscover(PRTTIMENANOTSDATA pData) { uint32_t iWorker; PSUPGLOBALINFOPAGE pGip = g_pSUPGlobalInfoPage; if ( pGip && pGip->u32Magic == SUPGLOBALINFOPAGE_MAGIC && ( pGip->u32Mode == SUPGIPMODE_SYNC_TSC || pGip->u32Mode == SUPGIPMODE_ASYNC_TSC)) { if (ASMCpuId_EDX(1) & X86_CPUID_FEATURE_EDX_SSE2) iWorker = pGip->u32Mode == SUPGIPMODE_SYNC_TSC ? RTTIMENANO_WORKER_SYNC_LFENCE : RTTIMENANO_WORKER_ASYNC_LFENCE; else iWorker = pGip->u32Mode == SUPGIPMODE_SYNC_TSC ? RTTIMENANO_WORKER_SYNC_CPUID : RTTIMENANO_WORKER_ASYNC_CPUID; } else iWorker = RTTIMENANO_WORKER_FALLBACK; ASMAtomicXchgU32((uint32_t volatile *)&g_iWorker, iWorker); return g_apfnWorkers[iWorker](pData); } #endif /* !IN_GUEST && !RT_NO_GIP */ /** * Internal worker for getting the current nanosecond timestamp. */ DECLINLINE(uint64_t) rtTimeNanoTSInternal(void) { #if !defined(IN_GUEST) && !defined(RT_NO_GIP) return g_apfnWorkers[g_iWorker](&g_TimeNanoTSData); #else return RTTimeSystemNanoTS(); #endif } /** * Gets the current nanosecond timestamp. * * @returns nanosecond timestamp. */ RTDECL(uint64_t) RTTimeNanoTS(void) { return rtTimeNanoTSInternal(); } /** * Gets the current millisecond timestamp. * * @returns millisecond timestamp. */ RTDECL(uint64_t) RTTimeMilliTS(void) { return rtTimeNanoTSInternal() / 1000000; } #if !defined(IN_GUEST) && !defined(RT_NO_GIP) /** * Debugging the time api. * * @returns the number of 1ns steps which has been applied by RTTimeNanoTS(). */ RTDECL(uint32_t) RTTimeDbgSteps(void) { return g_TimeNanoTSData.c1nsSteps; } /** * Debugging the time api. * * @returns the number of times the TSC interval expired RTTimeNanoTS(). */ RTDECL(uint32_t) RTTimeDbgExpired(void) { return g_TimeNanoTSData.cExpired; } /** * Debugging the time api. * * @returns the number of bad previous values encountered by RTTimeNanoTS(). */ RTDECL(uint32_t) RTTimeDbgBad(void) { return g_TimeNanoTSData.cBadPrev; } /** * Debugging the time api. * * @returns the number of update races in RTTimeNanoTS(). */ RTDECL(uint32_t) RTTimeDbgRaces(void) { return g_TimeNanoTSData.cUpdateRaces; } #endif /* !IN_GUEST && !RT_NO_GIP */