123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307 |
- /*
- * mptTime.cpp
- * -----------
- * Purpose: Various time utility functions.
- * Notes : (currently none)
- * Authors: OpenMPT Devs
- * The OpenMPT source code is released under the BSD license. Read LICENSE for more details.
- */
- #include "stdafx.h"
- #include "mptTime.h"
- #include "mptStringBuffer.h"
- #include <time.h>
- #if MPT_OS_WINDOWS
- #include <windows.h>
- #if defined(MODPLUG_TRACKER)
- #include <mmsystem.h>
- #endif
- #endif
- OPENMPT_NAMESPACE_BEGIN
- namespace mpt
- {
- namespace Date
- {
- #if defined(MODPLUG_TRACKER)
- #if MPT_OS_WINDOWS
- namespace ANSI
- {
- uint64 Now()
- {
- FILETIME filetime;
- GetSystemTimeAsFileTime(&filetime);
- return ((uint64)filetime.dwHighDateTime << 32 | filetime.dwLowDateTime);
- }
- mpt::ustring ToUString(uint64 time100ns)
- {
- constexpr std::size_t bufsize = 256;
- mpt::ustring result;
- FILETIME filetime;
- SYSTEMTIME systime;
- filetime.dwHighDateTime = (DWORD)(((uint64)time100ns) >> 32);
- filetime.dwLowDateTime = (DWORD)((uint64)time100ns);
- FileTimeToSystemTime(&filetime, &systime);
- TCHAR buf[bufsize];
- GetDateFormat(LOCALE_SYSTEM_DEFAULT, 0, &systime, TEXT("yyyy-MM-dd"), buf, bufsize);
- result.append(mpt::ToUnicode(mpt::String::ReadWinBuf(buf)));
- result.append(U_(" "));
- GetTimeFormat(LOCALE_SYSTEM_DEFAULT, TIME_FORCE24HOURFORMAT, &systime, TEXT("HH:mm:ss"), buf, bufsize);
- result.append(mpt::ToUnicode(mpt::String::ReadWinBuf(buf)));
- result.append(U_("."));
- result.append(mpt::ufmt::dec0<3>((unsigned)systime.wMilliseconds));
- return result;
- }
- } // namespace ANSI
- #endif // MPT_OS_WINDOWS
- #endif // MODPLUG_TRACKER
- Unix::Unix()
- : Value(0)
- {
- return;
- }
- Unix::Unix(int64 unixtime)
- : Value(unixtime)
- {
- return;
- }
- Unix::operator int64 () const
- {
- return Value;
- }
- static int32 ToDaynum(int32 year, int32 month, int32 day)
- {
- month = (month + 9) % 12;
- year = year - (month / 10);
- int32 daynum = year*365 + year/4 - year/100 + year/400 + (month*306 + 5)/10 + (day - 1);
- return daynum;
- }
- static void FromDaynum(int32 d, int32 & year, int32 & month, int32 & day)
- {
- int64 g = d;
- int64 y,ddd,mi,mm,dd;
- y = (10000*g + 14780)/3652425;
- ddd = g - (365*y + y/4 - y/100 + y/400);
- if(ddd < 0)
- {
- y = y - 1;
- ddd = g - (365*y + y/4 - y/100 + y/400);
- }
- mi = (100*ddd + 52)/3060;
- mm = (mi + 2)%12 + 1;
- y = y + (mi + 2)/12;
- dd = ddd - (mi*306 + 5)/10 + 1;
- year = static_cast<int32>(y);
- month = static_cast<int32>(mm);
- day = static_cast<int32>(dd);
- }
- mpt::Date::Unix Unix::FromUTC(tm timeUtc)
- {
- int32 daynum = ToDaynum(timeUtc.tm_year+1900, timeUtc.tm_mon+1, timeUtc.tm_mday);
- int64 seconds = static_cast<int64>(daynum - ToDaynum(1970,1,1))*24*60*60 + timeUtc.tm_hour*60*60 + timeUtc.tm_min*60 + timeUtc.tm_sec;
- return mpt::Date::Unix(seconds);
- }
- tm Unix::AsUTC() const
- {
- int64 tmp = Value;
- int64 seconds = tmp % 60; tmp /= 60;
- int64 minutes = tmp % 60; tmp /= 60;
- int64 hours = tmp % 24; tmp /= 24;
- int32 year = 0, month = 0, day = 0;
- FromDaynum(static_cast<int32>(tmp) + ToDaynum(1970,1,1), year, month, day);
- tm result = {};
- result.tm_year = year - 1900;
- result.tm_mon = month - 1;
- result.tm_mday = day;
- result.tm_hour = static_cast<int32>(hours);
- result.tm_min = static_cast<int32>(minutes);
- result.tm_sec = static_cast<int32>(seconds);
- return result;
- }
- mpt::ustring ToShortenedISO8601(tm date)
- {
- // We assume date in UTC here.
- // There are too many differences in supported format specifiers in strftime()
- // and strftime does not support reduced precision ISO8601 at all.
- // Just do the formatting ourselves.
- mpt::ustring result;
- mpt::ustring tz = U_("Z");
- if(date.tm_year == 0)
- {
- return result;
- }
- result += mpt::ufmt::dec0<4>(date.tm_year + 1900);
- if(date.tm_mon < 0 || date.tm_mon > 11)
- {
- return result;
- }
- result += U_("-") + mpt::ufmt::dec0<2>(date.tm_mon + 1);
- if(date.tm_mday < 1 || date.tm_mday > 31)
- {
- return result;
- }
- result += U_("-") + mpt::ufmt::dec0<2>(date.tm_mday);
- if(date.tm_hour == 0 && date.tm_min == 0 && date.tm_sec == 0)
- {
- return result;
- }
- if(date.tm_hour < 0 || date.tm_hour > 23)
- {
- return result;
- }
- if(date.tm_min < 0 || date.tm_min > 59)
- {
- return result;
- }
- result += U_("T");
- if(date.tm_isdst > 0)
- {
- tz = U_("+01:00");
- }
- result += mpt::ufmt::dec0<2>(date.tm_hour) + U_(":") + mpt::ufmt::dec0<2>(date.tm_min);
- if(date.tm_sec < 0 || date.tm_sec > 61)
- {
- return result + tz;
- }
- result += U_(":") + mpt::ufmt::dec0<2>(date.tm_sec);
- result += tz;
- return result;
- }
- } // namespace Date
- } // namespace mpt
- #ifdef MODPLUG_TRACKER
- namespace Util
- {
- #if MPT_OS_WINDOWS
- void MultimediaClock::Init()
- {
- m_CurrentPeriod = 0;
- }
- void MultimediaClock::SetPeriod(uint32 ms)
- {
- TIMECAPS caps = {};
- if(timeGetDevCaps(&caps, sizeof(caps)) != MMSYSERR_NOERROR)
- {
- return;
- }
- if((caps.wPeriodMax == 0) || (caps.wPeriodMin > caps.wPeriodMax))
- {
- return;
- }
- ms = std::clamp(mpt::saturate_cast<UINT>(ms), caps.wPeriodMin, caps.wPeriodMax);
- if(timeBeginPeriod(ms) != MMSYSERR_NOERROR)
- {
- return;
- }
- m_CurrentPeriod = ms;
- }
- void MultimediaClock::Cleanup()
- {
- if(m_CurrentPeriod > 0)
- {
- if(timeEndPeriod(m_CurrentPeriod) != MMSYSERR_NOERROR)
- {
- // should not happen
- MPT_ASSERT_NOTREACHED();
- }
- m_CurrentPeriod = 0;
- }
- }
- MultimediaClock::MultimediaClock()
- {
- Init();
- }
- MultimediaClock::MultimediaClock(uint32 ms)
- {
- Init();
- SetResolution(ms);
- }
- MultimediaClock::~MultimediaClock()
- {
- Cleanup();
- }
- uint32 MultimediaClock::SetResolution(uint32 ms)
- {
- if(m_CurrentPeriod == ms)
- {
- return m_CurrentPeriod;
- }
- Cleanup();
- if(ms != 0)
- {
- SetPeriod(ms);
- }
- return GetResolution();
- }
- uint32 MultimediaClock::GetResolution() const
- {
- return m_CurrentPeriod;
- }
- uint32 MultimediaClock::Now() const
- {
- return timeGetTime();
- }
- uint64 MultimediaClock::NowNanoseconds() const
- {
- return (uint64)timeGetTime() * (uint64)1000000;
- }
- #endif // MPT_OS_WINDOWS
- } // namespace Util
- #endif // MODPLUG_TRACKER
- OPENMPT_NAMESPACE_END
|