/* * Copyright (c) 2020, Liav A. * * SPDX-License-Identifier: BSD-2-Clause */ #pragma once #include #include #include #include #include #include #include #include #include namespace Kernel { #define OPTIMAL_TICKS_PER_SECOND_RATE 250 #define OPTIMAL_PROFILE_TICKS_PER_SECOND_RATE 1000 class HardwareTimerBase; enum class TimePrecision { Coarse = 0, Precise }; class TimeManagement { AK_MAKE_ETERNAL; public: TimeManagement(); static void initialize(u32 cpu); static bool is_initialized(); static TimeManagement& the(); static bool is_valid_clock_id(clockid_t); Time current_time(clockid_t) const; Time monotonic_time(TimePrecision = TimePrecision::Coarse) const; Time monotonic_time_raw() const { // TODO: implement return monotonic_time(TimePrecision::Precise); } Time epoch_time(TimePrecision = TimePrecision::Precise) const; void set_epoch_time(Time); time_t ticks_per_second() const; time_t boot_time() const; bool is_system_timer(const HardwareTimerBase&) const; static void update_time(const RegisterState&); static void update_time_hpet(const RegisterState&); void increment_time_since_boot_hpet(); void increment_time_since_boot(); static bool is_hpet_periodic_mode_allowed(); bool enable_profile_timer(); bool disable_profile_timer(); u64 uptime_ms() const; static Time now(); // FIXME: Should use AK::Time internally // FIXME: Also, most likely broken, because it does not check m_update[12] for in-progress updates. timespec remaining_epoch_time_adjustment() const { return m_remaining_epoch_time_adjustment; } // FIXME: Should use AK::Time internally // FIXME: Also, most likely broken, because it does not check m_update[12] for in-progress updates. void set_remaining_epoch_time_adjustment(const timespec& adjustment) { m_remaining_epoch_time_adjustment = adjustment; } bool can_query_precise_time() const { return m_can_query_precise_time; } Memory::VMObject& time_page_vmobject(); private: TimePage* time_page(); void update_time_page(); bool probe_and_set_legacy_hardware_timers(); bool probe_and_set_non_legacy_hardware_timers(); Vector scan_and_initialize_periodic_timers(); Vector scan_for_non_periodic_timers(); NonnullRefPtrVector m_hardware_timers; void set_system_timer(HardwareTimerBase&); static void system_timer_tick(const RegisterState&); static u64 scheduling_current_time(bool); // Variables between m_update1 and m_update2 are synchronized Atomic m_update1 { 0 }; u32 m_ticks_this_second { 0 }; u64 m_seconds_since_boot { 0 }; // FIXME: Should use AK::Time internally timespec m_epoch_time { 0, 0 }; timespec m_remaining_epoch_time_adjustment { 0, 0 }; Atomic m_update2 { 0 }; u32 m_time_ticks_per_second { 0 }; // may be different from interrupts/second (e.g. hpet) bool m_can_query_precise_time { false }; bool m_updating_time { false }; // may only be accessed from the BSP! RefPtr m_system_timer; RefPtr m_time_keeper_timer; Atomic m_profile_enable_count { 0 }; RefPtr m_profile_timer; OwnPtr m_time_page_region; }; }