224 lines
		
	
	
		
			4.5 KiB
		
	
	
	
		
			V
		
	
	
			
		
		
	
	
			224 lines
		
	
	
		
			4.5 KiB
		
	
	
	
		
			V
		
	
	
| // Copyright (c) 2019-2020 Alexander Medvednikov. All rights reserved.
 | |
| // Use of this source code is governed by an MIT license
 | |
| // that can be found in the LICENSE file.
 | |
| module time
 | |
| 
 | |
| #include <time.h>
 | |
| #include <sysinfoapi.h>
 | |
| 
 | |
| struct C.tm {
 | |
| 	tm_year int
 | |
| 	tm_mon  int
 | |
| 	tm_mday int
 | |
| 	tm_hour int
 | |
| 	tm_min  int
 | |
| 	tm_sec  int
 | |
| }
 | |
| 
 | |
| struct C._FILETIME
 | |
| 
 | |
| struct SystemTime {
 | |
|   year 			u16
 | |
|   month 		u16
 | |
|   day_of_week 	u16
 | |
|   day  			u16
 | |
|   hour 			u16
 | |
|   minute 		u16
 | |
|   second 		u16
 | |
|   millisecond 	u16
 | |
| }
 | |
| 
 | |
| fn C.GetSystemTimeAsFileTime(lpSystemTimeAsFileTime C._FILETIME)
 | |
| fn C.FileTimeToSystemTime()
 | |
| fn C.SystemTimeToTzSpecificLocalTime()
 | |
| fn C.localtime_s(t &C.time_t, tm &C.tm )
 | |
| 
 | |
| const (
 | |
| 	// start_time is needed on Darwin and Windows because of potential overflows
 | |
| 	start_time 		 	= init_win_time_start()
 | |
| 	freq_time  		 	= init_win_time_freq()
 | |
| 	start_local_time 	= local_as_unix_time()
 | |
| )
 | |
| 
 | |
| // in most systems, these are __quad_t, which is an i64
 | |
| struct C.timespec {
 | |
| 	tv_sec  i64
 | |
| 	tv_nsec i64
 | |
| }
 | |
| 
 | |
| 
 | |
| fn C._mkgmtime(&C.tm) time_t
 | |
| 
 | |
| fn C.QueryPerformanceCounter(&u64) C.BOOL
 | |
| 
 | |
| fn C.QueryPerformanceFrequency(&u64) C.BOOL
 | |
| 
 | |
| fn make_unix_time(t C.tm) int {
 | |
| 	return int(C._mkgmtime(&t))
 | |
| }
 | |
| 
 | |
| fn init_win_time_freq() u64 {
 | |
| 	f := u64(0)
 | |
| 	C.QueryPerformanceFrequency(&f)
 | |
| 	return f
 | |
| }
 | |
| 
 | |
| fn init_win_time_start() u64 {
 | |
| 	s := u64(0)
 | |
| 	C.QueryPerformanceCounter(&s)
 | |
| 	return s
 | |
| }
 | |
| 
 | |
| pub fn sys_mono_now() u64 {
 | |
| 	tm := u64(0)
 | |
| 	C.QueryPerformanceCounter(&tm) // XP or later never fail
 | |
| 	return (tm - start_time) * 1_000_000_000 / freq_time
 | |
| }
 | |
| 
 | |
| // 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 {
 | |
| 	tm := u64(0)
 | |
| 	C.QueryPerformanceCounter(&tm)
 | |
| 	return tm
 | |
| }
 | |
| 
 | |
| // local_as_unix_time returns the current local time as unix time
 | |
| fn local_as_unix_time() int {
 | |
| 	t := C.time(0)
 | |
| 	tm := C.localtime(&t)
 | |
| 
 | |
| 	return make_unix_time(tm)
 | |
| }
 | |
| 
 | |
| fn to_local_time(t Time) Time {
 | |
| 	st_utc := SystemTime{
 | |
| 		year: u16(t.year)
 | |
| 		month: u16(t.month)
 | |
| 		day: u16(t.day)
 | |
| 		hour: u16(t.hour)
 | |
| 		minute: u16(t.minute)
 | |
| 		second: u16(t.second)
 | |
| 	}
 | |
| 	st_local := SystemTime{}
 | |
| 	C.SystemTimeToTzSpecificLocalTime(voidptr(0), &st_utc, &st_local)
 | |
| 
 | |
| 	t_local := Time {
 | |
| 			year: st_local.year
 | |
| 			month: st_local.month
 | |
| 			day: st_local.day
 | |
| 			hour: st_local.hour
 | |
| 			minute: st_local.minute
 | |
| 			second: st_local.second
 | |
| 			// These are the same
 | |
| 			microsecond: t.microsecond
 | |
| 			unix: t.unix
 | |
| 	}
 | |
| 	return t_local
 | |
| }
 | |
