// Copyright 2015 The Crashpad Authors // // Licensed 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 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. #include "util/misc/time.h" #include #include "base/check_op.h" namespace crashpad { namespace { constexpr uint64_t kMicrosecondsPerSecond = static_cast(1E6); constexpr uint64_t kNanosecondsPerFiletimeInterval = static_cast(100); constexpr uint64_t kFiletimeIntervalsPerSecond = kNanosecondsPerSecond / kNanosecondsPerFiletimeInterval; constexpr uint64_t kFiletimeIntervalsPerMicrosecond = kFiletimeIntervalsPerSecond / kMicrosecondsPerSecond; // Windows epoch is 1601-01-01, and FILETIME ticks are 100 nanoseconds. // 1601 to 1970 is 369 years + 89 leap days = 134774 days * 86400 seconds per // day. It's not entirely clear, but it appears that these are solar seconds, // not SI seconds, so there are no leap seconds to be considered. constexpr uint64_t kNumSecondsFrom1601To1970 = (369 * 365 + 89) * 86400ULL; uint64_t FiletimeToMicroseconds(const FILETIME& filetime) { uint64_t t = (static_cast(filetime.dwHighDateTime) << 32) | filetime.dwLowDateTime; return t / kFiletimeIntervalsPerMicrosecond; } timeval MicrosecondsToTimeval(uint64_t microseconds) { timeval tv; tv.tv_sec = static_cast(microseconds / kMicrosecondsPerSecond); tv.tv_usec = static_cast(microseconds % kMicrosecondsPerSecond); return tv; } } // namespace FILETIME TimespecToFiletimeEpoch(const timespec& ts) { uint64_t intervals = (kNumSecondsFrom1601To1970 + ts.tv_sec) * kFiletimeIntervalsPerSecond + ts.tv_nsec / kNanosecondsPerFiletimeInterval; FILETIME filetime; filetime.dwLowDateTime = intervals & 0xffffffff; filetime.dwHighDateTime = intervals >> 32; return filetime; } timespec FiletimeToTimespecEpoch(const FILETIME& filetime) { uint64_t intervals = (uint64_t{filetime.dwHighDateTime} << 32) | filetime.dwLowDateTime; timespec result; result.tv_sec = (intervals / kFiletimeIntervalsPerSecond) - kNumSecondsFrom1601To1970; result.tv_nsec = static_cast(intervals % kFiletimeIntervalsPerSecond) * kNanosecondsPerFiletimeInterval; return result; } timeval FiletimeToTimevalEpoch(const FILETIME& filetime) { uint64_t microseconds = FiletimeToMicroseconds(filetime); DCHECK_GE(microseconds, kNumSecondsFrom1601To1970 * kMicrosecondsPerSecond); microseconds -= kNumSecondsFrom1601To1970 * kMicrosecondsPerSecond; return MicrosecondsToTimeval(microseconds); } timeval FiletimeToTimevalInterval(const FILETIME& filetime) { return MicrosecondsToTimeval(FiletimeToMicroseconds(filetime)); } void GetTimeOfDay(timeval* tv) { FILETIME filetime; GetSystemTimeAsFileTime(&filetime); *tv = FiletimeToTimevalEpoch(filetime); } } // namespace crashpad