2021-01-18 13:20:06 +01:00
|
|
|
// Copyright (c) 2019-2021 Alexander Medvednikov. All rights reserved.
|
2020-01-01 12:01:03 +01:00
|
|
|
// Use of this source code is governed by an MIT license
|
|
|
|
// that can be found in the LICENSE file.
|
|
|
|
module time
|
|
|
|
|
2020-06-07 15:19:09 +02:00
|
|
|
#include <time.h>
|
2021-02-01 14:50:41 +01:00
|
|
|
|
2020-01-01 12:01:03 +01:00
|
|
|
struct C.tm {
|
2020-12-06 15:19:39 +01:00
|
|
|
tm_sec int
|
|
|
|
tm_min int
|
|
|
|
tm_hour int
|
|
|
|
tm_mday int
|
|
|
|
tm_mon int
|
|
|
|
tm_year int
|
|
|
|
tm_wday int
|
|
|
|
tm_yday int
|
2020-06-10 11:14:55 +02:00
|
|
|
tm_isdst int
|
2020-01-01 12:01:03 +01:00
|
|
|
}
|
|
|
|
|
2020-02-05 06:13:11 +01:00
|
|
|
fn C.timegm(&tm) time_t
|
2020-12-06 15:19:39 +01:00
|
|
|
|
2020-12-26 02:10:47 +01:00
|
|
|
// fn C.gmtime_r(&tm, &gbuf)
|
2020-12-06 15:19:39 +01:00
|
|
|
fn C.localtime_r(t &C.time_t, tm &C.tm)
|
2020-01-01 12:01:03 +01:00
|
|
|
|
2020-03-03 15:06:21 +01:00
|
|
|
fn make_unix_time(t C.tm) int {
|
2020-02-07 22:10:48 +01:00
|
|
|
return int(C.timegm(&t))
|
2020-02-04 12:17:04 +01:00
|
|
|
}
|
2020-04-24 07:33:25 +02:00
|
|
|
|
2020-12-26 02:10:47 +01:00
|
|
|
// local returns t with the location set to local time.
|
|
|
|
pub fn (t Time) local() Time {
|
2020-06-10 11:14:55 +02:00
|
|
|
loc_tm := C.tm{}
|
2020-07-14 00:16:31 +02:00
|
|
|
C.localtime_r(time_t(&t.unix), &loc_tm)
|
2020-06-10 11:14:55 +02:00
|
|
|
return convert_ctime(loc_tm, t.microsecond)
|
|
|
|
}
|
|
|
|
|
2020-09-25 12:02:32 +02:00
|
|
|
type time_t = voidptr
|
2020-04-24 07:33:25 +02:00
|
|
|
|
|
|
|
// in most systems, these are __quad_t, which is an i64
|
|
|
|
struct C.timespec {
|
2020-07-15 10:22:33 +02:00
|
|
|
mut:
|
2020-04-24 07:33:25 +02:00
|
|
|
tv_sec i64
|
|
|
|
tv_nsec i64
|
|
|
|
}
|
|
|
|
|
|
|
|
// the first arg is defined in include/bits/types.h as `__S32_TYPE`, which is `int`
|
|
|
|
fn C.clock_gettime(int, &C.timespec)
|
|
|
|
|
2021-02-21 16:05:03 +01:00
|
|
|
fn C.nanosleep(req &C.timespec, rem &C.timespec) int
|
|
|
|
|
2021-01-13 15:30:54 +01:00
|
|
|
// sys_mono_now returns a *monotonically increasing time*, NOT a time adjusted for daylight savings, location etc.
|
2020-05-18 22:54:08 +02:00
|
|
|
pub fn sys_mono_now() u64 {
|
2020-04-24 07:33:25 +02:00
|
|
|
$if macos {
|
|
|
|
return sys_mono_now_darwin()
|
|
|
|
} $else {
|
|
|
|
ts := C.timespec{}
|
|
|
|
C.clock_gettime(C.CLOCK_MONOTONIC, &ts)
|
2020-12-06 15:19:39 +01:00
|
|
|
return u64(ts.tv_sec) * 1000000000 + u64(ts.tv_nsec)
|
2020-04-24 07:33:25 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-26 19:36:38 +02:00
|
|
|
// NB: vpc_now is used by `v -profile` .
|
|
|
|
// It should NOT call *any other v function*, just C functions and casts.
|
|
|
|
[inline]
|
|
|
|
fn vpc_now() u64 {
|
2020-05-05 16:19:25 +02:00
|
|
|
ts := C.timespec{}
|
|
|
|
C.clock_gettime(C.CLOCK_MONOTONIC, &ts)
|
2020-12-06 15:19:39 +01:00
|
|
|
return u64(ts.tv_sec) * 1000000000 + u64(ts.tv_nsec)
|
2020-04-26 19:36:38 +02:00
|
|
|
}
|
2020-06-07 15:19:09 +02:00
|
|
|
|
2020-09-25 17:02:02 +02:00
|
|
|
// The linux_* functions are placed here, since they're used on Android as well
|
|
|
|
// TODO: should `$if linux {}` be parsed on Android as well? (Android runs under the Linux kernel)
|
|
|
|
// linux_now returns the local time with high precision for most os:es
|
|
|
|
// this should be implemented properly with support for leap seconds.
|
|
|
|
// It uses the realtime clock to get and converts it to local time
|
|
|
|
fn linux_now() Time {
|
|
|
|
// get the high precision time as UTC realtime clock
|
|
|
|
// and use the nanoseconds part
|
|
|
|
mut ts := C.timespec{}
|
|
|
|
C.clock_gettime(C.CLOCK_REALTIME, &ts)
|
|
|
|
loc_tm := C.tm{}
|
|
|
|
C.localtime_r(&ts.tv_sec, &loc_tm)
|
2020-12-06 15:19:39 +01:00
|
|
|
return convert_ctime(loc_tm, int(ts.tv_nsec / 1000))
|
2020-09-25 17:02:02 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
fn linux_utc() Time {
|
|
|
|
// get the high precision time as UTC realtime clock
|
|
|
|
// and use the nanoseconds part
|
|
|
|
mut ts := C.timespec{}
|
|
|
|
C.clock_gettime(C.CLOCK_REALTIME, &ts)
|
2020-12-06 15:19:39 +01:00
|
|
|
return unix2(int(ts.tv_sec), int(ts.tv_nsec / 1000))
|
2020-09-25 17:02:02 +02:00
|
|
|
}
|
2020-06-10 11:14:55 +02:00
|
|
|
|
2020-06-07 15:19:09 +02:00
|
|
|
// dummy to compile with all compilers
|
|
|
|
pub fn win_now() Time {
|
|
|
|
return Time{}
|
|
|
|
}
|
|
|
|
|
2020-06-10 11:14:55 +02:00
|
|
|
// dummy to compile with all compilers
|
|
|
|
pub fn win_utc() Time {
|
|
|
|
return Time{}
|
|
|
|
}
|
2020-06-07 15:19:09 +02:00
|
|
|
|
|
|
|
// dummy to compile with all compilers
|
|
|
|
pub struct C.timeval {
|
|
|
|
tv_sec u64
|
|
|
|
tv_usec u64
|
2020-06-10 11:14:55 +02:00
|
|
|
}
|
2020-07-15 10:22:33 +02:00
|
|
|
|
|
|
|
// return absolute timespec for now()+d
|
|
|
|
pub fn (d Duration) timespec() C.timespec {
|
|
|
|
mut ts := C.timespec{}
|
|
|
|
C.clock_gettime(C.CLOCK_REALTIME, &ts)
|
|
|
|
d_sec := d / second
|
|
|
|
d_nsec := d % second
|
|
|
|
ts.tv_sec += d_sec
|
|
|
|
ts.tv_nsec += d_nsec
|
2020-12-21 09:27:06 +01:00
|
|
|
if ts.tv_nsec > i64(second) {
|
|
|
|
ts.tv_nsec -= i64(second)
|
2020-07-15 10:22:33 +02:00
|
|
|
ts.tv_sec++
|
|
|
|
}
|
|
|
|
return ts
|
|
|
|
}
|
|
|
|
|
|
|
|
// return timespec of 1970/1/1
|
|
|
|
pub fn zero_timespec() C.timespec {
|
|
|
|
ts := C.timespec{
|
2020-12-06 15:19:39 +01:00
|
|
|
tv_sec: 0
|
2020-07-15 10:22:33 +02:00
|
|
|
tv_nsec: 0
|
|
|
|
}
|
|
|
|
return ts
|
|
|
|
}
|
2021-02-21 16:05:03 +01:00
|
|
|
|
|
|
|
// wait makes the calling thread sleep for a given duration (in nanoseconds).
|
|
|
|
pub fn wait(duration Duration) {
|
|
|
|
ts := &C.timespec{duration / second, duration % second}
|
|
|
|
C.nanosleep(ts, C.NULL)
|
|
|
|
}
|