| 
 | |
| // win_now calculates current time using winapi to get higher resolution on windows
 | |
| // GetSystemTimeAsFileTime is used and converted to local time. It can resolve time
 | |
| // down to millisecond. Other more precice methods can be implemented in the future
 | |
| [inline]
 | |
| fn win_now() Time {
 | |
| 
 | |
| 	ft_utc := C._FILETIME{}
 | |
| 	C.GetSystemTimeAsFileTime(&ft_utc)
 | |
| 
 | |
| 	st_utc := SystemTime{}
 | |
| 	C.FileTimeToSystemTime(&ft_utc, &st_utc)
 | |
| 
 | |
| 	st_local := SystemTime{}
 | |
| 	C.SystemTimeToTzSpecificLocalTime(voidptr(0), &st_utc, &st_local)
 | |
| 
 | |
| 	t := Time {
 | |
| 		year: st_local.year
 | |
| 		month: st_local.month
 | |
| 		day: st_local.day
 | |
| 		hour: st_local.hour
 | |
| 		minute: st_local.minute
 | |
| 		second: st_local.second
 | |
| 		microsecond: st_local.millisecond*1000
 | |
| 		unix: u64(st_local.unix_time())
 | |
| 	}
 | |
| 
 | |
| 	return t
 | |
| }
 | |
| 
 | |
| // win_utc calculates current time using winapi to get higher resolution on windows
 | |
| // GetSystemTimeAsFileTime is used. It can resolve time down to millisecond
 | |
| // other more precice methods can be implemented in the future
 | |
| [inline]
 | |
| fn win_utc() Time {
 | |
| 
 | |
| 	ft_utc := C._FILETIME{}
 | |
| 	C.GetSystemTimeAsFileTime(&ft_utc)
 | |
| 
 | |
| 	st_utc := SystemTime{}
 | |
| 	C.FileTimeToSystemTime(&ft_utc, &st_utc)
 | |
| 
 | |
| 	t := Time {
 | |
| 		year: st_utc.year
 | |
| 		month: st_utc.month
 | |
| 		day: st_utc.day
 | |
| 		hour: st_utc.hour
 | |
| 		minute: st_utc.minute
 | |
| 		second: st_utc.second
 | |
| 		microsecond: st_utc.millisecond*1000
 | |
| 		unix: u64(st_utc.unix_time())
 | |
| 	}
 | |
| 
 | |
| 	return t
 | |
| }
 | |
| 
 | |
| // unix_time returns Unix time.
 | |
| pub fn (st SystemTime) unix_time() int {
 | |
| 	tt := C.tm{
 | |
| 		tm_sec: st.second
 | |
| 		tm_min: st.minute
 | |
| 		tm_hour: st.hour
 | |
| 		tm_mday: st.day
 | |
| 		tm_mon: st.month - 1
 | |
| 		tm_year: st.year - 1900
 | |
| 	}
 | |
| 	return make_unix_time(tt)
 | |
| }
 | |
| 
 | |
| // dummy to compile with all compilers
 | |
| pub fn darwin_now() Time {
 | |
| 	return Time{}
 | |
| }
 | |
| 
 | |
| // dummy to compile with all compilers
 | |
| pub fn linux_now() Time {
 | |
| 	return Time{}
 | |
| }
 | |
| 
 | |
| // dummy to compile with all compilers
 | |
| pub fn solaris_now() Time {
 | |
| 	return Time{}
 | |
| }
 | |
| 
 | |
| pub fn darwin_utc() Time {
 | |
| 	return Time{}
 | |
| }
 | |
| 
 | |
| // dummy to compile with all compilers
 | |
| pub fn linux_utc() Time {
 | |
| 	return Time{}
 | |
| }
 | |
| 
 | |
| // dummy to compile with all compilers
 | |
| pub fn solaris_utc() Time {
 | |
| 	return Time{}
 | |
| }
 | |
| 
 | |
| 
 | |
| // dummy to compile with all compilers
 | |
| pub struct C.timeval {
 | |
| 	tv_sec  u64
 | |
| 	tv_usec u64
 | |
| }
 |