2014-12-16 21:38:14 -08: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-09 22:11:18 -04:00
|
|
|
|
2018-10-26 10:27:13 -04:00
|
|
|
#include "core/hle/kernel/client_port.h"
|
2018-10-25 10:27:42 -04:00
|
|
|
#include "core/hle/kernel/config_mem.h"
|
2017-05-29 16:45:42 -07:00
|
|
|
#include "core/hle/kernel/handle_table.h"
|
2019-07-22 20:10:47 +08:00
|
|
|
#include "core/hle/kernel/ipc_debugger/recorder.h"
|
2016-09-20 23:52:38 -07:00
|
|
|
#include "core/hle/kernel/kernel.h"
|
2015-08-05 21:26:52 -03:00
|
|
|
#include "core/hle/kernel/memory.h"
|
2015-05-04 00:01:16 -03:00
|
|
|
#include "core/hle/kernel/process.h"
|
2015-08-05 21:26:52 -03:00
|
|
|
#include "core/hle/kernel/resource_limit.h"
|
2018-10-25 11:23:52 -04:00
|
|
|
#include "core/hle/kernel/shared_page.h"
|
2014-05-09 22:11:18 -04:00
|
|
|
#include "core/hle/kernel/thread.h"
|
2014-12-04 14:45:47 -05:00
|
|
|
#include "core/hle/kernel/timer.h"
|
2014-05-09 22:11:18 -04:00
|
|
|
|
2014-05-20 18:13:25 -04:00
|
|
|
namespace Kernel {
|
2014-05-09 22:11:18 -04:00
|
|
|
|
2014-06-10 22:43:50 -04:00
|
|
|
/// Initialize the kernel
|
2019-02-01 12:19:20 -05:00
|
|
|
KernelSystem::KernelSystem(Memory::MemorySystem& memory, Core::Timing& timing,
|
2020-02-21 19:31:32 +01:00
|
|
|
std::function<void()> prepare_reschedule_callback, u32 system_mode,
|
|
|
|
u32 num_cores)
|
2019-02-01 12:19:20 -05:00
|
|
|
: memory(memory), timing(timing),
|
|
|
|
prepare_reschedule_callback(std::move(prepare_reschedule_callback)) {
|
2018-10-25 10:51:00 -04:00
|
|
|
MemoryInit(system_mode);
|
2015-08-05 21:26:52 -03:00
|
|
|
|
2018-10-13 16:41:34 -04:00
|
|
|
resource_limits = std::make_unique<ResourceLimitList>(*this);
|
2020-02-21 19:31:32 +01:00
|
|
|
for (u32 core_id = 0; core_id < num_cores; ++core_id) {
|
|
|
|
thread_managers.push_back(std::make_unique<ThreadManager>(*this, core_id));
|
|
|
|
}
|
2019-02-01 11:54:52 -05:00
|
|
|
timer_manager = std::make_unique<TimerManager>(timing);
|
2019-07-22 20:10:47 +08:00
|
|
|
ipc_recorder = std::make_unique<IPCDebugger::Recorder>();
|
2020-02-21 19:31:32 +01:00
|
|
|
stored_processes.assign(num_cores, nullptr);
|
|
|
|
|
|
|
|
next_thread_id = 1;
|
2014-05-20 18:13:25 -04:00
|
|
|
}
|
|
|
|
|
2014-06-10 22:43:50 -04:00
|
|
|
/// Shutdown the kernel
|
2020-02-21 19:31:32 +01:00
|
|
|
KernelSystem::~KernelSystem() {
|
|
|
|
ResetThreadIDs();
|
|
|
|
};
|
2014-05-22 19:06:12 -04:00
|
|
|
|
2018-10-13 16:41:34 -04:00
|
|
|
ResourceLimitList& KernelSystem::ResourceLimit() {
|
|
|
|
return *resource_limits;
|
|
|
|
}
|
|
|
|
|
|
|
|
const ResourceLimitList& KernelSystem::ResourceLimit() const {
|
|
|
|
return *resource_limits;
|
|
|
|
}
|
|
|
|
|
2018-10-13 17:24:51 -04:00
|
|
|
u32 KernelSystem::GenerateObjectID() {
|
|
|
|
return next_object_id++;
|
|
|
|
}
|
|
|
|
|
2019-03-23 16:04:19 -04:00
|
|
|
std::shared_ptr<Process> KernelSystem::GetCurrentProcess() const {
|
2018-10-17 15:23:56 -04:00
|
|
|
return current_process;
|
|
|
|
}
|
|
|
|
|
2019-03-23 16:04:19 -04:00
|
|
|
void KernelSystem::SetCurrentProcess(std::shared_ptr<Process> process) {
|
2019-06-26 00:39:11 +02:00
|
|
|
current_process = process;
|
2019-06-27 16:33:55 +02:00
|
|
|
SetCurrentMemoryPageTable(&process->vm_manager.page_table);
|
2019-06-26 00:39:11 +02:00
|
|
|
}
|
|
|
|
|
2020-02-21 19:31:32 +01:00
|
|
|
void KernelSystem::SetCurrentProcessForCPU(std::shared_ptr<Process> 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;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-06-27 16:33:55 +02:00
|
|
|
void KernelSystem::SetCurrentMemoryPageTable(Memory::PageTable* page_table) {
|
2019-06-26 00:39:11 +02:00
|
|
|
memory.SetCurrentPageTable(page_table);
|
2019-06-26 11:51:42 +02:00
|
|
|
if (current_cpu != nullptr) {
|
|
|
|
current_cpu->PageTableChanged(); // notify the CPU the page table in memory has changed
|
|
|
|
}
|
2019-06-26 00:39:11 +02:00
|
|
|
}
|
|
|
|
|
2020-02-21 19:31:32 +01:00
|
|
|
void KernelSystem::SetCPUs(std::vector<std::shared_ptr<ARM_Interface>> 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<ARM_Interface> cpu) {
|
|
|
|
if (current_process) {
|
|
|
|
stored_processes[current_cpu->GetID()] = current_process;
|
|
|
|
}
|
2019-06-26 00:39:11 +02:00
|
|
|
current_cpu = cpu;
|
2020-02-21 19:31:32 +01:00
|
|
|
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];
|
2018-10-17 15:23:56 -04:00
|
|
|
}
|
|
|
|
|
2020-02-21 19:31:32 +01:00
|
|
|
ThreadManager& KernelSystem::GetCurrentThreadManager() {
|
|
|
|
return *thread_managers[current_cpu->GetID()];
|
2018-10-23 09:57:59 -04:00
|
|
|
}
|
|
|
|
|
2020-02-21 19:31:32 +01:00
|
|
|
const ThreadManager& KernelSystem::GetCurrentThreadManager() const {
|
|
|
|
return *thread_managers[current_cpu->GetID()];
|
2018-10-23 09:57:59 -04:00
|
|
|
}
|
|
|
|
|
2018-10-23 14:17:30 -04:00
|
|
|
TimerManager& KernelSystem::GetTimerManager() {
|
|
|
|
return *timer_manager;
|
|
|
|
}
|
|
|
|
|
|
|
|
const TimerManager& KernelSystem::GetTimerManager() const {
|
|
|
|
return *timer_manager;
|
|
|
|
}
|
|
|
|
|
2018-10-25 11:23:52 -04:00
|
|
|
SharedPage::Handler& KernelSystem::GetSharedPageHandler() {
|
|
|
|
return *shared_page_handler;
|
|
|
|
}
|
|
|
|
|
|
|
|
const SharedPage::Handler& KernelSystem::GetSharedPageHandler() const {
|
|
|
|
return *shared_page_handler;
|
|
|
|
}
|
|
|
|
|
2019-07-22 20:10:47 +08:00
|
|
|
IPCDebugger::Recorder& KernelSystem::GetIPCRecorder() {
|
|
|
|
return *ipc_recorder;
|
|
|
|
}
|
|
|
|
|
|
|
|
const IPCDebugger::Recorder& KernelSystem::GetIPCRecorder() const {
|
|
|
|
return *ipc_recorder;
|
|
|
|
}
|
|
|
|
|
2019-03-23 16:04:19 -04:00
|
|
|
void KernelSystem::AddNamedPort(std::string name, std::shared_ptr<ClientPort> port) {
|
2018-10-26 10:27:13 -04:00
|
|
|
named_ports.emplace(std::move(name), std::move(port));
|
|
|
|
}
|
|
|
|
|
2020-02-21 19:31:32 +01:00
|
|
|
u32 KernelSystem::NewThreadId() {
|
|
|
|
return next_thread_id++;
|
|
|
|
}
|
|
|
|
|
|
|
|
void KernelSystem::ResetThreadIDs() {
|
|
|
|
next_thread_id = 0;
|
|
|
|
}
|
|
|
|
|
2018-03-09 10:54:43 -07:00
|
|
|
} // namespace Kernel
|