summaryrefslogtreecommitdiffstats
path: root/third_party/rust/zeitstempel/src/fallback.rs
blob: 157f4c1a77e4e882fd62347a187180e3bf3bd33c (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
use std::convert::TryInto;
use std::time::Instant;

use once_cell::sync::Lazy;

static INIT_TIME: Lazy<Instant> = Lazy::new(Instant::now);

pub fn now_including_suspend() -> u64 {
    // For Windows:
    // Instead of relying on figuring out the underlying functions,
    // we can rely on the fact that `Instant::now` maps to [QueryPerformanceCounter] on Windows,
    // so by comparing it to another arbitrary timestamp we will get a duration that will include
    // suspend time.
    // If we use that as a timestamp we can compare later timestamps to it and that will also
    // include suspend time.
    //
    // [QueryPerformanceCounter]: https://docs.microsoft.com/en-us/windows/win32/api/profileapi/nf-profileapi-queryperformancecounter
    //
    // For other operating systems:
    // This fallback is not used on Linux, where it maps to `CLOCK_MONOTONIC`, which does NOT
    // include suspend time. But we don't use it there, so no problem.
    //
    // This fallback is not used on macOS, where it maps to `mach_absolute_time`, which does NOT
    // include suspend time. But we don't use it there, so no problem.
    //
    // For other operating systems we make no guarantees, other than that we won't panic.
    let now = Instant::now();
    now.checked_duration_since(*INIT_TIME)
        .and_then(|diff| diff.as_nanos().try_into().ok())
        .unwrap_or(0)
}