1use crate::sys::time::TimeSpec;
3#[cfg(any(freebsdlike, linux_android, target_os = "emscripten"))]
4#[cfg(feature = "process")]
5use crate::unistd::Pid;
6use crate::{Errno, Result};
7use libc::{self, clockid_t};
8use std::mem::MaybeUninit;
9
10#[derive(Debug, Copy, Clone, Eq, PartialEq, Ord, PartialOrd, Hash)]
14pub struct ClockId(clockid_t);
15
16impl ClockId {
17 pub const fn from_raw(clk_id: clockid_t) -> Self {
19 ClockId(clk_id)
20 }
21
22 feature! {
23 #![feature = "process"]
24 #[cfg(any(freebsdlike, linux_android, target_os = "emscripten"))]
26 pub fn pid_cpu_clock_id(pid: Pid) -> Result<Self> {
27 clock_getcpuclockid(pid)
28 }
29 }
30
31 #[cfg(not(target_os = "redox"))]
33 pub fn res(self) -> Result<TimeSpec> {
34 clock_getres(self)
35 }
36
37 pub fn now(self) -> Result<TimeSpec> {
39 clock_gettime(self)
40 }
41
42 #[cfg(not(any(
44 target_os = "ios",
45 target_os = "tvos",
46 target_os = "watchos",
47 target_os = "redox",
48 target_os = "hermit"
49 )))]
50 pub fn set_time(self, timespec: TimeSpec) -> Result<()> {
51 clock_settime(self, timespec)
52 }
53
54 pub const fn as_raw(self) -> clockid_t {
56 self.0
57 }
58
59 #[cfg(any(linux_android, target_os = "emscripten", target_os = "fuchsia"))]
60 pub const CLOCK_BOOTTIME: ClockId = ClockId(libc::CLOCK_BOOTTIME);
63 #[cfg(any(linux_android, target_os = "emscripten", target_os = "fuchsia"))]
66 pub const CLOCK_BOOTTIME_ALARM: ClockId =
67 ClockId(libc::CLOCK_BOOTTIME_ALARM);
68 pub const CLOCK_MONOTONIC: ClockId = ClockId(libc::CLOCK_MONOTONIC);
70 #[cfg(any(linux_android, target_os = "emscripten", target_os = "fuchsia"))]
72 pub const CLOCK_MONOTONIC_COARSE: ClockId =
73 ClockId(libc::CLOCK_MONOTONIC_COARSE);
74 #[cfg(freebsdlike)]
75 pub const CLOCK_MONOTONIC_FAST: ClockId =
77 ClockId(libc::CLOCK_MONOTONIC_FAST);
78 #[cfg(freebsdlike)]
79 pub const CLOCK_MONOTONIC_PRECISE: ClockId =
81 ClockId(libc::CLOCK_MONOTONIC_PRECISE);
82 #[cfg(any(linux_android, target_os = "emscripten", target_os = "fuchsia"))]
85 pub const CLOCK_MONOTONIC_RAW: ClockId = ClockId(libc::CLOCK_MONOTONIC_RAW);
86 #[cfg(any(
87 linux_android,
88 apple_targets,
89 freebsdlike,
90 target_os = "emscripten",
91 target_os = "fuchsia",
92 target_os = "redox",
93 ))]
94 pub const CLOCK_PROCESS_CPUTIME_ID: ClockId =
96 ClockId(libc::CLOCK_PROCESS_CPUTIME_ID);
97 #[cfg(freebsdlike)]
98 pub const CLOCK_PROF: ClockId = ClockId(libc::CLOCK_PROF);
100 pub const CLOCK_REALTIME: ClockId = ClockId(libc::CLOCK_REALTIME);
102 #[cfg(any(linux_android, target_os = "emscripten", target_os = "fuchsia"))]
104 pub const CLOCK_REALTIME_ALARM: ClockId =
105 ClockId(libc::CLOCK_REALTIME_ALARM);
106 #[cfg(any(linux_android, target_os = "emscripten", target_os = "fuchsia"))]
108 pub const CLOCK_REALTIME_COARSE: ClockId =
109 ClockId(libc::CLOCK_REALTIME_COARSE);
110 #[cfg(freebsdlike)]
111 pub const CLOCK_REALTIME_FAST: ClockId = ClockId(libc::CLOCK_REALTIME_FAST);
113 #[cfg(freebsdlike)]
114 pub const CLOCK_REALTIME_PRECISE: ClockId =
116 ClockId(libc::CLOCK_REALTIME_PRECISE);
117 #[cfg(freebsdlike)]
118 pub const CLOCK_SECOND: ClockId = ClockId(libc::CLOCK_SECOND);
121 #[allow(missing_docs)] #[cfg(any(
123 target_os = "emscripten",
124 target_os = "fuchsia",
125 all(target_os = "linux", target_env = "musl")
126 ))]
127 pub const CLOCK_SGI_CYCLE: ClockId = ClockId(libc::CLOCK_SGI_CYCLE);
128 #[cfg(any(linux_android, target_os = "emscripten", target_os = "fuchsia"))]
132 pub const CLOCK_TAI: ClockId = ClockId(libc::CLOCK_TAI);
133 #[cfg(any(
134 linux_android,
135 apple_targets,
136 freebsdlike,
137 target_os = "emscripten",
138 target_os = "fuchsia",
139 ))]
140 pub const CLOCK_THREAD_CPUTIME_ID: ClockId =
142 ClockId(libc::CLOCK_THREAD_CPUTIME_ID);
143 #[cfg(freebsdlike)]
144 pub const CLOCK_UPTIME: ClockId = ClockId(libc::CLOCK_UPTIME);
147 #[cfg(freebsdlike)]
148 pub const CLOCK_UPTIME_FAST: ClockId = ClockId(libc::CLOCK_UPTIME_FAST);
150 #[cfg(freebsdlike)]
151 pub const CLOCK_UPTIME_PRECISE: ClockId =
153 ClockId(libc::CLOCK_UPTIME_PRECISE);
154 #[cfg(freebsdlike)]
155 pub const CLOCK_VIRTUAL: ClockId = ClockId(libc::CLOCK_VIRTUAL);
157}
158
159impl From<ClockId> for clockid_t {
160 fn from(clock_id: ClockId) -> Self {
161 clock_id.as_raw()
162 }
163}
164
165impl From<clockid_t> for ClockId {
166 fn from(clk_id: clockid_t) -> Self {
167 ClockId::from_raw(clk_id)
168 }
169}
170
171impl std::fmt::Display for ClockId {
172 fn fmt(&self, f: &mut std::fmt::Formatter) -> std::fmt::Result {
173 std::fmt::Display::fmt(&self.0, f)
174 }
175}
176
177#[cfg(not(target_os = "redox"))]
180pub fn clock_getres(clock_id: ClockId) -> Result<TimeSpec> {
181 let mut c_time: MaybeUninit<libc::timespec> = MaybeUninit::uninit();
182 let ret =
183 unsafe { libc::clock_getres(clock_id.as_raw(), c_time.as_mut_ptr()) };
184 Errno::result(ret)?;
185 let res = unsafe { c_time.assume_init() };
186 Ok(TimeSpec::from(res))
187}
188
189pub fn clock_gettime(clock_id: ClockId) -> Result<TimeSpec> {
192 let mut c_time: MaybeUninit<libc::timespec> = MaybeUninit::uninit();
193 let ret =
194 unsafe { libc::clock_gettime(clock_id.as_raw(), c_time.as_mut_ptr()) };
195 Errno::result(ret)?;
196 let res = unsafe { c_time.assume_init() };
197 Ok(TimeSpec::from(res))
198}
199
200#[cfg(not(any(
203 target_os = "ios",
204 target_os = "tvos",
205 target_os = "watchos",
206 target_os = "redox",
207 target_os = "hermit"
208)))]
209pub fn clock_settime(clock_id: ClockId, timespec: TimeSpec) -> Result<()> {
210 let ret =
211 unsafe { libc::clock_settime(clock_id.as_raw(), timespec.as_ref()) };
212 Errno::result(ret).map(drop)
213}
214
215#[cfg(any(freebsdlike, linux_android, target_os = "emscripten"))]
218#[cfg(feature = "process")]
219#[cfg_attr(docsrs, doc(cfg(feature = "process")))]
220pub fn clock_getcpuclockid(pid: Pid) -> Result<ClockId> {
221 let mut clk_id: MaybeUninit<libc::clockid_t> = MaybeUninit::uninit();
222 let ret =
223 unsafe { libc::clock_getcpuclockid(pid.into(), clk_id.as_mut_ptr()) };
224 if ret == 0 {
225 let res = unsafe { clk_id.assume_init() };
226 Ok(ClockId::from(res))
227 } else {
228 Err(Errno::from_raw(ret))
229 }
230}
231
232#[cfg(any(
233 linux_android,
234 solarish,
235 freebsdlike,
236 target_os = "netbsd",
237 target_os = "hurd",
238 target_os = "aix"
239))]
240libc_bitflags! {
241 pub struct ClockNanosleepFlags: libc::c_int {
243 TIMER_ABSTIME;
246 }
247}
248
249#[cfg(any(
259 linux_android,
260 solarish,
261 freebsdlike,
262 target_os = "netbsd",
263 target_os = "hurd",
264 target_os = "aix"
265))]
266pub fn clock_nanosleep(
267 clock_id: ClockId,
268 flags: ClockNanosleepFlags,
269 request: &TimeSpec,
270) -> Result<TimeSpec> {
271 let mut remain = TimeSpec::new(0, 0);
272 let ret = unsafe {
273 libc::clock_nanosleep(
274 clock_id.as_raw(),
275 flags.bits(),
276 request.as_ref() as *const _,
277 remain.as_mut() as *mut _,
278 )
279 };
280 if ret == 0 {
281 Ok(remain)
282 } else {
283 Err(Errno::from_raw(ret))
284 }
285}