mirror of
https://github.com/yuzu-emu/yuzu.git
synced 2024-07-04 23:31:19 +01:00
560 lines
19 KiB
C++
560 lines
19 KiB
C++
// Copyright 2014 Citra Emulator Project
|
|
// Licensed under GPLv2 or any later version
|
|
// Refer to the license.txt file included.
|
|
|
|
#include <atomic>
|
|
#include <bitset>
|
|
#include <functional>
|
|
#include <memory>
|
|
#include <mutex>
|
|
#include <thread>
|
|
#include <unordered_map>
|
|
#include <utility>
|
|
|
|
#include "common/assert.h"
|
|
#include "common/logging/log.h"
|
|
#include "core/arm/arm_interface.h"
|
|
#include "core/arm/exclusive_monitor.h"
|
|
#include "core/core.h"
|
|
#include "core/core_timing.h"
|
|
#include "core/core_timing_util.h"
|
|
#include "core/device_memory.h"
|
|
#include "core/hardware_properties.h"
|
|
#include "core/hle/kernel/client_port.h"
|
|
#include "core/hle/kernel/errors.h"
|
|
#include "core/hle/kernel/handle_table.h"
|
|
#include "core/hle/kernel/kernel.h"
|
|
#include "core/hle/kernel/memory/memory_layout.h"
|
|
#include "core/hle/kernel/memory/memory_manager.h"
|
|
#include "core/hle/kernel/memory/slab_heap.h"
|
|
#include "core/hle/kernel/physical_core.h"
|
|
#include "core/hle/kernel/process.h"
|
|
#include "core/hle/kernel/resource_limit.h"
|
|
#include "core/hle/kernel/scheduler.h"
|
|
#include "core/hle/kernel/shared_memory.h"
|
|
#include "core/hle/kernel/synchronization.h"
|
|
#include "core/hle/kernel/thread.h"
|
|
#include "core/hle/kernel/time_manager.h"
|
|
#include "core/hle/lock.h"
|
|
#include "core/hle/result.h"
|
|
#include "core/memory.h"
|
|
|
|
namespace Kernel {
|
|
|
|
/**
|
|
* Callback that will wake up the thread it was scheduled for
|
|
* @param thread_handle The handle of the thread that's been awoken
|
|
* @param cycles_late The number of CPU cycles that have passed since the desired wakeup time
|
|
*/
|
|
static void ThreadWakeupCallback(u64 thread_handle, [[maybe_unused]] s64 cycles_late) {
|
|
const auto proper_handle = static_cast<Handle>(thread_handle);
|
|
const auto& system = Core::System::GetInstance();
|
|
|
|
// Lock the global kernel mutex when we enter the kernel HLE.
|
|
std::lock_guard lock{HLE::g_hle_lock};
|
|
|
|
std::shared_ptr<Thread> thread =
|
|
system.Kernel().RetrieveThreadFromGlobalHandleTable(proper_handle);
|
|
if (thread == nullptr) {
|
|
LOG_CRITICAL(Kernel, "Callback fired for invalid thread {:08X}", proper_handle);
|
|
return;
|
|
}
|
|
|
|
bool resume = true;
|
|
|
|
if (thread->GetStatus() == ThreadStatus::WaitSynch ||
|
|
thread->GetStatus() == ThreadStatus::WaitHLEEvent) {
|
|
// Remove the thread from each of its waiting objects' waitlists
|
|
for (const auto& object : thread->GetSynchronizationObjects()) {
|
|
object->RemoveWaitingThread(thread);
|
|
}
|
|
thread->ClearSynchronizationObjects();
|
|
|
|
// Invoke the wakeup callback before clearing the wait objects
|
|
if (thread->HasWakeupCallback()) {
|
|
resume = thread->InvokeWakeupCallback(ThreadWakeupReason::Timeout, thread, nullptr, 0);
|
|
}
|
|
} else if (thread->GetStatus() == ThreadStatus::WaitMutex ||
|
|
thread->GetStatus() == ThreadStatus::WaitCondVar) {
|
|
thread->SetMutexWaitAddress(0);
|
|
thread->SetWaitHandle(0);
|
|
if (thread->GetStatus() == ThreadStatus::WaitCondVar) {
|
|
thread->GetOwnerProcess()->RemoveConditionVariableThread(thread);
|
|
thread->SetCondVarWaitAddress(0);
|
|
}
|
|
|
|
auto* const lock_owner = thread->GetLockOwner();
|
|
// Threads waking up by timeout from WaitProcessWideKey do not perform priority inheritance
|
|
// and don't have a lock owner unless SignalProcessWideKey was called first and the thread
|
|
// wasn't awakened due to the mutex already being acquired.
|
|
if (lock_owner != nullptr) {
|
|
lock_owner->RemoveMutexWaiter(thread);
|
|
}
|
|
}
|
|
|
|
if (thread->GetStatus() == ThreadStatus::WaitArb) {
|
|
auto& address_arbiter = thread->GetOwnerProcess()->GetAddressArbiter();
|
|
address_arbiter.HandleWakeupThread(thread);
|
|
}
|
|
|
|
if (resume) {
|
|
if (thread->GetStatus() == ThreadStatus::WaitCondVar ||
|
|
thread->GetStatus() == ThreadStatus::WaitArb) {
|
|
thread->SetWaitSynchronizationResult(RESULT_TIMEOUT);
|
|
}
|
|
thread->ResumeFromWait();
|
|
}
|
|
}
|
|
|
|
struct KernelCore::Impl {
|
|
explicit Impl(Core::System& system, KernelCore& kernel)
|
|
: global_scheduler{kernel}, synchronization{system}, time_manager{system}, system{system} {}
|
|
|
|
void Initialize(KernelCore& kernel) {
|
|
Shutdown();
|
|
|
|
InitializePhysicalCores();
|
|
InitializeSystemResourceLimit(kernel);
|
|
InitializeMemoryLayout();
|
|
InitializeThreads();
|
|
InitializePreemption();
|
|
}
|
|
|
|
void Shutdown() {
|
|
next_object_id = 0;
|
|
next_kernel_process_id = Process::InitialKIPIDMin;
|
|
next_user_process_id = Process::ProcessIDMin;
|
|
next_thread_id = 1;
|
|
|
|
process_list.clear();
|
|
current_process = nullptr;
|
|
|
|
system_resource_limit = nullptr;
|
|
|
|
global_handle_table.Clear();
|
|
thread_wakeup_event_type = nullptr;
|
|
preemption_event = nullptr;
|
|
|
|
global_scheduler.Shutdown();
|
|
|
|
named_ports.clear();
|
|
|
|
for (auto& core : cores) {
|
|
core.Shutdown();
|
|
}
|
|
cores.clear();
|
|
|
|
exclusive_monitor.reset();
|
|
}
|
|
|
|
void InitializePhysicalCores() {
|
|
exclusive_monitor =
|
|
Core::MakeExclusiveMonitor(system.Memory(), Core::Hardware::NUM_CPU_CORES);
|
|
for (std::size_t i = 0; i < Core::Hardware::NUM_CPU_CORES; i++) {
|
|
cores.emplace_back(system, i, *exclusive_monitor);
|
|
}
|
|
}
|
|
|
|
// Creates the default system resource limit
|
|
void InitializeSystemResourceLimit(KernelCore& kernel) {
|
|
system_resource_limit = ResourceLimit::Create(kernel);
|
|
|
|
// If setting the default system values fails, then something seriously wrong has occurred.
|
|
ASSERT(system_resource_limit->SetLimitValue(ResourceType::PhysicalMemory, 0x100000000)
|
|
.IsSuccess());
|
|
ASSERT(system_resource_limit->SetLimitValue(ResourceType::Threads, 800).IsSuccess());
|
|
ASSERT(system_resource_limit->SetLimitValue(ResourceType::Events, 700).IsSuccess());
|
|
ASSERT(system_resource_limit->SetLimitValue(ResourceType::TransferMemory, 200).IsSuccess());
|
|
ASSERT(system_resource_limit->SetLimitValue(ResourceType::Sessions, 900).IsSuccess());
|
|
|
|
if (!system_resource_limit->Reserve(ResourceType::PhysicalMemory, 0) ||
|
|
!system_resource_limit->Reserve(ResourceType::PhysicalMemory, 0x60000)) {
|
|
UNREACHABLE();
|
|
}
|
|
}
|
|
|
|
void InitializeThreads() {
|
|
thread_wakeup_event_type =
|
|
Core::Timing::CreateEvent("ThreadWakeupCallback", ThreadWakeupCallback);
|
|
}
|
|
|
|
void InitializePreemption() {
|
|
preemption_event =
|
|
Core::Timing::CreateEvent("PreemptionCallback", [this](u64 userdata, s64 cycles_late) {
|
|
global_scheduler.PreemptThreads();
|
|
s64 time_interval = Core::Timing::msToCycles(std::chrono::milliseconds(10));
|
|
system.CoreTiming().ScheduleEvent(time_interval, preemption_event);
|
|
});
|
|
|
|
s64 time_interval = Core::Timing::msToCycles(std::chrono::milliseconds(10));
|
|
system.CoreTiming().ScheduleEvent(time_interval, preemption_event);
|
|
}
|
|
|
|
void MakeCurrentProcess(Process* process) {
|
|
current_process = process;
|
|
|
|
if (process == nullptr) {
|
|
return;
|
|
}
|
|
|
|
for (auto& core : cores) {
|
|
core.SetIs64Bit(process->Is64BitProcess());
|
|
}
|
|
|
|
system.Memory().SetCurrentPageTable(*process);
|
|
}
|
|
|
|
void RegisterCoreThread(std::size_t core_id) {
|
|
std::unique_lock lock{register_thread_mutex};
|
|
const std::thread::id this_id = std::this_thread::get_id();
|
|
const auto it = host_thread_ids.find(this_id);
|
|
ASSERT(core_id < Core::Hardware::NUM_CPU_CORES);
|
|
ASSERT(it == host_thread_ids.end());
|
|
ASSERT(!registered_core_threads[core_id]);
|
|
host_thread_ids[this_id] = static_cast<u32>(core_id);
|
|
registered_core_threads.set(core_id);
|
|
}
|
|
|
|
void RegisterHostThread() {
|
|
std::unique_lock lock{register_thread_mutex};
|
|
const std::thread::id this_id = std::this_thread::get_id();
|
|
const auto it = host_thread_ids.find(this_id);
|
|
ASSERT(it == host_thread_ids.end());
|
|
host_thread_ids[this_id] = registered_thread_ids++;
|
|
}
|
|
|
|
u32 GetCurrentHostThreadID() const {
|
|
const std::thread::id this_id = std::this_thread::get_id();
|
|
const auto it = host_thread_ids.find(this_id);
|
|
if (it == host_thread_ids.end()) {
|
|
return Core::INVALID_HOST_THREAD_ID;
|
|
}
|
|
return it->second;
|
|
}
|
|
|
|
Core::EmuThreadHandle GetCurrentEmuThreadID() const {
|
|
Core::EmuThreadHandle result = Core::EmuThreadHandle::InvalidHandle();
|
|
result.host_handle = GetCurrentHostThreadID();
|
|
if (result.host_handle >= Core::Hardware::NUM_CPU_CORES) {
|
|
return result;
|
|
}
|
|
const Kernel::Scheduler& sched = cores[result.host_handle].Scheduler();
|
|
const Kernel::Thread* current = sched.GetCurrentThread();
|
|
if (current != nullptr) {
|
|
result.guest_handle = current->GetGlobalHandle();
|
|
} else {
|
|
result.guest_handle = InvalidHandle;
|
|
}
|
|
return result;
|
|
}
|
|
|
|
void InitializeMemoryLayout() {
|
|
// Initialize memory layout
|
|
constexpr Memory::MemoryLayout layout{Memory::MemoryLayout::GetDefaultLayout()};
|
|
constexpr std::size_t hid_size{0x40000};
|
|
constexpr std::size_t font_size{0x1100000};
|
|
constexpr std::size_t irs_size{0x8000};
|
|
constexpr std::size_t time_size{0x1000};
|
|
constexpr PAddr hid_addr{layout.System().StartAddress()};
|
|
constexpr PAddr font_pa{layout.System().StartAddress() + hid_size};
|
|
constexpr PAddr irs_addr{layout.System().StartAddress() + hid_size + font_size};
|
|
constexpr PAddr time_addr{layout.System().StartAddress() + hid_size + font_size + irs_size};
|
|
|
|
// Initialize memory manager
|
|
memory_manager = std::make_unique<Memory::MemoryManager>();
|
|
memory_manager->InitializeManager(Memory::MemoryManager::Pool::Application,
|
|
layout.Application().StartAddress(),
|
|
layout.Application().EndAddress());
|
|
memory_manager->InitializeManager(Memory::MemoryManager::Pool::Applet,
|
|
layout.Applet().StartAddress(),
|
|
layout.Applet().EndAddress());
|
|
memory_manager->InitializeManager(Memory::MemoryManager::Pool::System,
|
|
layout.System().StartAddress(),
|
|
layout.System().EndAddress());
|
|
|
|
hid_shared_mem = Kernel::SharedMemory::Create(
|
|
system.Kernel(), system.DeviceMemory(), nullptr,
|
|
{hid_addr, hid_size / Memory::PageSize}, Memory::MemoryPermission::None,
|
|
Memory::MemoryPermission::Read, hid_addr, hid_size, "HID:SharedMemory");
|
|
font_shared_mem = Kernel::SharedMemory::Create(
|
|
system.Kernel(), system.DeviceMemory(), nullptr,
|
|
{font_pa, font_size / Memory::PageSize}, Memory::MemoryPermission::None,
|
|
Memory::MemoryPermission::Read, font_pa, font_size, "Font:SharedMemory");
|
|
irs_shared_mem = Kernel::SharedMemory::Create(
|
|
system.Kernel(), system.DeviceMemory(), nullptr,
|
|
{irs_addr, irs_size / Memory::PageSize}, Memory::MemoryPermission::None,
|
|
Memory::MemoryPermission::Read, irs_addr, irs_size, "IRS:SharedMemory");
|
|
time_shared_mem = Kernel::SharedMemory::Create(
|
|
system.Kernel(), system.DeviceMemory(), nullptr,
|
|
{time_addr, time_size / Memory::PageSize}, Memory::MemoryPermission::None,
|
|
Memory::MemoryPermission::Read, time_addr, time_size, "Time:SharedMemory");
|
|
|
|
// Allocate slab heaps
|
|
user_slab_heap_pages = std::make_unique<Memory::SlabHeap<Memory::Page>>();
|
|
|
|
// Initialize slab heaps
|
|
constexpr u64 user_slab_heap_size{0x3de000};
|
|
user_slab_heap_pages->Initialize(
|
|
system.DeviceMemory().GetPointer(Core::DramMemoryMap::SlabHeapBase),
|
|
user_slab_heap_size);
|
|
}
|
|
|
|
std::atomic<u32> next_object_id{0};
|
|
std::atomic<u64> next_kernel_process_id{Process::InitialKIPIDMin};
|
|
std::atomic<u64> next_user_process_id{Process::ProcessIDMin};
|
|
std::atomic<u64> next_thread_id{1};
|
|
|
|
// Lists all processes that exist in the current session.
|
|
std::vector<std::shared_ptr<Process>> process_list;
|
|
Process* current_process = nullptr;
|
|
Kernel::GlobalScheduler global_scheduler;
|
|
Kernel::Synchronization synchronization;
|
|
Kernel::TimeManager time_manager;
|
|
|
|
std::shared_ptr<ResourceLimit> system_resource_limit;
|
|
|
|
std::shared_ptr<Core::Timing::EventType> thread_wakeup_event_type;
|
|
std::shared_ptr<Core::Timing::EventType> preemption_event;
|
|
|
|
// This is the kernel's handle table or supervisor handle table which
|
|
// stores all the objects in place.
|
|
Kernel::HandleTable global_handle_table;
|
|
|
|
/// Map of named ports managed by the kernel, which can be retrieved using
|
|
/// the ConnectToPort SVC.
|
|
NamedPortTable named_ports;
|
|
|
|
std::unique_ptr<Core::ExclusiveMonitor> exclusive_monitor;
|
|
std::vector<Kernel::PhysicalCore> cores;
|
|
|
|
// 0-3 IDs represent core threads, >3 represent others
|
|
std::unordered_map<std::thread::id, u32> host_thread_ids;
|
|
u32 registered_thread_ids{Core::Hardware::NUM_CPU_CORES};
|
|
std::bitset<Core::Hardware::NUM_CPU_CORES> registered_core_threads;
|
|
std::mutex register_thread_mutex;
|
|
|
|
// Kernel memory management
|
|
std::unique_ptr<Memory::MemoryManager> memory_manager;
|
|
std::unique_ptr<Memory::SlabHeap<Memory::Page>> user_slab_heap_pages;
|
|
|
|
// Shared memory for services
|
|
std::shared_ptr<Kernel::SharedMemory> hid_shared_mem;
|
|
std::shared_ptr<Kernel::SharedMemory> font_shared_mem;
|
|
std::shared_ptr<Kernel::SharedMemory> irs_shared_mem;
|
|
std::shared_ptr<Kernel::SharedMemory> time_shared_mem;
|
|
|
|
// System context
|
|
Core::System& system;
|
|
};
|
|
|
|
KernelCore::KernelCore(Core::System& system) : impl{std::make_unique<Impl>(system, *this)} {}
|
|
KernelCore::~KernelCore() {
|
|
Shutdown();
|
|
}
|
|
|
|
void KernelCore::Initialize() {
|
|
impl->Initialize(*this);
|
|
}
|
|
|
|
void KernelCore::Shutdown() {
|
|
impl->Shutdown();
|
|
}
|
|
|
|
std::shared_ptr<ResourceLimit> KernelCore::GetSystemResourceLimit() const {
|
|
return impl->system_resource_limit;
|
|
}
|
|
|
|
std::shared_ptr<Thread> KernelCore::RetrieveThreadFromGlobalHandleTable(Handle handle) const {
|
|
return impl->global_handle_table.Get<Thread>(handle);
|
|
}
|
|
|
|
void KernelCore::AppendNewProcess(std::shared_ptr<Process> process) {
|
|
impl->process_list.push_back(std::move(process));
|
|
}
|
|
|
|
void KernelCore::MakeCurrentProcess(Process* process) {
|
|
impl->MakeCurrentProcess(process);
|
|
}
|
|
|
|
Process* KernelCore::CurrentProcess() {
|
|
return impl->current_process;
|
|
}
|
|
|
|
const Process* KernelCore::CurrentProcess() const {
|
|
return impl->current_process;
|
|
}
|
|
|
|
const std::vector<std::shared_ptr<Process>>& KernelCore::GetProcessList() const {
|
|
return impl->process_list;
|
|
}
|
|
|
|
Kernel::GlobalScheduler& KernelCore::GlobalScheduler() {
|
|
return impl->global_scheduler;
|
|
}
|
|
|
|
const Kernel::GlobalScheduler& KernelCore::GlobalScheduler() const {
|
|
return impl->global_scheduler;
|
|
}
|
|
|
|
Kernel::Scheduler& KernelCore::Scheduler(std::size_t id) {
|
|
return impl->cores[id].Scheduler();
|
|
}
|
|
|
|
const Kernel::Scheduler& KernelCore::Scheduler(std::size_t id) const {
|
|
return impl->cores[id].Scheduler();
|
|
}
|
|
|
|
Kernel::PhysicalCore& KernelCore::PhysicalCore(std::size_t id) {
|
|
return impl->cores[id];
|
|
}
|
|
|
|
const Kernel::PhysicalCore& KernelCore::PhysicalCore(std::size_t id) const {
|
|
return impl->cores[id];
|
|
}
|
|
|
|
Kernel::Synchronization& KernelCore::Synchronization() {
|
|
return impl->synchronization;
|
|
}
|
|
|
|
const Kernel::Synchronization& KernelCore::Synchronization() const {
|
|
return impl->synchronization;
|
|
}
|
|
|
|
Kernel::TimeManager& KernelCore::TimeManager() {
|
|
return impl->time_manager;
|
|
}
|
|
|
|
const Kernel::TimeManager& KernelCore::TimeManager() const {
|
|
return impl->time_manager;
|
|
}
|
|
|
|
Core::ExclusiveMonitor& KernelCore::GetExclusiveMonitor() {
|
|
return *impl->exclusive_monitor;
|
|
}
|
|
|
|
const Core::ExclusiveMonitor& KernelCore::GetExclusiveMonitor() const {
|
|
return *impl->exclusive_monitor;
|
|
}
|
|
|
|
void KernelCore::InvalidateAllInstructionCaches() {
|
|
for (std::size_t i = 0; i < impl->global_scheduler.CpuCoresCount(); i++) {
|
|
PhysicalCore(i).ArmInterface().ClearInstructionCache();
|
|
}
|
|
}
|
|
|
|
void KernelCore::PrepareReschedule(std::size_t id) {
|
|
if (id < impl->global_scheduler.CpuCoresCount()) {
|
|
impl->cores[id].Stop();
|
|
}
|
|
}
|
|
|
|
void KernelCore::AddNamedPort(std::string name, std::shared_ptr<ClientPort> port) {
|
|
impl->named_ports.emplace(std::move(name), std::move(port));
|
|
}
|
|
|
|
KernelCore::NamedPortTable::iterator KernelCore::FindNamedPort(const std::string& name) {
|
|
return impl->named_ports.find(name);
|
|
}
|
|
|
|
KernelCore::NamedPortTable::const_iterator KernelCore::FindNamedPort(
|
|
const std::string& name) const {
|
|
return impl->named_ports.find(name);
|
|
}
|
|
|
|
bool KernelCore::IsValidNamedPort(NamedPortTable::const_iterator port) const {
|
|
return port != impl->named_ports.cend();
|
|
}
|
|
|
|
u32 KernelCore::CreateNewObjectID() {
|
|
return impl->next_object_id++;
|
|
}
|
|
|
|
u64 KernelCore::CreateNewThreadID() {
|
|
return impl->next_thread_id++;
|
|
}
|
|
|
|
u64 KernelCore::CreateNewKernelProcessID() {
|
|
return impl->next_kernel_process_id++;
|
|
}
|
|
|
|
u64 KernelCore::CreateNewUserProcessID() {
|
|
return impl->next_user_process_id++;
|
|
}
|
|
|
|
const std::shared_ptr<Core::Timing::EventType>& KernelCore::ThreadWakeupCallbackEventType() const {
|
|
return impl->thread_wakeup_event_type;
|
|
}
|
|
|
|
Kernel::HandleTable& KernelCore::GlobalHandleTable() {
|
|
return impl->global_handle_table;
|
|
}
|
|
|
|
const Kernel::HandleTable& KernelCore::GlobalHandleTable() const {
|
|
return impl->global_handle_table;
|
|
}
|
|
|
|
void KernelCore::RegisterCoreThread(std::size_t core_id) {
|
|
impl->RegisterCoreThread(core_id);
|
|
}
|
|
|
|
void KernelCore::RegisterHostThread() {
|
|
impl->RegisterHostThread();
|
|
}
|
|
|
|
u32 KernelCore::GetCurrentHostThreadID() const {
|
|
return impl->GetCurrentHostThreadID();
|
|
}
|
|
|
|
Core::EmuThreadHandle KernelCore::GetCurrentEmuThreadID() const {
|
|
return impl->GetCurrentEmuThreadID();
|
|
}
|
|
|
|
Memory::MemoryManager& KernelCore::MemoryManager() {
|
|
return *impl->memory_manager;
|
|
}
|
|
|
|
const Memory::MemoryManager& KernelCore::MemoryManager() const {
|
|
return *impl->memory_manager;
|
|
}
|
|
|
|
Memory::SlabHeap<Memory::Page>& KernelCore::GetUserSlabHeapPages() {
|
|
return *impl->user_slab_heap_pages;
|
|
}
|
|
|
|
const Memory::SlabHeap<Memory::Page>& KernelCore::GetUserSlabHeapPages() const {
|
|
return *impl->user_slab_heap_pages;
|
|
}
|
|
|
|
Kernel::SharedMemory& KernelCore::GetHidSharedMem() {
|
|
return *impl->hid_shared_mem;
|
|
}
|
|
|
|
const Kernel::SharedMemory& KernelCore::GetHidSharedMem() const {
|
|
return *impl->hid_shared_mem;
|
|
}
|
|
|
|
Kernel::SharedMemory& KernelCore::GetFontSharedMem() {
|
|
return *impl->font_shared_mem;
|
|
}
|
|
|
|
const Kernel::SharedMemory& KernelCore::GetFontSharedMem() const {
|
|
return *impl->font_shared_mem;
|
|
}
|
|
|
|
Kernel::SharedMemory& KernelCore::GetIrsSharedMem() {
|
|
return *impl->irs_shared_mem;
|
|
}
|
|
|
|
const Kernel::SharedMemory& KernelCore::GetIrsSharedMem() const {
|
|
return *impl->irs_shared_mem;
|
|
}
|
|
|
|
Kernel::SharedMemory& KernelCore::GetTimeSharedMem() {
|
|
return *impl->time_shared_mem;
|
|
}
|
|
|
|
const Kernel::SharedMemory& KernelCore::GetTimeSharedMem() const {
|
|
return *impl->time_shared_mem;
|
|
}
|
|
|
|
} // namespace Kernel
|