luaur-vm 0.1.4

The Luau register virtual machine and standard library (Rust).
Documentation
#[allow(non_camel_case_types)]
pub type time_t = i64;

#[repr(C)]
#[derive(Debug, Copy, Clone)]
#[allow(non_camel_case_types)]
pub struct tm {
    pub tm_sec: core::ffi::c_int,
    pub tm_min: core::ffi::c_int,
    pub tm_hour: core::ffi::c_int,
    pub tm_mday: core::ffi::c_int,
    pub tm_mon: core::ffi::c_int,
    pub tm_year: core::ffi::c_int,
    pub tm_wday: core::ffi::c_int,
    pub tm_yday: core::ffi::c_int,
    pub tm_isdst: core::ffi::c_int,
}

pub fn os_timegm(timep: *const tm) -> time_t {
    let timep = unsafe { &*timep };

    // Compute the Julian-day arithmetic in i64. The `tm_*` fields are C `int`
    // (i32), and an out-of-range date table from Lua (e.g. `os.time{year = 2^31}`)
    // makes intermediate products like `365 * y` and `julianday * 86400` exceed
    // i32 — with overflow-checks on (debug / the fuzz profile) that aborts; in
    // plain release it silently wraps to a garbage timestamp. i64 holds every
    // value reachable from i32 inputs, so the result is well-defined for valid
    // dates (identical) and never overflows for hostile ones. (Found by the fuzzer
    // running a mutated datetime conformance test.)
    let day = timep.tm_mday as i64;
    let month = (timep.tm_mon as i64) + 1;
    let year = (timep.tm_year as i64) + 1900;

    let a: i64 = if timep.tm_mon % 12 < 2 { 1 } else { 0 };
    let a = a - (timep.tm_mon as i64 / 12);

    let y = year + 4800 - a;
    let m = month + (12 * a) - 3;

    let julianday = day + ((153 * m + 2) / 5) + (365 * y) + (y / 4) - (y / 100) + (y / 400) - 32045;

    let utcstartasjulianday: i64 = 2440588;
    let utcstartasjuliansecond: i64 = utcstartasjulianday * 86400;

    if julianday < utcstartasjulianday {
        return -1;
    }

    let daysecond =
        (timep.tm_hour as i64) * 3600 + (timep.tm_min as i64) * 60 + (timep.tm_sec as i64);
    let julianseconds = julianday * 86400 + daysecond;

    if julianseconds < utcstartasjuliansecond {
        return -1;
    }

    julianseconds - utcstartasjuliansecond
}