~ chicken-core (chicken-5) 051234429d8cfccdad8609cdcaf8c431acb2e81f
commit 051234429d8cfccdad8609cdcaf8c431acb2e81f Author: Peter Bex <peter@more-magic.net> AuthorDate: Sun May 3 12:33:20 2020 +0200 Commit: felix <felix@call-with-current-continuation.org> CommitDate: Wed May 6 11:22:03 2020 +0200 Fix a few platform-specific issues with current-process-millisecond On Windows, clock() will return (clock_t)(-1) after about 24.8 days (see https://docs.microsoft.com/en-us/cpp/c-runtime-library/reference/clock?view=vs-2019) Instead, use GetTickCount64(), which returns the milliseconds since system startup time as a 64-bit quantity, which does not have these problems. Unfortunately, the old mingw32 (which is close to unmaintained) does not support this. For compatibility, we will use GetTickCount() there. The C_NONUNIX macro is defined only on Mingw32 in chicken.h, so change the #ifdef guard around this code to check on __MINGW32__, to make it clear what it is checking for (for example, Haiku is not UNIX, but we don't define that macro there). If we need to support other non-UNIX platforms, we should add specific guards per platform. While we're at it, on Windows, also record the current value at startup, so that we can tighten the definition of current-process-milliseconds to state that it always returns the number of milliseconds since process startup. This means less variance across platforms and makes the functionality less "vague". Finally, split up the startup time into seconds and milliseconds, to make the time since startup more precise; otherwise you get seemingly random values when retrieving this due to only looking at the tv_sec part of the gettimeofday return value. Signed-off-by: felix <felix@call-with-current-continuation.org> diff --git a/NEWS b/NEWS index 67cc9f55..67b81974 100644 --- a/NEWS +++ b/NEWS @@ -7,7 +7,12 @@ - current-milliseconds has been deprecated in favor of the name current-process-milliseconds, to avoid confusion due to naming of current-milliseconds versus current-seconds, which do something - quite different. + quite different. Also, its definition has been tweaked to + guarantee that it returns the time since process startup. On + Windows, it will now return sane values after 24 days of uptime. + On UNIX, the values returned will be precise to the millisecond + instead of rounded down to the nearest second at startup (which + would result in erratic startup times). - Runtime system - Sleeping primordial thread doesn't forget mutations made to diff --git a/manual/Module (chicken time) b/manual/Module (chicken time) index 2f302c4e..4422ce64 100644 --- a/manual/Module (chicken time) +++ b/manual/Module (chicken time) @@ -23,7 +23,7 @@ system time will be always be 0. <procedure>(current-process-milliseconds)</procedure> -Returns the number of milliseconds since process- or machine startup. +Returns the number of milliseconds since process startup. ==== current-seconds diff --git a/runtime.c b/runtime.c index b01cdd06..ab14f66e 100644 --- a/runtime.c +++ b/runtime.c @@ -362,7 +362,8 @@ C_TLS C_uword C_heap_shrinkage_used = DEFAULT_HEAP_SHRINKAGE_USED, C_maximal_heap_size = DEFAULT_MAXIMAL_HEAP_SIZE; C_TLS time_t - C_startup_time_seconds, + C_startup_time_sec, + C_startup_time_msec, profile_frequency = 10000; C_TLS char **C_main_argv, @@ -684,17 +685,23 @@ int CHICKEN_initialize(int heap, int stack, int symbols, void *toplevel) struct sigaction sa; #endif - /*FIXME Should have C_tzset in chicken.h? */ -#ifdef C_NONUNIX - C_startup_time_seconds = (time_t)0; -# if defined(__MINGW32__) + /* FIXME Should have C_tzset in chicken.h? */ +#if defined(__MINGW32__) +# if defined(__MINGW64_VERSION_MAJOR) + ULONGLONG tick_count = GetTickCount64(); +# else + /* mingw32 doesn't yet have GetTickCount64 support */ + ULONGLONG tick_count = GetTickCount(); +# endif + C_startup_time_sec = tick_count / 1000; + C_startup_time_msec = tick_count % 1000; /* Make sure _tzname, _timezone, and _daylight are set */ _tzset(); -# endif #else struct timeval tv; C_gettimeofday(&tv, NULL); - C_startup_time_seconds = tv.tv_sec; + C_startup_time_sec = tv.tv_sec; + C_startup_time_msec = tv.tv_usec / 1000; /* Make sure tzname, timezone, and daylight are set */ tzset(); #endif @@ -2033,14 +2040,18 @@ C_regparm C_u64 C_fcall C_milliseconds(void) C_regparm C_u64 C_fcall C_current_process_milliseconds(void) { -#ifdef C_NONUNIX - if(CLOCKS_PER_SEC == 1000) return clock(); - else return (C_u64)clock() / (C_u64)CLOCKS_PER_SEC * 1000; +#if defined(__MINGW32__) +# if defined(__MINGW64_VERSION_MAJOR) + ULONGLONG tick_count = GetTickCount64(); +# else + ULONGLONG tick_count = GetTickCount(); +# endif + return tick_count - (C_startup_time_sec * 1000) - C_startup_time_msec; #else struct timeval tv; if(C_gettimeofday(&tv, NULL) == -1) return 0; - else return (tv.tv_sec - C_startup_time_seconds) * 1000 + tv.tv_usec / 1000; + else return (tv.tv_sec - C_startup_time_sec) * 1000 + tv.tv_usec / 1000 - C_startup_time_msec; #endif }Trap