From f4feb729918b1e04a0d04103359d6cb2ecd24fec Mon Sep 17 00:00:00 2001 From: David Venhoek Date: Thu, 14 Sep 2023 15:46:24 +0200 Subject: [PATCH 1/4] Fixed bug in opening ptp hardware clocks. --- src/unix.rs | 6 ++---- 1 file changed, 2 insertions(+), 4 deletions(-) diff --git a/src/unix.rs b/src/unix.rs index ed53bac..d572672 100644 --- a/src/unix.rs +++ b/src/unix.rs @@ -1,7 +1,6 @@ use crate::{Clock, LeapIndicator, TimeOffset, Timestamp}; use std::{ - os::unix::io::IntoRawFd, - os::unix::prelude::{FromRawFd, RawFd}, + os::unix::io::{FromRawFd, IntoRawFd, RawFd}, path::Path, time::Duration, }; @@ -47,12 +46,11 @@ impl UnixClock { /// ``` pub fn open(path: impl AsRef) -> std::io::Result { let file = std::fs::File::open(path)?; - // we need an owned fd. the file will be closed when the process exits. Ok(Self::safe_from_raw_fd(file.into_raw_fd())) } - // Consume the fd and produce a clock id. Clock id is only valid + // Consume an fd and produce a clock id. Clock id is only valid // so long as the fd is open, so the RawFd here should // not be borrowed. fn safe_from_raw_fd(fd: RawFd) -> Self { From 1ab191a1bc8a30bea90f896f0d894435497b73bb Mon Sep 17 00:00:00 2001 From: David Venhoek Date: Thu, 21 Sep 2023 10:48:50 +0200 Subject: [PATCH 2/4] Added call for measuring offset to system clock. --- src/unix.rs | 81 ++++++++++++++++++++++++++++++++++++++++++++++++++++- 1 file changed, 80 insertions(+), 1 deletion(-) diff --git a/src/unix.rs b/src/unix.rs index d572672..dec4a46 100644 --- a/src/unix.rs +++ b/src/unix.rs @@ -9,6 +9,7 @@ use std::{ #[derive(Debug, Clone, Copy)] pub struct UnixClock { clock: libc::clockid_t, + fd: Option, } impl UnixClock { @@ -28,6 +29,7 @@ impl UnixClock { /// ``` pub const CLOCK_REALTIME: Self = UnixClock { clock: libc::CLOCK_REALTIME, + fd: None, }; /// Open a clock device. @@ -56,7 +58,84 @@ impl UnixClock { fn safe_from_raw_fd(fd: RawFd) -> Self { let clock = ((!(fd as libc::clockid_t)) << 3) | 3; - Self { clock } + Self { + clock, + fd: Some(fd), + } + } + + /// Determine offset between file clock and system clock (if any) + /// Returns two system timestamps sandwhiching a timestamp from the + /// hardware clock. + #[cfg(target_os = "linux")] + pub fn system_offset(&self) -> Result<(Timestamp, Timestamp, Timestamp), Error> { + let Some(fd) = self.fd else { + return Err(Error::Invalid); + }; + + // TODO: remove type and constant definitions once libc updates + #[repr(C)] + #[derive(Default, Clone, Copy, PartialEq, Eq)] + #[allow(non_camel_case_types)] + struct ptp_clock_time { + sec: libc::__s64, + nsec: libc::__u32, + reserved: libc::__u32, + } + + #[repr(C)] + #[derive(Clone, Copy, PartialEq, Eq)] + #[allow(non_camel_case_types)] + struct ptp_sys_offset { + n_samples: libc::c_uint, + rsv: [libc::c_uint; 3], + ts: [ptp_clock_time; 51], + } + + // Needed as arrays larger than 32 elements don't implement default. + impl Default for ptp_sys_offset { + fn default() -> Self { + Self { + n_samples: Default::default(), + rsv: Default::default(), + ts: [Default::default(); 51], + } + } + } + + const PTP_SYS_OFFSET: libc::c_uint = 0x43403d05; + + let mut offset = ptp_sys_offset { + n_samples: 1, + ..Default::default() + }; + + // # Safety + // Safe since PTP_SYS_OFFSET expects as argument a mutable pointer to + // ptp_sys_offset and offset is valid during the call + if unsafe { libc::ioctl(fd, PTP_SYS_OFFSET as _, &mut offset as *mut ptp_sys_offset) } != 0 + { + let t1 = Self::CLOCK_REALTIME.now(); + let tp = self.now(); + let t2 = Self::CLOCK_REALTIME.now(); + + Ok((t1?, tp?, t2?)) + } else { + Ok(( + Timestamp { + seconds: offset.ts[0].sec as _, + nanos: offset.ts[0].nsec as _, + }, + Timestamp { + seconds: offset.ts[1].sec as _, + nanos: offset.ts[1].nsec as _, + }, + Timestamp { + seconds: offset.ts[2].sec as _, + nanos: offset.ts[2].nsec as _, + }, + )) + } } fn clock_adjtime(&self, timex: &mut libc::timex) -> Result<(), Error> { From 4b15caa2672d7efcde3124014354e30ee63c5a31 Mon Sep 17 00:00:00 2001 From: David Venhoek Date: Thu, 28 Sep 2023 16:24:51 +0200 Subject: [PATCH 3/4] Fixed incorrect permissions on clocks from files. --- src/unix.rs | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/src/unix.rs b/src/unix.rs index dec4a46..3898673 100644 --- a/src/unix.rs +++ b/src/unix.rs @@ -47,7 +47,11 @@ impl UnixClock { /// } /// ``` pub fn open(path: impl AsRef) -> std::io::Result { - let file = std::fs::File::open(path)?; + let file = std::fs::OpenOptions::new() + .write(true) + .read(true) + .open(path)?; + // we need an owned fd. the file will be closed when the process exits. Ok(Self::safe_from_raw_fd(file.into_raw_fd())) } From 4bb24bb3c533acf7df9c1f0ae6fc4c35aa223b44 Mon Sep 17 00:00:00 2001 From: David Venhoek Date: Wed, 4 Oct 2023 11:31:35 +0200 Subject: [PATCH 4/4] Hide file descriptor on non-linux platforms. --- src/unix.rs | 15 +++++++-------- 1 file changed, 7 insertions(+), 8 deletions(-) diff --git a/src/unix.rs b/src/unix.rs index 3898673..d85e3a0 100644 --- a/src/unix.rs +++ b/src/unix.rs @@ -1,14 +1,16 @@ use crate::{Clock, LeapIndicator, TimeOffset, Timestamp}; +use std::time::Duration; +#[cfg(target_os = "linux")] use std::{ - os::unix::io::{FromRawFd, IntoRawFd, RawFd}, + os::unix::io::{IntoRawFd, RawFd}, path::Path, - time::Duration, }; /// A Unix OS clock #[derive(Debug, Clone, Copy)] pub struct UnixClock { clock: libc::clockid_t, + #[cfg(target_os = "linux")] fd: Option, } @@ -29,6 +31,7 @@ impl UnixClock { /// ``` pub const CLOCK_REALTIME: Self = UnixClock { clock: libc::CLOCK_REALTIME, + #[cfg(target_os = "linux")] fd: None, }; @@ -46,6 +49,7 @@ impl UnixClock { /// Ok(()) /// } /// ``` + #[cfg(target_os = "linux")] pub fn open(path: impl AsRef) -> std::io::Result { let file = std::fs::OpenOptions::new() .write(true) @@ -59,6 +63,7 @@ impl UnixClock { // Consume an fd and produce a clock id. Clock id is only valid // so long as the fd is open, so the RawFd here should // not be borrowed. + #[cfg(target_os = "linux")] fn safe_from_raw_fd(fd: RawFd) -> Self { let clock = ((!(fd as libc::clockid_t)) << 3) | 3; @@ -430,12 +435,6 @@ impl UnixClock { } } -impl FromRawFd for UnixClock { - unsafe fn from_raw_fd(fd: RawFd) -> Self { - Self::safe_from_raw_fd(fd) - } -} - impl Clock for UnixClock { type Error = Error;