// Copyright (c) 2016-2018 Kiwano - Nomango // // Permission is hereby granted, free of charge, to any person obtaining a copy // of this software and associated documentation files (the "Software"), to deal // in the Software without restriction, including without limitation the rights // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell // copies of the Software, and to permit persons to whom the Software is // furnished to do so, subject to the following conditions: // // The above copyright notice and this permission notice shall be included in // all copies or substantial portions of the Software. // // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN // THE SOFTWARE. #include #include #include #include #include #include #include namespace kiwano { //------------------------------------------------------- // Time //------------------------------------------------------- Time::Time() : dur_(0) { } Time::Time(int64_t dur) : dur_(dur) { } const Time Time::operator+(const Duration& dur) const { return Time{ dur_ + dur.Milliseconds() }; } const Time Time::operator-(const Duration& dur) const { return Time{ dur_ - dur.Milliseconds() }; } Time& Time::operator+=(const Duration& other) { dur_ += other.Milliseconds(); return (*this); } Time& Time::operator-=(const Duration& other) { dur_ -= other.Milliseconds(); return (*this); } const Duration Time::operator-(const Time& other) const { return Duration(dur_ - other.dur_); } Time Time::Now() noexcept { #if defined(KGE_PLATFORM_WINDOWS) static double millisecs_per_count = {}; if (millisecs_per_count == 0) { LARGE_INTEGER freq = {}; // the Function will always succceed on systems that run Windows XP or later QueryPerformanceFrequency(&freq); millisecs_per_count = 1000.0 / static_cast(freq.QuadPart); } LARGE_INTEGER count; QueryPerformanceCounter(&count); return Time{ static_cast(count.QuadPart * millisecs_per_count) }; #else using std::chrono::steady_clock; using std::chrono::duration_cast; using std::chrono::milliseconds; const auto now = steady_clock::now(); const auto count = duration_cast(now.time_since_epoch()).count(); return Time{ static_cast(count) }; #endif } //------------------------------------------------------- // ClockTime //------------------------------------------------------- ClockTime::ClockTime() : ms_since_epoch_(0) { } int64_t ClockTime::GetTimeStamp() const { using std::chrono::duration_cast; using std::chrono::milliseconds; using std::chrono::seconds; const auto timestamp = duration_cast(milliseconds(ms_since_epoch_)).count(); return static_cast(timestamp); } int64_t ClockTime::GetMillisecondsSinceEpoch() const { return ms_since_epoch_; } std::time_t ClockTime::GetCTime() const { return static_cast(GetTimeStamp()); } ClockTime::ClockTime(int64_t ms_since_epoch) : ms_since_epoch_(ms_since_epoch) { } ClockTime ClockTime::FromTimeStamp(int64_t timestamp) noexcept { using std::chrono::duration_cast; using std::chrono::milliseconds; using std::chrono::seconds; const auto ms = duration_cast(seconds(timestamp)).count(); return ClockTime(static_cast(ms)); } ClockTime ClockTime::Now() noexcept { using std::chrono::duration_cast; using std::chrono::milliseconds; using std::chrono::system_clock; const auto now = system_clock::now(); const auto count = duration_cast(now.time_since_epoch()).count(); return ClockTime{ static_cast(count) }; } const Duration ClockTime::operator-(const ClockTime& other) const { return Duration(ms_since_epoch_ - other.ms_since_epoch_); } const ClockTime ClockTime::operator+(const Duration& dur) const { return ClockTime{ ms_since_epoch_ + dur.Milliseconds() }; } const ClockTime ClockTime::operator-(const Duration& dur) const { return ClockTime{ ms_since_epoch_ - dur.Milliseconds() }; } ClockTime& ClockTime::operator+=(const Duration& other) { ms_since_epoch_ += other.Milliseconds(); return (*this); } ClockTime& ClockTime::operator-=(const Duration& other) { ms_since_epoch_ -= other.Milliseconds(); return (*this); } //------------------------------------------------------- // Duration //------------------------------------------------------- const Duration Duration::Ms = 1L; const Duration Duration::Second = 1000 * Duration::Ms; const Duration Duration::Minute = 60 * Duration::Second; const Duration Duration::Hour = 60 * Duration::Minute; namespace { const auto duration_regex = std::regex(R"(^[-+]?([0-9]*(\.[0-9]*)?(h|m|s|ms)+)+$)"); typedef std::unordered_map UnitMap; const auto unit_map = UnitMap{ { "ms", Duration::Ms }, { "s", Duration::Second }, { "m", Duration::Minute }, { "h", Duration::Hour } }; } // namespace Duration::Duration() : milliseconds_(0) { } Duration::Duration(int64_t milliseconds) : milliseconds_(milliseconds) { } float Duration::Seconds() const { auto sec = milliseconds_ / Second.milliseconds_; auto ms = milliseconds_ % Second.milliseconds_; return static_cast(sec + ms) / 1000.f; } float Duration::Minutes() const { auto min = milliseconds_ / Minute.milliseconds_; auto ms = milliseconds_ % Minute.milliseconds_; return static_cast(min + ms) / (60 * 1000.f); } float Duration::Hours() const { auto hour = milliseconds_ / Hour.milliseconds_; auto ms = milliseconds_ % Hour.milliseconds_; return static_cast(hour + ms) / (60 * 60 * 1000.f); } void Duration::Sleep() const { using std::chrono::milliseconds; using std::this_thread::sleep_for; if (milliseconds_) { sleep_for(milliseconds(milliseconds_)); } } String Duration::ToString() const { if (IsZero()) { return String("0s"); } StringStream stream; int64_t total_ms = milliseconds_; if (total_ms < 0) { stream << "-"; total_ms = -total_ms; } int64_t hour = total_ms / Hour.milliseconds_; int64_t min = total_ms / Minute.milliseconds_ - hour * 60; int64_t sec = total_ms / Second.milliseconds_ - (hour * 60 * 60 + min * 60); int64_t ms = total_ms % Second.milliseconds_; if (hour) { stream << hour << 'h' << min << 'm'; } else if (min) { stream << min << 'm'; } if (ms != 0) { stream << float(sec) + float(ms) / 1000.f << 's'; } else if (sec != 0) { stream << sec << 's'; } return stream.str(); } bool Duration::operator==(const Duration& other) const { return milliseconds_ == other.milliseconds_; } bool Duration::operator!=(const Duration& other) const { return milliseconds_ != other.milliseconds_; } bool Duration::operator>(const Duration& other) const { return milliseconds_ > other.milliseconds_; } bool Duration::operator>=(const Duration& other) const { return milliseconds_ >= other.milliseconds_; } bool Duration::operator<(const Duration& other) const { return milliseconds_ < other.milliseconds_; } bool Duration::operator<=(const Duration& other) const { return milliseconds_ <= other.milliseconds_; } float Duration::operator/(const Duration& other) const { return static_cast(milliseconds_) / other.milliseconds_; } const Duration Duration::operator+(const Duration& other) const { return Duration(milliseconds_ + other.milliseconds_); } const Duration Duration::operator-(const Duration& other) const { return Duration(milliseconds_ - other.milliseconds_); } const Duration Duration::operator-() const { return Duration(-milliseconds_); } const Duration Duration::operator*(int val) const { return Duration(milliseconds_ * val); } const Duration Duration::operator*(unsigned long long val) const { return Duration(static_cast(milliseconds_ * val)); } const Duration Duration::operator*(float val) const { return Duration(static_cast(milliseconds_ * val)); } const Duration Duration::operator*(double val) const { return Duration(static_cast(milliseconds_ * val)); } const Duration Duration::operator*(long double val) const { return Duration(static_cast(milliseconds_ * val)); } const Duration Duration::operator/(int val) const { return Duration(milliseconds_ / val); } const Duration Duration::operator/(float val) const { return Duration(static_cast(milliseconds_ / val)); } const Duration Duration::operator/(double val) const { return Duration(static_cast(milliseconds_ / val)); } Duration& Duration::operator+=(const Duration& other) { milliseconds_ += other.milliseconds_; return (*this); } Duration& Duration::operator-=(const Duration& other) { milliseconds_ -= other.milliseconds_; return (*this); } Duration& Duration::operator*=(int val) { milliseconds_ *= val; return (*this); } Duration& Duration::operator/=(int val) { milliseconds_ = static_cast(milliseconds_ / val); return (*this); } Duration& Duration::operator*=(float val) { milliseconds_ = static_cast(milliseconds_ * val); return (*this); } Duration& Duration::operator/=(float val) { milliseconds_ = static_cast(milliseconds_ / val); return (*this); } Duration& Duration::operator*=(double val) { milliseconds_ = static_cast(milliseconds_ * val); return (*this); } Duration& Duration::operator/=(double val) { milliseconds_ = static_cast(milliseconds_ / val); return (*this); } const Duration operator*(int val, const Duration& dur) { return dur * val; } const Duration operator/(int val, const Duration& dur) { return dur / val; } const Duration operator*(float val, const Duration& dur) { return dur * val; } const Duration operator/(float val, const Duration& dur) { return dur / val; } const Duration operator*(double val, const Duration& dur) { return dur * val; } const Duration operator/(double val, const Duration& dur) { return dur / val; } const Duration operator*(long double val, const Duration& dur) { return dur * val; } Duration Duration::Parse(const String& format) { bool negative = false; size_t len = format.length(); size_t pos = 0; Duration ret; if (!std::regex_match(format.c_str(), duration_regex)) { KGE_THROW("Duration::Parse failed, invalid duration"); } if (format.empty() || format == "0") { return ret; } // 符号位 if (format[0] == '-' || format[0] == '+') { negative = (format[0] == '-'); pos++; } while (pos < len) { // 数值 size_t i = pos; for (; i < len; ++i) { wchar_t ch = format[i]; if (!(ch == '.' || '0' <= ch && ch <= '9')) { break; } } String num_str = format.substr(pos, i - pos); pos = i; if (num_str.empty() || num_str == ".") KGE_THROW("Duration::Parse failed, invalid duration"); // 单位 for (; i < len; ++i) { wchar_t ch = format[i]; if (ch == '.' || '0' <= ch && ch <= '9') { break; } } String unit_str = format.substr(pos, i - pos); pos = i; if (unit_map.find(unit_str) == unit_map.end()) KGE_THROW("Duration::Parse failed, invalid duration"); double num = std::stod(num_str.c_str()); Duration unit = unit_map.at(unit_str); ret += unit * num; } if (negative) { ret = -ret; } return ret; } } // namespace kiwano