use raw_cpuid::CpuId;
#[cfg(feature = "irq")]
use int_ratio::Ratio;
#[cfg(feature = "irq")]
const LAPIC_TICKS_PER_SEC: u64 = 1_000_000_000; #[cfg(feature = "irq")]
static mut NANOS_TO_LAPIC_TICKS_RATIO: Ratio = Ratio::zero();
static mut INIT_TICK: u64 = 0;
static mut CPU_FREQ_MHZ: u64 = axconfig::TIMER_FREQUENCY as u64 / 1_000_000;
static mut RTC_EPOCHOFFSET_NANOS: u64 = 0;
pub fn current_ticks() -> u64 {
unsafe { core::arch::x86_64::_rdtsc() - INIT_TICK }
}
pub fn ticks_to_nanos(ticks: u64) -> u64 {
ticks * 1_000 / unsafe { CPU_FREQ_MHZ }
}
pub fn nanos_to_ticks(nanos: u64) -> u64 {
nanos * unsafe { CPU_FREQ_MHZ } / 1_000
}
pub fn epochoffset_nanos() -> u64 {
unsafe { RTC_EPOCHOFFSET_NANOS }
}
#[cfg(feature = "irq")]
pub fn set_oneshot_timer(deadline_ns: u64) {
let lapic = super::apic::local_apic();
let now_ns = crate::time::monotonic_time_nanos();
unsafe {
if now_ns < deadline_ns {
let apic_ticks = NANOS_TO_LAPIC_TICKS_RATIO.mul_trunc(deadline_ns - now_ns);
assert!(apic_ticks <= u32::MAX as u64);
lapic.set_timer_initial(apic_ticks.max(1) as u32);
} else {
lapic.set_timer_initial(1);
}
}
}
pub(super) fn init_early() {
if let Some(freq) = CpuId::new()
.get_processor_frequency_info()
.map(|info| info.processor_base_frequency())
{
if freq > 0 {
axlog::ax_println!("Got TSC frequency by CPUID: {} MHz", freq);
unsafe { CPU_FREQ_MHZ = freq as u64 }
}
}
unsafe {
INIT_TICK = core::arch::x86_64::_rdtsc();
}
#[cfg(feature = "rtc")]
{
use x86_rtc::Rtc;
let eopch_time_nanos = Rtc::new().get_unix_timestamp() * 1_000_000_000;
unsafe {
RTC_EPOCHOFFSET_NANOS = eopch_time_nanos - ticks_to_nanos(INIT_TICK);
}
}
}
pub(super) fn init_primary() {
#[cfg(feature = "irq")]
unsafe {
use x2apic::lapic::{TimerDivide, TimerMode};
let lapic = super::apic::local_apic();
lapic.set_timer_mode(TimerMode::OneShot);
lapic.set_timer_divide(TimerDivide::Div256); lapic.enable_timer();
NANOS_TO_LAPIC_TICKS_RATIO = Ratio::new(
LAPIC_TICKS_PER_SEC as u32,
crate::time::NANOS_PER_SEC as u32,
);
}
}
#[cfg(feature = "smp")]
pub(super) fn init_secondary() {
#[cfg(feature = "irq")]
unsafe {
super::apic::local_apic().enable_timer();
}
}