committed by
GitHub
13 changed files with 290 additions and 109 deletions
-
6src/core/CMakeLists.txt
-
1src/core/hle/kernel/k_address_arbiter.cpp
-
74src/core/hle/kernel/k_hardware_timer.cpp
-
54src/core/hle/kernel/k_hardware_timer.h
-
92src/core/hle/kernel/k_hardware_timer_base.h
-
4src/core/hle/kernel/k_scoped_scheduler_lock_and_sleep.h
-
8src/core/hle/kernel/k_thread.h
-
6src/core/hle/kernel/k_thread_queue.cpp
-
40src/core/hle/kernel/k_timer_task.h
-
20src/core/hle/kernel/kernel.cpp
-
9src/core/hle/kernel/kernel.h
-
44src/core/hle/kernel/time_manager.cpp
-
41src/core/hle/kernel/time_manager.h
@ -0,0 +1,74 @@ |
|||||
|
// SPDX-FileCopyrightText: Copyright 2022 yuzu Emulator Project
|
||||
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||
|
|
||||
|
#include "core/core.h"
|
||||
|
#include "core/core_timing.h"
|
||||
|
#include "core/hle/kernel/k_hardware_timer.h"
|
||||
|
#include "core/hle/kernel/k_scheduler.h"
|
||||
|
|
||||
|
namespace Kernel { |
||||
|
|
||||
|
void KHardwareTimer::Initialize() { |
||||
|
// Create the timing callback to register with CoreTiming.
|
||||
|
m_event_type = Core::Timing::CreateEvent( |
||||
|
"KHardwareTimer::Callback", [](std::uintptr_t timer_handle, s64, std::chrono::nanoseconds) { |
||||
|
reinterpret_cast<KHardwareTimer*>(timer_handle)->DoTask(); |
||||
|
return std::nullopt; |
||||
|
}); |
||||
|
} |
||||
|
|
||||
|
void KHardwareTimer::Finalize() { |
||||
|
this->DisableInterrupt(); |
||||
|
m_event_type.reset(); |
||||
|
} |
||||
|
|
||||
|
void KHardwareTimer::DoTask() { |
||||
|
// Handle the interrupt.
|
||||
|
{ |
||||
|
KScopedSchedulerLock slk{m_kernel}; |
||||
|
KScopedSpinLock lk(this->GetLock()); |
||||
|
|
||||
|
//! Ignore this event if needed.
|
||||
|
if (!this->GetInterruptEnabled()) { |
||||
|
return; |
||||
|
} |
||||
|
|
||||
|
// Disable the timer interrupt while we handle this.
|
||||
|
this->DisableInterrupt(); |
||||
|
|
||||
|
if (const s64 next_time = this->DoInterruptTaskImpl(GetTick()); |
||||
|
0 < next_time && next_time <= m_wakeup_time) { |
||||
|
// We have a next time, so we should set the time to interrupt and turn the interrupt
|
||||
|
// on.
|
||||
|
this->EnableInterrupt(next_time); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
// Clear the timer interrupt.
|
||||
|
// Kernel::GetInterruptManager().ClearInterrupt(KInterruptName_NonSecurePhysicalTimer,
|
||||
|
// GetCurrentCoreId());
|
||||
|
} |
||||
|
|
||||
|
void KHardwareTimer::EnableInterrupt(s64 wakeup_time) { |
||||
|
this->DisableInterrupt(); |
||||
|
|
||||
|
m_wakeup_time = wakeup_time; |
||||
|
m_kernel.System().CoreTiming().ScheduleEvent(std::chrono::nanoseconds{m_wakeup_time}, |
||||
|
m_event_type, reinterpret_cast<uintptr_t>(this), |
||||
|
true); |
||||
|
} |
||||
|
|
||||
|
void KHardwareTimer::DisableInterrupt() { |
||||
|
m_kernel.System().CoreTiming().UnscheduleEvent(m_event_type, reinterpret_cast<uintptr_t>(this)); |
||||
|
m_wakeup_time = std::numeric_limits<s64>::max(); |
||||
|
} |
||||
|
|
||||
|
s64 KHardwareTimer::GetTick() const { |
||||
|
return m_kernel.System().CoreTiming().GetGlobalTimeNs().count(); |
||||
|
} |
||||
|
|
||||
|
bool KHardwareTimer::GetInterruptEnabled() { |
||||
|
return m_wakeup_time != std::numeric_limits<s64>::max(); |
||||
|
} |
||||
|
|
||||
|
} // namespace Kernel
|
||||
@ -0,0 +1,54 @@ |
|||||
|
// SPDX-FileCopyrightText: Copyright 2022 yuzu Emulator Project |
||||
|
// SPDX-License-Identifier: GPL-2.0-or-later |
||||
|
|
||||
|
#pragma once |
||||
|
|
||||
|
#include "core/hle/kernel/k_hardware_timer_base.h" |
||||
|
|
||||
|
namespace Core::Timing { |
||||
|
struct EventType; |
||||
|
} // namespace Core::Timing |
||||
|
|
||||
|
namespace Kernel { |
||||
|
|
||||
|
class KHardwareTimer : /* public KInterruptTask, */ public KHardwareTimerBase { |
||||
|
public: |
||||
|
explicit KHardwareTimer(KernelCore& kernel) : KHardwareTimerBase{kernel} {} |
||||
|
|
||||
|
// Public API. |
||||
|
void Initialize(); |
||||
|
void Finalize(); |
||||
|
|
||||
|
s64 GetCount() const { |
||||
|
return GetTick(); |
||||
|
} |
||||
|
|
||||
|
void RegisterTask(KTimerTask* task, s64 time_from_now) { |
||||
|
this->RegisterAbsoluteTask(task, GetTick() + time_from_now); |
||||
|
} |
||||
|
|
||||
|
void RegisterAbsoluteTask(KTimerTask* task, s64 task_time) { |
||||
|
KScopedDisableDispatch dd{m_kernel}; |
||||
|
KScopedSpinLock lk{this->GetLock()}; |
||||
|
|
||||
|
if (this->RegisterAbsoluteTaskImpl(task, task_time)) { |
||||
|
if (task_time <= m_wakeup_time) { |
||||
|
this->EnableInterrupt(task_time); |
||||
|
} |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
private: |
||||
|
void EnableInterrupt(s64 wakeup_time); |
||||
|
void DisableInterrupt(); |
||||
|
bool GetInterruptEnabled(); |
||||
|
s64 GetTick() const; |
||||
|
void DoTask(); |
||||
|
|
||||
|
private: |
||||
|
// Absolute time in nanoseconds |
||||
|
s64 m_wakeup_time{std::numeric_limits<s64>::max()}; |
||||
|
std::shared_ptr<Core::Timing::EventType> m_event_type{}; |
||||
|
}; |
||||
|
|
||||
|
} // namespace Kernel |
||||
@ -0,0 +1,92 @@ |
|||||
|
// SPDX-FileCopyrightText: Copyright 2022 yuzu Emulator Project |
||||
|
// SPDX-License-Identifier: GPL-2.0-or-later |
||||
|
|
||||
|
#pragma once |
||||
|
|
||||
|
#include "core/hle/kernel/k_spin_lock.h" |
||||
|
#include "core/hle/kernel/k_thread.h" |
||||
|
#include "core/hle/kernel/k_timer_task.h" |
||||
|
|
||||
|
namespace Kernel { |
||||
|
|
||||
|
class KHardwareTimerBase { |
||||
|
public: |
||||
|
explicit KHardwareTimerBase(KernelCore& kernel) : m_kernel{kernel} {} |
||||
|
|
||||
|
void CancelTask(KTimerTask* task) { |
||||
|
KScopedDisableDispatch dd{m_kernel}; |
||||
|
KScopedSpinLock lk{m_lock}; |
||||
|
|
||||
|
if (const s64 task_time = task->GetTime(); task_time > 0) { |
||||
|
this->RemoveTaskFromTree(task); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
protected: |
||||
|
KSpinLock& GetLock() { |
||||
|
return m_lock; |
||||
|
} |
||||
|
|
||||
|
s64 DoInterruptTaskImpl(s64 cur_time) { |
||||
|
// We want to handle all tasks, returning the next time that a task is scheduled. |
||||
|
while (true) { |
||||
|
// Get the next task. If there isn't one, return 0. |
||||
|
KTimerTask* task = m_next_task; |
||||
|
if (task == nullptr) { |
||||
|
return 0; |
||||
|
} |
||||
|
|
||||
|
// If the task needs to be done in the future, do it in the future and not now. |
||||
|
if (const s64 task_time = task->GetTime(); task_time > cur_time) { |
||||
|
return task_time; |
||||
|
} |
||||
|
|
||||
|
// Remove the task from the tree of tasks, and update our next task. |
||||
|
this->RemoveTaskFromTree(task); |
||||
|
|
||||
|
// Handle the task. |
||||
|
task->OnTimer(); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
bool RegisterAbsoluteTaskImpl(KTimerTask* task, s64 task_time) { |
||||
|
ASSERT(task_time > 0); |
||||
|
|
||||
|
// Set the task's time, and insert it into our tree. |
||||
|
task->SetTime(task_time); |
||||
|
m_task_tree.insert(*task); |
||||
|
|
||||
|
// Update our next task if relevant. |
||||
|
if (m_next_task != nullptr && m_next_task->GetTime() <= task_time) { |
||||
|
return false; |
||||
|
} |
||||
|
m_next_task = task; |
||||
|
return true; |
||||
|
} |
||||
|
|
||||
|
private: |
||||
|
void RemoveTaskFromTree(KTimerTask* task) { |
||||
|
// Erase from the tree. |
||||
|
auto it = m_task_tree.erase(m_task_tree.iterator_to(*task)); |
||||
|
|
||||
|
// Clear the task's scheduled time. |
||||
|
task->SetTime(0); |
||||
|
|
||||
|
// Update our next task if relevant. |
||||
|
if (m_next_task == task) { |
||||
|
m_next_task = (it != m_task_tree.end()) ? std::addressof(*it) : nullptr; |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
protected: |
||||
|
KernelCore& m_kernel; |
||||
|
|
||||
|
private: |
||||
|
using TimerTaskTree = Common::IntrusiveRedBlackTreeBaseTraits<KTimerTask>::TreeType<KTimerTask>; |
||||
|
|
||||
|
KSpinLock m_lock{}; |
||||
|
TimerTaskTree m_task_tree{}; |
||||
|
KTimerTask* m_next_task{}; |
||||
|
}; |
||||
|
|
||||
|
} // namespace Kernel |
||||
@ -0,0 +1,40 @@ |
|||||
|
// SPDX-FileCopyrightText: Copyright 2022 yuzu Emulator Project |
||||
|
// SPDX-License-Identifier: GPL-2.0-or-later |
||||
|
|
||||
|
#pragma once |
||||
|
|
||||
|
#include "common/intrusive_red_black_tree.h" |
||||
|
|
||||
|
namespace Kernel { |
||||
|
|
||||
|
class KTimerTask : public Common::IntrusiveRedBlackTreeBaseNode<KTimerTask> { |
||||
|
public: |
||||
|
static constexpr int Compare(const KTimerTask& lhs, const KTimerTask& rhs) { |
||||
|
if (lhs.GetTime() < rhs.GetTime()) { |
||||
|
return -1; |
||||
|
} else { |
||||
|
return 1; |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
constexpr explicit KTimerTask() = default; |
||||
|
|
||||
|
constexpr void SetTime(s64 t) { |
||||
|
m_time = t; |
||||
|
} |
||||
|
|
||||
|
constexpr s64 GetTime() const { |
||||
|
return m_time; |
||||
|
} |
||||
|
|
||||
|
// NOTE: This is virtual in Nintendo's kernel. Prior to 13.0.0, KWaitObject was also a |
||||
|
// TimerTask; this is no longer the case. Since this is now KThread exclusive, we have |
||||
|
// devirtualized (see inline declaration for this inside k_thread.h). |
||||
|
void OnTimer(); |
||||
|
|
||||
|
private: |
||||
|
// Absolute time in nanoseconds |
||||
|
s64 m_time{}; |
||||
|
}; |
||||
|
|
||||
|
} // namespace Kernel |
||||
@ -1,44 +0,0 @@ |
|||||
// SPDX-FileCopyrightText: Copyright 2020 yuzu Emulator Project
|
|
||||
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
||||
|
|
||||
#include "common/assert.h"
|
|
||||
#include "core/core.h"
|
|
||||
#include "core/core_timing.h"
|
|
||||
#include "core/hle/kernel/k_scheduler.h"
|
|
||||
#include "core/hle/kernel/k_thread.h"
|
|
||||
#include "core/hle/kernel/time_manager.h"
|
|
||||
|
|
||||
namespace Kernel { |
|
||||
|
|
||||
TimeManager::TimeManager(Core::System& system_) : system{system_} { |
|
||||
time_manager_event_type = Core::Timing::CreateEvent( |
|
||||
"Kernel::TimeManagerCallback", |
|
||||
[this](std::uintptr_t thread_handle, s64 time, |
|
||||
std::chrono::nanoseconds) -> std::optional<std::chrono::nanoseconds> { |
|
||||
KThread* thread = reinterpret_cast<KThread*>(thread_handle); |
|
||||
{ |
|
||||
KScopedSchedulerLock sl(system.Kernel()); |
|
||||
thread->OnTimer(); |
|
||||
} |
|
||||
return std::nullopt; |
|
||||
}); |
|
||||
} |
|
||||
|
|
||||
void TimeManager::ScheduleTimeEvent(KThread* thread, s64 nanoseconds) { |
|
||||
std::scoped_lock lock{mutex}; |
|
||||
if (nanoseconds > 0) { |
|
||||
ASSERT(thread); |
|
||||
ASSERT(thread->GetState() != ThreadState::Runnable); |
|
||||
system.CoreTiming().ScheduleEvent(std::chrono::nanoseconds{nanoseconds}, |
|
||||
time_manager_event_type, |
|
||||
reinterpret_cast<uintptr_t>(thread)); |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
void TimeManager::UnscheduleTimeEvent(KThread* thread) { |
|
||||
std::scoped_lock lock{mutex}; |
|
||||
system.CoreTiming().UnscheduleEvent(time_manager_event_type, |
|
||||
reinterpret_cast<uintptr_t>(thread)); |
|
||||
} |
|
||||
|
|
||||
} // namespace Kernel
|
|
||||
@ -1,41 +0,0 @@ |
|||||
// SPDX-FileCopyrightText: Copyright 2020 yuzu Emulator Project |
|
||||
// SPDX-License-Identifier: GPL-2.0-or-later |
|
||||
|
|
||||
#pragma once |
|
||||
|
|
||||
#include <memory> |
|
||||
#include <mutex> |
|
||||
|
|
||||
namespace Core { |
|
||||
class System; |
|
||||
} // namespace Core |
|
||||
|
|
||||
namespace Core::Timing { |
|
||||
struct EventType; |
|
||||
} // namespace Core::Timing |
|
||||
|
|
||||
namespace Kernel { |
|
||||
|
|
||||
class KThread; |
|
||||
|
|
||||
/** |
|
||||
* The `TimeManager` takes care of scheduling time events on threads and executes their TimeUp |
|
||||
* method when the event is triggered. |
|
||||
*/ |
|
||||
class TimeManager { |
|
||||
public: |
|
||||
explicit TimeManager(Core::System& system); |
|
||||
|
|
||||
/// Schedule a time event on `timetask` thread that will expire in 'nanoseconds' |
|
||||
void ScheduleTimeEvent(KThread* time_task, s64 nanoseconds); |
|
||||
|
|
||||
/// Unschedule an existing time event |
|
||||
void UnscheduleTimeEvent(KThread* thread); |
|
||||
|
|
||||
private: |
|
||||
Core::System& system; |
|
||||
std::shared_ptr<Core::Timing::EventType> time_manager_event_type; |
|
||||
std::mutex mutex; |
|
||||
}; |
|
||||
|
|
||||
} // namespace Kernel |
|
||||
Write
Preview
Loading…
Cancel
Save
Reference in new issue