mod hardware_timer;
mod rtc;
mod tsc;
use core::fmt;
use alloc::{sync::Arc, vec::Vec};
use tracing::info;
use crate::{
acpi::tables::{self, BiosTables, Facp},
cpu,
sync::{once::OnceLock, spin::rwlock::RwLock},
};
use self::rtc::Rtc;
pub const NANOS_PER_SEC: u64 = 1_000_000_000;
pub const FEMTOS_PER_SEC: u64 = 1_000_000_000_000_000;
pub const NANOS_PER_FEMTO: u64 = 1_000_000;
static CLOCKS: OnceLock<Clock> = OnceLock::new();
pub fn clocks() -> &'static Clock {
CLOCKS.get()
}
#[derive(Debug, Default, Clone, Copy, PartialEq, Eq)]
pub struct ClockTime {
pub nanoseconds: u64,
pub seconds: u64,
}
#[allow(dead_code)]
impl ClockTime {
pub fn as_nanos(&self) -> u64 {
self.seconds * NANOS_PER_SEC + self.nanoseconds
}
}
impl Ord for ClockTime {
fn cmp(&self, other: &Self) -> core::cmp::Ordering {
self.seconds
.cmp(&other.seconds)
.then(self.nanoseconds.cmp(&other.nanoseconds))
}
}
impl PartialOrd for ClockTime {
fn partial_cmp(&self, other: &Self) -> Option<core::cmp::Ordering> {
Some(self.cmp(other))
}
}
impl core::ops::Sub for ClockTime {
type Output = Self;
fn sub(self, rhs: Self) -> Self::Output {
let nanoseconds = if self.nanoseconds < rhs.nanoseconds {
self.nanoseconds + 1_000_000_000 - rhs.nanoseconds
} else {
self.nanoseconds - rhs.nanoseconds
};
let seconds = self.seconds
- rhs.seconds
- if self.nanoseconds < rhs.nanoseconds {
1
} else {
0
};
Self {
nanoseconds,
seconds,
}
}
}
impl core::ops::Add for ClockTime {
type Output = Self;
fn add(self, rhs: Self) -> Self::Output {
let nanoseconds = self.nanoseconds + rhs.nanoseconds;
let seconds = self.seconds + rhs.seconds + nanoseconds / 1_000_000_000;
Self {
nanoseconds: nanoseconds % 1_000_000_000,
seconds,
}
}
}
impl core::ops::AddAssign for ClockTime {
fn add_assign(&mut self, rhs: Self) {
*self = *self + rhs;
}
}
trait ClockDevice: Send + Sync {
fn name(&self) -> &'static str;
fn get_time(&self) -> ClockTime;
fn granularity(&self) -> u64;
fn require_calibration(&self) -> bool;
fn rating(&self) -> u64 {
1
}
}
struct SystemTime {
start_unix: ClockTime,
last_tick: ClockTime,
startup_offset: ClockTime,
device: Option<Arc<dyn ClockDevice>>,
}
impl SystemTime {
fn new(rtc: &Rtc) -> Self {
let time = rtc.get_time();
let timestamp = time.seconds_since_unix_epoch().expect("Must be after 1970");
info!("Time now: {time} - UTC");
info!("System start timestamp: {}", timestamp);
let start_unix = ClockTime {
nanoseconds: 0,
seconds: timestamp,
};
Self {
start_unix,
last_tick: ClockTime {
nanoseconds: 0,
seconds: 0,
},
startup_offset: ClockTime {
nanoseconds: 0,
seconds: 0,
},
device: None,
}
}
fn tick(&mut self) {
if let Some(device) = &self.device {
let time = device.get_time();
let diff = time - self.last_tick;
self.startup_offset += diff;
self.last_tick = time;
}
}
fn update_device(&mut self, device: Arc<dyn ClockDevice>, rtc: &Rtc) {
if let Some(current_device) = &self.device {
if Arc::ptr_eq(&device, current_device) {
return;
}
let time = current_device.get_time();
let new_time = device.get_time();
let diff = time - self.last_tick;
self.startup_offset += diff;
self.device = Some(device);
self.last_tick = new_time
} else {
cpu::cpu().push_cli();
let mut rtc_time = rtc.get_time();
loop {
let new_rtc_time = rtc.get_time();
if new_rtc_time.seconds != rtc_time.seconds {
rtc_time = new_rtc_time;
break;
}
}
let device_time = device.get_time();
let timestamp = rtc_time
.seconds_since_unix_epoch()
.expect("Must be after 1970");
info!("Adjusted Time now: {rtc_time} - UTC");
info!("Adjusted System start timestamp: {}", timestamp);
self.last_tick = device_time;
self.device = Some(device);
self.startup_offset = ClockTime {
nanoseconds: 0,
seconds: 0,
};
self.start_unix = ClockTime {
nanoseconds: 0,
seconds: rtc_time
.seconds_since_unix_epoch()
.expect("Must be after 1970"),
};
cpu::cpu().pop_cli();
}
}
fn time_since_startup(&self) -> ClockTime {
self.startup_offset
}
fn time_since_unix_epoch(&self) -> ClockTime {
self.start_unix + self.startup_offset
}
}
#[allow(dead_code)]
pub struct Clock {
devices: RwLock<Vec<Arc<dyn ClockDevice>>>,
rtc: Rtc,
system_time: RwLock<SystemTime>,
}
impl fmt::Debug for Clock {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
f.debug_struct("Clock").finish()
}
}
impl Clock {
fn new(rtc: Rtc) -> Self {
Self {
devices: RwLock::new(Vec::new()),
system_time: RwLock::new(SystemTime::new(&rtc)),
rtc,
}
}
fn add_device(&self, device: Arc<dyn ClockDevice>) {
info!(
"Adding clock device: {}, rating: {}",
device.name(),
device.rating()
);
let mut devs = self.devices.write();
devs.push(device);
devs.sort_unstable_by_key(|device| -(device.rating() as i64));
self.system_time
.write()
.update_device(devs[0].clone(), &self.rtc);
}
#[allow(dead_code)]
fn get_best_clock(&self) -> Option<Arc<dyn ClockDevice>> {
self.devices.read().first().map(Arc::clone)
}
fn get_best_for_calibration(&self) -> Option<Arc<dyn ClockDevice>> {
self.devices
.read()
.iter()
.find(|device| !device.require_calibration())
.map(Arc::clone)
}
#[allow(dead_code)]
pub fn tick_system_time(&self) {
self.system_time.write().tick();
}
#[allow(dead_code)]
pub fn time_since_startup(&self) -> ClockTime {
let mut time = self.system_time.write();
time.tick();
time.time_since_startup()
}
#[allow(dead_code)]
pub fn time_since_unix_epoch(&self) -> ClockTime {
let mut time = self.system_time.write();
time.tick();
time.time_since_unix_epoch()
}
}
pub fn init(bios_tables: &BiosTables) {
let facp = bios_tables.rsdt.get_table::<Facp>();
let century_reg = facp.map(|facp| facp.century);
CLOCKS
.set(Clock::new(Rtc::new(century_reg)))
.expect("Clock is already initialized");
let hpet_table = bios_tables.rsdt.get_table::<tables::Hpet>();
let hardware_timer = hardware_timer::HardwareTimer::init(hpet_table);
clocks().add_device(hardware_timer);
if let Some(tsc) = tsc::Tsc::new(
clocks()
.get_best_for_calibration()
.expect("Have a clock that can be used as a base for TSC calibration")
.as_ref(),
) {
clocks().add_device(Arc::new(tsc));
}
}