mirror of
https://github.com/jerryscript-project/jerryscript.git
synced 2026-02-07 19:26:25 +00:00
PR #4513 caused a measurable slowdown by removing the GNU specific TZA calculation. This patch reverts this removal but also keeps the general implementation introduced in #4513 JerryScript-DCO-1.0-Signed-off-by: Robert Fancsik robert.fancsik@h-lab.eu
159 lines
4.8 KiB
C
159 lines
4.8 KiB
C
/* Copyright JS Foundation and other contributors, http://js.foundation
|
|
*
|
|
* 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 <time.h>
|
|
|
|
#ifdef _WIN32
|
|
#include <windows.h>
|
|
#include <winbase.h>
|
|
#include <winnt.h>
|
|
#endif /* _WIN32 */
|
|
|
|
#if defined (__GNUC__) || defined (__clang__)
|
|
#include <sys/time.h>
|
|
#endif /* __GNUC__ || __clang__ */
|
|
|
|
#include "jerryscript-port.h"
|
|
#include "jerryscript-port-default.h"
|
|
|
|
#ifdef _WIN32
|
|
static const LONGLONG UnixEpochInTicks = 116444736000000000; /* difference between 1970 and 1601 */
|
|
static const LONGLONG TicksPerMs = 10000; /* 1 tick is 100 nanoseconds */
|
|
|
|
/* https://support.microsoft.com/en-us/help/167296/how-to-convert-a-unix-time-t-to-a-win32-filetime-or-systemtime */
|
|
static void UnixTimeMsToFileTime (double t, LPFILETIME pft)
|
|
{
|
|
LONGLONG ll = (LONGLONG) t * TicksPerMs + UnixEpochInTicks;
|
|
pft->dwLowDateTime = (DWORD) ll;
|
|
pft->dwHighDateTime = (DWORD) (ll >> 32);
|
|
} /* UnixTimeMsToFileTime */
|
|
|
|
static double FileTimeToUnixTimeMs (FILETIME ft)
|
|
{
|
|
ULARGE_INTEGER date;
|
|
date.HighPart = ft.dwHighDateTime;
|
|
date.LowPart = ft.dwLowDateTime;
|
|
return (double) (((LONGLONG) date.QuadPart - UnixEpochInTicks) / TicksPerMs);
|
|
} /* FileTimeToUnixTimeMs */
|
|
|
|
#endif /* _WIN32 */
|
|
|
|
/**
|
|
* Default implementation of jerry_port_get_local_time_zone_adjustment.
|
|
*
|
|
* @return offset between UTC and local time at the given unix timestamp, if
|
|
* available. Otherwise, returns 0, assuming UTC time.
|
|
*/
|
|
double jerry_port_get_local_time_zone_adjustment (double unix_ms, /**< ms since unix epoch */
|
|
bool is_utc) /**< is the time above in UTC? */
|
|
{
|
|
#if defined (HAVE_TM_GMTOFF)
|
|
struct tm tm;
|
|
time_t now = (time_t) (unix_ms / 1000);
|
|
localtime_r (&now, &tm);
|
|
|
|
if (!is_utc)
|
|
{
|
|
now -= tm.tm_gmtoff;
|
|
localtime_r (&now, &tm);
|
|
}
|
|
|
|
return ((double) tm.tm_gmtoff) * 1000;
|
|
#elif defined (_WIN32)
|
|
FILETIME fileTime, localFileTime;
|
|
SYSTEMTIME systemTime, localSystemTime;
|
|
ULARGE_INTEGER time, localTime;
|
|
|
|
UnixTimeMsToFileTime (unix_ms, &fileTime);
|
|
/* If time is earlier than year 1601, then always using year 1601 to query time zone adjustment */
|
|
if (fileTime.dwHighDateTime >= 0x80000000)
|
|
{
|
|
fileTime.dwHighDateTime = 0;
|
|
fileTime.dwLowDateTime = 0;
|
|
}
|
|
|
|
if (FileTimeToSystemTime (&fileTime, &systemTime)
|
|
&& SystemTimeToTzSpecificLocalTime (0, &systemTime, &localSystemTime)
|
|
&& SystemTimeToFileTime (&localSystemTime, &localFileTime))
|
|
{
|
|
time.LowPart = fileTime.dwLowDateTime;
|
|
time.HighPart = fileTime.dwHighDateTime;
|
|
localTime.LowPart = localFileTime.dwLowDateTime;
|
|
localTime.HighPart = localFileTime.dwHighDateTime;
|
|
return (double) (((LONGLONG) localTime.QuadPart - (LONGLONG) time.QuadPart) / TicksPerMs);
|
|
}
|
|
return 0.0;
|
|
#elif defined (__GNUC__) || defined (__clang__)
|
|
time_t now_time = (time_t) (unix_ms / 1000);
|
|
double tza_s = 0.0;
|
|
|
|
while (true)
|
|
{
|
|
struct tm now_tm;
|
|
if (!gmtime_r (&now_time, &now_tm))
|
|
{
|
|
break;
|
|
}
|
|
now_tm.tm_isdst = -1; /* if not overridden, DST will not be taken into account */
|
|
time_t local_time = mktime (&now_tm);
|
|
if (local_time == (time_t) -1)
|
|
{
|
|
break;
|
|
}
|
|
tza_s = difftime (now_time, local_time);
|
|
|
|
if (is_utc)
|
|
{
|
|
break;
|
|
}
|
|
now_time -= (time_t) tza_s;
|
|
is_utc = true;
|
|
}
|
|
|
|
return tza_s * 1000;
|
|
#else /* !HAVE_TM_GMTOFF && !_WIN32 && !__GNUC__ && !__clang__ */
|
|
(void) unix_ms; /* unused */
|
|
(void) is_utc; /* unused */
|
|
return 0.0;
|
|
#endif /* HAVE_TM_GMTOFF */
|
|
} /* jerry_port_get_local_time_zone_adjustment */
|
|
|
|
/**
|
|
* Default implementation of jerry_port_get_current_time. Uses 'gettimeofday' if
|
|
* available on the system, does nothing otherwise.
|
|
*
|
|
* @return milliseconds since Unix epoch - if 'gettimeofday' is available and
|
|
* executed successfully,
|
|
* 0 - otherwise.
|
|
*/
|
|
double jerry_port_get_current_time (void)
|
|
{
|
|
#ifdef _WIN32
|
|
FILETIME ft;
|
|
GetSystemTimeAsFileTime (&ft);
|
|
return FileTimeToUnixTimeMs (ft);
|
|
#elif defined (__GNUC__) || defined (__clang__)
|
|
struct timeval tv;
|
|
|
|
if (gettimeofday (&tv, NULL) == 0)
|
|
{
|
|
return ((double) tv.tv_sec) * 1000.0 + ((double) tv.tv_usec) / 1000.0;
|
|
}
|
|
return 0.0;
|
|
#else /* !_WIN32 && !__GNUC__ && !__clang__ */
|
|
return 0.0;
|
|
#endif /* _WIN32 */
|
|
} /* jerry_port_get_current_time */
|