2018-01-13 21:22:39 +00:00
|
|
|
// Copyright 2018 yuzu emulator team
|
2014-12-17 05:38:14 +00:00
|
|
|
// Licensed under GPLv2 or any later version
|
2014-11-19 08:49:13 +00:00
|
|
|
// Refer to the license.txt file included.
|
2014-04-11 00:58:28 +01:00
|
|
|
|
2018-01-12 03:36:56 +00:00
|
|
|
#include <algorithm>
|
2018-02-25 12:40:22 +00:00
|
|
|
#include <cinttypes>
|
2018-01-12 03:36:56 +00:00
|
|
|
|
2015-05-06 08:06:12 +01:00
|
|
|
#include "common/logging/log.h"
|
2015-08-17 22:25:21 +01:00
|
|
|
#include "common/microprofile.h"
|
2018-01-05 00:45:15 +00:00
|
|
|
#include "common/string_util.h"
|
2018-03-13 21:49:59 +00:00
|
|
|
#include "core/core.h"
|
2016-09-18 01:38:01 +01:00
|
|
|
#include "core/core_timing.h"
|
2016-05-22 18:30:13 +01:00
|
|
|
#include "core/hle/kernel/client_port.h"
|
2016-06-15 00:03:30 +01:00
|
|
|
#include "core/hle/kernel/client_session.h"
|
2018-01-09 02:41:37 +00:00
|
|
|
#include "core/hle/kernel/condition_variable.h"
|
2018-01-08 02:24:19 +00:00
|
|
|
#include "core/hle/kernel/event.h"
|
2017-05-30 00:45:42 +01:00
|
|
|
#include "core/hle/kernel/handle_table.h"
|
2018-01-01 19:38:34 +00:00
|
|
|
#include "core/hle/kernel/mutex.h"
|
|
|
|
#include "core/hle/kernel/object_address_table.h"
|
2015-05-11 15:15:10 +01:00
|
|
|
#include "core/hle/kernel/process.h"
|
2017-12-31 20:58:16 +00:00
|
|
|
#include "core/hle/kernel/resource_limit.h"
|
2018-01-14 22:15:31 +00:00
|
|
|
#include "core/hle/kernel/shared_memory.h"
|
2018-01-03 01:40:30 +00:00
|
|
|
#include "core/hle/kernel/svc.h"
|
|
|
|
#include "core/hle/kernel/svc_wrap.h"
|
2017-10-15 03:18:42 +01:00
|
|
|
#include "core/hle/kernel/thread.h"
|
2017-10-14 22:30:07 +01:00
|
|
|
#include "core/hle/lock.h"
|
2014-10-23 04:20:01 +01:00
|
|
|
#include "core/hle/result.h"
|
2014-04-13 02:55:36 +01:00
|
|
|
#include "core/hle/service/service.h"
|
2014-04-11 00:58:28 +01:00
|
|
|
|
2018-01-03 01:40:30 +00:00
|
|
|
namespace Kernel {
|
2014-04-11 04:26:12 +01:00
|
|
|
|
2017-12-28 20:29:52 +00:00
|
|
|
/// Set the process heap to a given Size. It can both extend and shrink the heap.
|
|
|
|
static ResultCode SetHeapSize(VAddr* heap_addr, u64 heap_size) {
|
2017-12-29 02:38:38 +00:00
|
|
|
LOG_TRACE(Kernel_SVC, "called, heap_size=0x%llx", heap_size);
|
2018-03-13 21:49:59 +00:00
|
|
|
auto& process = *Core::CurrentProcess();
|
2018-01-03 03:24:12 +00:00
|
|
|
CASCADE_RESULT(*heap_addr,
|
|
|
|
process.HeapAllocate(Memory::HEAP_VADDR, heap_size, VMAPermission::ReadWrite));
|
2017-12-28 20:29:52 +00:00
|
|
|
return RESULT_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2018-01-08 02:23:42 +00:00
|
|
|
static ResultCode SetMemoryAttribute(VAddr addr, u64 size, u32 state0, u32 state1) {
|
2018-03-19 16:27:04 +00:00
|
|
|
LOG_WARNING(Kernel_SVC, "(STUBBED) called, addr=0x%lx", addr);
|
2018-01-08 02:23:42 +00:00
|
|
|
return RESULT_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2017-12-29 02:38:38 +00:00
|
|
|
/// Maps a memory range into a different range.
|
|
|
|
static ResultCode MapMemory(VAddr dst_addr, VAddr src_addr, u64 size) {
|
|
|
|
LOG_TRACE(Kernel_SVC, "called, dst_addr=0x%llx, src_addr=0x%llx, size=0x%llx", dst_addr,
|
|
|
|
src_addr, size);
|
2018-03-13 21:49:59 +00:00
|
|
|
return Core::CurrentProcess()->MirrorMemory(dst_addr, src_addr, size);
|
2017-12-29 02:38:38 +00:00
|
|
|
}
|
|
|
|
|
2017-12-31 20:22:49 +00:00
|
|
|
/// Unmaps a region that was previously mapped with svcMapMemory
|
|
|
|
static ResultCode UnmapMemory(VAddr dst_addr, VAddr src_addr, u64 size) {
|
|
|
|
LOG_TRACE(Kernel_SVC, "called, dst_addr=0x%llx, src_addr=0x%llx, size=0x%llx", dst_addr,
|
2018-01-01 19:38:34 +00:00
|
|
|
src_addr, size);
|
2018-03-13 21:49:59 +00:00
|
|
|
return Core::CurrentProcess()->UnmapMemory(dst_addr, src_addr, size);
|
2017-12-31 20:22:49 +00:00
|
|
|
}
|
|
|
|
|
2014-04-13 02:55:36 +01:00
|
|
|
/// Connect to an OS service given the port name, returns the handle to the port to out
|
2018-01-18 01:34:52 +00:00
|
|
|
static ResultCode ConnectToNamedPort(Handle* out_handle, VAddr port_name_address) {
|
2017-10-14 22:30:07 +01:00
|
|
|
if (!Memory::IsValidVirtualAddress(port_name_address))
|
2018-01-03 01:40:30 +00:00
|
|
|
return ERR_NOT_FOUND;
|
2017-10-14 22:30:07 +01:00
|
|
|
|
|
|
|
static constexpr std::size_t PortNameMaxLength = 11;
|
|
|
|
// Read 1 char beyond the max allowed port name to detect names that are too long.
|
|
|
|
std::string port_name = Memory::ReadCString(port_name_address, PortNameMaxLength + 1);
|
|
|
|
if (port_name.size() > PortNameMaxLength)
|
2018-01-03 01:40:30 +00:00
|
|
|
return ERR_PORT_NAME_TOO_LONG;
|
2014-06-02 01:48:29 +01:00
|
|
|
|
2017-10-14 22:35:21 +01:00
|
|
|
LOG_TRACE(Kernel_SVC, "called port_name=%s", port_name.c_str());
|
2014-06-02 01:48:29 +01:00
|
|
|
|
2015-01-30 18:07:04 +00:00
|
|
|
auto it = Service::g_kernel_named_ports.find(port_name);
|
|
|
|
if (it == Service::g_kernel_named_ports.end()) {
|
2017-10-14 22:30:07 +01:00
|
|
|
LOG_WARNING(Kernel_SVC, "tried to connect to unknown port: %s", port_name.c_str());
|
2018-01-03 01:40:30 +00:00
|
|
|
return ERR_NOT_FOUND;
|
2015-01-30 18:07:04 +00:00
|
|
|
}
|
2014-06-02 01:48:29 +01:00
|
|
|
|
2016-12-05 16:02:08 +00:00
|
|
|
auto client_port = it->second;
|
2016-06-15 00:03:30 +01:00
|
|
|
|
2018-01-03 01:40:30 +00:00
|
|
|
SharedPtr<ClientSession> client_session;
|
2016-12-05 18:59:57 +00:00
|
|
|
CASCADE_RESULT(client_session, client_port->Connect());
|
2016-06-15 00:03:30 +01:00
|
|
|
|
|
|
|
// Return the client session
|
2018-01-03 01:40:30 +00:00
|
|
|
CASCADE_RESULT(*out_handle, g_handle_table.Create(client_session));
|
2015-01-23 05:36:58 +00:00
|
|
|
return RESULT_SUCCESS;
|
2014-04-13 02:55:36 +01:00
|
|
|
}
|
|
|
|
|
2016-12-08 16:06:19 +00:00
|
|
|
/// Makes a blocking IPC call to an OS service.
|
2018-01-03 01:40:30 +00:00
|
|
|
static ResultCode SendSyncRequest(Handle handle) {
|
2018-01-23 21:13:19 +00:00
|
|
|
SharedPtr<ClientSession> session = g_handle_table.Get<ClientSession>(handle);
|
2017-12-30 18:40:28 +00:00
|
|
|
if (!session) {
|
2017-10-14 22:30:07 +01:00
|
|
|
LOG_ERROR(Kernel_SVC, "called with invalid handle=0x%08X", handle);
|
2015-01-23 05:44:52 +00:00
|
|
|
return ERR_INVALID_HANDLE;
|
2014-10-23 04:20:01 +01:00
|
|
|
}
|
2014-05-27 03:12:46 +01:00
|
|
|
|
2014-12-14 05:30:11 +00:00
|
|
|
LOG_TRACE(Kernel_SVC, "called handle=0x%08X(%s)", handle, session->GetName().c_str());
|
2014-05-27 03:12:46 +01:00
|
|
|
|
2017-01-01 16:57:02 +00:00
|
|
|
Core::System::GetInstance().PrepareReschedule();
|
|
|
|
|
2016-12-14 17:33:49 +00:00
|
|
|
// TODO(Subv): svcSendSyncRequest should put the caller thread to sleep while the server
|
|
|
|
// responds and cause a reschedule.
|
2018-01-03 01:40:30 +00:00
|
|
|
return session->SendSyncRequest(GetCurrentThread());
|
2014-04-11 00:58:28 +01:00
|
|
|
}
|
|
|
|
|
2017-10-23 05:15:45 +01:00
|
|
|
/// Get the ID for the specified thread.
|
2018-01-03 01:40:30 +00:00
|
|
|
static ResultCode GetThreadId(u32* thread_id, Handle thread_handle) {
|
2017-12-30 18:40:28 +00:00
|
|
|
LOG_TRACE(Kernel_SVC, "called thread=0x%08X", thread_handle);
|
2017-10-23 05:15:45 +01:00
|
|
|
|
2018-01-03 03:24:12 +00:00
|
|
|
const SharedPtr<Thread> thread = g_handle_table.Get<Thread>(thread_handle);
|
2017-12-30 18:40:28 +00:00
|
|
|
if (!thread) {
|
2017-10-23 05:15:45 +01:00
|
|
|
return ERR_INVALID_HANDLE;
|
|
|
|
}
|
|
|
|
|
|
|
|
*thread_id = thread->GetThreadId();
|
|
|
|
return RESULT_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Get the ID of the specified process
|
2018-01-03 01:40:30 +00:00
|
|
|
static ResultCode GetProcessId(u32* process_id, Handle process_handle) {
|
2017-10-23 05:15:45 +01:00
|
|
|
LOG_TRACE(Kernel_SVC, "called process=0x%08X", process_handle);
|
|
|
|
|
2018-01-03 03:24:12 +00:00
|
|
|
const SharedPtr<Process> process = g_handle_table.Get<Process>(process_handle);
|
2017-12-30 18:40:28 +00:00
|
|
|
if (!process) {
|
2017-10-23 05:15:45 +01:00
|
|
|
return ERR_INVALID_HANDLE;
|
|
|
|
}
|
|
|
|
|
|
|
|
*process_id = process->process_id;
|
|
|
|
return RESULT_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2018-01-06 19:19:28 +00:00
|
|
|
/// Default thread wakeup callback for WaitSynchronization
|
2018-01-08 16:35:03 +00:00
|
|
|
static bool DefaultThreadWakeupCallback(ThreadWakeupReason reason, SharedPtr<Thread> thread,
|
|
|
|
SharedPtr<WaitObject> object, size_t index) {
|
2018-01-06 19:19:28 +00:00
|
|
|
ASSERT(thread->status == THREADSTATUS_WAIT_SYNCH_ANY);
|
|
|
|
|
|
|
|
if (reason == ThreadWakeupReason::Timeout) {
|
|
|
|
thread->SetWaitSynchronizationResult(RESULT_TIMEOUT);
|
2018-01-08 16:35:03 +00:00
|
|
|
return true;
|
2018-01-06 19:19:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
ASSERT(reason == ThreadWakeupReason::Signal);
|
|
|
|
thread->SetWaitSynchronizationResult(RESULT_SUCCESS);
|
2018-01-09 20:02:43 +00:00
|
|
|
thread->SetWaitSynchronizationOutput(static_cast<u32>(index));
|
2018-01-08 16:35:03 +00:00
|
|
|
return true;
|
2018-01-06 19:19:28 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/// Wait for a kernel object to synchronize, timeout after the specified nanoseconds
|
|
|
|
static ResultCode WaitSynchronization1(
|
|
|
|
SharedPtr<WaitObject> object, Thread* thread, s64 nano_seconds = -1,
|
|
|
|
std::function<Thread::WakeupCallback> wakeup_callback = DefaultThreadWakeupCallback) {
|
|
|
|
|
|
|
|
if (!object) {
|
|
|
|
return ERR_INVALID_HANDLE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (object->ShouldWait(thread)) {
|
|
|
|
if (nano_seconds == 0) {
|
|
|
|
return RESULT_TIMEOUT;
|
|
|
|
}
|
|
|
|
|
|
|
|
thread->wait_objects = {object};
|
|
|
|
object->AddWaitingThread(thread);
|
|
|
|
thread->status = THREADSTATUS_WAIT_SYNCH_ANY;
|
|
|
|
|
|
|
|
// Create an event to wake the thread up after the specified nanosecond delay has passed
|
|
|
|
thread->WakeAfterDelay(nano_seconds);
|
|
|
|
thread->wakeup_callback = wakeup_callback;
|
|
|
|
|
|
|
|
Core::System::GetInstance().PrepareReschedule();
|
|
|
|
} else {
|
|
|
|
object->Acquire(thread);
|
|
|
|
}
|
|
|
|
|
|
|
|
return RESULT_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2018-01-01 19:47:57 +00:00
|
|
|
/// Wait for the given handles to synchronize, timeout after the specified nanoseconds
|
2018-01-09 16:53:50 +00:00
|
|
|
static ResultCode WaitSynchronization(Handle* index, VAddr handles_address, u64 handle_count,
|
|
|
|
s64 nano_seconds) {
|
2018-01-06 19:34:32 +00:00
|
|
|
LOG_TRACE(Kernel_SVC, "called handles_address=0x%llx, handle_count=%d, nano_seconds=%d",
|
|
|
|
handles_address, handle_count, nano_seconds);
|
|
|
|
|
|
|
|
if (!Memory::IsValidVirtualAddress(handles_address))
|
|
|
|
return ERR_INVALID_POINTER;
|
|
|
|
|
2018-01-09 20:02:43 +00:00
|
|
|
static constexpr u64 MaxHandles = 0x40;
|
|
|
|
|
|
|
|
if (handle_count > MaxHandles)
|
|
|
|
return ResultCode(ErrorModule::Kernel, ErrCodes::TooLarge);
|
2018-01-06 19:34:32 +00:00
|
|
|
|
2018-01-09 16:53:50 +00:00
|
|
|
auto thread = GetCurrentThread();
|
|
|
|
|
2018-01-06 19:34:32 +00:00
|
|
|
using ObjectPtr = SharedPtr<WaitObject>;
|
|
|
|
std::vector<ObjectPtr> objects(handle_count);
|
|
|
|
|
|
|
|
for (int i = 0; i < handle_count; ++i) {
|
|
|
|
Handle handle = Memory::Read32(handles_address + i * sizeof(Handle));
|
|
|
|
auto object = g_handle_table.Get<WaitObject>(handle);
|
|
|
|
if (object == nullptr)
|
|
|
|
return ERR_INVALID_HANDLE;
|
|
|
|
objects[i] = object;
|
|
|
|
}
|
|
|
|
|
2018-01-09 16:53:50 +00:00
|
|
|
// Find the first object that is acquirable in the provided list of objects
|
|
|
|
auto itr = std::find_if(objects.begin(), objects.end(), [thread](const ObjectPtr& object) {
|
|
|
|
return !object->ShouldWait(thread);
|
|
|
|
});
|
|
|
|
|
|
|
|
if (itr != objects.end()) {
|
|
|
|
// We found a ready object, acquire it and set the result value
|
|
|
|
WaitObject* object = itr->get();
|
|
|
|
object->Acquire(thread);
|
|
|
|
*index = static_cast<s32>(std::distance(objects.begin(), itr));
|
|
|
|
return RESULT_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
// No objects were ready to be acquired, prepare to suspend the thread.
|
|
|
|
|
|
|
|
// If a timeout value of 0 was provided, just return the Timeout error code instead of
|
|
|
|
// suspending the thread.
|
|
|
|
if (nano_seconds == 0)
|
|
|
|
return RESULT_TIMEOUT;
|
|
|
|
|
2018-01-09 20:02:43 +00:00
|
|
|
for (auto& object : objects)
|
|
|
|
object->AddWaitingThread(thread);
|
|
|
|
|
|
|
|
thread->wait_objects = std::move(objects);
|
|
|
|
thread->status = THREADSTATUS_WAIT_SYNCH_ANY;
|
|
|
|
|
|
|
|
// Create an event to wake the thread up after the specified nanosecond delay has passed
|
|
|
|
thread->WakeAfterDelay(nano_seconds);
|
|
|
|
thread->wakeup_callback = DefaultThreadWakeupCallback;
|
|
|
|
|
|
|
|
Core::System::GetInstance().PrepareReschedule();
|
|
|
|
|
|
|
|
return RESULT_TIMEOUT;
|
2018-01-01 19:47:57 +00:00
|
|
|
}
|
|
|
|
|
2018-01-09 20:02:04 +00:00
|
|
|
/// Resumes a thread waiting on WaitSynchronization
|
|
|
|
static ResultCode CancelSynchronization(Handle thread_handle) {
|
|
|
|
LOG_TRACE(Kernel_SVC, "called thread=0x%08X", thread_handle);
|
|
|
|
|
|
|
|
const SharedPtr<Thread> thread = g_handle_table.Get<Thread>(thread_handle);
|
|
|
|
if (!thread) {
|
|
|
|
return ERR_INVALID_HANDLE;
|
|
|
|
}
|
|
|
|
|
|
|
|
ASSERT(thread->status == THREADSTATUS_WAIT_SYNCH_ANY);
|
|
|
|
thread->SetWaitSynchronizationResult(
|
|
|
|
ResultCode(ErrorModule::Kernel, ErrCodes::SynchronizationCanceled));
|
|
|
|
thread->ResumeFromWait();
|
|
|
|
return RESULT_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2018-01-01 19:02:26 +00:00
|
|
|
/// Attempts to locks a mutex, creating it if it does not already exist
|
2018-01-18 01:34:52 +00:00
|
|
|
static ResultCode ArbitrateLock(Handle holding_thread_handle, VAddr mutex_addr,
|
|
|
|
Handle requesting_thread_handle) {
|
2018-01-20 07:48:02 +00:00
|
|
|
LOG_TRACE(Kernel_SVC,
|
|
|
|
"called holding_thread_handle=0x%08X, mutex_addr=0x%llx, "
|
|
|
|
"requesting_current_thread_handle=0x%08X",
|
2018-01-01 19:02:26 +00:00
|
|
|
holding_thread_handle, mutex_addr, requesting_thread_handle);
|
|
|
|
|
2018-01-03 03:24:12 +00:00
|
|
|
SharedPtr<Thread> holding_thread = g_handle_table.Get<Thread>(holding_thread_handle);
|
|
|
|
SharedPtr<Thread> requesting_thread = g_handle_table.Get<Thread>(requesting_thread_handle);
|
2018-01-01 19:02:26 +00:00
|
|
|
|
|
|
|
ASSERT(requesting_thread);
|
2018-02-03 18:29:18 +00:00
|
|
|
ASSERT(requesting_thread == GetCurrentThread());
|
2018-01-01 19:02:26 +00:00
|
|
|
|
2018-01-03 01:40:30 +00:00
|
|
|
SharedPtr<Mutex> mutex = g_object_address_table.Get<Mutex>(mutex_addr);
|
2018-01-01 19:02:26 +00:00
|
|
|
if (!mutex) {
|
|
|
|
// Create a new mutex for the specified address if one does not already exist
|
2018-01-03 01:40:30 +00:00
|
|
|
mutex = Mutex::Create(holding_thread, mutex_addr);
|
2018-01-01 19:02:26 +00:00
|
|
|
mutex->name = Common::StringFromFormat("mutex-%llx", mutex_addr);
|
|
|
|
}
|
|
|
|
|
2018-01-08 19:12:03 +00:00
|
|
|
ASSERT(holding_thread == mutex->GetHoldingThread());
|
|
|
|
|
2018-01-06 19:19:28 +00:00
|
|
|
return WaitSynchronization1(mutex, requesting_thread.get());
|
2018-01-01 19:02:26 +00:00
|
|
|
}
|
|
|
|
|
2018-01-01 19:04:36 +00:00
|
|
|
/// Unlock a mutex
|
2018-01-18 01:34:52 +00:00
|
|
|
static ResultCode ArbitrateUnlock(VAddr mutex_addr) {
|
2018-01-01 19:04:36 +00:00
|
|
|
LOG_TRACE(Kernel_SVC, "called mutex_addr=0x%llx", mutex_addr);
|
|
|
|
|
2018-01-03 01:40:30 +00:00
|
|
|
SharedPtr<Mutex> mutex = g_object_address_table.Get<Mutex>(mutex_addr);
|
2018-01-01 19:04:36 +00:00
|
|
|
ASSERT(mutex);
|
|
|
|
|
2018-01-03 01:40:30 +00:00
|
|
|
return mutex->Release(GetCurrentThread());
|
2018-01-01 19:04:36 +00:00
|
|
|
}
|
|
|
|
|
2017-10-14 22:30:07 +01:00
|
|
|
/// Break program execution
|
|
|
|
static void Break(u64 unk_0, u64 unk_1, u64 unk_2) {
|
|
|
|
LOG_CRITICAL(Debug_Emulated, "Emulated program broke execution!");
|
|
|
|
ASSERT(false);
|
2014-04-17 01:41:33 +01:00
|
|
|
}
|
|
|
|
|
2017-10-14 22:30:07 +01:00
|
|
|
/// Used to output a message on a debug hardware unit - does nothing on a retail unit
|
2018-01-01 19:38:34 +00:00
|
|
|
static void OutputDebugString(VAddr address, s32 len) {
|
2017-10-14 22:30:07 +01:00
|
|
|
std::vector<char> string(len);
|
|
|
|
Memory::ReadBlock(address, string.data(), len);
|
|
|
|
LOG_DEBUG(Debug_Emulated, "%.*s", len, string.data());
|
2014-05-18 04:37:25 +01:00
|
|
|
}
|
|
|
|
|
2018-01-01 21:01:06 +00:00
|
|
|
/// Gets system/memory information for the current process
|
2017-10-14 22:30:07 +01:00
|
|
|
static ResultCode GetInfo(u64* result, u64 info_id, u64 handle, u64 info_sub_id) {
|
2018-01-01 21:01:06 +00:00
|
|
|
LOG_TRACE(Kernel_SVC, "called info_id=0x%X, info_sub_id=0x%X, handle=0x%08X", info_id,
|
|
|
|
info_sub_id, handle);
|
|
|
|
|
2018-03-13 21:49:59 +00:00
|
|
|
auto& vm_manager = Core::CurrentProcess()->vm_manager;
|
2018-01-10 05:58:25 +00:00
|
|
|
|
2018-01-01 21:01:06 +00:00
|
|
|
switch (static_cast<GetInfoType>(info_id)) {
|
2018-01-10 05:58:25 +00:00
|
|
|
case GetInfoType::AllowedCpuIdBitmask:
|
2018-03-13 21:49:59 +00:00
|
|
|
*result = Core::CurrentProcess()->allowed_processor_mask;
|
2018-01-10 05:58:25 +00:00
|
|
|
break;
|
2018-01-16 22:06:45 +00:00
|
|
|
case GetInfoType::AllowedThreadPrioBitmask:
|
2018-03-13 21:49:59 +00:00
|
|
|
*result = Core::CurrentProcess()->allowed_thread_priority_mask;
|
2018-01-16 22:06:45 +00:00
|
|
|
break;
|
|
|
|
case GetInfoType::MapRegionBaseAddr:
|
2018-03-15 02:09:22 +00:00
|
|
|
*result = Memory::MAP_REGION_VADDR;
|
2018-01-16 22:06:45 +00:00
|
|
|
break;
|
|
|
|
case GetInfoType::MapRegionSize:
|
2018-03-15 02:09:22 +00:00
|
|
|
*result = Memory::MAP_REGION_SIZE;
|
2018-01-16 22:06:45 +00:00
|
|
|
break;
|
2018-01-15 20:42:57 +00:00
|
|
|
case GetInfoType::HeapRegionBaseAddr:
|
2018-03-15 02:09:22 +00:00
|
|
|
*result = Memory::HEAP_VADDR;
|
2018-01-15 20:42:57 +00:00
|
|
|
break;
|
|
|
|
case GetInfoType::HeapRegionSize:
|
|
|
|
*result = Memory::HEAP_SIZE;
|
|
|
|
break;
|
2018-01-01 21:01:06 +00:00
|
|
|
case GetInfoType::TotalMemoryUsage:
|
|
|
|
*result = vm_manager.GetTotalMemoryUsage();
|
|
|
|
break;
|
|
|
|
case GetInfoType::TotalHeapUsage:
|
|
|
|
*result = vm_manager.GetTotalHeapUsage();
|
|
|
|
break;
|
2018-02-04 17:34:45 +00:00
|
|
|
case GetInfoType::IsCurrentProcessBeingDebugged:
|
|
|
|
*result = 0;
|
|
|
|
break;
|
2018-01-01 21:01:06 +00:00
|
|
|
case GetInfoType::RandomEntropy:
|
|
|
|
*result = 0;
|
|
|
|
break;
|
|
|
|
case GetInfoType::AddressSpaceBaseAddr:
|
|
|
|
*result = vm_manager.GetAddressSpaceBaseAddr();
|
|
|
|
break;
|
|
|
|
case GetInfoType::AddressSpaceSize:
|
|
|
|
*result = vm_manager.GetAddressSpaceSize();
|
|
|
|
break;
|
|
|
|
case GetInfoType::NewMapRegionBaseAddr:
|
2018-03-15 02:09:22 +00:00
|
|
|
*result = Memory::NEW_MAP_REGION_VADDR;
|
2018-01-01 21:01:06 +00:00
|
|
|
break;
|
|
|
|
case GetInfoType::NewMapRegionSize:
|
2018-03-15 02:09:22 +00:00
|
|
|
*result = Memory::NEW_MAP_REGION_SIZE;
|
2018-01-01 21:01:06 +00:00
|
|
|
break;
|
2018-01-16 22:06:45 +00:00
|
|
|
case GetInfoType::IsVirtualAddressMemoryEnabled:
|
2018-03-13 21:49:59 +00:00
|
|
|
*result = Core::CurrentProcess()->is_virtual_address_memory_enabled;
|
2018-01-16 22:06:45 +00:00
|
|
|
break;
|
2018-01-15 20:42:57 +00:00
|
|
|
case GetInfoType::TitleId:
|
|
|
|
LOG_WARNING(Kernel_SVC, "(STUBBED) Attempted to query titleid, returned 0");
|
|
|
|
*result = 0;
|
|
|
|
break;
|
|
|
|
case GetInfoType::PrivilegedProcessId:
|
|
|
|
LOG_WARNING(Kernel_SVC,
|
|
|
|
"(STUBBED) Attempted to query priviledged process id bounds, returned 0");
|
|
|
|
*result = 0;
|
|
|
|
break;
|
2018-01-01 21:01:06 +00:00
|
|
|
default:
|
|
|
|
UNIMPLEMENTED();
|
2015-05-17 06:06:59 +01:00
|
|
|
}
|
2018-01-01 21:01:06 +00:00
|
|
|
|
2015-01-23 05:36:58 +00:00
|
|
|
return RESULT_SUCCESS;
|
2014-05-01 23:50:36 +01:00
|
|
|
}
|
|
|
|
|
2014-06-02 03:12:54 +01:00
|
|
|
/// Gets the priority for the specified thread
|
2017-12-31 21:06:11 +00:00
|
|
|
static ResultCode GetThreadPriority(u32* priority, Handle handle) {
|
2018-01-03 01:40:30 +00:00
|
|
|
const SharedPtr<Thread> thread = g_handle_table.Get<Thread>(handle);
|
2017-12-31 21:06:11 +00:00
|
|
|
if (!thread)
|
|
|
|
return ERR_INVALID_HANDLE;
|
|
|
|
|
|
|
|
*priority = thread->GetPriority();
|
2015-01-23 05:36:58 +00:00
|
|
|
return RESULT_SUCCESS;
|
2014-12-03 23:49:51 +00:00
|
|
|
}
|
|
|
|
|
2017-12-31 20:58:16 +00:00
|
|
|
/// Sets the priority for the specified thread
|
|
|
|
static ResultCode SetThreadPriority(Handle handle, u32 priority) {
|
|
|
|
if (priority > THREADPRIO_LOWEST) {
|
2018-01-03 01:40:30 +00:00
|
|
|
return ERR_OUT_OF_RANGE;
|
2017-12-31 20:58:16 +00:00
|
|
|
}
|
|
|
|
|
2018-01-03 01:40:30 +00:00
|
|
|
SharedPtr<Thread> thread = g_handle_table.Get<Thread>(handle);
|
2017-12-31 20:58:16 +00:00
|
|
|
if (!thread)
|
2018-01-03 01:40:30 +00:00
|
|
|
return ERR_INVALID_HANDLE;
|
2017-12-31 20:58:16 +00:00
|
|
|
|
|
|
|
// Note: The kernel uses the current process's resource limit instead of
|
|
|
|
// the one from the thread owner's resource limit.
|
2018-03-13 21:49:59 +00:00
|
|
|
SharedPtr<ResourceLimit>& resource_limit = Core::CurrentProcess()->resource_limit;
|
2018-01-03 01:40:30 +00:00
|
|
|
if (resource_limit->GetMaxResourceValue(ResourceTypes::PRIORITY) > priority) {
|
|
|
|
return ERR_NOT_AUTHORIZED;
|
2017-12-31 20:58:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
thread->SetPriority(priority);
|
|
|
|
thread->UpdatePriority();
|
|
|
|
|
|
|
|
// Update the mutexes that this thread is waiting for
|
|
|
|
for (auto& mutex : thread->pending_mutexes)
|
|
|
|
mutex->UpdatePriority();
|
|
|
|
|
|
|
|
Core::System::GetInstance().PrepareReschedule();
|
|
|
|
return RESULT_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2017-12-31 21:01:04 +00:00
|
|
|
/// Get which CPU core is executing the current thread
|
|
|
|
static u32 GetCurrentProcessorNumber() {
|
|
|
|
LOG_WARNING(Kernel_SVC, "(STUBBED) called, defaulting to processor 0");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-01-14 22:15:31 +00:00
|
|
|
static ResultCode MapSharedMemory(Handle shared_memory_handle, VAddr addr, u64 size,
|
|
|
|
u32 permissions) {
|
|
|
|
LOG_TRACE(Kernel_SVC,
|
|
|
|
"called, shared_memory_handle=0x%08X, addr=0x%llx, size=0x%llx, permissions=0x%08X",
|
|
|
|
shared_memory_handle, addr, size, permissions);
|
|
|
|
|
2018-02-03 18:36:54 +00:00
|
|
|
SharedPtr<SharedMemory> shared_memory = g_handle_table.Get<SharedMemory>(shared_memory_handle);
|
2018-01-14 22:15:31 +00:00
|
|
|
if (!shared_memory) {
|
|
|
|
return ERR_INVALID_HANDLE;
|
|
|
|
}
|
|
|
|
|
|
|
|
MemoryPermission permissions_type = static_cast<MemoryPermission>(permissions);
|
|
|
|
switch (permissions_type) {
|
|
|
|
case MemoryPermission::Read:
|
|
|
|
case MemoryPermission::Write:
|
|
|
|
case MemoryPermission::ReadWrite:
|
|
|
|
case MemoryPermission::Execute:
|
|
|
|
case MemoryPermission::ReadExecute:
|
|
|
|
case MemoryPermission::WriteExecute:
|
|
|
|
case MemoryPermission::ReadWriteExecute:
|
|
|
|
case MemoryPermission::DontCare:
|
2018-03-13 21:49:59 +00:00
|
|
|
return shared_memory->Map(Core::CurrentProcess().get(), addr, permissions_type,
|
2018-01-14 22:15:31 +00:00
|
|
|
MemoryPermission::DontCare);
|
|
|
|
default:
|
|
|
|
LOG_ERROR(Kernel_SVC, "unknown permissions=0x%08X", permissions);
|
|
|
|
}
|
|
|
|
|
|
|
|
return RESULT_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2018-02-22 19:16:43 +00:00
|
|
|
static ResultCode UnmapSharedMemory(Handle shared_memory_handle, VAddr addr, u64 size) {
|
|
|
|
LOG_WARNING(Kernel_SVC,
|
|
|
|
"called, shared_memory_handle=0x%08X, addr=0x%" PRIx64 ", size=0x%" PRIx64 "",
|
|
|
|
shared_memory_handle, addr, size);
|
|
|
|
|
|
|
|
SharedPtr<SharedMemory> shared_memory = g_handle_table.Get<SharedMemory>(shared_memory_handle);
|
|
|
|
|
2018-03-13 21:49:59 +00:00
|
|
|
return shared_memory->Unmap(Core::CurrentProcess().get(), addr);
|
2018-02-22 19:16:43 +00:00
|
|
|
}
|
|
|
|
|
2015-07-17 20:45:12 +01:00
|
|
|
/// Query process memory
|
2017-10-14 22:30:07 +01:00
|
|
|
static ResultCode QueryProcessMemory(MemoryInfo* memory_info, PageInfo* /*page_info*/,
|
2018-01-03 01:40:30 +00:00
|
|
|
Handle process_handle, u64 addr) {
|
|
|
|
SharedPtr<Process> process = g_handle_table.Get<Process>(process_handle);
|
2017-12-30 18:40:28 +00:00
|
|
|
if (!process) {
|
2015-07-17 20:45:12 +01:00
|
|
|
return ERR_INVALID_HANDLE;
|
2017-10-20 04:00:46 +01:00
|
|
|
}
|
2015-07-18 03:19:16 +01:00
|
|
|
auto vma = process->vm_manager.FindVMA(addr);
|
2017-10-20 04:00:46 +01:00
|
|
|
memory_info->attributes = 0;
|
2018-03-13 21:49:59 +00:00
|
|
|
if (vma == Core::CurrentProcess()->vm_manager.vma_map.end()) {
|
2017-10-14 22:30:07 +01:00
|
|
|
memory_info->base_address = 0;
|
2018-01-03 01:40:30 +00:00
|
|
|
memory_info->permission = static_cast<u32>(VMAPermission::None);
|
2017-10-14 22:30:07 +01:00
|
|
|
memory_info->size = 0;
|
2018-03-10 22:46:23 +00:00
|
|
|
memory_info->type = static_cast<u32>(MemoryState::Unmapped);
|
2017-10-20 04:00:46 +01:00
|
|
|
} else {
|
|
|
|
memory_info->base_address = vma->second.base;
|
|
|
|
memory_info->permission = static_cast<u32>(vma->second.permissions);
|
|
|
|
memory_info->size = vma->second.size;
|
|
|
|
memory_info->type = static_cast<u32>(vma->second.meminfo_state);
|
2017-10-14 22:30:07 +01:00
|
|
|
}
|
2017-12-30 18:40:28 +00:00
|
|
|
|
2017-10-14 22:30:07 +01:00
|
|
|
LOG_TRACE(Kernel_SVC, "called process=0x%08X addr=%llx", process_handle, addr);
|
2015-01-23 05:36:58 +00:00
|
|
|
return RESULT_SUCCESS;
|
2014-05-16 01:17:30 +01:00
|
|
|
}
|
|
|
|
|
2015-07-17 20:45:12 +01:00
|
|
|
/// Query memory
|
2017-10-14 22:30:07 +01:00
|
|
|
static ResultCode QueryMemory(MemoryInfo* memory_info, PageInfo* page_info, VAddr addr) {
|
2017-10-14 22:35:21 +01:00
|
|
|
LOG_TRACE(Kernel_SVC, "called, addr=%llx", addr);
|
2018-01-03 01:40:30 +00:00
|
|
|
return QueryProcessMemory(memory_info, page_info, CurrentProcess, addr);
|
2015-07-17 20:45:12 +01:00
|
|
|
}
|
|
|
|
|
2018-01-01 19:38:34 +00:00
|
|
|
/// Exits the current process
|
|
|
|
static void ExitProcess() {
|
2018-03-13 21:49:59 +00:00
|
|
|
LOG_INFO(Kernel_SVC, "Process %u exiting", Core::CurrentProcess()->process_id);
|
2018-01-01 19:38:34 +00:00
|
|
|
|
2018-03-13 21:49:59 +00:00
|
|
|
ASSERT_MSG(Core::CurrentProcess()->status == ProcessStatus::Running,
|
|
|
|
"Process has already exited");
|
2018-01-01 19:38:34 +00:00
|
|
|
|
2018-03-13 21:49:59 +00:00
|
|
|
Core::CurrentProcess()->status = ProcessStatus::Exited;
|
2018-01-01 19:38:34 +00:00
|
|
|
|
|
|
|
// Stop all the process threads that are currently waiting for objects.
|
2018-02-18 20:17:16 +00:00
|
|
|
auto& thread_list = Core::System::GetInstance().Scheduler().GetThreadList();
|
2018-01-01 19:38:34 +00:00
|
|
|
for (auto& thread : thread_list) {
|
2018-03-13 21:49:59 +00:00
|
|
|
if (thread->owner_process != Core::CurrentProcess())
|
2018-01-01 19:38:34 +00:00
|
|
|
continue;
|
|
|
|
|
2018-01-03 01:40:30 +00:00
|
|
|
if (thread == GetCurrentThread())
|
2018-01-01 19:38:34 +00:00
|
|
|
continue;
|
|
|
|
|
|
|
|
// TODO(Subv): When are the other running/ready threads terminated?
|
|
|
|
ASSERT_MSG(thread->status == THREADSTATUS_WAIT_SYNCH_ANY ||
|
|
|
|
thread->status == THREADSTATUS_WAIT_SYNCH_ALL,
|
|
|
|
"Exiting processes with non-waiting threads is currently unimplemented");
|
|
|
|
|
|
|
|
thread->Stop();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Kill the current thread
|
2018-01-03 01:40:30 +00:00
|
|
|
GetCurrentThread()->Stop();
|
2018-01-01 19:38:34 +00:00
|
|
|
|
|
|
|
Core::System::GetInstance().PrepareReschedule();
|
|
|
|
}
|
|
|
|
|
2017-12-31 21:10:01 +00:00
|
|
|
/// Creates a new thread
|
|
|
|
static ResultCode CreateThread(Handle* out_handle, VAddr entry_point, u64 arg, VAddr stack_top,
|
|
|
|
u32 priority, s32 processor_id) {
|
2018-01-01 20:48:08 +00:00
|
|
|
std::string name = Common::StringFromFormat("unknown-%llx", entry_point);
|
2017-12-31 21:10:01 +00:00
|
|
|
|
|
|
|
if (priority > THREADPRIO_LOWEST) {
|
2018-01-03 01:40:30 +00:00
|
|
|
return ERR_OUT_OF_RANGE;
|
2017-12-31 21:10:01 +00:00
|
|
|
}
|
|
|
|
|
2018-03-13 21:49:59 +00:00
|
|
|
SharedPtr<ResourceLimit>& resource_limit = Core::CurrentProcess()->resource_limit;
|
2018-01-03 01:40:30 +00:00
|
|
|
if (resource_limit->GetMaxResourceValue(ResourceTypes::PRIORITY) > priority) {
|
|
|
|
return ERR_NOT_AUTHORIZED;
|
2017-12-31 21:10:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (processor_id == THREADPROCESSORID_DEFAULT) {
|
|
|
|
// Set the target CPU to the one specified in the process' exheader.
|
2018-03-13 21:49:59 +00:00
|
|
|
processor_id = Core::CurrentProcess()->ideal_processor;
|
2017-12-31 21:10:01 +00:00
|
|
|
ASSERT(processor_id != THREADPROCESSORID_DEFAULT);
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (processor_id) {
|
|
|
|
case THREADPROCESSORID_0:
|
|
|
|
break;
|
|
|
|
case THREADPROCESSORID_1:
|
2018-01-10 05:58:25 +00:00
|
|
|
case THREADPROCESSORID_2:
|
|
|
|
case THREADPROCESSORID_3:
|
|
|
|
// TODO(bunnei): Implement support for other processor IDs
|
2017-12-31 21:10:01 +00:00
|
|
|
LOG_ERROR(Kernel_SVC,
|
2018-01-10 05:58:25 +00:00
|
|
|
"Newly created thread must run in another thread (%u), unimplemented.",
|
|
|
|
processor_id);
|
2017-12-31 21:10:01 +00:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
ASSERT_MSG(false, "Unsupported thread processor ID: %d", processor_id);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2018-01-03 01:40:30 +00:00
|
|
|
CASCADE_RESULT(SharedPtr<Thread> thread,
|
|
|
|
Thread::Create(name, entry_point, priority, arg, processor_id, stack_top,
|
2018-03-13 21:49:59 +00:00
|
|
|
Core::CurrentProcess()));
|
2018-01-03 01:40:30 +00:00
|
|
|
CASCADE_RESULT(thread->guest_handle, g_handle_table.Create(thread));
|
2017-12-31 22:23:36 +00:00
|
|
|
*out_handle = thread->guest_handle;
|
2017-12-31 21:10:01 +00:00
|
|
|
|
|
|
|
Core::System::GetInstance().PrepareReschedule();
|
|
|
|
|
2018-01-20 07:48:02 +00:00
|
|
|
LOG_TRACE(Kernel_SVC,
|
|
|
|
"called entrypoint=0x%08X (%s), arg=0x%08X, stacktop=0x%08X, "
|
|
|
|
"threadpriority=0x%08X, processorid=0x%08X : created handle=0x%08X",
|
2017-12-31 21:10:01 +00:00
|
|
|
entry_point, name.c_str(), arg, stack_top, priority, processor_id, *out_handle);
|
|
|
|
|
|
|
|
return RESULT_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2017-12-30 18:40:28 +00:00
|
|
|
/// Starts the thread for the provided handle
|
2017-12-30 18:37:07 +00:00
|
|
|
static ResultCode StartThread(Handle thread_handle) {
|
|
|
|
LOG_TRACE(Kernel_SVC, "called thread=0x%08X", thread_handle);
|
|
|
|
|
2018-01-03 03:24:12 +00:00
|
|
|
const SharedPtr<Thread> thread = g_handle_table.Get<Thread>(thread_handle);
|
2017-12-30 18:37:07 +00:00
|
|
|
if (!thread) {
|
|
|
|
return ERR_INVALID_HANDLE;
|
|
|
|
}
|
|
|
|
|
|
|
|
thread->ResumeFromWait();
|
|
|
|
|
|
|
|
return RESULT_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2017-12-31 21:11:27 +00:00
|
|
|
/// Called when a thread exits
|
|
|
|
static void ExitThread() {
|
|
|
|
LOG_TRACE(Kernel_SVC, "called, pc=0x%08X", Core::CPU().GetPC());
|
|
|
|
|
2018-01-03 01:40:30 +00:00
|
|
|
ExitCurrentThread();
|
2017-12-31 21:11:27 +00:00
|
|
|
Core::System::GetInstance().PrepareReschedule();
|
|
|
|
}
|
|
|
|
|
2014-06-01 15:37:19 +01:00
|
|
|
/// Sleep the current thread
|
2014-11-17 03:58:39 +00:00
|
|
|
static void SleepThread(s64 nanoseconds) {
|
2014-12-06 01:53:49 +00:00
|
|
|
LOG_TRACE(Kernel_SVC, "called nanoseconds=%lld", nanoseconds);
|
2014-11-26 05:34:14 +00:00
|
|
|
|
2017-01-05 19:14:22 +00:00
|
|
|
// Don't attempt to yield execution if there are no available threads to run,
|
|
|
|
// this way we avoid a useless reschedule to the idle thread.
|
2018-02-18 20:17:16 +00:00
|
|
|
if (nanoseconds == 0 && !Core::System::GetInstance().Scheduler().HaveReadyThreads())
|
2017-01-05 19:14:22 +00:00
|
|
|
return;
|
|
|
|
|
2014-12-20 07:32:19 +00:00
|
|
|
// Sleep current thread and check for next thread to schedule
|
2018-01-03 01:40:30 +00:00
|
|
|
WaitCurrentThread_Sleep();
|
2015-01-07 21:40:08 +00:00
|
|
|
|
|
|
|
// Create an event to wake the thread up after the specified nanosecond delay has passed
|
2018-01-03 01:40:30 +00:00
|
|
|
GetCurrentThread()->WakeAfterDelay(nanoseconds);
|
2017-01-01 16:57:02 +00:00
|
|
|
|
|
|
|
Core::System::GetInstance().PrepareReschedule();
|
2014-06-01 15:37:19 +01:00
|
|
|
}
|
|
|
|
|
2018-01-06 21:14:12 +00:00
|
|
|
/// Signal process wide key atomic
|
2018-01-09 02:41:37 +00:00
|
|
|
static ResultCode WaitProcessWideKeyAtomic(VAddr mutex_addr, VAddr condition_variable_addr,
|
2018-01-06 21:14:12 +00:00
|
|
|
Handle thread_handle, s64 nano_seconds) {
|
2018-01-09 02:41:37 +00:00
|
|
|
LOG_TRACE(
|
|
|
|
Kernel_SVC,
|
|
|
|
"called mutex_addr=%llx, condition_variable_addr=%llx, thread_handle=0x%08X, timeout=%d",
|
|
|
|
mutex_addr, condition_variable_addr, thread_handle, nano_seconds);
|
2018-01-06 21:14:12 +00:00
|
|
|
|
|
|
|
SharedPtr<Thread> thread = g_handle_table.Get<Thread>(thread_handle);
|
|
|
|
ASSERT(thread);
|
|
|
|
|
|
|
|
SharedPtr<Mutex> mutex = g_object_address_table.Get<Mutex>(mutex_addr);
|
|
|
|
if (!mutex) {
|
|
|
|
// Create a new mutex for the specified address if one does not already exist
|
|
|
|
mutex = Mutex::Create(thread, mutex_addr);
|
|
|
|
mutex->name = Common::StringFromFormat("mutex-%llx", mutex_addr);
|
|
|
|
}
|
|
|
|
|
2018-01-09 02:41:37 +00:00
|
|
|
SharedPtr<ConditionVariable> condition_variable =
|
|
|
|
g_object_address_table.Get<ConditionVariable>(condition_variable_addr);
|
|
|
|
if (!condition_variable) {
|
|
|
|
// Create a new condition_variable for the specified address if one does not already exist
|
2018-02-04 17:30:51 +00:00
|
|
|
condition_variable = ConditionVariable::Create(condition_variable_addr).Unwrap();
|
2018-01-09 02:41:37 +00:00
|
|
|
condition_variable->name =
|
|
|
|
Common::StringFromFormat("condition-variable-%llx", condition_variable_addr);
|
2018-01-06 21:14:12 +00:00
|
|
|
}
|
|
|
|
|
2018-02-04 17:30:51 +00:00
|
|
|
if (condition_variable->mutex_addr) {
|
|
|
|
// Previously created the ConditionVariable using WaitProcessWideKeyAtomic, verify
|
|
|
|
// everything is correct
|
|
|
|
ASSERT(condition_variable->mutex_addr == mutex_addr);
|
|
|
|
} else {
|
|
|
|
// Previously created the ConditionVariable using SignalProcessWideKey, set the mutex
|
|
|
|
// associated with it
|
|
|
|
condition_variable->mutex_addr = mutex_addr;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (mutex->GetOwnerHandle()) {
|
|
|
|
// Release the mutex if the current thread is holding it
|
|
|
|
mutex->Release(thread.get());
|
|
|
|
}
|
2018-01-06 21:14:12 +00:00
|
|
|
|
2018-01-08 16:35:03 +00:00
|
|
|
auto wakeup_callback = [mutex, nano_seconds](ThreadWakeupReason reason,
|
|
|
|
SharedPtr<Thread> thread,
|
|
|
|
SharedPtr<WaitObject> object, size_t index) {
|
|
|
|
ASSERT(thread->status == THREADSTATUS_WAIT_SYNCH_ANY);
|
|
|
|
|
|
|
|
if (reason == ThreadWakeupReason::Timeout) {
|
|
|
|
thread->SetWaitSynchronizationResult(RESULT_TIMEOUT);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
ASSERT(reason == ThreadWakeupReason::Signal);
|
|
|
|
|
|
|
|
// Now try to acquire the mutex and don't resume if it's not available.
|
|
|
|
if (!mutex->ShouldWait(thread.get())) {
|
|
|
|
mutex->Acquire(thread.get());
|
|
|
|
thread->SetWaitSynchronizationResult(RESULT_SUCCESS);
|
|
|
|
return true;
|
|
|
|
}
|
2018-01-06 21:14:12 +00:00
|
|
|
|
2018-01-08 16:35:03 +00:00
|
|
|
if (nano_seconds == 0) {
|
|
|
|
thread->SetWaitSynchronizationResult(RESULT_TIMEOUT);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
thread->wait_objects = {mutex};
|
|
|
|
mutex->AddWaitingThread(thread);
|
|
|
|
thread->status = THREADSTATUS_WAIT_SYNCH_ANY;
|
|
|
|
|
|
|
|
// Create an event to wake the thread up after the
|
|
|
|
// specified nanosecond delay has passed
|
|
|
|
thread->WakeAfterDelay(nano_seconds);
|
|
|
|
thread->wakeup_callback = DefaultThreadWakeupCallback;
|
|
|
|
|
|
|
|
Core::System::GetInstance().PrepareReschedule();
|
2018-01-06 21:14:12 +00:00
|
|
|
|
2018-01-08 16:35:03 +00:00
|
|
|
return false;
|
|
|
|
};
|
2018-01-09 02:41:37 +00:00
|
|
|
CASCADE_CODE(
|
|
|
|
WaitSynchronization1(condition_variable, thread.get(), nano_seconds, wakeup_callback));
|
2018-01-06 21:14:12 +00:00
|
|
|
|
|
|
|
return RESULT_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2017-10-14 22:30:07 +01:00
|
|
|
/// Signal process wide key
|
2018-01-09 02:41:37 +00:00
|
|
|
static ResultCode SignalProcessWideKey(VAddr condition_variable_addr, s32 target) {
|
|
|
|
LOG_TRACE(Kernel_SVC, "called, condition_variable_addr=0x%llx, target=0x%08x",
|
|
|
|
condition_variable_addr, target);
|
2018-01-07 21:55:17 +00:00
|
|
|
|
|
|
|
// Wakeup all or one thread - Any other value is unimplemented
|
|
|
|
ASSERT(target == -1 || target == 1);
|
|
|
|
|
2018-01-09 02:41:37 +00:00
|
|
|
SharedPtr<ConditionVariable> condition_variable =
|
|
|
|
g_object_address_table.Get<ConditionVariable>(condition_variable_addr);
|
|
|
|
if (!condition_variable) {
|
|
|
|
// Create a new condition_variable for the specified address if one does not already exist
|
|
|
|
condition_variable = ConditionVariable::Create(condition_variable_addr).Unwrap();
|
|
|
|
condition_variable->name =
|
|
|
|
Common::StringFromFormat("condition-variable-%llx", condition_variable_addr);
|
2018-01-07 21:55:17 +00:00
|
|
|
}
|
|
|
|
|
2018-01-09 02:41:37 +00:00
|
|
|
CASCADE_CODE(condition_variable->Release(target));
|
2018-01-07 21:55:17 +00:00
|
|
|
|
2018-01-09 02:41:37 +00:00
|
|
|
if (condition_variable->mutex_addr) {
|
|
|
|
// If a mutex was created for this condition_variable, wait the current thread on it
|
|
|
|
SharedPtr<Mutex> mutex = g_object_address_table.Get<Mutex>(condition_variable->mutex_addr);
|
2018-01-07 21:55:17 +00:00
|
|
|
return WaitSynchronization1(mutex, GetCurrentThread());
|
|
|
|
}
|
|
|
|
|
2015-01-23 05:36:58 +00:00
|
|
|
return RESULT_SUCCESS;
|
2014-12-09 04:52:27 +00:00
|
|
|
}
|
|
|
|
|
2018-01-12 02:59:31 +00:00
|
|
|
/// This returns the total CPU ticks elapsed since the CPU was powered-on
|
|
|
|
static u64 GetSystemTick() {
|
|
|
|
const u64 result{CoreTiming::GetTicks()};
|
|
|
|
|
|
|
|
// Advance time to defeat dumb games that busy-wait for the frame to end.
|
|
|
|
CoreTiming::AddTicks(400);
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2017-10-14 22:30:07 +01:00
|
|
|
/// Close a handle
|
2018-01-03 01:40:30 +00:00
|
|
|
static ResultCode CloseHandle(Handle handle) {
|
2017-10-14 22:30:07 +01:00
|
|
|
LOG_TRACE(Kernel_SVC, "Closing handle 0x%08X", handle);
|
2018-01-03 01:40:30 +00:00
|
|
|
return g_handle_table.Close(handle);
|
2015-08-06 01:39:53 +01:00
|
|
|
}
|
|
|
|
|
2018-01-08 02:24:19 +00:00
|
|
|
/// Reset an event
|
|
|
|
static ResultCode ResetSignal(Handle handle) {
|
|
|
|
LOG_WARNING(Kernel_SVC, "(STUBBED) called handle 0x%08X", handle);
|
|
|
|
auto event = g_handle_table.Get<Event>(handle);
|
|
|
|
ASSERT(event != nullptr);
|
|
|
|
event->Clear();
|
|
|
|
return RESULT_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Creates a TransferMemory object
|
|
|
|
static ResultCode CreateTransferMemory(Handle* handle, VAddr addr, u64 size, u32 permissions) {
|
2018-03-19 16:07:08 +00:00
|
|
|
LOG_WARNING(Kernel_SVC, "(STUBBED) called addr=0x%lx, size=0x%lx, perms=%08X", addr, size,
|
2018-01-12 02:59:31 +00:00
|
|
|
permissions);
|
2018-01-08 02:24:19 +00:00
|
|
|
*handle = 0;
|
|
|
|
return RESULT_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2018-03-30 02:07:49 +01:00
|
|
|
static ResultCode GetThreadCoreMask(Handle handle, u32* mask, u64* unknown) {
|
|
|
|
LOG_WARNING(Kernel_SVC, "(STUBBED) called, handle=0x%08X", handle);
|
|
|
|
*mask = 0x0;
|
|
|
|
*unknown = 0xf;
|
|
|
|
return RESULT_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
static ResultCode SetThreadCoreMask(Handle handle, u32 mask, u64 unknown) {
|
|
|
|
LOG_WARNING(Kernel_SVC, "(STUBBED) called, handle=0x%08X, mask=0x%08X, unknown=0x%lx", handle,
|
|
|
|
mask, unknown);
|
2018-01-16 22:23:53 +00:00
|
|
|
return RESULT_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2018-02-03 18:36:54 +00:00
|
|
|
static ResultCode CreateSharedMemory(Handle* handle, u64 size, u32 local_permissions,
|
2018-01-20 00:35:25 +00:00
|
|
|
u32 remote_permissions) {
|
2018-02-03 18:36:54 +00:00
|
|
|
LOG_TRACE(Kernel_SVC, "called, size=0x%llx, localPerms=0x%08x, remotePerms=0x%08x", size,
|
2018-01-20 00:35:25 +00:00
|
|
|
local_permissions, remote_permissions);
|
2018-02-03 18:36:54 +00:00
|
|
|
auto sharedMemHandle =
|
|
|
|
SharedMemory::Create(g_handle_table.Get<Process>(KernelHandle::CurrentProcess), size,
|
|
|
|
static_cast<MemoryPermission>(local_permissions),
|
|
|
|
static_cast<MemoryPermission>(remote_permissions));
|
2018-01-20 00:35:25 +00:00
|
|
|
|
|
|
|
CASCADE_RESULT(*handle, g_handle_table.Create(sharedMemHandle));
|
|
|
|
return RESULT_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2018-02-22 14:28:15 +00:00
|
|
|
static ResultCode ClearEvent(Handle handle) {
|
|
|
|
LOG_TRACE(Kernel_SVC, "called, event=0xX", handle);
|
|
|
|
|
|
|
|
SharedPtr<Event> evt = g_handle_table.Get<Event>(handle);
|
|
|
|
if (evt == nullptr)
|
|
|
|
return ERR_INVALID_HANDLE;
|
|
|
|
evt->Clear();
|
|
|
|
return RESULT_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2015-05-06 04:04:25 +01:00
|
|
|
namespace {
|
2016-09-18 01:38:01 +01:00
|
|
|
struct FunctionDef {
|
|
|
|
using Func = void();
|
2015-05-06 04:04:25 +01:00
|
|
|
|
2016-09-18 01:38:01 +01:00
|
|
|
u32 id;
|
|
|
|
Func* func;
|
|
|
|
const char* name;
|
|
|
|
};
|
2017-10-14 22:30:07 +01:00
|
|
|
} // namespace
|
2015-05-06 04:04:25 +01:00
|
|
|
|
|
|
|
static const FunctionDef SVC_Table[] = {
|
2016-09-18 01:38:01 +01:00
|
|
|
{0x00, nullptr, "Unknown"},
|
2018-01-03 01:47:26 +00:00
|
|
|
{0x01, SvcWrap<SetHeapSize>, "SetHeapSize"},
|
|
|
|
{0x02, nullptr, "SetMemoryPermission"},
|
2018-01-08 02:24:19 +00:00
|
|
|
{0x03, SvcWrap<SetMemoryAttribute>, "SetMemoryAttribute"},
|
2018-01-03 01:47:26 +00:00
|
|
|
{0x04, SvcWrap<MapMemory>, "MapMemory"},
|
|
|
|
{0x05, SvcWrap<UnmapMemory>, "UnmapMemory"},
|
|
|
|
{0x06, SvcWrap<QueryMemory>, "QueryMemory"},
|
|
|
|
{0x07, SvcWrap<ExitProcess>, "ExitProcess"},
|
|
|
|
{0x08, SvcWrap<CreateThread>, "CreateThread"},
|
|
|
|
{0x09, SvcWrap<StartThread>, "StartThread"},
|
|
|
|
{0x0A, SvcWrap<ExitThread>, "ExitThread"},
|
|
|
|
{0x0B, SvcWrap<SleepThread>, "SleepThread"},
|
|
|
|
{0x0C, SvcWrap<GetThreadPriority>, "GetThreadPriority"},
|
|
|
|
{0x0D, SvcWrap<SetThreadPriority>, "SetThreadPriority"},
|
2018-03-30 02:07:49 +01:00
|
|
|
{0x0E, SvcWrap<GetThreadCoreMask>, "GetThreadCoreMask"},
|
2018-01-16 22:23:53 +00:00
|
|
|
{0x0F, SvcWrap<SetThreadCoreMask>, "SetThreadCoreMask"},
|
2018-01-03 01:47:26 +00:00
|
|
|
{0x10, SvcWrap<GetCurrentProcessorNumber>, "GetCurrentProcessorNumber"},
|
|
|
|
{0x11, nullptr, "SignalEvent"},
|
2018-02-22 14:28:15 +00:00
|
|
|
{0x12, SvcWrap<ClearEvent>, "ClearEvent"},
|
2018-01-14 22:15:31 +00:00
|
|
|
{0x13, SvcWrap<MapSharedMemory>, "MapSharedMemory"},
|
2018-02-22 19:16:43 +00:00
|
|
|
{0x14, SvcWrap<UnmapSharedMemory>, "UnmapSharedMemory"},
|
2018-01-08 02:24:19 +00:00
|
|
|
{0x15, SvcWrap<CreateTransferMemory>, "CreateTransferMemory"},
|
2018-01-03 01:47:26 +00:00
|
|
|
{0x16, SvcWrap<CloseHandle>, "CloseHandle"},
|
2018-01-08 02:24:19 +00:00
|
|
|
{0x17, SvcWrap<ResetSignal>, "ResetSignal"},
|
2018-01-03 01:47:26 +00:00
|
|
|
{0x18, SvcWrap<WaitSynchronization>, "WaitSynchronization"},
|
2018-01-09 20:02:04 +00:00
|
|
|
{0x19, SvcWrap<CancelSynchronization>, "CancelSynchronization"},
|
2018-01-18 01:34:52 +00:00
|
|
|
{0x1A, SvcWrap<ArbitrateLock>, "ArbitrateLock"},
|
|
|
|
{0x1B, SvcWrap<ArbitrateUnlock>, "ArbitrateUnlock"},
|
2018-01-06 21:14:12 +00:00
|
|
|
{0x1C, SvcWrap<WaitProcessWideKeyAtomic>, "WaitProcessWideKeyAtomic"},
|
2018-01-03 01:47:26 +00:00
|
|
|
{0x1D, SvcWrap<SignalProcessWideKey>, "SignalProcessWideKey"},
|
2018-01-12 02:59:31 +00:00
|
|
|
{0x1E, SvcWrap<GetSystemTick>, "GetSystemTick"},
|
2018-01-18 01:34:52 +00:00
|
|
|
{0x1F, SvcWrap<ConnectToNamedPort>, "ConnectToNamedPort"},
|
2018-01-03 01:47:26 +00:00
|
|
|
{0x20, nullptr, "SendSyncRequestLight"},
|
|
|
|
{0x21, SvcWrap<SendSyncRequest>, "SendSyncRequest"},
|
|
|
|
{0x22, nullptr, "SendSyncRequestWithUserBuffer"},
|
|
|
|
{0x23, nullptr, "SendAsyncRequestWithUserBuffer"},
|
|
|
|
{0x24, SvcWrap<GetProcessId>, "GetProcessId"},
|
|
|
|
{0x25, SvcWrap<GetThreadId>, "GetThreadId"},
|
|
|
|
{0x26, SvcWrap<Break>, "Break"},
|
|
|
|
{0x27, SvcWrap<OutputDebugString>, "OutputDebugString"},
|
|
|
|
{0x28, nullptr, "ReturnFromException"},
|
|
|
|
{0x29, SvcWrap<GetInfo>, "GetInfo"},
|
|
|
|
{0x2A, nullptr, "FlushEntireDataCache"},
|
|
|
|
{0x2B, nullptr, "FlushDataCache"},
|
|
|
|
{0x2C, nullptr, "MapPhysicalMemory"},
|
|
|
|
{0x2D, nullptr, "UnmapPhysicalMemory"},
|
2017-10-14 22:30:07 +01:00
|
|
|
{0x2E, nullptr, "Unknown"},
|
2018-01-03 01:47:26 +00:00
|
|
|
{0x2F, nullptr, "GetLastThreadInfo"},
|
|
|
|
{0x30, nullptr, "GetResourceLimitLimitValue"},
|
|
|
|
{0x31, nullptr, "GetResourceLimitCurrentValue"},
|
|
|
|
{0x32, nullptr, "SetThreadActivity"},
|
|
|
|
{0x33, nullptr, "GetThreadContext"},
|
2017-10-14 22:30:07 +01:00
|
|
|
{0x34, nullptr, "Unknown"},
|
|
|
|
{0x35, nullptr, "Unknown"},
|
|
|
|
{0x36, nullptr, "Unknown"},
|
|
|
|
{0x37, nullptr, "Unknown"},
|
|
|
|
{0x38, nullptr, "Unknown"},
|
|
|
|
{0x39, nullptr, "Unknown"},
|
|
|
|
{0x3A, nullptr, "Unknown"},
|
|
|
|
{0x3B, nullptr, "Unknown"},
|
2018-01-03 01:47:26 +00:00
|
|
|
{0x3C, nullptr, "DumpInfo"},
|
2017-10-14 22:30:07 +01:00
|
|
|
{0x3D, nullptr, "Unknown"},
|
|
|
|
{0x3E, nullptr, "Unknown"},
|
2016-09-18 01:38:01 +01:00
|
|
|
{0x3F, nullptr, "Unknown"},
|
2018-01-03 01:47:26 +00:00
|
|
|
{0x40, nullptr, "CreateSession"},
|
|
|
|
{0x41, nullptr, "AcceptSession"},
|
|
|
|
{0x42, nullptr, "ReplyAndReceiveLight"},
|
|
|
|
{0x43, nullptr, "ReplyAndReceive"},
|
|
|
|
{0x44, nullptr, "ReplyAndReceiveWithUserBuffer"},
|
|
|
|
{0x45, nullptr, "CreateEvent"},
|
2016-09-18 01:38:01 +01:00
|
|
|
{0x46, nullptr, "Unknown"},
|
2017-10-14 22:30:07 +01:00
|
|
|
{0x47, nullptr, "Unknown"},
|
|
|
|
{0x48, nullptr, "Unknown"},
|
|
|
|
{0x49, nullptr, "Unknown"},
|
|
|
|
{0x4A, nullptr, "Unknown"},
|
2018-01-18 01:32:38 +00:00
|
|
|
{0x4B, nullptr, "CreateJitMemory"},
|
|
|
|
{0x4C, nullptr, "MapJitMemory"},
|
2018-01-03 01:47:26 +00:00
|
|
|
{0x4D, nullptr, "SleepSystem"},
|
|
|
|
{0x4E, nullptr, "ReadWriteRegister"},
|
|
|
|
{0x4F, nullptr, "SetProcessActivity"},
|
2018-01-20 00:35:25 +00:00
|
|
|
{0x50, SvcWrap<CreateSharedMemory>, "CreateSharedMemory"},
|
2018-01-03 01:47:26 +00:00
|
|
|
{0x51, nullptr, "MapTransferMemory"},
|
|
|
|
{0x52, nullptr, "UnmapTransferMemory"},
|
|
|
|
{0x53, nullptr, "CreateInterruptEvent"},
|
|
|
|
{0x54, nullptr, "QueryPhysicalAddress"},
|
|
|
|
{0x55, nullptr, "QueryIoMapping"},
|
|
|
|
{0x56, nullptr, "CreateDeviceAddressSpace"},
|
|
|
|
{0x57, nullptr, "AttachDeviceAddressSpace"},
|
|
|
|
{0x58, nullptr, "DetachDeviceAddressSpace"},
|
|
|
|
{0x59, nullptr, "MapDeviceAddressSpaceByForce"},
|
|
|
|
{0x5A, nullptr, "MapDeviceAddressSpaceAligned"},
|
|
|
|
{0x5B, nullptr, "MapDeviceAddressSpace"},
|
|
|
|
{0x5C, nullptr, "UnmapDeviceAddressSpace"},
|
|
|
|
{0x5D, nullptr, "InvalidateProcessDataCache"},
|
|
|
|
{0x5E, nullptr, "StoreProcessDataCache"},
|
|
|
|
{0x5F, nullptr, "FlushProcessDataCache"},
|
|
|
|
{0x60, nullptr, "DebugActiveProcess"},
|
|
|
|
{0x61, nullptr, "BreakDebugProcess"},
|
|
|
|
{0x62, nullptr, "TerminateDebugProcess"},
|
|
|
|
{0x63, nullptr, "GetDebugEvent"},
|
|
|
|
{0x64, nullptr, "ContinueDebugEvent"},
|
|
|
|
{0x65, nullptr, "GetProcessList"},
|
|
|
|
{0x66, nullptr, "GetThreadList"},
|
|
|
|
{0x67, nullptr, "GetDebugThreadContext"},
|
|
|
|
{0x68, nullptr, "SetDebugThreadContext"},
|
|
|
|
{0x69, nullptr, "QueryDebugProcessMemory"},
|
|
|
|
{0x6A, nullptr, "ReadDebugProcessMemory"},
|
|
|
|
{0x6B, nullptr, "WriteDebugProcessMemory"},
|
|
|
|
{0x6C, nullptr, "SetHardwareBreakPoint"},
|
|
|
|
{0x6D, nullptr, "GetDebugThreadParam"},
|
2016-09-18 01:38:01 +01:00
|
|
|
{0x6E, nullptr, "Unknown"},
|
|
|
|
{0x6F, nullptr, "Unknown"},
|
2018-01-03 01:47:26 +00:00
|
|
|
{0x70, nullptr, "CreatePort"},
|
|
|
|
{0x71, nullptr, "ManageNamedPort"},
|
|
|
|
{0x72, nullptr, "ConnectToPort"},
|
|
|
|
{0x73, nullptr, "SetProcessMemoryPermission"},
|
|
|
|
{0x74, nullptr, "MapProcessMemory"},
|
|
|
|
{0x75, nullptr, "UnmapProcessMemory"},
|
|
|
|
{0x76, nullptr, "QueryProcessMemory"},
|
|
|
|
{0x77, nullptr, "MapProcessCodeMemory"},
|
|
|
|
{0x78, nullptr, "UnmapProcessCodeMemory"},
|
|
|
|
{0x79, nullptr, "CreateProcess"},
|
|
|
|
{0x7A, nullptr, "StartProcess"},
|
|
|
|
{0x7B, nullptr, "TerminateProcess"},
|
|
|
|
{0x7C, nullptr, "GetProcessInfo"},
|
|
|
|
{0x7D, nullptr, "CreateResourceLimit"},
|
|
|
|
{0x7E, nullptr, "SetResourceLimitLimitValue"},
|
|
|
|
{0x7F, nullptr, "CallSecureMonitor"},
|
2014-04-11 00:58:28 +01:00
|
|
|
};
|
|
|
|
|
2015-07-21 08:51:36 +01:00
|
|
|
static const FunctionDef* GetSVCInfo(u32 func_num) {
|
2015-05-06 04:04:25 +01:00
|
|
|
if (func_num >= ARRAY_SIZE(SVC_Table)) {
|
|
|
|
LOG_ERROR(Kernel_SVC, "unknown svc=0x%02X", func_num);
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
return &SVC_Table[func_num];
|
|
|
|
}
|
|
|
|
|
2015-08-17 22:25:21 +01:00
|
|
|
MICROPROFILE_DEFINE(Kernel_SVC, "Kernel", "SVC", MP_RGB(70, 200, 70));
|
|
|
|
|
2015-07-21 08:51:36 +01:00
|
|
|
void CallSVC(u32 immediate) {
|
2015-08-17 22:25:21 +01:00
|
|
|
MICROPROFILE_SCOPE(Kernel_SVC);
|
2015-05-06 04:04:25 +01:00
|
|
|
|
2017-10-14 22:30:07 +01:00
|
|
|
// Lock the global kernel mutex when we enter the kernel HLE.
|
|
|
|
std::lock_guard<std::recursive_mutex> lock(HLE::g_hle_lock);
|
|
|
|
|
2015-07-21 08:51:36 +01:00
|
|
|
const FunctionDef* info = GetSVCInfo(immediate);
|
2015-05-06 04:04:25 +01:00
|
|
|
if (info) {
|
|
|
|
if (info->func) {
|
|
|
|
info->func();
|
|
|
|
} else {
|
2017-10-14 22:30:07 +01:00
|
|
|
LOG_CRITICAL(Kernel_SVC, "unimplemented SVC function %s(..)", info->name);
|
2015-05-06 04:04:25 +01:00
|
|
|
}
|
2017-10-14 22:30:07 +01:00
|
|
|
} else {
|
|
|
|
LOG_CRITICAL(Kernel_SVC, "unknown SVC function 0x%x", immediate);
|
2015-05-06 04:04:25 +01:00
|
|
|
}
|
2014-04-11 00:58:28 +01:00
|
|
|
}
|
2014-04-11 23:44:21 +01:00
|
|
|
|
2018-01-03 01:40:30 +00:00
|
|
|
} // namespace Kernel
|