/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ /* * This file is part of the LibreOffice project. * * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. * * This file incorporates work covered by the following license notice: * * Licensed to the Apache Software Foundation (ASF) under one or more * contributor license agreements. See the NOTICE file distributed * with this work for additional information regarding copyright * ownership. The ASF licenses this file to you under the Apache * License, Version 2.0 (the "License"); you may not use this file * except in compliance with the License. You may obtain a copy of * the License at http://www.apache.org/licenses/LICENSE-2.0 . */ #include #include #if defined(_WIN32) #if !defined WIN32_LEAN_AND_MEAN # define WIN32_LEAN_AND_MEAN #endif #include #include #elif defined UNX #include #include #endif #include #ifdef __MACH__ #include #include #include #endif #include #include #include #include #if defined(__sun) && defined(__GNUC__) extern long altzone; #endif namespace tools { Time::Time( TimeInitSystem ) { if ( !GetSystemDateTime( nullptr, &nTime ) ) nTime = 0; } Time::Time( sal_uInt32 nHour, sal_uInt32 nMin, sal_uInt32 nSec, sal_uInt64 nNanoSec ) { // normalize time nSec += nNanoSec / nanoSecPerSec; nNanoSec %= nanoSecPerSec; nMin += nSec / secondPerMinute; nSec %= secondPerMinute; nHour += nMin / minutePerHour; nMin %= minutePerHour; // 922337 * HOUR_MASK = 9223370000000000000 largest possible value, 922338 // would be -9223364073709551616. assert(HOUR_MASK * nHour >= 0 && "use tools::Duration with days instead!"); if (HOUR_MASK * nHour < 0) nHour = 922337; // But as is, GetHour() retrieves only sal_uInt16. Though retrieving in // nanoseconds or milliseconds might be possible this is all crap. assert(nHour <= SAL_MAX_UINT16 && "use tools::Duration with days instead!"); if (nHour > SAL_MAX_UINT16) nHour = SAL_MAX_UINT16; // construct time nTime = assemble(nHour, nMin, nSec, nNanoSec); } Time::Time( const css::util::Time &_rTime ) : Time(_rTime.Hours, _rTime.Minutes, _rTime.Seconds, _rTime.NanoSeconds) { } Time::Time( const css::util::DateTime &_rDateTime ) : Time(_rDateTime.Hours, _rDateTime.Minutes, _rDateTime.Seconds, _rDateTime.NanoSeconds) { } // static sal_Int64 tools::Time::assemble(sal_uInt32 h, sal_uInt32 m, sal_uInt32 s, sal_uInt64 ns) { return ns + s * SEC_MASK + m * MIN_MASK + h * HOUR_MASK; } void tools::Time::SetHour( sal_uInt16 nNewHour ) { nTime = GetSign() * assemble(nNewHour, GetMin(), GetSec(), GetNanoSec()); } void tools::Time::SetMin( sal_uInt16 nNewMin ) { // no overflow nTime = GetSign() * assemble(GetHour(), nNewMin % minutePerHour, GetSec(), GetNanoSec()); } void tools::Time::SetSec( sal_uInt16 nNewSec ) { // no overflow nTime = GetSign() * assemble(GetHour(), GetMin(), nNewSec % secondPerMinute, GetNanoSec()); } void tools::Time::SetNanoSec( sal_uInt32 nNewNanoSec ) { // no overflow nTime = GetSign() * assemble(GetHour(), GetMin(), GetSec(), nNewNanoSec % nanoSecPerSec); } sal_Int64 tools::Time::GetNSFromTime() const { return GetSign() * ( GetNanoSec() + GetSec() * nanoSecPerSec + GetMin() * nanoSecPerMinute + GetHour() * nanoSecPerHour ); } void tools::Time::MakeTimeFromNS( sal_Int64 nNS ) { short nSign; if ( nNS < 0 ) { nNS *= -1; nSign = -1; } else nSign = 1; tools::Time aTime(0, 0, 0, nNS); SetTime( aTime.GetTime() * nSign ); } sal_Int32 tools::Time::GetMSFromTime() const { return GetNSFromTime() / nanoPerMilli; } void tools::Time::MakeTimeFromMS( sal_Int32 nMS ) { MakeTimeFromNS(nMS * nanoPerMilli); } double tools::Time::GetTimeInDays() const { return GetNSFromTime() / double(nanoSecPerDay); } // static void tools::Time::GetClock( double fTimeInDays, sal_uInt16& nHour, sal_uInt16& nMinute, sal_uInt16& nSecond, double& fFractionOfSecond, int nFractionDecimals ) { const double fTime = fTimeInDays - rtl::math::approxFloor(fTimeInDays); // date part absent // If 0 then full day (or no day), shortcut. // If < 0 then approxFloor() effectively returned the ceiling (note this // also holds for negative fTimeInDays values) because of a near identical // value, shortcut this to a full day as well. // If >= 1.0 (actually == 1.0) then fTimeInDays is a negative small value // not significant for a representable time and approxFloor() returned -1, // shortcut to 0:0:0, otherwise it would become 24:0:0. if (fTime <= 0.0 || fTime >= 1.0) { nHour = nMinute = nSecond = 0; fFractionOfSecond = 0.0; return; } // In seconds, including milli and nano. const double fRawSeconds = fTime * tools::Time::secondPerDay; // Round to nanoseconds most, which is the highest resolution this could be // influenced by, but if the original value included a date round to at // most 14 significant digits (including adding 4 for *86400), otherwise we // might end up with a fake precision of h:m:s.999999986 which in fact // should had been h:m:s+1 // BUT, leave at least 2 decimals to round. Which shouldn't be a problem in // practice because class Date can calculate only 8-digit days for it's // sal_Int16 year range, which exactly leaves us with 14-4-8=2. int nDec = 9; const double fAbsTimeInDays = fabs( fTimeInDays); if (fAbsTimeInDays >= 1.0) { const int nDig = static_cast(ceil( log10( fAbsTimeInDays))); nDec = std::clamp( 10 - nDig, 2, 9 ); } double fSeconds = rtl::math::round( fRawSeconds, nDec); // If this ended up as a full day the original value was very very close // but not quite. Take that. if (fSeconds >= tools::Time::secondPerDay) fSeconds = fRawSeconds; // Now do not round values (specifically not up), but truncate to the next // magnitude, so 23:59:59.99 is still 23:59:59 and not 24:00:00 (or even // 00:00:00 which Excel does). nHour = fSeconds / tools::Time::secondPerHour; fSeconds -= nHour * tools::Time::secondPerHour; nMinute = fSeconds / tools::Time::secondPerMinute; fSeconds -= nMinute * tools::Time::secondPerMinute; nSecond = fSeconds; fSeconds -= nSecond; assert(fSeconds < 1.0); // or back to the drawing board... if (nFractionDecimals > 0) { // Do not simply round the fraction, otherwise .999 would end up as .00 // again. Truncate instead if rounding would round up into an integer // value. fFractionOfSecond = rtl::math::round( fSeconds, nFractionDecimals); if (fFractionOfSecond >= 1.0) fFractionOfSecond = rtl::math::pow10Exp( std::trunc( rtl::math::pow10Exp( fSeconds, nFractionDecimals)), -nFractionDecimals); } else fFractionOfSecond = fSeconds; } Time& tools::Time::operator +=( const tools::Time& rTime ) { MakeTimeFromNS(GetNSFromTime() + rTime.GetNSFromTime()); return *this; } Time& tools::Time::operator -=( const tools::Time& rTime ) { MakeTimeFromNS(GetNSFromTime() - rTime.GetNSFromTime()); return *this; } Time operator +( const tools::Time& rTime1, const tools::Time& rTime2 ) { tools::Time result(rTime1); return result += rTime2; } Time operator -( const tools::Time& rTime1, const tools::Time& rTime2 ) { tools::Time result(rTime1); return result -= rTime2; } bool tools::Time::IsEqualIgnoreNanoSec( const tools::Time& rTime ) const { return nTime / SEC_MASK == rTime.nTime / SEC_MASK; } Time tools::Time::GetUTCOffset() { #if defined(_WIN32) TIME_ZONE_INFORMATION aTimeZone; aTimeZone.Bias = 0; DWORD nTimeZoneRet = GetTimeZoneInformation( &aTimeZone ); sal_Int32 nTempTime = aTimeZone.Bias; if ( nTimeZoneRet == TIME_ZONE_ID_STANDARD ) nTempTime += aTimeZone.StandardBias; else if ( nTimeZoneRet == TIME_ZONE_ID_DAYLIGHT ) nTempTime += aTimeZone.DaylightBias; tools::Time aTime( 0, static_cast(abs( nTempTime )) ); if ( nTempTime > 0 ) aTime = -aTime; return aTime; #else static sal_uInt64 nCacheTicks = 0; static sal_Int32 nCacheSecOffset = -1; sal_uInt64 nTicks = tools::Time::GetSystemTicks(); time_t nTime; tm aTM; short nTempTime; // determine value again if needed if ( (nCacheSecOffset == -1) || ((nTicks - nCacheTicks) > 360000) || ( nTicks < nCacheTicks ) // handle overflow ) { nTime = time( nullptr ); localtime_r( &nTime, &aTM ); auto nLocalTime = mktime( &aTM ); #if defined(__sun) // Solaris gmtime_r() seems not to handle daylight saving time // flags correctly auto nUTC = nLocalTime + ( aTM.tm_isdst == 0 ? timezone : altzone ); #elif defined( LINUX ) // Linux mktime() seems not to handle tm_isdst correctly auto nUTC = nLocalTime - aTM.tm_gmtoff; #else gmtime_r( &nTime, &aTM ); auto nUTC = mktime( &aTM ); #endif nCacheTicks = nTicks; nCacheSecOffset = (nLocalTime-nUTC) / 60; } nTempTime = abs( nCacheSecOffset ); tools::Time aTime( 0, static_cast(nTempTime) ); if ( nCacheSecOffset < 0 ) aTime = -aTime; return aTime; #endif } sal_uInt64 tools::Time::GetSystemTicks() { return tools::Time::GetMonotonicTicks() / 1000; } #ifdef _WIN32 static LARGE_INTEGER initPerformanceFrequency() { LARGE_INTEGER nTicksPerSecond = { 0, 0 }; if (!QueryPerformanceFrequency(&nTicksPerSecond)) nTicksPerSecond.QuadPart = 0; return nTicksPerSecond; } #endif sal_uInt64 tools::Time::GetMonotonicTicks() { #ifdef _WIN32 static const LARGE_INTEGER nTicksPerSecond = initPerformanceFrequency(); if (nTicksPerSecond.QuadPart > 0) { LARGE_INTEGER nPerformanceCount; QueryPerformanceCounter(&nPerformanceCount); return static_cast( ( nPerformanceCount.QuadPart * 1000 * 1000 ) / nTicksPerSecond.QuadPart ); } else { return static_cast( timeGetTime() * 1000 ); } #else sal_uInt64 nMicroSeconds; #ifdef __MACH__ static mach_timebase_info_data_t info = { 0, 0 }; if ( 0 == info.numer ) mach_timebase_info( &info ); nMicroSeconds = mach_absolute_time() * static_cast(info.numer / info.denom) / 1000; #else #if defined(_POSIX_TIMERS) struct timespec currentTime; clock_gettime( CLOCK_MONOTONIC, ¤tTime ); nMicroSeconds = static_cast(currentTime.tv_sec) * 1000 * 1000 + currentTime.tv_nsec / 1000; #else struct timeval currentTime; gettimeofday( ¤tTime, nullptr ); nMicroSeconds = static_cast(currentTime.tv_sec) * 1000 * 1000 + currentTime.tv_usec; #endif #endif // __MACH__ return nMicroSeconds; #endif // _WIN32 } } /* namespace tools */ /* vim:set shiftwidth=4 softtabstop=4 expandtab: */