// Copyright 2014 Citra Emulator Project // Licensed under GPLv2 or any later version // Refer to the license.txt file included. #include "common/archives.h" #include "common/serialization/atomic.h" #include "core/hle/kernel/client_port.h" #include "core/hle/kernel/config_mem.h" #include "core/hle/kernel/handle_table.h" #include "core/hle/kernel/ipc_debugger/recorder.h" #include "core/hle/kernel/kernel.h" #include "core/hle/kernel/memory.h" #include "core/hle/kernel/process.h" #include "core/hle/kernel/resource_limit.h" #include "core/hle/kernel/shared_page.h" #include "core/hle/kernel/thread.h" #include "core/hle/kernel/timer.h" namespace Kernel { /// Initialize the kernel KernelSystem::KernelSystem(Memory::MemorySystem& memory, Core::Timing& timing, std::function prepare_reschedule_callback, u32 system_mode, u32 num_cores, u8 n3ds_mode) : memory(memory), timing(timing), prepare_reschedule_callback(std::move(prepare_reschedule_callback)) { std::generate(memory_regions.begin(), memory_regions.end(), [] { return std::make_shared(); }); memory_regions[i] = std::make_shared(); } MemoryInit(system_mode, n3ds_mode); resource_limits = std::make_unique(*this); for (u32 core_id = 0; core_id < num_cores; ++core_id) { thread_managers.push_back(std::make_unique(*this, core_id)); } timer_manager = std::make_unique(timing); ipc_recorder = std::make_unique(); stored_processes.assign(num_cores, nullptr); next_thread_id = 1; } /// Shutdown the kernel KernelSystem::~KernelSystem() { ResetThreadIDs(); }; ResourceLimitList& KernelSystem::ResourceLimit() { return *resource_limits; } const ResourceLimitList& KernelSystem::ResourceLimit() const { return *resource_limits; } u32 KernelSystem::GenerateObjectID() { return next_object_id++; } std::shared_ptr KernelSystem::GetCurrentProcess() const { return current_process; } void KernelSystem::SetCurrentProcess(std::shared_ptr process) { current_process = process; SetCurrentMemoryPageTable(process->vm_manager.page_table); } void KernelSystem::SetCurrentProcessForCPU(std::shared_ptr process, u32 core_id) { if (current_cpu->GetID() == core_id) { current_process = process; SetCurrentMemoryPageTable(process->vm_manager.page_table); } else { stored_processes[core_id] = process; } } void KernelSystem::SetCurrentMemoryPageTable(std::shared_ptr page_table) { memory.SetCurrentPageTable(page_table); if (current_cpu != nullptr) { current_cpu->SetPageTable(page_table); } } void KernelSystem::SetCPUs(std::vector> cpus) { ASSERT(cpus.size() == thread_managers.size()); u32 i = 0; for (const auto& cpu : cpus) { thread_managers[i++]->SetCPU(*cpu); } } void KernelSystem::SetRunningCPU(std::shared_ptr cpu) { if (current_process) { stored_processes[current_cpu->GetID()] = current_process; } current_cpu = cpu; timing.SetCurrentTimer(cpu->GetID()); if (stored_processes[current_cpu->GetID()]) { SetCurrentProcess(stored_processes[current_cpu->GetID()]); } } ThreadManager& KernelSystem::GetThreadManager(u32 core_id) { return *thread_managers[core_id]; } const ThreadManager& KernelSystem::GetThreadManager(u32 core_id) const { return *thread_managers[core_id]; } ThreadManager& KernelSystem::GetCurrentThreadManager() { return *thread_managers[current_cpu->GetID()]; } const ThreadManager& KernelSystem::GetCurrentThreadManager() const { return *thread_managers[current_cpu->GetID()]; } TimerManager& KernelSystem::GetTimerManager() { return *timer_manager; } const TimerManager& KernelSystem::GetTimerManager() const { return *timer_manager; } SharedPage::Handler& KernelSystem::GetSharedPageHandler() { return *shared_page_handler; } const SharedPage::Handler& KernelSystem::GetSharedPageHandler() const { return *shared_page_handler; } IPCDebugger::Recorder& KernelSystem::GetIPCRecorder() { return *ipc_recorder; } const IPCDebugger::Recorder& KernelSystem::GetIPCRecorder() const { return *ipc_recorder; } void KernelSystem::AddNamedPort(std::string name, std::shared_ptr port) { named_ports.emplace(std::move(name), std::move(port)); } u32 KernelSystem::NewThreadId() { return next_thread_id++; } void KernelSystem::ResetThreadIDs() { next_thread_id = 0; } template void KernelSystem::serialize(Archive& ar, const unsigned int file_version) { ar& memory_regions; ar& named_ports; // current_cpu set externally // NB: subsystem references and prepare_reschedule_callback are constant ar&* resource_limits.get(); ar& next_object_id; ar&* timer_manager.get(); ar& next_process_id; ar& process_list; ar& current_process; // NB: core count checked in 'core' for (auto& thread_manager : thread_managers) { ar&* thread_manager.get(); } ar& config_mem_handler; ar& shared_page_handler; ar& stored_processes; ar& next_thread_id; // Deliberately don't include debugger info to allow debugging through loads } SERIALIZE_IMPL(KernelSystem) } // namespace Kernel