2014-12-17 05:38:14 +00:00
|
|
|
// Copyright 2014 Citra Emulator Project
|
|
|
|
// Licensed under GPLv2 or any later version
|
2014-11-19 08:49:13 +00:00
|
|
|
// Refer to the license.txt file included.
|
2014-05-10 03:11:18 +01:00
|
|
|
|
2019-08-11 00:20:09 +01:00
|
|
|
#include "common/archives.h"
|
|
|
|
#include "common/serialization/atomic.h"
|
2018-10-26 15:27:13 +01:00
|
|
|
#include "core/hle/kernel/client_port.h"
|
2018-10-25 15:27:42 +01:00
|
|
|
#include "core/hle/kernel/config_mem.h"
|
2017-05-30 00:45:42 +01:00
|
|
|
#include "core/hle/kernel/handle_table.h"
|
2019-07-22 13:10:47 +01:00
|
|
|
#include "core/hle/kernel/ipc_debugger/recorder.h"
|
2016-09-21 07:52:38 +01:00
|
|
|
#include "core/hle/kernel/kernel.h"
|
2015-08-06 01:26:52 +01:00
|
|
|
#include "core/hle/kernel/memory.h"
|
2015-05-04 04:01:16 +01:00
|
|
|
#include "core/hle/kernel/process.h"
|
2015-08-06 01:26:52 +01:00
|
|
|
#include "core/hle/kernel/resource_limit.h"
|
2018-10-25 16:23:52 +01:00
|
|
|
#include "core/hle/kernel/shared_page.h"
|
2014-05-10 03:11:18 +01:00
|
|
|
#include "core/hle/kernel/thread.h"
|
2014-12-04 19:45:47 +00:00
|
|
|
#include "core/hle/kernel/timer.h"
|
2014-05-10 03:11:18 +01:00
|
|
|
|
2014-05-20 23:13:25 +01:00
|
|
|
namespace Kernel {
|
2014-05-10 03:11:18 +01:00
|
|
|
|
2014-06-11 03:43:50 +01:00
|
|
|
/// Initialize the kernel
|
2019-02-01 17:19:20 +00:00
|
|
|
KernelSystem::KernelSystem(Memory::MemorySystem& memory, Core::Timing& timing,
|
|
|
|
std::function<void()> prepare_reschedule_callback, u32 system_mode)
|
|
|
|
: memory(memory), timing(timing),
|
|
|
|
prepare_reschedule_callback(std::move(prepare_reschedule_callback)) {
|
2018-10-25 15:51:00 +01:00
|
|
|
MemoryInit(system_mode);
|
2015-08-06 01:26:52 +01:00
|
|
|
|
2018-10-13 21:41:34 +01:00
|
|
|
resource_limits = std::make_unique<ResourceLimitList>(*this);
|
2018-11-21 04:50:00 +00:00
|
|
|
thread_manager = std::make_unique<ThreadManager>(*this);
|
2019-02-01 16:54:52 +00:00
|
|
|
timer_manager = std::make_unique<TimerManager>(timing);
|
2019-07-22 13:10:47 +01:00
|
|
|
ipc_recorder = std::make_unique<IPCDebugger::Recorder>();
|
2014-05-20 23:13:25 +01:00
|
|
|
}
|
|
|
|
|
2014-06-11 03:43:50 +01:00
|
|
|
/// Shutdown the kernel
|
2018-10-26 02:07:15 +01:00
|
|
|
KernelSystem::~KernelSystem() = default;
|
2014-05-23 00:06:12 +01:00
|
|
|
|
2018-10-13 21:41:34 +01:00
|
|
|
ResourceLimitList& KernelSystem::ResourceLimit() {
|
|
|
|
return *resource_limits;
|
|
|
|
}
|
|
|
|
|
|
|
|
const ResourceLimitList& KernelSystem::ResourceLimit() const {
|
|
|
|
return *resource_limits;
|
|
|
|
}
|
|
|
|
|
2018-10-13 22:24:51 +01:00
|
|
|
u32 KernelSystem::GenerateObjectID() {
|
|
|
|
return next_object_id++;
|
|
|
|
}
|
|
|
|
|
2019-03-23 20:04:19 +00:00
|
|
|
std::shared_ptr<Process> KernelSystem::GetCurrentProcess() const {
|
2018-10-17 20:23:56 +01:00
|
|
|
return current_process;
|
|
|
|
}
|
|
|
|
|
2019-03-23 20:04:19 +00:00
|
|
|
void KernelSystem::SetCurrentProcess(std::shared_ptr<Process> process) {
|
2019-06-25 23:39:11 +01:00
|
|
|
current_process = process;
|
2020-01-05 16:35:01 +00:00
|
|
|
SetCurrentMemoryPageTable(process->vm_manager.page_table);
|
2019-06-25 23:39:11 +01:00
|
|
|
}
|
|
|
|
|
2020-01-05 16:35:01 +00:00
|
|
|
void KernelSystem::SetCurrentMemoryPageTable(std::shared_ptr<Memory::PageTable> page_table) {
|
2019-06-25 23:39:11 +01:00
|
|
|
memory.SetCurrentPageTable(page_table);
|
2019-06-26 10:51:42 +01:00
|
|
|
if (current_cpu != nullptr) {
|
|
|
|
current_cpu->PageTableChanged(); // notify the CPU the page table in memory has changed
|
|
|
|
}
|
2019-06-25 23:39:11 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void KernelSystem::SetCPU(std::shared_ptr<ARM_Interface> cpu) {
|
|
|
|
current_cpu = cpu;
|
|
|
|
thread_manager->SetCPU(*cpu);
|
2018-10-17 20:23:56 +01:00
|
|
|
}
|
|
|
|
|
2018-10-23 14:57:59 +01:00
|
|
|
ThreadManager& KernelSystem::GetThreadManager() {
|
|
|
|
return *thread_manager;
|
|
|
|
}
|
|
|
|
|
|
|
|
const ThreadManager& KernelSystem::GetThreadManager() const {
|
|
|
|
return *thread_manager;
|
|
|
|
}
|
|
|
|
|
2018-10-23 19:17:30 +01:00
|
|
|
TimerManager& KernelSystem::GetTimerManager() {
|
|
|
|
return *timer_manager;
|
|
|
|
}
|
|
|
|
|
|
|
|
const TimerManager& KernelSystem::GetTimerManager() const {
|
|
|
|
return *timer_manager;
|
|
|
|
}
|
|
|
|
|
2018-10-25 16:23:52 +01:00
|
|
|
SharedPage::Handler& KernelSystem::GetSharedPageHandler() {
|
|
|
|
return *shared_page_handler;
|
|
|
|
}
|
|
|
|
|
|
|
|
const SharedPage::Handler& KernelSystem::GetSharedPageHandler() const {
|
|
|
|
return *shared_page_handler;
|
|
|
|
}
|
|
|
|
|
2019-07-22 13:10:47 +01:00
|
|
|
IPCDebugger::Recorder& KernelSystem::GetIPCRecorder() {
|
|
|
|
return *ipc_recorder;
|
|
|
|
}
|
|
|
|
|
|
|
|
const IPCDebugger::Recorder& KernelSystem::GetIPCRecorder() const {
|
|
|
|
return *ipc_recorder;
|
|
|
|
}
|
|
|
|
|
2019-03-23 20:04:19 +00:00
|
|
|
void KernelSystem::AddNamedPort(std::string name, std::shared_ptr<ClientPort> port) {
|
2018-10-26 15:27:13 +01:00
|
|
|
named_ports.emplace(std::move(name), std::move(port));
|
|
|
|
}
|
|
|
|
|
2019-08-11 00:20:09 +01:00
|
|
|
template <class Archive>
|
2019-12-27 21:07:29 +00:00
|
|
|
void KernelSystem::serialize(Archive& ar, const unsigned int file_version) {
|
|
|
|
ar& memory_regions;
|
|
|
|
ar& named_ports;
|
|
|
|
ar&* current_cpu.get();
|
2019-08-11 00:20:09 +01:00
|
|
|
// NB: subsystem references and prepare_reschedule_callback are constant
|
2019-12-27 21:07:29 +00:00
|
|
|
ar&* resource_limits.get();
|
|
|
|
ar& next_object_id;
|
|
|
|
ar&* timer_manager.get();
|
|
|
|
ar& next_process_id;
|
|
|
|
ar& process_list;
|
|
|
|
ar& current_process;
|
|
|
|
ar&* thread_manager.get();
|
2020-01-04 22:39:54 +00:00
|
|
|
ar& config_mem_handler;
|
|
|
|
ar& shared_page_handler;
|
2019-12-22 23:37:17 +00:00
|
|
|
// Deliberately don't include debugger info to allow debugging through loads
|
2019-08-11 00:20:09 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
SERIALIZE_IMPL(KernelSystem)
|
|
|
|
|
2018-03-09 17:54:43 +00:00
|
|
|
} // namespace Kernel
|