2018-12-11 17:06:20 +00:00
|
|
|
use super::utils::{copy_cstr_into_wasm, write_to_buf};
|
2018-11-26 21:15:49 +00:00
|
|
|
use libc::{
|
|
|
|
c_int,
|
2018-11-26 22:27:56 +00:00
|
|
|
c_long,
|
2018-11-26 21:20:10 +00:00
|
|
|
clock_gettime as libc_clock_gettime,
|
2018-11-26 21:15:49 +00:00
|
|
|
localtime,
|
2018-12-07 13:50:35 +00:00
|
|
|
localtime_r,
|
|
|
|
tm,
|
2018-11-27 04:29:26 +00:00
|
|
|
time,
|
2018-11-26 21:15:49 +00:00
|
|
|
time_t,
|
2018-11-27 04:29:26 +00:00
|
|
|
timespec,
|
2018-12-07 13:50:35 +00:00
|
|
|
c_char,
|
2018-11-26 21:15:49 +00:00
|
|
|
};
|
2018-11-27 04:29:26 +00:00
|
|
|
use std::mem;
|
2018-11-26 21:30:55 +00:00
|
|
|
use std::time::SystemTime;
|
2018-11-26 20:28:20 +00:00
|
|
|
|
|
|
|
use crate::webassembly::Instance;
|
|
|
|
|
|
|
|
/// emscripten: _gettimeofday
|
2018-11-27 04:29:26 +00:00
|
|
|
pub extern "C" fn _gettimeofday(tp: c_int, tz: c_int, instance: &mut Instance) -> c_int {
|
2018-11-29 06:02:25 +00:00
|
|
|
debug!("emscripten::_gettimeofday {} {}", tp, tz);
|
2018-11-26 21:45:38 +00:00
|
|
|
#[repr(C)]
|
|
|
|
struct GuestTimeVal {
|
|
|
|
tv_sec: i32,
|
|
|
|
tv_usec: i32,
|
|
|
|
}
|
2018-11-26 20:28:20 +00:00
|
|
|
|
2018-11-27 04:29:26 +00:00
|
|
|
assert!(
|
|
|
|
tz == 0,
|
|
|
|
"the timezone argument of `_gettimeofday` must be null"
|
|
|
|
);
|
2018-11-26 20:28:20 +00:00
|
|
|
unsafe {
|
2018-11-26 21:45:38 +00:00
|
|
|
let now = SystemTime::now();
|
|
|
|
let since_epoch = now.duration_since(SystemTime::UNIX_EPOCH).unwrap();
|
|
|
|
let timeval_struct_ptr = instance.memory_offset_addr(0, tp as _) as *mut GuestTimeVal;
|
2018-11-26 20:28:20 +00:00
|
|
|
|
2018-11-26 21:45:38 +00:00
|
|
|
(*timeval_struct_ptr).tv_sec = since_epoch.as_secs() as _;
|
|
|
|
(*timeval_struct_ptr).tv_usec = since_epoch.subsec_nanos() as _;
|
2018-11-26 20:28:20 +00:00
|
|
|
}
|
2018-11-26 21:45:38 +00:00
|
|
|
0
|
2018-11-26 20:28:20 +00:00
|
|
|
}
|
2018-11-26 20:42:47 +00:00
|
|
|
|
2018-11-26 20:28:20 +00:00
|
|
|
/// emscripten: _clock_gettime
|
2018-11-27 04:29:26 +00:00
|
|
|
pub extern "C" fn _clock_gettime(clk_id: c_int, tp: c_int, instance: &mut Instance) -> c_int {
|
2018-11-29 06:02:25 +00:00
|
|
|
debug!("emscripten::_clock_gettime {} {}", clk_id, tp);
|
2018-11-26 21:45:38 +00:00
|
|
|
#[repr(C)]
|
|
|
|
struct GuestTimeSpec {
|
|
|
|
tv_sec: i32,
|
|
|
|
tv_nsec: i32,
|
|
|
|
}
|
2018-11-26 20:28:20 +00:00
|
|
|
|
|
|
|
unsafe {
|
2018-11-26 21:45:38 +00:00
|
|
|
let mut timespec = timespec {
|
|
|
|
tv_sec: 0,
|
|
|
|
tv_nsec: 0,
|
|
|
|
};
|
|
|
|
let ret = libc_clock_gettime(clk_id as _, &mut timespec);
|
|
|
|
if ret != 0 {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
let timespec_struct_ptr = instance.memory_offset_addr(0, tp as _) as *mut GuestTimeSpec;
|
|
|
|
(*timespec_struct_ptr).tv_sec = timespec.tv_sec as _;
|
|
|
|
(*timespec_struct_ptr).tv_nsec = timespec.tv_nsec as _;
|
2018-11-26 20:28:20 +00:00
|
|
|
}
|
2018-11-26 21:45:38 +00:00
|
|
|
0
|
2018-11-26 20:28:20 +00:00
|
|
|
}
|
2018-11-26 21:15:49 +00:00
|
|
|
|
2018-12-07 13:50:35 +00:00
|
|
|
#[repr(C)]
|
|
|
|
struct guest_tm {
|
2018-12-07 15:37:14 +00:00
|
|
|
pub tm_sec: c_int, // 0
|
|
|
|
pub tm_min: c_int, // 4
|
|
|
|
pub tm_hour: c_int, // 8
|
|
|
|
pub tm_mday: c_int, // 12
|
|
|
|
pub tm_mon: c_int, // 16
|
|
|
|
pub tm_year: c_int, // 20
|
|
|
|
pub tm_wday: c_int, // 24
|
|
|
|
pub tm_yday: c_int, // 28
|
|
|
|
pub tm_isdst: c_int, // 32
|
|
|
|
pub tm_gmtoff: c_int, // 36
|
|
|
|
pub tm_zone: c_int, // 40
|
2018-12-07 13:50:35 +00:00
|
|
|
}
|
|
|
|
|
2018-12-11 18:04:46 +00:00
|
|
|
/// emscripten: _tvset
|
|
|
|
pub extern "C" fn _tvset() {
|
|
|
|
debug!("emscripten::_tvset UNIMPLEMENTED");
|
|
|
|
}
|
|
|
|
|
2018-12-11 17:06:20 +00:00
|
|
|
/// formats time as a C string
|
|
|
|
unsafe extern "C" fn fmt_time(time: u32, instance: &Instance) -> *const c_char {
|
|
|
|
let date = &*(instance.memory_offset_addr(0, time as _) as *mut guest_tm);
|
|
|
|
|
|
|
|
let days = vec!["Sun", "Mon", "Tue", "Wed", "Thu", "Fri", "Sat"];
|
|
|
|
let months = vec!["Jan", "Feb", "Mar", "Apr", "May", "Jun", "Jul", "Aug", "Sep", "Oct", "Nov", "Dec"];
|
|
|
|
let year = 1900 + date.tm_year;
|
|
|
|
|
|
|
|
let time_str = format!(
|
2018-12-11 18:04:46 +00:00
|
|
|
// NOTE: TODO: Hack! The 14 accompanying chars are needed for some reason
|
2018-12-11 17:06:20 +00:00
|
|
|
"{} {} {:2} {:02}:{:02}:{:02} {:4}\n\0\0\0\0\0\0\0\0\0\0\0\0\0",
|
|
|
|
days[date.tm_wday as usize],
|
|
|
|
months[date.tm_mon as usize],
|
|
|
|
date.tm_mday,
|
|
|
|
date.tm_hour,
|
|
|
|
date.tm_min,
|
|
|
|
date.tm_sec,
|
|
|
|
year
|
|
|
|
);
|
|
|
|
|
|
|
|
time_str[0..26].as_ptr() as _
|
|
|
|
}
|
|
|
|
|
2018-12-07 13:50:35 +00:00
|
|
|
/// emscripten: _asctime
|
|
|
|
pub extern "C" fn _asctime(time: u32, instance: &mut Instance) -> u32 {
|
|
|
|
debug!("emscripten::_asctime {}", time);
|
|
|
|
|
|
|
|
unsafe {
|
2018-12-11 17:06:20 +00:00
|
|
|
let time_str_ptr = fmt_time(time, instance);
|
|
|
|
copy_cstr_into_wasm(instance, time_str_ptr)
|
2018-12-11 16:05:07 +00:00
|
|
|
|
|
|
|
// let c_str = instance.memory_offset_addr(0, time_str_offset as _) as *mut i8;
|
|
|
|
// use std::ffi::CStr;
|
|
|
|
// debug!("#### cstr = {:?}", CStr::from_ptr(c_str));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// emscripten: _asctime_r
|
|
|
|
pub extern "C" fn _asctime_r(time: u32, buf: u32, instance: &mut Instance) -> u32 {
|
2018-12-11 18:04:46 +00:00
|
|
|
debug!("emscripten::_asctime_r {}", time);
|
2018-12-11 16:05:07 +00:00
|
|
|
|
|
|
|
unsafe {
|
2018-12-07 15:37:14 +00:00
|
|
|
// NOTE: asctime_r is specced to behave in an undefined manner if the algorithm would attempt
|
2018-12-07 13:50:35 +00:00
|
|
|
// to write out more than 26 bytes (including the null terminator).
|
|
|
|
// See http://pubs.opengroup.org/onlinepubs/9699919799/functions/asctime.html
|
|
|
|
// Our undefined behavior is to truncate the write to at most 26 bytes, including null terminator.
|
2018-12-11 17:06:20 +00:00
|
|
|
let time_str_ptr = fmt_time(time, instance);
|
2018-12-11 16:05:07 +00:00
|
|
|
write_to_buf(time_str_ptr, buf, 26, instance)
|
2018-12-07 13:50:35 +00:00
|
|
|
|
2018-12-07 17:19:28 +00:00
|
|
|
// let c_str = instance.memory_offset_addr(0, time_str_offset as _) as *mut i8;
|
|
|
|
// use std::ffi::CStr;
|
|
|
|
// debug!("#### cstr = {:?}", CStr::from_ptr(c_str));
|
2018-12-07 13:50:35 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-11-26 21:15:49 +00:00
|
|
|
/// emscripten: _localtime
|
2018-11-26 22:27:56 +00:00
|
|
|
pub extern "C" fn _localtime(time_p: u32, instance: &mut Instance) -> c_int {
|
2018-11-26 21:15:49 +00:00
|
|
|
debug!("emscripten::_localtime {}", time_p);
|
2018-12-07 13:50:35 +00:00
|
|
|
// NOTE: emscripten seems to want tzset() called in this function
|
|
|
|
// https://stackoverflow.com/questions/19170721/real-time-awareness-of-timezone-change-in-localtime-vs-localtime-r
|
2018-11-26 21:15:49 +00:00
|
|
|
|
2018-12-07 13:50:35 +00:00
|
|
|
unsafe {
|
|
|
|
let time_p_addr = instance.memory_offset_addr(0, time_p as _) as *mut i64;
|
2018-12-11 19:42:29 +00:00
|
|
|
let result_tm = &*localtime(time_p_addr);
|
2018-12-07 13:50:35 +00:00
|
|
|
let tm_struct_offset = (instance.emscripten_data.as_ref().unwrap().malloc)(
|
|
|
|
mem::size_of::<guest_tm>() as _,
|
|
|
|
instance,
|
|
|
|
);
|
|
|
|
|
|
|
|
let tm_struct_ptr = instance.memory_offset_addr(0, tm_struct_offset as _) as *mut guest_tm;
|
2018-12-11 19:42:29 +00:00
|
|
|
debug!(
|
|
|
|
">>>>>>> time = {}, {}, {}, {}, {}, {}, {}, {}",
|
|
|
|
result_tm.tm_sec, result_tm.tm_min, result_tm.tm_hour, result_tm.tm_mday,
|
|
|
|
result_tm.tm_mon, result_tm.tm_year, result_tm.tm_wday, result_tm.tm_yday,
|
|
|
|
);
|
2018-12-07 13:50:35 +00:00
|
|
|
|
2018-12-11 19:42:29 +00:00
|
|
|
(*tm_struct_ptr).tm_sec = result_tm.tm_sec;
|
|
|
|
(*tm_struct_ptr).tm_min = result_tm.tm_min;
|
|
|
|
(*tm_struct_ptr).tm_hour = result_tm.tm_hour;
|
|
|
|
(*tm_struct_ptr).tm_mday = result_tm.tm_mday;
|
|
|
|
(*tm_struct_ptr).tm_mon = result_tm.tm_mon;
|
|
|
|
(*tm_struct_ptr).tm_year = result_tm.tm_year;
|
|
|
|
(*tm_struct_ptr).tm_wday = result_tm.tm_wday;
|
|
|
|
(*tm_struct_ptr).tm_yday = result_tm.tm_yday;
|
|
|
|
(*tm_struct_ptr).tm_isdst = result_tm.tm_isdst;
|
|
|
|
(*tm_struct_ptr).tm_gmtoff = result_tm.tm_gmtoff as i32;
|
|
|
|
(*tm_struct_ptr).tm_zone = copy_cstr_into_wasm(instance, result_tm.tm_zone) as i32;
|
2018-12-07 13:50:35 +00:00
|
|
|
|
2018-12-11 16:05:07 +00:00
|
|
|
tm_struct_offset as _
|
2018-11-26 22:27:56 +00:00
|
|
|
}
|
2018-12-07 13:50:35 +00:00
|
|
|
}
|
|
|
|
/// emscripten: _localtime_r
|
|
|
|
pub extern "C" fn _localtime_r(time_p: u32, result: u32, instance: &mut Instance) -> c_int {
|
|
|
|
debug!("emscripten::_localtime_r {}", time_p);
|
|
|
|
|
|
|
|
// NOTE: emscripten seems to want tzset() called in this function
|
|
|
|
// https://stackoverflow.com/questions/19170721/real-time-awareness-of-timezone-change-in-localtime-vs-localtime-r
|
2018-11-26 22:27:56 +00:00
|
|
|
|
2018-11-26 21:15:49 +00:00
|
|
|
unsafe {
|
|
|
|
let time_p_addr = instance.memory_offset_addr(0, time_p as _) as *mut i64;
|
2018-12-11 18:24:47 +00:00
|
|
|
let result_addr = instance.memory_offset_addr(0, result as _) as *mut guest_tm;
|
2018-12-08 00:38:51 +00:00
|
|
|
|
2018-12-07 13:50:35 +00:00
|
|
|
let mut result_tm = tm {
|
|
|
|
tm_sec: (*result_addr).tm_sec,
|
|
|
|
tm_min: (*result_addr).tm_min,
|
|
|
|
tm_hour: (*result_addr).tm_hour,
|
|
|
|
tm_mday: (*result_addr).tm_mday,
|
|
|
|
tm_mon: (*result_addr).tm_mon,
|
|
|
|
tm_year: (*result_addr).tm_year,
|
|
|
|
tm_wday: (*result_addr).tm_wday,
|
|
|
|
tm_yday: (*result_addr).tm_yday,
|
|
|
|
tm_isdst: (*result_addr).tm_isdst,
|
|
|
|
tm_gmtoff: (*result_addr).tm_gmtoff as _,
|
|
|
|
tm_zone: instance.memory_offset_addr(0, (*result_addr).tm_zone as _) as _,
|
|
|
|
};
|
|
|
|
|
2018-12-11 18:04:46 +00:00
|
|
|
localtime_r(time_p_addr, &mut result_tm);
|
2018-12-11 19:42:29 +00:00
|
|
|
// let tm_struct = result_tm;
|
|
|
|
// debug!(
|
|
|
|
// ">>>>>>> time = {}, {}, {}, {}, {}, {}, {}, {}",
|
|
|
|
// result_tm.tm_sec, result_tm.tm_min, result_tm.tm_hour, result_tm.tm_mday,
|
|
|
|
// result_tm.tm_mon, result_tm.tm_year, result_tm.tm_wday, result_tm.tm_yday,
|
|
|
|
// );
|
|
|
|
|
|
|
|
(*result_addr).tm_sec = result_tm.tm_sec;
|
|
|
|
(*result_addr).tm_min = result_tm.tm_min;
|
|
|
|
(*result_addr).tm_hour = result_tm.tm_hour;
|
|
|
|
(*result_addr).tm_mday = result_tm.tm_mday;
|
|
|
|
(*result_addr).tm_mon = result_tm.tm_mon;
|
|
|
|
(*result_addr).tm_year = result_tm.tm_year;
|
|
|
|
(*result_addr).tm_wday = result_tm.tm_wday;
|
|
|
|
(*result_addr).tm_yday = result_tm.tm_yday;
|
|
|
|
(*result_addr).tm_isdst = result_tm.tm_isdst;
|
|
|
|
(*result_addr).tm_gmtoff = result_tm.tm_gmtoff as _;
|
|
|
|
(*result_addr).tm_zone = copy_cstr_into_wasm(instance, result_tm.tm_zone) as _;
|
|
|
|
|
2018-12-11 18:04:46 +00:00
|
|
|
result as _
|
2018-11-26 21:15:49 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// emscripten: _time
|
|
|
|
pub extern "C" fn _time(time_p: u32, instance: &mut Instance) -> time_t {
|
|
|
|
debug!("emscripten::_time {}", time_p);
|
|
|
|
|
|
|
|
unsafe {
|
|
|
|
let time_p_addr = instance.memory_offset_addr(0, time_p as _) as *mut i64;
|
|
|
|
time(time_p_addr)
|
|
|
|
}
|
|
|
|
}
|
2018-11-26 22:16:51 +00:00
|
|
|
|
|
|
|
/// emscripten: _strftime
|
2018-11-27 04:29:26 +00:00
|
|
|
pub extern "C" fn _strftime(
|
|
|
|
s_ptr: c_int,
|
|
|
|
maxsize: u32,
|
|
|
|
format_ptr: c_int,
|
|
|
|
tm_ptr: c_int,
|
|
|
|
_instance: &mut Instance,
|
|
|
|
) -> time_t {
|
|
|
|
debug!(
|
|
|
|
"emscripten::_strftime {} {} {} {}",
|
|
|
|
s_ptr, maxsize, format_ptr, tm_ptr
|
|
|
|
);
|
2018-11-26 22:16:51 +00:00
|
|
|
0
|
|
|
|
}
|