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-10 23:58:28 +00:00
|
|
|
|
2018-01-12 03:36:56 +00:00
|
|
|
#include <algorithm>
|
2018-02-25 12:40:22 +00:00
|
|
|
#include <cinttypes>
|
2018-04-20 02:36:48 +00:00
|
|
|
#include <iterator>
|
2018-07-31 12:06:09 +00:00
|
|
|
#include <mutex>
|
|
|
|
#include <vector>
|
2018-01-12 03:36:56 +00:00
|
|
|
|
2018-10-18 16:55:27 +00:00
|
|
|
#include "common/alignment.h"
|
2018-07-31 12:06:09 +00:00
|
|
|
#include "common/assert.h"
|
2015-05-06 07:06:12 +00:00
|
|
|
#include "common/logging/log.h"
|
2015-08-17 21:25:21 +00:00
|
|
|
#include "common/microprofile.h"
|
2018-01-05 00:45:15 +00:00
|
|
|
#include "common/string_util.h"
|
2018-08-31 16:21:34 +00:00
|
|
|
#include "core/arm/exclusive_monitor.h"
|
2018-03-13 21:49:59 +00:00
|
|
|
#include "core/core.h"
|
2018-08-31 16:21:34 +00:00
|
|
|
#include "core/core_cpu.h"
|
2016-09-18 00:38:01 +00:00
|
|
|
#include "core/core_timing.h"
|
2018-06-21 06:49:43 +00:00
|
|
|
#include "core/hle/kernel/address_arbiter.h"
|
2016-05-22 17:30:13 +00:00
|
|
|
#include "core/hle/kernel/client_port.h"
|
2016-06-14 23:03:30 +00:00
|
|
|
#include "core/hle/kernel/client_session.h"
|
2017-05-29 23:45:42 +00:00
|
|
|
#include "core/hle/kernel/handle_table.h"
|
2018-08-31 16:21:34 +00:00
|
|
|
#include "core/hle/kernel/kernel.h"
|
2018-01-01 19:38:34 +00:00
|
|
|
#include "core/hle/kernel/mutex.h"
|
2015-05-11 14:15:10 +00:00
|
|
|
#include "core/hle/kernel/process.h"
|
2018-11-26 23:34:07 +00:00
|
|
|
#include "core/hle/kernel/readable_event.h"
|
2017-12-31 20:58:16 +00:00
|
|
|
#include "core/hle/kernel/resource_limit.h"
|
2018-08-31 16:21:34 +00:00
|
|
|
#include "core/hle/kernel/scheduler.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 02:18:42 +00:00
|
|
|
#include "core/hle/kernel/thread.h"
|
2018-11-26 23:34:07 +00:00
|
|
|
#include "core/hle/kernel/writable_event.h"
|
2017-10-14 21:30:07 +00:00
|
|
|
#include "core/hle/lock.h"
|
2014-10-23 03:20:01 +00:00
|
|
|
#include "core/hle/result.h"
|
2014-04-13 01:55:36 +00:00
|
|
|
#include "core/hle/service/service.h"
|
svc: Handle memory writing explicitly within QueryProcessMemory
Moves the memory writes directly into QueryProcessMemory instead of
letting the wrapper function do it. It would be inaccurate to allow the
handler to do it because there's cases where memory shouldn't even be
written to. For example, if the given process handle is invalid.
HOWEVER, if the memory writing is within the wrapper, then we have no
control over if these memory writes occur, meaning in an error case, 68
bytes of memory randomly get trashed with zeroes, 64 of those being
written to wherever the memory info address points to, and the remaining
4 being written wherever the page info address points to.
One solution in this case would be to just conditionally check within
the handler itself, but this is kind of smelly, given the handler
shouldn't be performing conditional behavior itself, it's a behavior of
the managed function. In other words, if you remove the handler from the
equation entirely, does the function still retain its proper behavior?
In this case, no.
Now, we don't potentially trash memory from this function if an invalid
query is performed.
2018-12-12 16:48:06 +00:00
|
|
|
#include "core/memory.h"
|
2014-04-10 23:58:28 +00:00
|
|
|
|
2018-01-03 01:40:30 +00:00
|
|
|
namespace Kernel {
|
2018-09-13 23:14:50 +00:00
|
|
|
namespace {
|
2018-10-10 18:18:27 +00:00
|
|
|
|
|
|
|
// Checks if address + size is greater than the given address
|
|
|
|
// This can return false if the size causes an overflow of a 64-bit type
|
|
|
|
// or if the given size is zero.
|
|
|
|
constexpr bool IsValidAddressRange(VAddr address, u64 size) {
|
|
|
|
return address + size > address;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Checks if a given address range lies within a larger address range.
|
|
|
|
constexpr bool IsInsideAddressRange(VAddr address, u64 size, VAddr address_range_begin,
|
|
|
|
VAddr address_range_end) {
|
|
|
|
const VAddr end_address = address + size - 1;
|
|
|
|
return address_range_begin <= address && end_address <= address_range_end - 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool IsInsideAddressSpace(const VMManager& vm, VAddr address, u64 size) {
|
|
|
|
return IsInsideAddressRange(address, size, vm.GetAddressSpaceBaseAddress(),
|
|
|
|
vm.GetAddressSpaceEndAddress());
|
|
|
|
}
|
|
|
|
|
|
|
|
bool IsInsideNewMapRegion(const VMManager& vm, VAddr address, u64 size) {
|
|
|
|
return IsInsideAddressRange(address, size, vm.GetNewMapRegionBaseAddress(),
|
|
|
|
vm.GetNewMapRegionEndAddress());
|
|
|
|
}
|
|
|
|
|
2018-11-27 01:56:50 +00:00
|
|
|
// 8 GiB
|
2018-11-27 01:53:18 +00:00
|
|
|
constexpr u64 MAIN_MEMORY_SIZE = 0x200000000;
|
2018-11-27 01:29:06 +00:00
|
|
|
|
2018-10-10 18:18:27 +00:00
|
|
|
// Helper function that performs the common sanity checks for svcMapMemory
|
|
|
|
// and svcUnmapMemory. This is doable, as both functions perform their sanitizing
|
|
|
|
// in the same order.
|
|
|
|
ResultCode MapUnmapMemorySanityChecks(const VMManager& vm_manager, VAddr dst_addr, VAddr src_addr,
|
|
|
|
u64 size) {
|
2018-11-26 08:47:39 +00:00
|
|
|
if (!Common::Is4KBAligned(dst_addr)) {
|
|
|
|
LOG_ERROR(Kernel_SVC, "Destination address is not aligned to 4KB, 0x{:016X}", dst_addr);
|
2018-10-10 18:18:27 +00:00
|
|
|
return ERR_INVALID_ADDRESS;
|
|
|
|
}
|
|
|
|
|
2018-11-26 08:47:39 +00:00
|
|
|
if (!Common::Is4KBAligned(src_addr)) {
|
|
|
|
LOG_ERROR(Kernel_SVC, "Source address is not aligned to 4KB, 0x{:016X}", src_addr);
|
2018-11-27 01:29:06 +00:00
|
|
|
return ERR_INVALID_SIZE;
|
2018-11-26 08:47:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (size == 0) {
|
|
|
|
LOG_ERROR(Kernel_SVC, "Size is 0");
|
|
|
|
return ERR_INVALID_SIZE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!Common::Is4KBAligned(size)) {
|
|
|
|
LOG_ERROR(Kernel_SVC, "Size is not aligned to 4KB, 0x{:016X}", size);
|
2018-10-10 18:18:27 +00:00
|
|
|
return ERR_INVALID_SIZE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!IsValidAddressRange(dst_addr, size)) {
|
2018-11-26 08:47:39 +00:00
|
|
|
LOG_ERROR(Kernel_SVC,
|
|
|
|
"Destination is not a valid address range, addr=0x{:016X}, size=0x{:016X}",
|
|
|
|
dst_addr, size);
|
2018-10-10 18:18:27 +00:00
|
|
|
return ERR_INVALID_ADDRESS_STATE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!IsValidAddressRange(src_addr, size)) {
|
2018-11-26 08:47:39 +00:00
|
|
|
LOG_ERROR(Kernel_SVC, "Source is not a valid address range, addr=0x{:016X}, size=0x{:016X}",
|
|
|
|
src_addr, size);
|
2018-10-10 18:18:27 +00:00
|
|
|
return ERR_INVALID_ADDRESS_STATE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!IsInsideAddressSpace(vm_manager, src_addr, size)) {
|
2018-11-26 08:47:39 +00:00
|
|
|
LOG_ERROR(Kernel_SVC,
|
|
|
|
"Source is not within the address space, addr=0x{:016X}, size=0x{:016X}",
|
|
|
|
src_addr, size);
|
2018-10-10 18:18:27 +00:00
|
|
|
return ERR_INVALID_ADDRESS_STATE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!IsInsideNewMapRegion(vm_manager, dst_addr, size)) {
|
2018-11-26 08:47:39 +00:00
|
|
|
LOG_ERROR(Kernel_SVC,
|
|
|
|
"Destination is not within the new map region, addr=0x{:016X}, size=0x{:016X}",
|
|
|
|
dst_addr, size);
|
2018-10-10 18:18:27 +00:00
|
|
|
return ERR_INVALID_MEMORY_RANGE;
|
|
|
|
}
|
|
|
|
|
|
|
|
const VAddr dst_end_address = dst_addr + size;
|
|
|
|
if (dst_end_address > vm_manager.GetHeapRegionBaseAddress() &&
|
2018-10-12 05:43:15 +00:00
|
|
|
vm_manager.GetHeapRegionEndAddress() > dst_addr) {
|
2018-11-26 08:47:39 +00:00
|
|
|
LOG_ERROR(Kernel_SVC,
|
|
|
|
"Destination does not fit within the heap region, addr=0x{:016X}, "
|
|
|
|
"size=0x{:016X}, end_addr=0x{:016X}",
|
|
|
|
dst_addr, size, dst_end_address);
|
2018-10-10 18:18:27 +00:00
|
|
|
return ERR_INVALID_MEMORY_RANGE;
|
|
|
|
}
|
|
|
|
|
2018-10-12 05:43:15 +00:00
|
|
|
if (dst_end_address > vm_manager.GetMapRegionBaseAddress() &&
|
|
|
|
vm_manager.GetMapRegionEndAddress() > dst_addr) {
|
2018-11-26 08:47:39 +00:00
|
|
|
LOG_ERROR(Kernel_SVC,
|
|
|
|
"Destination does not fit within the map region, addr=0x{:016X}, "
|
|
|
|
"size=0x{:016X}, end_addr=0x{:016X}",
|
|
|
|
dst_addr, size, dst_end_address);
|
2018-10-10 18:18:27 +00:00
|
|
|
return ERR_INVALID_MEMORY_RANGE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return RESULT_SUCCESS;
|
|
|
|
}
|
2018-11-27 00:14:29 +00:00
|
|
|
|
|
|
|
enum class ResourceLimitValueType {
|
|
|
|
CurrentValue,
|
|
|
|
LimitValue,
|
|
|
|
};
|
|
|
|
|
|
|
|
ResultVal<s64> RetrieveResourceLimitValue(Handle resource_limit, u32 resource_type,
|
|
|
|
ResourceLimitValueType value_type) {
|
|
|
|
const auto type = static_cast<ResourceType>(resource_type);
|
|
|
|
if (!IsValidResourceType(type)) {
|
|
|
|
LOG_ERROR(Kernel_SVC, "Invalid resource limit type: '{}'", resource_type);
|
|
|
|
return ERR_INVALID_ENUM_VALUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
const auto& kernel = Core::System::GetInstance().Kernel();
|
|
|
|
const auto* const current_process = kernel.CurrentProcess();
|
|
|
|
ASSERT(current_process != nullptr);
|
|
|
|
|
|
|
|
const auto resource_limit_object =
|
|
|
|
current_process->GetHandleTable().Get<ResourceLimit>(resource_limit);
|
|
|
|
if (!resource_limit_object) {
|
|
|
|
LOG_ERROR(Kernel_SVC, "Handle to non-existent resource limit instance used. Handle={:08X}",
|
|
|
|
resource_limit);
|
|
|
|
return ERR_INVALID_HANDLE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (value_type == ResourceLimitValueType::CurrentValue) {
|
|
|
|
return MakeResult(resource_limit_object->GetCurrentResourceValue(type));
|
|
|
|
}
|
|
|
|
|
|
|
|
return MakeResult(resource_limit_object->GetMaxResourceValue(type));
|
|
|
|
}
|
2018-09-13 23:14:50 +00:00
|
|
|
} // Anonymous namespace
|
2014-04-11 03:26:12 +00: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) {
|
2018-07-02 16:13:26 +00:00
|
|
|
LOG_TRACE(Kernel_SVC, "called, heap_size=0x{:X}", heap_size);
|
2018-09-13 23:09:04 +00:00
|
|
|
|
2018-11-27 01:29:06 +00:00
|
|
|
// Size must be a multiple of 0x200000 (2MB) and be equal to or less than 8GB.
|
2018-11-27 01:53:18 +00:00
|
|
|
if ((heap_size % 0x200000) != 0) {
|
2018-11-27 01:29:06 +00:00
|
|
|
LOG_ERROR(Kernel_SVC, "The heap size is not a multiple of 2MB, heap_size=0x{:016X}",
|
|
|
|
heap_size);
|
|
|
|
return ERR_INVALID_SIZE;
|
|
|
|
}
|
|
|
|
|
2018-11-27 01:53:18 +00:00
|
|
|
if (heap_size >= 0x200000000) {
|
2018-11-27 01:29:06 +00:00
|
|
|
LOG_ERROR(Kernel_SVC, "The heap size is not less than 8GB, heap_size=0x{:016X}", heap_size);
|
2018-09-13 23:09:04 +00:00
|
|
|
return ERR_INVALID_SIZE;
|
|
|
|
}
|
|
|
|
|
2018-03-13 21:49:59 +00:00
|
|
|
auto& process = *Core::CurrentProcess();
|
2018-09-29 22:47:00 +00:00
|
|
|
const VAddr heap_base = process.VMManager().GetHeapRegionBaseAddress();
|
2018-01-03 03:24:12 +00:00
|
|
|
CASCADE_RESULT(*heap_addr,
|
2018-09-25 00:01:45 +00:00
|
|
|
process.HeapAllocate(heap_base, heap_size, VMAPermission::ReadWrite));
|
2017-12-28 20:29:52 +00:00
|
|
|
return RESULT_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2018-11-03 15:01:34 +00:00
|
|
|
static ResultCode SetMemoryPermission(VAddr addr, u64 size, u32 prot) {
|
2018-11-06 09:21:01 +00:00
|
|
|
LOG_TRACE(Kernel_SVC, "called, addr=0x{:X}, size=0x{:X}, prot=0x{:X}", addr, size, prot);
|
|
|
|
|
|
|
|
if (!Common::Is4KBAligned(addr)) {
|
2018-11-26 08:47:39 +00:00
|
|
|
LOG_ERROR(Kernel_SVC, "Address is not aligned to 4KB, addr=0x{:016X}", addr);
|
2018-11-06 09:21:01 +00:00
|
|
|
return ERR_INVALID_ADDRESS;
|
|
|
|
}
|
|
|
|
|
2018-11-26 08:47:39 +00:00
|
|
|
if (size == 0) {
|
|
|
|
LOG_ERROR(Kernel_SVC, "Size is 0");
|
|
|
|
return ERR_INVALID_SIZE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!Common::Is4KBAligned(size)) {
|
|
|
|
LOG_ERROR(Kernel_SVC, "Size is not aligned to 4KB, size=0x{:016X}", size);
|
2018-11-06 09:21:01 +00:00
|
|
|
return ERR_INVALID_SIZE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!IsValidAddressRange(addr, size)) {
|
2018-11-26 08:47:39 +00:00
|
|
|
LOG_ERROR(Kernel_SVC, "Region is not a valid address range, addr=0x{:016X}, size=0x{:016X}",
|
|
|
|
addr, size);
|
2018-11-06 09:21:01 +00:00
|
|
|
return ERR_INVALID_ADDRESS_STATE;
|
|
|
|
}
|
|
|
|
|
|
|
|
const auto permission = static_cast<MemoryPermission>(prot);
|
|
|
|
if (permission != MemoryPermission::None && permission != MemoryPermission::Read &&
|
|
|
|
permission != MemoryPermission::ReadWrite) {
|
2018-11-26 08:47:39 +00:00
|
|
|
LOG_ERROR(Kernel_SVC, "Invalid memory permission specified, Got memory permission=0x{:08X}",
|
|
|
|
static_cast<u32>(permission));
|
2018-11-06 09:21:01 +00:00
|
|
|
return ERR_INVALID_MEMORY_PERMISSIONS;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto* const current_process = Core::CurrentProcess();
|
|
|
|
auto& vm_manager = current_process->VMManager();
|
|
|
|
|
|
|
|
if (!IsInsideAddressSpace(vm_manager, addr, size)) {
|
2018-11-26 08:47:39 +00:00
|
|
|
LOG_ERROR(Kernel_SVC,
|
|
|
|
"Source is not within the address space, addr=0x{:016X}, size=0x{:016X}", addr,
|
|
|
|
size);
|
2018-11-06 09:21:01 +00:00
|
|
|
return ERR_INVALID_ADDRESS_STATE;
|
|
|
|
}
|
|
|
|
|
|
|
|
const VMManager::VMAHandle iter = vm_manager.FindVMA(addr);
|
2018-12-06 15:59:22 +00:00
|
|
|
if (!vm_manager.IsValidHandle(iter)) {
|
2018-11-26 08:47:39 +00:00
|
|
|
LOG_ERROR(Kernel_SVC, "Unable to find VMA for address=0x{:016X}", addr);
|
2018-11-06 09:21:01 +00:00
|
|
|
return ERR_INVALID_ADDRESS_STATE;
|
|
|
|
}
|
|
|
|
|
|
|
|
LOG_WARNING(Kernel_SVC, "Uniformity check on protected memory is not implemented.");
|
|
|
|
// TODO: Performs a uniformity check to make sure only protected memory is changed (it doesn't
|
|
|
|
// make sense to allow changing permissions on kernel memory itself, etc).
|
|
|
|
|
|
|
|
const auto converted_permissions = SharedMemory::ConvertPermissions(permission);
|
|
|
|
|
|
|
|
return vm_manager.ReprotectRange(addr, size, converted_permissions);
|
2018-11-03 15:01:34 +00:00
|
|
|
}
|
|
|
|
|
2018-12-15 20:21:41 +00:00
|
|
|
static ResultCode SetMemoryAttribute(VAddr address, u64 size, u32 mask, u32 attribute) {
|
|
|
|
LOG_DEBUG(Kernel_SVC,
|
|
|
|
"called, address=0x{:016X}, size=0x{:X}, mask=0x{:08X}, attribute=0x{:08X}", address,
|
|
|
|
size, mask, attribute);
|
|
|
|
|
|
|
|
if (!Common::Is4KBAligned(address)) {
|
|
|
|
LOG_ERROR(Kernel_SVC, "Address not page aligned (0x{:016X})", address);
|
|
|
|
return ERR_INVALID_ADDRESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (size == 0 || !Common::Is4KBAligned(size)) {
|
|
|
|
LOG_ERROR(Kernel_SVC, "Invalid size (0x{:X}). Size must be non-zero and page aligned.",
|
|
|
|
size);
|
|
|
|
return ERR_INVALID_ADDRESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!IsValidAddressRange(address, size)) {
|
|
|
|
LOG_ERROR(Kernel_SVC, "Address range overflowed (Address: 0x{:016X}, Size: 0x{:016X})",
|
|
|
|
address, size);
|
|
|
|
return ERR_INVALID_ADDRESS_STATE;
|
|
|
|
}
|
|
|
|
|
|
|
|
const auto mem_attribute = static_cast<MemoryAttribute>(attribute);
|
|
|
|
const auto mem_mask = static_cast<MemoryAttribute>(mask);
|
|
|
|
const auto attribute_with_mask = mem_attribute | mem_mask;
|
|
|
|
|
|
|
|
if (attribute_with_mask != mem_mask) {
|
|
|
|
LOG_ERROR(Kernel_SVC,
|
|
|
|
"Memory attribute doesn't match the given mask (Attribute: 0x{:X}, Mask: {:X}",
|
|
|
|
attribute, mask);
|
|
|
|
return ERR_INVALID_COMBINATION;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((attribute_with_mask | MemoryAttribute::Uncached) != MemoryAttribute::Uncached) {
|
|
|
|
LOG_ERROR(Kernel_SVC, "Specified attribute isn't equal to MemoryAttributeUncached (8).");
|
|
|
|
return ERR_INVALID_COMBINATION;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto& vm_manager = Core::CurrentProcess()->VMManager();
|
|
|
|
if (!IsInsideAddressSpace(vm_manager, address, size)) {
|
|
|
|
LOG_ERROR(Kernel_SVC,
|
|
|
|
"Given address (0x{:016X}) is outside the bounds of the address space.", address);
|
|
|
|
return ERR_INVALID_ADDRESS_STATE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return vm_manager.SetMemoryAttribute(address, size, mem_mask, mem_attribute);
|
2018-01-08 02:23:42 +00:00
|
|
|
}
|
|
|
|
|
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) {
|
2018-07-02 16:13:26 +00:00
|
|
|
LOG_TRACE(Kernel_SVC, "called, dst_addr=0x{:X}, src_addr=0x{:X}, size=0x{:X}", dst_addr,
|
2018-07-02 16:20:50 +00:00
|
|
|
src_addr, size);
|
2018-09-13 23:14:50 +00:00
|
|
|
|
2018-10-10 18:18:27 +00:00
|
|
|
auto* const current_process = Core::CurrentProcess();
|
|
|
|
const auto& vm_manager = current_process->VMManager();
|
2018-09-13 23:14:50 +00:00
|
|
|
|
2018-10-10 18:18:27 +00:00
|
|
|
const auto result = MapUnmapMemorySanityChecks(vm_manager, dst_addr, src_addr, size);
|
|
|
|
if (result != RESULT_SUCCESS) {
|
|
|
|
return result;
|
2018-09-13 23:14:50 +00:00
|
|
|
}
|
|
|
|
|
2018-12-12 15:08:46 +00:00
|
|
|
return current_process->MirrorMemory(dst_addr, src_addr, size, MemoryState::Stack);
|
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) {
|
2018-07-02 16:13:26 +00:00
|
|
|
LOG_TRACE(Kernel_SVC, "called, dst_addr=0x{:X}, src_addr=0x{:X}, size=0x{:X}", dst_addr,
|
2018-07-02 16:20:50 +00:00
|
|
|
src_addr, size);
|
2018-09-13 23:14:50 +00:00
|
|
|
|
2018-10-10 18:18:27 +00:00
|
|
|
auto* const current_process = Core::CurrentProcess();
|
|
|
|
const auto& vm_manager = current_process->VMManager();
|
2018-09-13 23:14:50 +00:00
|
|
|
|
2018-10-10 18:18:27 +00:00
|
|
|
const auto result = MapUnmapMemorySanityChecks(vm_manager, dst_addr, src_addr, size);
|
|
|
|
if (result != RESULT_SUCCESS) {
|
|
|
|
return result;
|
2018-09-13 23:14:50 +00:00
|
|
|
}
|
|
|
|
|
2018-10-10 18:18:27 +00:00
|
|
|
return current_process->UnmapMemory(dst_addr, src_addr, size);
|
2017-12-31 20:22:49 +00:00
|
|
|
}
|
|
|
|
|
2014-04-13 01:55:36 +00: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) {
|
2018-09-02 15:58:58 +00:00
|
|
|
if (!Memory::IsValidVirtualAddress(port_name_address)) {
|
2018-11-26 08:47:39 +00:00
|
|
|
LOG_ERROR(Kernel_SVC,
|
|
|
|
"Port Name Address is not a valid virtual address, port_name_address=0x{:016X}",
|
|
|
|
port_name_address);
|
2018-01-03 01:40:30 +00:00
|
|
|
return ERR_NOT_FOUND;
|
2018-09-02 15:58:58 +00:00
|
|
|
}
|
2017-10-14 21:30:07 +00: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);
|
2018-09-02 15:58:58 +00:00
|
|
|
if (port_name.size() > PortNameMaxLength) {
|
2018-11-26 08:47:39 +00:00
|
|
|
LOG_ERROR(Kernel_SVC, "Port name is too long, expected {} but got {}", PortNameMaxLength,
|
|
|
|
port_name.size());
|
2018-11-16 19:24:27 +00:00
|
|
|
return ERR_OUT_OF_RANGE;
|
2018-09-02 15:58:58 +00:00
|
|
|
}
|
2014-06-02 00:48:29 +00:00
|
|
|
|
2018-07-02 16:13:26 +00:00
|
|
|
LOG_TRACE(Kernel_SVC, "called port_name={}", port_name);
|
2014-06-02 00:48:29 +00:00
|
|
|
|
2018-09-02 15:58:58 +00:00
|
|
|
auto& kernel = Core::System::GetInstance().Kernel();
|
|
|
|
auto it = kernel.FindNamedPort(port_name);
|
|
|
|
if (!kernel.IsValidNamedPort(it)) {
|
2018-07-02 16:13:26 +00:00
|
|
|
LOG_WARNING(Kernel_SVC, "tried to connect to unknown port: {}", port_name);
|
2018-01-03 01:40:30 +00:00
|
|
|
return ERR_NOT_FOUND;
|
2015-01-30 18:07:04 +00:00
|
|
|
}
|
2014-06-02 00:48:29 +00:00
|
|
|
|
2016-12-05 16:02:08 +00:00
|
|
|
auto client_port = it->second;
|
2016-06-14 23:03:30 +00: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-14 23:03:30 +00:00
|
|
|
|
|
|
|
// Return the client session
|
2018-10-20 18:34:41 +00:00
|
|
|
auto& handle_table = Core::CurrentProcess()->GetHandleTable();
|
|
|
|
CASCADE_RESULT(*out_handle, handle_table.Create(client_session));
|
2015-01-23 05:36:58 +00:00
|
|
|
return RESULT_SUCCESS;
|
2014-04-13 01:55:36 +00: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-10-20 18:34:41 +00:00
|
|
|
const auto& handle_table = Core::CurrentProcess()->GetHandleTable();
|
|
|
|
SharedPtr<ClientSession> session = handle_table.Get<ClientSession>(handle);
|
2017-12-30 18:40:28 +00:00
|
|
|
if (!session) {
|
2018-07-02 16:13:26 +00: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 03:20:01 +00:00
|
|
|
}
|
2014-05-27 02:12:46 +00:00
|
|
|
|
2018-07-02 16:13:26 +00:00
|
|
|
LOG_TRACE(Kernel_SVC, "called handle=0x{:08X}({})", handle, session->GetName());
|
2014-05-27 02:12:46 +00: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-10 23:58:28 +00:00
|
|
|
}
|
|
|
|
|
2017-10-23 04:15:45 +00:00
|
|
|
/// Get the ID for the specified thread.
|
2018-12-19 03:38:22 +00:00
|
|
|
static ResultCode GetThreadId(u64* thread_id, Handle thread_handle) {
|
2018-07-02 16:13:26 +00:00
|
|
|
LOG_TRACE(Kernel_SVC, "called thread=0x{:08X}", thread_handle);
|
2017-10-23 04:15:45 +00:00
|
|
|
|
2018-10-20 18:34:41 +00:00
|
|
|
const auto& handle_table = Core::CurrentProcess()->GetHandleTable();
|
|
|
|
const SharedPtr<Thread> thread = handle_table.Get<Thread>(thread_handle);
|
2017-12-30 18:40:28 +00:00
|
|
|
if (!thread) {
|
2018-11-26 08:47:39 +00:00
|
|
|
LOG_ERROR(Kernel_SVC, "Thread handle does not exist, handle=0x{:08X}", thread_handle);
|
2017-10-23 04:15:45 +00:00
|
|
|
return ERR_INVALID_HANDLE;
|
|
|
|
}
|
|
|
|
|
2018-10-03 22:47:57 +00:00
|
|
|
*thread_id = thread->GetThreadID();
|
2017-10-23 04:15:45 +00:00
|
|
|
return RESULT_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2018-12-19 04:09:08 +00:00
|
|
|
/// Gets the ID of the specified process or a specified thread's owning process.
|
|
|
|
static ResultCode GetProcessId(u64* process_id, Handle handle) {
|
|
|
|
LOG_DEBUG(Kernel_SVC, "called handle=0x{:08X}", handle);
|
2017-10-23 04:15:45 +00:00
|
|
|
|
2018-10-20 18:34:41 +00:00
|
|
|
const auto& handle_table = Core::CurrentProcess()->GetHandleTable();
|
2018-12-19 04:09:08 +00:00
|
|
|
const SharedPtr<Process> process = handle_table.Get<Process>(handle);
|
|
|
|
if (process) {
|
|
|
|
*process_id = process->GetProcessID();
|
|
|
|
return RESULT_SUCCESS;
|
2017-10-23 04:15:45 +00:00
|
|
|
}
|
|
|
|
|
2018-12-19 04:09:08 +00:00
|
|
|
const SharedPtr<Thread> thread = handle_table.Get<Thread>(handle);
|
|
|
|
if (thread) {
|
|
|
|
const Process* const owner_process = thread->GetOwnerProcess();
|
|
|
|
if (!owner_process) {
|
|
|
|
LOG_ERROR(Kernel_SVC, "Non-existent owning process encountered.");
|
|
|
|
return ERR_INVALID_HANDLE;
|
|
|
|
}
|
|
|
|
|
|
|
|
*process_id = owner_process->GetProcessID();
|
|
|
|
return RESULT_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
// NOTE: This should also handle debug objects before returning.
|
|
|
|
|
|
|
|
LOG_ERROR(Kernel_SVC, "Handle does not exist, handle=0x{:08X}", handle);
|
|
|
|
return ERR_INVALID_HANDLE;
|
2017-10-23 04:15:45 +00:00
|
|
|
}
|
|
|
|
|
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,
|
2018-09-15 13:21:06 +00:00
|
|
|
SharedPtr<WaitObject> object, std::size_t index) {
|
2018-10-03 22:47:57 +00:00
|
|
|
ASSERT(thread->GetStatus() == ThreadStatus::WaitSynchAny);
|
2018-01-06 19:19:28 +00:00
|
|
|
|
|
|
|
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
|
|
|
};
|
|
|
|
|
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-07-02 16:13:26 +00:00
|
|
|
LOG_TRACE(Kernel_SVC, "called handles_address=0x{:X}, handle_count={}, nano_seconds={}",
|
2018-07-02 16:20:50 +00:00
|
|
|
handles_address, handle_count, nano_seconds);
|
2018-01-06 19:34:32 +00:00
|
|
|
|
2018-11-26 06:06:13 +00:00
|
|
|
if (!Memory::IsValidVirtualAddress(handles_address)) {
|
2018-11-26 08:47:39 +00:00
|
|
|
LOG_ERROR(Kernel_SVC,
|
|
|
|
"Handle address is not a valid virtual address, handle_address=0x{:016X}",
|
|
|
|
handles_address);
|
2018-01-06 19:34:32 +00:00
|
|
|
return ERR_INVALID_POINTER;
|
2018-11-26 06:06:13 +00:00
|
|
|
}
|
2018-01-06 19:34:32 +00:00
|
|
|
|
2018-01-09 20:02:43 +00:00
|
|
|
static constexpr u64 MaxHandles = 0x40;
|
|
|
|
|
2018-11-16 19:24:27 +00:00
|
|
|
if (handle_count > MaxHandles) {
|
2018-11-26 08:47:39 +00:00
|
|
|
LOG_ERROR(Kernel_SVC, "Handle count specified is too large, expected {} but got {}",
|
|
|
|
MaxHandles, handle_count);
|
2018-11-16 19:24:27 +00:00
|
|
|
return ERR_OUT_OF_RANGE;
|
|
|
|
}
|
2018-01-06 19:34:32 +00:00
|
|
|
|
2018-10-03 22:47:57 +00:00
|
|
|
auto* const thread = GetCurrentThread();
|
2018-01-09 16:53:50 +00:00
|
|
|
|
2018-10-03 22:47:57 +00:00
|
|
|
using ObjectPtr = Thread::ThreadWaitObjects::value_type;
|
|
|
|
Thread::ThreadWaitObjects objects(handle_count);
|
2018-10-20 18:34:41 +00:00
|
|
|
const auto& handle_table = Core::CurrentProcess()->GetHandleTable();
|
2018-01-06 19:34:32 +00:00
|
|
|
|
2018-07-24 13:55:15 +00:00
|
|
|
for (u64 i = 0; i < handle_count; ++i) {
|
|
|
|
const Handle handle = Memory::Read32(handles_address + i * sizeof(Handle));
|
2018-10-20 18:34:41 +00:00
|
|
|
const auto object = handle_table.Get<WaitObject>(handle);
|
2018-07-24 13:55:15 +00:00
|
|
|
|
|
|
|
if (object == nullptr) {
|
2018-11-26 06:06:13 +00:00
|
|
|
LOG_ERROR(Kernel_SVC, "Object is a nullptr");
|
2018-01-06 19:34:32 +00:00
|
|
|
return ERR_INVALID_HANDLE;
|
2018-07-24 13:55:15 +00:00
|
|
|
}
|
|
|
|
|
2018-01-06 19:34:32 +00:00
|
|
|
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.
|
2018-11-26 06:06:13 +00:00
|
|
|
if (nano_seconds == 0) {
|
2018-01-09 16:53:50 +00:00
|
|
|
return RESULT_TIMEOUT;
|
2018-11-26 06:06:13 +00:00
|
|
|
}
|
2018-01-09 16:53:50 +00:00
|
|
|
|
2018-11-26 06:06:13 +00:00
|
|
|
for (auto& object : objects) {
|
2018-01-09 20:02:43 +00:00
|
|
|
object->AddWaitingThread(thread);
|
2018-11-26 06:06:13 +00:00
|
|
|
}
|
2018-01-09 20:02:43 +00:00
|
|
|
|
2018-10-03 22:47:57 +00:00
|
|
|
thread->SetWaitObjects(std::move(objects));
|
|
|
|
thread->SetStatus(ThreadStatus::WaitSynchAny);
|
2018-01-09 20:02:43 +00:00
|
|
|
|
|
|
|
// Create an event to wake the thread up after the specified nanosecond delay has passed
|
|
|
|
thread->WakeAfterDelay(nano_seconds);
|
2018-10-03 22:47:57 +00:00
|
|
|
thread->SetWakeupCallback(DefaultThreadWakeupCallback);
|
2018-01-09 20:02:43 +00:00
|
|
|
|
2018-10-03 22:47:57 +00:00
|
|
|
Core::System::GetInstance().CpuCore(thread->GetProcessorID()).PrepareReschedule();
|
2018-01-09 20:02:43 +00:00
|
|
|
|
|
|
|
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) {
|
2018-07-02 16:13:26 +00:00
|
|
|
LOG_TRACE(Kernel_SVC, "called thread=0x{:X}", thread_handle);
|
2018-01-09 20:02:04 +00:00
|
|
|
|
2018-10-20 18:34:41 +00:00
|
|
|
const auto& handle_table = Core::CurrentProcess()->GetHandleTable();
|
|
|
|
const SharedPtr<Thread> thread = handle_table.Get<Thread>(thread_handle);
|
2018-01-09 20:02:04 +00:00
|
|
|
if (!thread) {
|
2018-11-26 08:47:39 +00:00
|
|
|
LOG_ERROR(Kernel_SVC, "Thread handle does not exist, thread_handle=0x{:08X}",
|
|
|
|
thread_handle);
|
2018-01-09 20:02:04 +00:00
|
|
|
return ERR_INVALID_HANDLE;
|
|
|
|
}
|
|
|
|
|
2018-10-03 22:47:57 +00:00
|
|
|
ASSERT(thread->GetStatus() == ThreadStatus::WaitSynchAny);
|
2018-11-16 19:24:27 +00:00
|
|
|
thread->SetWaitSynchronizationResult(ERR_SYNCHRONIZATION_CANCELED);
|
2018-01-09 20:02:04 +00:00
|
|
|
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-07-02 16:13:26 +00:00
|
|
|
LOG_TRACE(Kernel_SVC,
|
2018-07-02 16:20:50 +00:00
|
|
|
"called holding_thread_handle=0x{:08X}, mutex_addr=0x{:X}, "
|
|
|
|
"requesting_current_thread_handle=0x{:08X}",
|
|
|
|
holding_thread_handle, mutex_addr, requesting_thread_handle);
|
2018-01-01 19:02:26 +00:00
|
|
|
|
2018-09-17 22:49:51 +00:00
|
|
|
if (Memory::IsKernelVirtualAddress(mutex_addr)) {
|
2018-11-26 08:47:39 +00:00
|
|
|
LOG_ERROR(Kernel_SVC, "Mutex Address is a kernel virtual address, mutex_addr={:016X}",
|
|
|
|
mutex_addr);
|
2018-09-17 22:49:51 +00:00
|
|
|
return ERR_INVALID_ADDRESS_STATE;
|
|
|
|
}
|
|
|
|
|
2018-10-18 17:01:26 +00:00
|
|
|
if (!Common::IsWordAligned(mutex_addr)) {
|
2018-11-26 08:47:39 +00:00
|
|
|
LOG_ERROR(Kernel_SVC, "Mutex Address is not word aligned, mutex_addr={:016X}", mutex_addr);
|
2018-10-18 17:01:26 +00:00
|
|
|
return ERR_INVALID_ADDRESS;
|
|
|
|
}
|
|
|
|
|
2018-10-20 18:34:41 +00:00
|
|
|
auto& handle_table = Core::CurrentProcess()->GetHandleTable();
|
2018-08-28 16:30:33 +00:00
|
|
|
return Mutex::TryAcquire(handle_table, mutex_addr, holding_thread_handle,
|
|
|
|
requesting_thread_handle);
|
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-07-02 16:13:26 +00:00
|
|
|
LOG_TRACE(Kernel_SVC, "called mutex_addr=0x{:X}", mutex_addr);
|
2018-01-01 19:04:36 +00:00
|
|
|
|
2018-09-17 22:49:51 +00:00
|
|
|
if (Memory::IsKernelVirtualAddress(mutex_addr)) {
|
2018-11-26 08:47:39 +00:00
|
|
|
LOG_ERROR(Kernel_SVC, "Mutex Address is a kernel virtual address, mutex_addr={:016X}",
|
|
|
|
mutex_addr);
|
2018-09-17 22:49:51 +00:00
|
|
|
return ERR_INVALID_ADDRESS_STATE;
|
|
|
|
}
|
|
|
|
|
2018-10-18 17:01:26 +00:00
|
|
|
if (!Common::IsWordAligned(mutex_addr)) {
|
2018-11-26 08:47:39 +00:00
|
|
|
LOG_ERROR(Kernel_SVC, "Mutex Address is not word aligned, mutex_addr={:016X}", mutex_addr);
|
2018-10-18 17:01:26 +00:00
|
|
|
return ERR_INVALID_ADDRESS;
|
|
|
|
}
|
|
|
|
|
2018-04-20 17:01:14 +00:00
|
|
|
return Mutex::Release(mutex_addr);
|
2018-01-01 19:04:36 +00:00
|
|
|
}
|
|
|
|
|
2018-10-23 04:17:13 +00:00
|
|
|
enum class BreakType : u32 {
|
2018-10-23 04:03:59 +00:00
|
|
|
Panic = 0,
|
2018-10-23 04:17:13 +00:00
|
|
|
AssertionFailed = 1,
|
2018-10-23 04:03:59 +00:00
|
|
|
PreNROLoad = 3,
|
|
|
|
PostNROLoad = 4,
|
|
|
|
PreNROUnload = 5,
|
|
|
|
PostNROUnload = 6,
|
|
|
|
};
|
|
|
|
|
2018-10-09 01:11:14 +00:00
|
|
|
struct BreakReason {
|
|
|
|
union {
|
2018-10-10 01:23:50 +00:00
|
|
|
u32 raw;
|
2018-10-23 04:03:59 +00:00
|
|
|
BitField<0, 30, BreakType> break_type;
|
2018-10-10 01:27:44 +00:00
|
|
|
BitField<31, 1, u32> signal_debugger;
|
2018-10-09 01:11:14 +00:00
|
|
|
};
|
|
|
|
};
|
|
|
|
|
2017-10-14 21:30:07 +00:00
|
|
|
/// Break program execution
|
2018-10-10 01:23:50 +00:00
|
|
|
static void Break(u32 reason, u64 info1, u64 info2) {
|
2018-10-09 01:11:14 +00:00
|
|
|
BreakReason break_reason{reason};
|
2018-11-08 04:43:54 +00:00
|
|
|
bool has_dumped_buffer{};
|
2018-10-23 04:03:59 +00:00
|
|
|
|
2018-11-08 04:43:54 +00:00
|
|
|
const auto handle_debug_buffer = [&](VAddr addr, u64 sz) {
|
|
|
|
if (sz == 0 || addr == 0 || has_dumped_buffer) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// This typically is an error code so we're going to assume this is the case
|
|
|
|
if (sz == sizeof(u32)) {
|
|
|
|
LOG_CRITICAL(Debug_Emulated, "debug_buffer_err_code={:X}", Memory::Read32(addr));
|
|
|
|
} else {
|
|
|
|
// We don't know what's in here so we'll hexdump it
|
|
|
|
std::vector<u8> debug_buffer(sz);
|
|
|
|
Memory::ReadBlock(addr, debug_buffer.data(), sz);
|
|
|
|
std::string hexdump;
|
|
|
|
for (std::size_t i = 0; i < debug_buffer.size(); i++) {
|
|
|
|
hexdump += fmt::format("{:02X} ", debug_buffer[i]);
|
|
|
|
if (i != 0 && i % 16 == 0) {
|
|
|
|
hexdump += '\n';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
LOG_CRITICAL(Debug_Emulated, "debug_buffer=\n{}", hexdump);
|
|
|
|
}
|
|
|
|
has_dumped_buffer = true;
|
|
|
|
};
|
2018-10-23 04:03:59 +00:00
|
|
|
switch (break_reason.break_type) {
|
|
|
|
case BreakType::Panic:
|
2018-10-23 04:17:13 +00:00
|
|
|
LOG_CRITICAL(Debug_Emulated, "Signalling debugger, PANIC! info1=0x{:016X}, info2=0x{:016X}",
|
|
|
|
info1, info2);
|
2018-11-08 04:43:54 +00:00
|
|
|
handle_debug_buffer(info1, info2);
|
2018-10-23 04:17:13 +00:00
|
|
|
break;
|
|
|
|
case BreakType::AssertionFailed:
|
|
|
|
LOG_CRITICAL(Debug_Emulated,
|
|
|
|
"Signalling debugger, Assertion failed! info1=0x{:016X}, info2=0x{:016X}",
|
|
|
|
info1, info2);
|
2018-11-08 04:43:54 +00:00
|
|
|
handle_debug_buffer(info1, info2);
|
2018-10-23 04:03:59 +00:00
|
|
|
break;
|
|
|
|
case BreakType::PreNROLoad:
|
2018-10-23 04:17:13 +00:00
|
|
|
LOG_WARNING(
|
2018-10-09 00:10:30 +00:00
|
|
|
Debug_Emulated,
|
2018-10-23 04:17:13 +00:00
|
|
|
"Signalling debugger, Attempting to load an NRO at 0x{:016X} with size 0x{:016X}",
|
|
|
|
info1, info2);
|
2018-10-23 04:03:59 +00:00
|
|
|
break;
|
|
|
|
case BreakType::PostNROLoad:
|
2018-10-23 04:17:13 +00:00
|
|
|
LOG_WARNING(Debug_Emulated,
|
|
|
|
"Signalling debugger, Loaded an NRO at 0x{:016X} with size 0x{:016X}", info1,
|
|
|
|
info2);
|
2018-10-23 04:03:59 +00:00
|
|
|
break;
|
|
|
|
case BreakType::PreNROUnload:
|
2018-10-23 04:17:13 +00:00
|
|
|
LOG_WARNING(
|
2018-10-09 00:10:30 +00:00
|
|
|
Debug_Emulated,
|
2018-10-23 04:03:59 +00:00
|
|
|
"Signalling debugger, Attempting to unload an NRO at 0x{:016X} with size 0x{:016X}",
|
|
|
|
info1, info2);
|
|
|
|
break;
|
|
|
|
case BreakType::PostNROUnload:
|
2018-10-23 04:17:13 +00:00
|
|
|
LOG_WARNING(Debug_Emulated,
|
|
|
|
"Signalling debugger, Unloaded an NRO at 0x{:016X} with size 0x{:016X}", info1,
|
|
|
|
info2);
|
2018-10-23 04:03:59 +00:00
|
|
|
break;
|
|
|
|
default:
|
2018-10-23 04:17:13 +00:00
|
|
|
LOG_WARNING(
|
|
|
|
Debug_Emulated,
|
|
|
|
"Signalling debugger, Unknown break reason {}, info1=0x{:016X}, info2=0x{:016X}",
|
|
|
|
static_cast<u32>(break_reason.break_type.Value()), info1, info2);
|
2018-11-08 04:43:54 +00:00
|
|
|
handle_debug_buffer(info1, info2);
|
2018-10-23 04:03:59 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!break_reason.signal_debugger) {
|
2018-10-09 01:11:14 +00:00
|
|
|
LOG_CRITICAL(
|
2018-10-09 00:10:30 +00:00
|
|
|
Debug_Emulated,
|
|
|
|
"Emulated program broke execution! reason=0x{:016X}, info1=0x{:016X}, info2=0x{:016X}",
|
|
|
|
reason, info1, info2);
|
2018-11-08 04:43:54 +00:00
|
|
|
handle_debug_buffer(info1, info2);
|
2018-10-09 01:11:14 +00:00
|
|
|
ASSERT(false);
|
2018-10-14 07:14:51 +00:00
|
|
|
|
|
|
|
Core::CurrentProcess()->PrepareForTermination();
|
|
|
|
|
|
|
|
// Kill the current thread
|
|
|
|
GetCurrentThread()->Stop();
|
|
|
|
Core::System::GetInstance().PrepareReschedule();
|
2018-10-09 00:10:30 +00:00
|
|
|
}
|
2014-04-17 00:41:33 +00:00
|
|
|
}
|
|
|
|
|
2017-10-14 21:30:07 +00:00
|
|
|
/// Used to output a message on a debug hardware unit - does nothing on a retail unit
|
2018-09-12 08:49:08 +00:00
|
|
|
static void OutputDebugString(VAddr address, u64 len) {
|
2018-09-12 08:51:41 +00:00
|
|
|
if (len == 0) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-04-25 23:11:22 +00:00
|
|
|
std::string str(len, '\0');
|
|
|
|
Memory::ReadBlock(address, str.data(), str.size());
|
2018-07-02 16:13:26 +00:00
|
|
|
LOG_DEBUG(Debug_Emulated, "{}", str);
|
2014-05-18 03:37:25 +00:00
|
|
|
}
|
|
|
|
|
2018-01-01 21:01:06 +00:00
|
|
|
/// Gets system/memory information for the current process
|
2017-10-14 21:30:07 +00:00
|
|
|
static ResultCode GetInfo(u64* result, u64 info_id, u64 handle, u64 info_sub_id) {
|
2018-07-02 16:13:26 +00:00
|
|
|
LOG_TRACE(Kernel_SVC, "called info_id=0x{:X}, info_sub_id=0x{:X}, handle=0x{:08X}", info_id,
|
2018-07-02 16:20:50 +00:00
|
|
|
info_sub_id, handle);
|
2018-01-01 21:01:06 +00:00
|
|
|
|
2018-10-26 04:37:14 +00:00
|
|
|
enum class GetInfoType : u64 {
|
|
|
|
// 1.0.0+
|
|
|
|
AllowedCpuIdBitmask = 0,
|
|
|
|
AllowedThreadPrioBitmask = 1,
|
|
|
|
MapRegionBaseAddr = 2,
|
|
|
|
MapRegionSize = 3,
|
|
|
|
HeapRegionBaseAddr = 4,
|
|
|
|
HeapRegionSize = 5,
|
|
|
|
TotalMemoryUsage = 6,
|
|
|
|
TotalHeapUsage = 7,
|
|
|
|
IsCurrentProcessBeingDebugged = 8,
|
2018-12-04 05:29:15 +00:00
|
|
|
RegisterResourceLimit = 9,
|
2018-10-26 04:37:14 +00:00
|
|
|
IdleTickCount = 10,
|
|
|
|
RandomEntropy = 11,
|
|
|
|
PerformanceCounter = 0xF0000002,
|
|
|
|
// 2.0.0+
|
|
|
|
ASLRRegionBaseAddr = 12,
|
|
|
|
ASLRRegionSize = 13,
|
|
|
|
NewMapRegionBaseAddr = 14,
|
|
|
|
NewMapRegionSize = 15,
|
|
|
|
// 3.0.0+
|
|
|
|
IsVirtualAddressMemoryEnabled = 16,
|
|
|
|
PersonalMmHeapUsage = 17,
|
|
|
|
TitleId = 18,
|
|
|
|
// 4.0.0+
|
|
|
|
PrivilegedProcessId = 19,
|
|
|
|
// 5.0.0+
|
|
|
|
UserExceptionContextAddr = 20,
|
|
|
|
ThreadTickCount = 0xF0000002,
|
|
|
|
};
|
|
|
|
|
2018-12-02 06:37:15 +00:00
|
|
|
const auto info_id_type = static_cast<GetInfoType>(info_id);
|
2018-01-10 05:58:25 +00:00
|
|
|
|
2018-12-02 06:37:15 +00:00
|
|
|
switch (info_id_type) {
|
2018-01-10 05:58:25 +00:00
|
|
|
case GetInfoType::AllowedCpuIdBitmask:
|
2018-01-16 22:06:45 +00:00
|
|
|
case GetInfoType::AllowedThreadPrioBitmask:
|
|
|
|
case GetInfoType::MapRegionBaseAddr:
|
|
|
|
case GetInfoType::MapRegionSize:
|
2018-01-15 20:42:57 +00:00
|
|
|
case GetInfoType::HeapRegionBaseAddr:
|
|
|
|
case GetInfoType::HeapRegionSize:
|
2018-12-02 06:37:15 +00:00
|
|
|
case GetInfoType::ASLRRegionBaseAddr:
|
|
|
|
case GetInfoType::ASLRRegionSize:
|
|
|
|
case GetInfoType::NewMapRegionBaseAddr:
|
|
|
|
case GetInfoType::NewMapRegionSize:
|
2018-01-01 21:01:06 +00:00
|
|
|
case GetInfoType::TotalMemoryUsage:
|
|
|
|
case GetInfoType::TotalHeapUsage:
|
2018-12-02 06:37:15 +00:00
|
|
|
case GetInfoType::IsVirtualAddressMemoryEnabled:
|
|
|
|
case GetInfoType::PersonalMmHeapUsage:
|
|
|
|
case GetInfoType::TitleId:
|
|
|
|
case GetInfoType::UserExceptionContextAddr: {
|
|
|
|
if (info_sub_id != 0) {
|
|
|
|
return ERR_INVALID_ENUM_VALUE;
|
|
|
|
}
|
|
|
|
|
svc: Use the current process' handle table for retrieving the process instance to act upon
The kernel uses the handle table of the current process to retrieve the
process that should be used to retrieve certain information. To someone
not familiar with the kernel, this might raise the question of "Ok,
sounds nice, but doesn't this make it impossible to retrieve information
about the current process?".
No, it doesn't, because HandleTable instances in the kernel have the
notion of a "pseudo-handle", where certain values allow the kernel to
lookup objects outside of a given handle table. Currently, there's only
a pseudo-handle for the current process (0xFFFF8001) and a pseudo-handle
for the current thread (0xFFFF8000), so to retrieve the current process,
one would just pass 0xFFFF8001 into svcGetInfo.
The lookup itself in the handle table would be something like:
template <typename T>
T* Lookup(Handle handle) {
if (handle == PSEUDO_HANDLE_CURRENT_PROCESS) {
return CurrentProcess();
}
if (handle == PSUEDO_HANDLE_CURRENT_THREAD) {
return CurrentThread();
}
return static_cast<T*>(&objects[handle]);
}
which, as is shown, allows accessing the current process or current
thread, even if those two objects aren't actually within the HandleTable
instance.
2018-12-02 07:00:11 +00:00
|
|
|
const auto& current_process_handle_table = Core::CurrentProcess()->GetHandleTable();
|
|
|
|
const auto process = current_process_handle_table.Get<Process>(static_cast<Handle>(handle));
|
2018-12-02 06:37:15 +00:00
|
|
|
if (!process) {
|
|
|
|
return ERR_INVALID_HANDLE;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (info_id_type) {
|
|
|
|
case GetInfoType::AllowedCpuIdBitmask:
|
|
|
|
*result = process->GetAllowedProcessorMask();
|
|
|
|
return RESULT_SUCCESS;
|
|
|
|
|
|
|
|
case GetInfoType::AllowedThreadPrioBitmask:
|
|
|
|
*result = process->GetAllowedThreadPriorityMask();
|
|
|
|
return RESULT_SUCCESS;
|
|
|
|
|
|
|
|
case GetInfoType::MapRegionBaseAddr:
|
|
|
|
*result = process->VMManager().GetMapRegionBaseAddress();
|
|
|
|
return RESULT_SUCCESS;
|
|
|
|
|
|
|
|
case GetInfoType::MapRegionSize:
|
|
|
|
*result = process->VMManager().GetMapRegionSize();
|
|
|
|
return RESULT_SUCCESS;
|
|
|
|
|
|
|
|
case GetInfoType::HeapRegionBaseAddr:
|
|
|
|
*result = process->VMManager().GetHeapRegionBaseAddress();
|
|
|
|
return RESULT_SUCCESS;
|
|
|
|
|
|
|
|
case GetInfoType::HeapRegionSize:
|
|
|
|
*result = process->VMManager().GetHeapRegionSize();
|
|
|
|
return RESULT_SUCCESS;
|
|
|
|
|
|
|
|
case GetInfoType::ASLRRegionBaseAddr:
|
|
|
|
*result = process->VMManager().GetASLRRegionBaseAddress();
|
|
|
|
return RESULT_SUCCESS;
|
|
|
|
|
|
|
|
case GetInfoType::ASLRRegionSize:
|
|
|
|
*result = process->VMManager().GetASLRRegionSize();
|
|
|
|
return RESULT_SUCCESS;
|
|
|
|
|
|
|
|
case GetInfoType::NewMapRegionBaseAddr:
|
|
|
|
*result = process->VMManager().GetNewMapRegionBaseAddress();
|
|
|
|
return RESULT_SUCCESS;
|
|
|
|
|
|
|
|
case GetInfoType::NewMapRegionSize:
|
|
|
|
*result = process->VMManager().GetNewMapRegionSize();
|
|
|
|
return RESULT_SUCCESS;
|
|
|
|
|
|
|
|
case GetInfoType::TotalMemoryUsage:
|
|
|
|
*result = process->VMManager().GetTotalMemoryUsage();
|
|
|
|
return RESULT_SUCCESS;
|
|
|
|
|
|
|
|
case GetInfoType::TotalHeapUsage:
|
|
|
|
*result = process->VMManager().GetTotalHeapUsage();
|
|
|
|
return RESULT_SUCCESS;
|
|
|
|
|
|
|
|
case GetInfoType::IsVirtualAddressMemoryEnabled:
|
|
|
|
*result = process->IsVirtualMemoryEnabled();
|
|
|
|
return RESULT_SUCCESS;
|
|
|
|
|
|
|
|
case GetInfoType::TitleId:
|
|
|
|
*result = process->GetTitleID();
|
|
|
|
return RESULT_SUCCESS;
|
|
|
|
|
|
|
|
case GetInfoType::UserExceptionContextAddr:
|
|
|
|
LOG_WARNING(Kernel_SVC,
|
|
|
|
"(STUBBED) Attempted to query user exception context address, returned 0");
|
|
|
|
*result = 0;
|
|
|
|
return RESULT_SUCCESS;
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
LOG_WARNING(Kernel_SVC, "(STUBBED) Unimplemented svcGetInfo id=0x{:016X}", info_id);
|
|
|
|
return ERR_INVALID_ENUM_VALUE;
|
|
|
|
}
|
|
|
|
|
2018-02-04 17:34:45 +00:00
|
|
|
case GetInfoType::IsCurrentProcessBeingDebugged:
|
|
|
|
*result = 0;
|
2018-12-02 06:37:15 +00:00
|
|
|
return RESULT_SUCCESS;
|
|
|
|
|
2018-12-04 05:29:15 +00:00
|
|
|
case GetInfoType::RegisterResourceLimit: {
|
|
|
|
if (handle != 0) {
|
|
|
|
return ERR_INVALID_HANDLE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (info_sub_id != 0) {
|
|
|
|
return ERR_INVALID_COMBINATION;
|
|
|
|
}
|
|
|
|
|
|
|
|
Process* const current_process = Core::CurrentProcess();
|
|
|
|
HandleTable& handle_table = current_process->GetHandleTable();
|
|
|
|
const auto resource_limit = current_process->GetResourceLimit();
|
|
|
|
if (!resource_limit) {
|
|
|
|
*result = KernelHandle::InvalidHandle;
|
|
|
|
// Yes, the kernel considers this a successful operation.
|
|
|
|
return RESULT_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
const auto table_result = handle_table.Create(resource_limit);
|
|
|
|
if (table_result.Failed()) {
|
|
|
|
return table_result.Code();
|
|
|
|
}
|
|
|
|
|
|
|
|
*result = *table_result;
|
|
|
|
return RESULT_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2018-01-01 21:01:06 +00:00
|
|
|
case GetInfoType::RandomEntropy:
|
2018-11-13 17:25:43 +00:00
|
|
|
if (handle != 0) {
|
2018-11-26 08:47:39 +00:00
|
|
|
LOG_ERROR(Kernel_SVC, "Process Handle is non zero, expected 0 result but got {:016X}",
|
|
|
|
handle);
|
2018-11-13 17:25:43 +00:00
|
|
|
return ERR_INVALID_HANDLE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (info_sub_id >= Process::RANDOM_ENTROPY_SIZE) {
|
2018-11-26 08:47:39 +00:00
|
|
|
LOG_ERROR(Kernel_SVC, "Entropy size is out of range, expected {} but got {}",
|
|
|
|
Process::RANDOM_ENTROPY_SIZE, info_sub_id);
|
2018-11-16 19:24:27 +00:00
|
|
|
return ERR_INVALID_COMBINATION;
|
2018-11-13 17:25:43 +00:00
|
|
|
}
|
|
|
|
|
2018-12-02 06:37:15 +00:00
|
|
|
*result = Core::CurrentProcess()->GetRandomEntropy(info_sub_id);
|
2018-11-13 17:25:43 +00:00
|
|
|
return RESULT_SUCCESS;
|
2018-12-02 06:37:15 +00:00
|
|
|
|
2018-01-15 20:42:57 +00:00
|
|
|
case GetInfoType::PrivilegedProcessId:
|
2018-07-02 16:13:26 +00:00
|
|
|
LOG_WARNING(Kernel_SVC,
|
2018-07-02 16:20:50 +00:00
|
|
|
"(STUBBED) Attempted to query privileged process id bounds, returned 0");
|
2018-01-15 20:42:57 +00:00
|
|
|
*result = 0;
|
2018-12-02 06:37:15 +00:00
|
|
|
return RESULT_SUCCESS;
|
|
|
|
|
2018-10-25 22:42:50 +00:00
|
|
|
case GetInfoType::ThreadTickCount: {
|
|
|
|
constexpr u64 num_cpus = 4;
|
|
|
|
if (info_sub_id != 0xFFFFFFFFFFFFFFFF && info_sub_id >= num_cpus) {
|
2018-11-26 08:47:39 +00:00
|
|
|
LOG_ERROR(Kernel_SVC, "Core count is out of range, expected {} but got {}", num_cpus,
|
|
|
|
info_sub_id);
|
2018-11-16 19:24:27 +00:00
|
|
|
return ERR_INVALID_COMBINATION;
|
2018-10-25 22:42:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
const auto thread =
|
2018-12-02 06:37:15 +00:00
|
|
|
Core::CurrentProcess()->GetHandleTable().Get<Thread>(static_cast<Handle>(handle));
|
2018-10-25 22:42:50 +00:00
|
|
|
if (!thread) {
|
2018-11-26 08:47:39 +00:00
|
|
|
LOG_ERROR(Kernel_SVC, "Thread handle does not exist, handle=0x{:08X}",
|
|
|
|
static_cast<Handle>(handle));
|
2018-10-25 22:42:50 +00:00
|
|
|
return ERR_INVALID_HANDLE;
|
|
|
|
}
|
|
|
|
|
2018-10-28 21:44:46 +00:00
|
|
|
const auto& system = Core::System::GetInstance();
|
2018-10-25 22:42:50 +00:00
|
|
|
const auto& scheduler = system.CurrentScheduler();
|
|
|
|
const auto* const current_thread = scheduler.GetCurrentThread();
|
|
|
|
const bool same_thread = current_thread == thread;
|
|
|
|
|
|
|
|
const u64 prev_ctx_ticks = scheduler.GetLastContextSwitchTicks();
|
|
|
|
u64 out_ticks = 0;
|
|
|
|
if (same_thread && info_sub_id == 0xFFFFFFFFFFFFFFFF) {
|
|
|
|
const u64 thread_ticks = current_thread->GetTotalCPUTimeTicks();
|
|
|
|
|
|
|
|
out_ticks = thread_ticks + (CoreTiming::GetTicks() - prev_ctx_ticks);
|
|
|
|
} else if (same_thread && info_sub_id == system.CurrentCoreIndex()) {
|
|
|
|
out_ticks = CoreTiming::GetTicks() - prev_ctx_ticks;
|
|
|
|
}
|
|
|
|
|
|
|
|
*result = out_ticks;
|
2018-12-02 06:37:15 +00:00
|
|
|
return RESULT_SUCCESS;
|
2018-10-25 22:42:50 +00:00
|
|
|
}
|
2018-12-02 06:37:15 +00:00
|
|
|
|
2018-01-01 21:01:06 +00:00
|
|
|
default:
|
2018-11-25 21:48:44 +00:00
|
|
|
LOG_WARNING(Kernel_SVC, "(STUBBED) Unimplemented svcGetInfo id=0x{:016X}", info_id);
|
|
|
|
return ERR_INVALID_ENUM_VALUE;
|
2015-05-17 05:06:59 +00:00
|
|
|
}
|
2014-05-01 22:50:36 +00:00
|
|
|
}
|
|
|
|
|
2018-04-03 03:50:17 +00:00
|
|
|
/// Sets the thread activity
|
|
|
|
static ResultCode SetThreadActivity(Handle handle, u32 unknown) {
|
2018-07-02 16:20:50 +00:00
|
|
|
LOG_WARNING(Kernel_SVC, "(STUBBED) called, handle=0x{:08X}, unknown=0x{:08X}", handle, unknown);
|
2018-04-03 03:50:17 +00:00
|
|
|
return RESULT_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Gets the thread context
|
2018-09-29 23:58:21 +00:00
|
|
|
static ResultCode GetThreadContext(VAddr thread_context, Handle handle) {
|
|
|
|
LOG_DEBUG(Kernel_SVC, "called, context=0x{:08X}, thread=0x{:X}", thread_context, handle);
|
|
|
|
|
2018-10-20 18:34:41 +00:00
|
|
|
const auto* current_process = Core::CurrentProcess();
|
|
|
|
const SharedPtr<Thread> thread = current_process->GetHandleTable().Get<Thread>(handle);
|
2018-09-29 23:58:21 +00:00
|
|
|
if (!thread) {
|
2018-11-26 08:47:39 +00:00
|
|
|
LOG_ERROR(Kernel_SVC, "Thread handle does not exist, handle=0x{:08X}", handle);
|
2018-09-29 23:58:21 +00:00
|
|
|
return ERR_INVALID_HANDLE;
|
|
|
|
}
|
|
|
|
|
2018-10-03 22:47:57 +00:00
|
|
|
if (thread->GetOwnerProcess() != current_process) {
|
2018-11-26 08:47:39 +00:00
|
|
|
LOG_ERROR(Kernel_SVC,
|
|
|
|
"The current process does not own the current thread, thread_handle={:08X} "
|
|
|
|
"thread_pid={}, "
|
|
|
|
"current_process_pid={}",
|
|
|
|
handle, thread->GetOwnerProcess()->GetProcessID(),
|
|
|
|
current_process->GetProcessID());
|
2018-09-29 23:58:21 +00:00
|
|
|
return ERR_INVALID_HANDLE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (thread == GetCurrentThread()) {
|
2018-11-26 08:47:39 +00:00
|
|
|
LOG_ERROR(Kernel_SVC, "The thread handle specified is the current running thread");
|
2018-09-29 23:58:21 +00:00
|
|
|
return ERR_ALREADY_REGISTERED;
|
|
|
|
}
|
|
|
|
|
2018-10-03 22:47:57 +00:00
|
|
|
Core::ARM_Interface::ThreadContext ctx = thread->GetContext();
|
2018-09-29 23:58:21 +00:00
|
|
|
// Mask away mode bits, interrupt bits, IL bit, and other reserved bits.
|
|
|
|
ctx.pstate &= 0xFF0FFE20;
|
|
|
|
|
|
|
|
// If 64-bit, we can just write the context registers directly and we're good.
|
|
|
|
// However, if 32-bit, we have to ensure some registers are zeroed out.
|
|
|
|
if (!current_process->Is64BitProcess()) {
|
|
|
|
std::fill(ctx.cpu_registers.begin() + 15, ctx.cpu_registers.end(), 0);
|
|
|
|
std::fill(ctx.vector_registers.begin() + 16, ctx.vector_registers.end(), u128{});
|
|
|
|
}
|
|
|
|
|
|
|
|
Memory::WriteBlock(thread_context, &ctx, sizeof(ctx));
|
2018-04-03 03:50:17 +00:00
|
|
|
return RESULT_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2014-06-02 02:12:54 +00:00
|
|
|
/// Gets the priority for the specified thread
|
2017-12-31 21:06:11 +00:00
|
|
|
static ResultCode GetThreadPriority(u32* priority, Handle handle) {
|
2018-11-26 06:06:13 +00:00
|
|
|
LOG_TRACE(Kernel_SVC, "called");
|
|
|
|
|
2018-10-20 18:34:41 +00:00
|
|
|
const auto& handle_table = Core::CurrentProcess()->GetHandleTable();
|
|
|
|
const SharedPtr<Thread> thread = handle_table.Get<Thread>(handle);
|
|
|
|
if (!thread) {
|
2018-11-26 08:47:39 +00:00
|
|
|
LOG_ERROR(Kernel_SVC, "Thread handle does not exist, handle=0x{:08X}", handle);
|
2017-12-31 21:06:11 +00:00
|
|
|
return ERR_INVALID_HANDLE;
|
2018-10-20 18:34:41 +00:00
|
|
|
}
|
2017-12-31 21:06:11 +00:00
|
|
|
|
|
|
|
*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) {
|
2018-11-26 06:06:13 +00:00
|
|
|
LOG_TRACE(Kernel_SVC, "called");
|
|
|
|
|
2017-12-31 20:58:16 +00:00
|
|
|
if (priority > THREADPRIO_LOWEST) {
|
2018-11-26 08:47:39 +00:00
|
|
|
LOG_ERROR(
|
|
|
|
Kernel_SVC,
|
|
|
|
"An invalid priority was specified, expected {} but got {} for thread_handle={:08X}",
|
|
|
|
THREADPRIO_LOWEST, priority, handle);
|
2018-09-12 08:25:53 +00:00
|
|
|
return ERR_INVALID_THREAD_PRIORITY;
|
2017-12-31 20:58:16 +00:00
|
|
|
}
|
|
|
|
|
2018-10-20 18:34:41 +00:00
|
|
|
const auto* const current_process = Core::CurrentProcess();
|
2017-12-31 20:58:16 +00:00
|
|
|
|
2018-10-24 18:07:53 +00:00
|
|
|
SharedPtr<Thread> thread = current_process->GetHandleTable().Get<Thread>(handle);
|
|
|
|
if (!thread) {
|
2018-11-26 08:47:39 +00:00
|
|
|
LOG_ERROR(Kernel_SVC, "Thread handle does not exist, handle=0x{:08X}", handle);
|
2018-10-24 18:07:53 +00:00
|
|
|
return ERR_INVALID_HANDLE;
|
2017-12-31 20:58:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
thread->SetPriority(priority);
|
|
|
|
|
2018-10-03 22:47:57 +00:00
|
|
|
Core::System::GetInstance().CpuCore(thread->GetProcessorID()).PrepareReschedule();
|
2017-12-31 20:58:16 +00:00
|
|
|
return RESULT_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2017-12-31 21:01:04 +00:00
|
|
|
/// Get which CPU core is executing the current thread
|
|
|
|
static u32 GetCurrentProcessorNumber() {
|
2018-07-02 16:13:26 +00:00
|
|
|
LOG_TRACE(Kernel_SVC, "called");
|
2018-10-03 22:47:57 +00:00
|
|
|
return GetCurrentThread()->GetProcessorID();
|
2017-12-31 21:01:04 +00:00
|
|
|
}
|
|
|
|
|
2018-01-14 22:15:31 +00:00
|
|
|
static ResultCode MapSharedMemory(Handle shared_memory_handle, VAddr addr, u64 size,
|
|
|
|
u32 permissions) {
|
2018-07-02 16:20:50 +00:00
|
|
|
LOG_TRACE(Kernel_SVC,
|
|
|
|
"called, shared_memory_handle=0x{:X}, addr=0x{:X}, size=0x{:X}, permissions=0x{:08X}",
|
|
|
|
shared_memory_handle, addr, size, permissions);
|
2018-01-14 22:15:31 +00:00
|
|
|
|
2018-10-18 16:55:27 +00:00
|
|
|
if (!Common::Is4KBAligned(addr)) {
|
2018-11-26 08:47:39 +00:00
|
|
|
LOG_ERROR(Kernel_SVC, "Address is not aligned to 4KB, addr=0x{:016X}", addr);
|
2018-09-14 00:16:43 +00:00
|
|
|
return ERR_INVALID_ADDRESS;
|
|
|
|
}
|
|
|
|
|
2018-11-26 08:47:39 +00:00
|
|
|
if (size == 0) {
|
|
|
|
LOG_ERROR(Kernel_SVC, "Size is 0");
|
|
|
|
return ERR_INVALID_SIZE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!Common::Is4KBAligned(size)) {
|
|
|
|
LOG_ERROR(Kernel_SVC, "Size is not aligned to 4KB, size=0x{:016X}", size);
|
2018-09-14 00:16:43 +00:00
|
|
|
return ERR_INVALID_SIZE;
|
|
|
|
}
|
|
|
|
|
2018-10-18 02:39:21 +00:00
|
|
|
if (!IsValidAddressRange(addr, size)) {
|
2018-11-26 08:47:39 +00:00
|
|
|
LOG_ERROR(Kernel_SVC, "Region is not a valid address range, addr=0x{:016X}, size=0x{:016X}",
|
|
|
|
addr, size);
|
2018-10-18 02:39:21 +00:00
|
|
|
return ERR_INVALID_ADDRESS_STATE;
|
|
|
|
}
|
|
|
|
|
2018-09-14 00:16:43 +00:00
|
|
|
const auto permissions_type = static_cast<MemoryPermission>(permissions);
|
|
|
|
if (permissions_type != MemoryPermission::Read &&
|
|
|
|
permissions_type != MemoryPermission::ReadWrite) {
|
2018-11-26 08:47:39 +00:00
|
|
|
LOG_ERROR(Kernel_SVC, "Expected Read or ReadWrite permission but got permissions=0x{:08X}",
|
|
|
|
permissions);
|
2018-09-14 00:16:43 +00:00
|
|
|
return ERR_INVALID_MEMORY_PERMISSIONS;
|
|
|
|
}
|
|
|
|
|
2018-10-20 18:34:41 +00:00
|
|
|
auto* const current_process = Core::CurrentProcess();
|
|
|
|
auto shared_memory = current_process->GetHandleTable().Get<SharedMemory>(shared_memory_handle);
|
2018-01-14 22:15:31 +00:00
|
|
|
if (!shared_memory) {
|
2018-11-26 08:47:39 +00:00
|
|
|
LOG_ERROR(Kernel_SVC, "Shared memory does not exist, shared_memory_handle=0x{:08X}",
|
|
|
|
shared_memory_handle);
|
2018-01-14 22:15:31 +00:00
|
|
|
return ERR_INVALID_HANDLE;
|
|
|
|
}
|
|
|
|
|
2018-10-18 02:39:21 +00:00
|
|
|
const auto& vm_manager = current_process->VMManager();
|
|
|
|
if (!vm_manager.IsWithinASLRRegion(addr, size)) {
|
2018-11-26 08:47:39 +00:00
|
|
|
LOG_ERROR(Kernel_SVC, "Region is not within the ASLR region. addr=0x{:016X}, size={:016X}",
|
|
|
|
addr, size);
|
2018-10-18 02:39:21 +00:00
|
|
|
return ERR_INVALID_MEMORY_RANGE;
|
|
|
|
}
|
|
|
|
|
2018-11-19 14:05:04 +00:00
|
|
|
return shared_memory->Map(*current_process, addr, permissions_type, MemoryPermission::DontCare);
|
2018-01-14 22:15:31 +00:00
|
|
|
}
|
|
|
|
|
2018-02-22 19:16:43 +00:00
|
|
|
static ResultCode UnmapSharedMemory(Handle shared_memory_handle, VAddr addr, u64 size) {
|
2018-07-02 16:13:26 +00:00
|
|
|
LOG_WARNING(Kernel_SVC, "called, shared_memory_handle=0x{:08X}, addr=0x{:X}, size=0x{:X}",
|
2018-07-02 16:20:50 +00:00
|
|
|
shared_memory_handle, addr, size);
|
2018-02-22 19:16:43 +00:00
|
|
|
|
2018-10-18 16:55:27 +00:00
|
|
|
if (!Common::Is4KBAligned(addr)) {
|
2018-11-26 08:47:39 +00:00
|
|
|
LOG_ERROR(Kernel_SVC, "Address is not aligned to 4KB, addr=0x{:016X}", addr);
|
2018-09-14 00:16:43 +00:00
|
|
|
return ERR_INVALID_ADDRESS;
|
|
|
|
}
|
|
|
|
|
2018-11-26 08:47:39 +00:00
|
|
|
if (size == 0) {
|
|
|
|
LOG_ERROR(Kernel_SVC, "Size is 0");
|
|
|
|
return ERR_INVALID_SIZE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!Common::Is4KBAligned(size)) {
|
|
|
|
LOG_ERROR(Kernel_SVC, "Size is not aligned to 4KB, size=0x{:016X}", size);
|
2018-09-14 00:16:43 +00:00
|
|
|
return ERR_INVALID_SIZE;
|
|
|
|
}
|
|
|
|
|
2018-10-18 02:39:21 +00:00
|
|
|
if (!IsValidAddressRange(addr, size)) {
|
2018-11-26 08:47:39 +00:00
|
|
|
LOG_ERROR(Kernel_SVC, "Region is not a valid address range, addr=0x{:016X}, size=0x{:016X}",
|
|
|
|
addr, size);
|
2018-10-18 02:39:21 +00:00
|
|
|
return ERR_INVALID_ADDRESS_STATE;
|
|
|
|
}
|
|
|
|
|
2018-10-20 18:34:41 +00:00
|
|
|
auto* const current_process = Core::CurrentProcess();
|
|
|
|
auto shared_memory = current_process->GetHandleTable().Get<SharedMemory>(shared_memory_handle);
|
2018-10-18 02:39:21 +00:00
|
|
|
if (!shared_memory) {
|
2018-11-26 08:47:39 +00:00
|
|
|
LOG_ERROR(Kernel_SVC, "Shared memory does not exist, shared_memory_handle=0x{:08X}",
|
|
|
|
shared_memory_handle);
|
2018-10-18 02:39:21 +00:00
|
|
|
return ERR_INVALID_HANDLE;
|
|
|
|
}
|
|
|
|
|
|
|
|
const auto& vm_manager = current_process->VMManager();
|
|
|
|
if (!vm_manager.IsWithinASLRRegion(addr, size)) {
|
2018-11-26 08:47:39 +00:00
|
|
|
LOG_ERROR(Kernel_SVC, "Region is not within the ASLR region. addr=0x{:016X}, size={:016X}",
|
|
|
|
addr, size);
|
2018-10-18 02:39:21 +00:00
|
|
|
return ERR_INVALID_MEMORY_RANGE;
|
|
|
|
}
|
2018-02-22 19:16:43 +00:00
|
|
|
|
2018-11-19 14:05:04 +00:00
|
|
|
return shared_memory->Unmap(*current_process, addr);
|
2018-02-22 19:16:43 +00:00
|
|
|
}
|
|
|
|
|
svc: Handle memory writing explicitly within QueryProcessMemory
Moves the memory writes directly into QueryProcessMemory instead of
letting the wrapper function do it. It would be inaccurate to allow the
handler to do it because there's cases where memory shouldn't even be
written to. For example, if the given process handle is invalid.
HOWEVER, if the memory writing is within the wrapper, then we have no
control over if these memory writes occur, meaning in an error case, 68
bytes of memory randomly get trashed with zeroes, 64 of those being
written to wherever the memory info address points to, and the remaining
4 being written wherever the page info address points to.
One solution in this case would be to just conditionally check within
the handler itself, but this is kind of smelly, given the handler
shouldn't be performing conditional behavior itself, it's a behavior of
the managed function. In other words, if you remove the handler from the
equation entirely, does the function still retain its proper behavior?
In this case, no.
Now, we don't potentially trash memory from this function if an invalid
query is performed.
2018-12-12 16:48:06 +00:00
|
|
|
static ResultCode QueryProcessMemory(VAddr memory_info_address, VAddr page_info_address,
|
|
|
|
Handle process_handle, VAddr address) {
|
2018-12-12 16:34:01 +00:00
|
|
|
LOG_TRACE(Kernel_SVC, "called process=0x{:08X} address={:X}", process_handle, address);
|
2018-10-20 18:34:41 +00:00
|
|
|
const auto& handle_table = Core::CurrentProcess()->GetHandleTable();
|
|
|
|
SharedPtr<Process> process = handle_table.Get<Process>(process_handle);
|
2017-12-30 18:40:28 +00:00
|
|
|
if (!process) {
|
2018-11-26 08:47:39 +00:00
|
|
|
LOG_ERROR(Kernel_SVC, "Process handle does not exist, process_handle=0x{:08X}",
|
|
|
|
process_handle);
|
2015-07-17 19:45:12 +00:00
|
|
|
return ERR_INVALID_HANDLE;
|
2017-10-20 03:00:46 +00:00
|
|
|
}
|
2018-12-06 15:59:22 +00:00
|
|
|
|
|
|
|
const auto& vm_manager = process->VMManager();
|
svc: Handle memory writing explicitly within QueryProcessMemory
Moves the memory writes directly into QueryProcessMemory instead of
letting the wrapper function do it. It would be inaccurate to allow the
handler to do it because there's cases where memory shouldn't even be
written to. For example, if the given process handle is invalid.
HOWEVER, if the memory writing is within the wrapper, then we have no
control over if these memory writes occur, meaning in an error case, 68
bytes of memory randomly get trashed with zeroes, 64 of those being
written to wherever the memory info address points to, and the remaining
4 being written wherever the page info address points to.
One solution in this case would be to just conditionally check within
the handler itself, but this is kind of smelly, given the handler
shouldn't be performing conditional behavior itself, it's a behavior of
the managed function. In other words, if you remove the handler from the
equation entirely, does the function still retain its proper behavior?
In this case, no.
Now, we don't potentially trash memory from this function if an invalid
query is performed.
2018-12-12 16:48:06 +00:00
|
|
|
const MemoryInfo memory_info = vm_manager.QueryMemory(address);
|
2018-12-06 15:59:22 +00:00
|
|
|
|
svc: Handle memory writing explicitly within QueryProcessMemory
Moves the memory writes directly into QueryProcessMemory instead of
letting the wrapper function do it. It would be inaccurate to allow the
handler to do it because there's cases where memory shouldn't even be
written to. For example, if the given process handle is invalid.
HOWEVER, if the memory writing is within the wrapper, then we have no
control over if these memory writes occur, meaning in an error case, 68
bytes of memory randomly get trashed with zeroes, 64 of those being
written to wherever the memory info address points to, and the remaining
4 being written wherever the page info address points to.
One solution in this case would be to just conditionally check within
the handler itself, but this is kind of smelly, given the handler
shouldn't be performing conditional behavior itself, it's a behavior of
the managed function. In other words, if you remove the handler from the
equation entirely, does the function still retain its proper behavior?
In this case, no.
Now, we don't potentially trash memory from this function if an invalid
query is performed.
2018-12-12 16:48:06 +00:00
|
|
|
Memory::Write64(memory_info_address, memory_info.base_address);
|
|
|
|
Memory::Write64(memory_info_address + 8, memory_info.size);
|
|
|
|
Memory::Write32(memory_info_address + 16, memory_info.state);
|
|
|
|
Memory::Write32(memory_info_address + 20, memory_info.attributes);
|
|
|
|
Memory::Write32(memory_info_address + 24, memory_info.permission);
|
svc: Write out the complete MemoryInfo structure in QueryProcessMemory
In the previous change, the memory writing was moved into the service
function itself, however it still had a problem, in that the entire
MemoryInfo structure wasn't being written out, only the first 32 bytes
of it were being written out. We still need to write out the trailing
two reference count members and zero out the padding bits.
Not doing this can result in wrong behavior in userland code in the following
scenario:
MemoryInfo info; // Put on the stack, not quaranteed to be zeroed out.
svcQueryMemory(&info, ...);
if (info.device_refcount == ...) // Whoops, uninitialized read.
This can also cause the wrong thing to happen if the user code uses
std::memcmp to compare the struct, with another one (questionable, but
allowed), as the padding bits are not guaranteed to be a deterministic
value. Note that the kernel itself also fully zeroes out the structure
before writing it out including the padding bits.
2018-12-12 17:52:31 +00:00
|
|
|
Memory::Write32(memory_info_address + 32, memory_info.ipc_ref_count);
|
|
|
|
Memory::Write32(memory_info_address + 28, memory_info.device_ref_count);
|
|
|
|
Memory::Write32(memory_info_address + 36, 0);
|
svc: Handle memory writing explicitly within QueryProcessMemory
Moves the memory writes directly into QueryProcessMemory instead of
letting the wrapper function do it. It would be inaccurate to allow the
handler to do it because there's cases where memory shouldn't even be
written to. For example, if the given process handle is invalid.
HOWEVER, if the memory writing is within the wrapper, then we have no
control over if these memory writes occur, meaning in an error case, 68
bytes of memory randomly get trashed with zeroes, 64 of those being
written to wherever the memory info address points to, and the remaining
4 being written wherever the page info address points to.
One solution in this case would be to just conditionally check within
the handler itself, but this is kind of smelly, given the handler
shouldn't be performing conditional behavior itself, it's a behavior of
the managed function. In other words, if you remove the handler from the
equation entirely, does the function still retain its proper behavior?
In this case, no.
Now, we don't potentially trash memory from this function if an invalid
query is performed.
2018-12-12 16:48:06 +00:00
|
|
|
|
|
|
|
// Page info appears to be currently unused by the kernel and is always set to zero.
|
|
|
|
Memory::Write32(page_info_address, 0);
|
2018-12-06 15:59:22 +00:00
|
|
|
|
2015-01-23 05:36:58 +00:00
|
|
|
return RESULT_SUCCESS;
|
2014-05-16 00:17:30 +00:00
|
|
|
}
|
|
|
|
|
svc: Handle memory writing explicitly within QueryProcessMemory
Moves the memory writes directly into QueryProcessMemory instead of
letting the wrapper function do it. It would be inaccurate to allow the
handler to do it because there's cases where memory shouldn't even be
written to. For example, if the given process handle is invalid.
HOWEVER, if the memory writing is within the wrapper, then we have no
control over if these memory writes occur, meaning in an error case, 68
bytes of memory randomly get trashed with zeroes, 64 of those being
written to wherever the memory info address points to, and the remaining
4 being written wherever the page info address points to.
One solution in this case would be to just conditionally check within
the handler itself, but this is kind of smelly, given the handler
shouldn't be performing conditional behavior itself, it's a behavior of
the managed function. In other words, if you remove the handler from the
equation entirely, does the function still retain its proper behavior?
In this case, no.
Now, we don't potentially trash memory from this function if an invalid
query is performed.
2018-12-12 16:48:06 +00:00
|
|
|
static ResultCode QueryMemory(VAddr memory_info_address, VAddr page_info_address,
|
|
|
|
VAddr query_address) {
|
|
|
|
LOG_TRACE(Kernel_SVC,
|
|
|
|
"called, memory_info_address=0x{:016X}, page_info_address=0x{:016X}, "
|
|
|
|
"query_address=0x{:016X}",
|
|
|
|
memory_info_address, page_info_address, query_address);
|
|
|
|
|
|
|
|
return QueryProcessMemory(memory_info_address, page_info_address, CurrentProcess,
|
|
|
|
query_address);
|
2015-07-17 19:45:12 +00:00
|
|
|
}
|
|
|
|
|
2018-01-01 19:38:34 +00:00
|
|
|
/// Exits the current process
|
|
|
|
static void ExitProcess() {
|
2018-10-10 04:42:10 +00:00
|
|
|
auto* current_process = Core::CurrentProcess();
|
2018-01-01 19:38:34 +00:00
|
|
|
|
2018-09-21 06:06:47 +00:00
|
|
|
LOG_INFO(Kernel_SVC, "Process {} exiting", current_process->GetProcessID());
|
|
|
|
ASSERT_MSG(current_process->GetStatus() == ProcessStatus::Running,
|
2018-03-13 21:49:59 +00:00
|
|
|
"Process has already exited");
|
2018-01-01 19:38:34 +00:00
|
|
|
|
2018-09-21 06:06:47 +00:00
|
|
|
current_process->PrepareForTermination();
|
2018-01-01 19:38:34 +00:00
|
|
|
|
|
|
|
// 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-11-26 06:06:13 +00:00
|
|
|
LOG_TRACE(Kernel_SVC,
|
2018-12-04 03:13:50 +00:00
|
|
|
"called entrypoint=0x{:08X}, arg=0x{:08X}, stacktop=0x{:08X}, "
|
2018-11-26 06:06:13 +00:00
|
|
|
"threadpriority=0x{:08X}, processorid=0x{:08X} : created handle=0x{:08X}",
|
2018-12-01 07:05:19 +00:00
|
|
|
entry_point, arg, stack_top, priority, processor_id, *out_handle);
|
2018-11-26 06:06:13 +00:00
|
|
|
|
2017-12-31 21:10:01 +00:00
|
|
|
if (priority > THREADPRIO_LOWEST) {
|
2018-11-26 08:47:39 +00:00
|
|
|
LOG_ERROR(Kernel_SVC, "An invalid priority was specified, expected {} but got {}",
|
|
|
|
THREADPRIO_LOWEST, priority);
|
2018-09-12 08:25:53 +00:00
|
|
|
return ERR_INVALID_THREAD_PRIORITY;
|
2017-12-31 21:10:01 +00:00
|
|
|
}
|
|
|
|
|
2018-10-20 18:34:41 +00:00
|
|
|
auto* const current_process = Core::CurrentProcess();
|
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-10-20 18:34:41 +00:00
|
|
|
processor_id = current_process->GetDefaultProcessorID();
|
2017-12-31 21:10:01 +00:00
|
|
|
ASSERT(processor_id != THREADPROCESSORID_DEFAULT);
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (processor_id) {
|
|
|
|
case THREADPROCESSORID_0:
|
|
|
|
case THREADPROCESSORID_1:
|
2018-01-10 05:58:25 +00:00
|
|
|
case THREADPROCESSORID_2:
|
|
|
|
case THREADPROCESSORID_3:
|
2017-12-31 21:10:01 +00:00
|
|
|
break;
|
|
|
|
default:
|
2018-09-12 08:27:35 +00:00
|
|
|
LOG_ERROR(Kernel_SVC, "Invalid thread processor ID: {}", processor_id);
|
|
|
|
return ERR_INVALID_PROCESSOR_ID;
|
2017-12-31 21:10:01 +00:00
|
|
|
}
|
|
|
|
|
2018-10-20 18:34:41 +00:00
|
|
|
const std::string name = fmt::format("thread-{:X}", entry_point);
|
2018-08-28 16:30:33 +00:00
|
|
|
auto& kernel = Core::System::GetInstance().Kernel();
|
2018-01-03 01:40:30 +00:00
|
|
|
CASCADE_RESULT(SharedPtr<Thread> thread,
|
2018-08-28 16:30:33 +00:00
|
|
|
Thread::Create(kernel, name, entry_point, priority, arg, processor_id, stack_top,
|
2018-10-20 18:34:41 +00:00
|
|
|
*current_process));
|
|
|
|
|
|
|
|
const auto new_guest_handle = current_process->GetHandleTable().Create(thread);
|
2018-10-03 22:47:57 +00:00
|
|
|
if (new_guest_handle.Failed()) {
|
2018-11-26 06:06:13 +00:00
|
|
|
LOG_ERROR(Kernel_SVC, "Failed to create handle with error=0x{:X}",
|
|
|
|
new_guest_handle.Code().raw);
|
2018-10-03 22:47:57 +00:00
|
|
|
return new_guest_handle.Code();
|
|
|
|
}
|
|
|
|
thread->SetGuestHandle(*new_guest_handle);
|
|
|
|
*out_handle = *new_guest_handle;
|
2017-12-31 21:10:01 +00:00
|
|
|
|
2018-10-03 22:47:57 +00:00
|
|
|
Core::System::GetInstance().CpuCore(thread->GetProcessorID()).PrepareReschedule();
|
2017-12-31 21:10:01 +00:00
|
|
|
|
|
|
|
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) {
|
2018-07-02 16:13:26 +00:00
|
|
|
LOG_TRACE(Kernel_SVC, "called thread=0x{:08X}", thread_handle);
|
2017-12-30 18:37:07 +00:00
|
|
|
|
2018-10-20 18:34:41 +00:00
|
|
|
const auto& handle_table = Core::CurrentProcess()->GetHandleTable();
|
|
|
|
const SharedPtr<Thread> thread = handle_table.Get<Thread>(thread_handle);
|
2017-12-30 18:37:07 +00:00
|
|
|
if (!thread) {
|
2018-11-26 08:47:39 +00:00
|
|
|
LOG_ERROR(Kernel_SVC, "Thread handle does not exist, thread_handle=0x{:08X}",
|
|
|
|
thread_handle);
|
2017-12-30 18:37:07 +00:00
|
|
|
return ERR_INVALID_HANDLE;
|
|
|
|
}
|
|
|
|
|
2018-10-03 22:47:57 +00:00
|
|
|
ASSERT(thread->GetStatus() == ThreadStatus::Dormant);
|
2018-05-19 21:57:44 +00:00
|
|
|
|
2017-12-30 18:37:07 +00:00
|
|
|
thread->ResumeFromWait();
|
2018-10-03 22:47:57 +00:00
|
|
|
Core::System::GetInstance().CpuCore(thread->GetProcessorID()).PrepareReschedule();
|
2017-12-30 18:37:07 +00:00
|
|
|
|
|
|
|
return RESULT_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2017-12-31 21:11:27 +00:00
|
|
|
/// Called when a thread exits
|
|
|
|
static void ExitThread() {
|
2018-07-02 16:13:26 +00:00
|
|
|
LOG_TRACE(Kernel_SVC, "called, pc=0x{:08X}", Core::CurrentArmInterface().GetPC());
|
2017-12-31 21:11:27 +00:00
|
|
|
|
2018-01-03 01:40:30 +00:00
|
|
|
ExitCurrentThread();
|
2017-12-31 21:11:27 +00:00
|
|
|
Core::System::GetInstance().PrepareReschedule();
|
|
|
|
}
|
|
|
|
|
2014-06-01 14:37:19 +00:00
|
|
|
/// Sleep the current thread
|
2014-11-17 03:58:39 +00:00
|
|
|
static void SleepThread(s64 nanoseconds) {
|
2018-07-02 16:13:26 +00:00
|
|
|
LOG_TRACE(Kernel_SVC, "called nanoseconds={}", nanoseconds);
|
2014-11-26 05:34:14 +00:00
|
|
|
|
2018-11-22 05:33:53 +00:00
|
|
|
enum class SleepType : s64 {
|
|
|
|
YieldWithoutLoadBalancing = 0,
|
2018-12-02 05:44:40 +00:00
|
|
|
YieldWithLoadBalancing = -1,
|
|
|
|
YieldAndWaitForLoadBalancing = -2,
|
2018-11-22 05:33:53 +00:00
|
|
|
};
|
2017-01-05 19:14:22 +00:00
|
|
|
|
2018-11-19 04:44:19 +00:00
|
|
|
if (nanoseconds <= 0) {
|
2018-11-22 05:33:53 +00:00
|
|
|
auto& scheduler{Core::System::GetInstance().CurrentScheduler()};
|
|
|
|
switch (static_cast<SleepType>(nanoseconds)) {
|
|
|
|
case SleepType::YieldWithoutLoadBalancing:
|
|
|
|
scheduler.YieldWithoutLoadBalancing(GetCurrentThread());
|
2018-11-19 04:44:19 +00:00
|
|
|
break;
|
2018-11-22 05:33:53 +00:00
|
|
|
case SleepType::YieldWithLoadBalancing:
|
|
|
|
scheduler.YieldWithLoadBalancing(GetCurrentThread());
|
2018-11-19 04:44:19 +00:00
|
|
|
break;
|
2018-11-22 05:33:53 +00:00
|
|
|
case SleepType::YieldAndWaitForLoadBalancing:
|
|
|
|
scheduler.YieldAndWaitForLoadBalancing(GetCurrentThread());
|
2018-11-19 04:44:19 +00:00
|
|
|
break;
|
|
|
|
default:
|
2018-12-03 22:29:21 +00:00
|
|
|
UNREACHABLE_MSG("Unimplemented sleep yield type '{:016X}'!", nanoseconds);
|
2018-11-19 04:44:19 +00:00
|
|
|
}
|
2018-12-03 22:29:21 +00:00
|
|
|
} else {
|
|
|
|
// Sleep current thread and check for next thread to schedule
|
|
|
|
WaitCurrentThread_Sleep();
|
2015-01-07 21:40:08 +00:00
|
|
|
|
2018-12-03 22:29:21 +00:00
|
|
|
// Create an event to wake the thread up after the specified nanosecond delay has passed
|
|
|
|
GetCurrentThread()->WakeAfterDelay(nanoseconds);
|
2018-11-19 04:44:19 +00:00
|
|
|
}
|
2017-01-01 16:57:02 +00:00
|
|
|
|
2018-12-04 02:22:09 +00:00
|
|
|
// Reschedule all CPU cores
|
2018-12-05 03:11:32 +00:00
|
|
|
for (std::size_t i = 0; i < Core::NUM_CPU_CORES; ++i)
|
2018-12-04 02:22:09 +00:00
|
|
|
Core::System::GetInstance().CpuCore(i).PrepareReschedule();
|
2014-06-01 14:37:19 +00:00
|
|
|
}
|
|
|
|
|
2018-06-21 06:49:43 +00:00
|
|
|
/// Wait 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-07-02 16:13:26 +00:00
|
|
|
LOG_TRACE(
|
2018-01-09 02:41:37 +00:00
|
|
|
Kernel_SVC,
|
2018-05-02 13:14:28 +00:00
|
|
|
"called mutex_addr={:X}, condition_variable_addr={:X}, thread_handle=0x{:08X}, timeout={}",
|
2018-01-09 02:41:37 +00:00
|
|
|
mutex_addr, condition_variable_addr, thread_handle, nano_seconds);
|
2018-01-06 21:14:12 +00:00
|
|
|
|
2018-10-20 18:34:41 +00:00
|
|
|
const auto& handle_table = Core::CurrentProcess()->GetHandleTable();
|
|
|
|
SharedPtr<Thread> thread = handle_table.Get<Thread>(thread_handle);
|
2018-01-06 21:14:12 +00:00
|
|
|
ASSERT(thread);
|
|
|
|
|
2018-04-20 19:39:28 +00:00
|
|
|
CASCADE_CODE(Mutex::Release(mutex_addr));
|
2018-01-08 16:35:03 +00:00
|
|
|
|
2018-04-20 19:39:28 +00:00
|
|
|
SharedPtr<Thread> current_thread = GetCurrentThread();
|
2018-10-03 22:47:57 +00:00
|
|
|
current_thread->SetCondVarWaitAddress(condition_variable_addr);
|
|
|
|
current_thread->SetMutexWaitAddress(mutex_addr);
|
|
|
|
current_thread->SetWaitHandle(thread_handle);
|
|
|
|
current_thread->SetStatus(ThreadStatus::WaitMutex);
|
|
|
|
current_thread->InvalidateWakeupCallback();
|
2018-01-08 16:35:03 +00:00
|
|
|
|
2018-04-20 19:39:28 +00:00
|
|
|
current_thread->WakeAfterDelay(nano_seconds);
|
2018-01-06 21:14:12 +00:00
|
|
|
|
2018-04-21 01:15:16 +00:00
|
|
|
// Note: Deliberately don't attempt to inherit the lock owner's priority.
|
2018-01-06 21:14:12 +00:00
|
|
|
|
2018-10-03 22:47:57 +00:00
|
|
|
Core::System::GetInstance().CpuCore(current_thread->GetProcessorID()).PrepareReschedule();
|
2018-01-06 21:14:12 +00:00
|
|
|
return RESULT_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2017-10-14 21:30:07 +00:00
|
|
|
/// Signal process wide key
|
2018-01-09 02:41:37 +00:00
|
|
|
static ResultCode SignalProcessWideKey(VAddr condition_variable_addr, s32 target) {
|
2018-07-02 16:13:26 +00:00
|
|
|
LOG_TRACE(Kernel_SVC, "called, condition_variable_addr=0x{:X}, target=0x{:08X}",
|
2018-07-02 16:20:50 +00:00
|
|
|
condition_variable_addr, target);
|
2018-01-07 21:55:17 +00:00
|
|
|
|
2018-10-03 22:47:57 +00:00
|
|
|
const auto RetrieveWaitingThreads = [](std::size_t core_index,
|
|
|
|
std::vector<SharedPtr<Thread>>& waiting_threads,
|
|
|
|
VAddr condvar_addr) {
|
2018-09-15 13:21:06 +00:00
|
|
|
const auto& scheduler = Core::System::GetInstance().Scheduler(core_index);
|
2018-10-15 13:25:11 +00:00
|
|
|
const auto& thread_list = scheduler.GetThreadList();
|
2018-05-19 21:58:30 +00:00
|
|
|
|
2018-10-03 22:47:57 +00:00
|
|
|
for (const auto& thread : thread_list) {
|
|
|
|
if (thread->GetCondVarWaitAddress() == condvar_addr)
|
2018-09-15 13:21:06 +00:00
|
|
|
waiting_threads.push_back(thread);
|
|
|
|
}
|
|
|
|
};
|
2018-05-19 21:58:30 +00:00
|
|
|
|
|
|
|
// Retrieve a list of all threads that are waiting for this condition variable.
|
|
|
|
std::vector<SharedPtr<Thread>> waiting_threads;
|
|
|
|
RetrieveWaitingThreads(0, waiting_threads, condition_variable_addr);
|
|
|
|
RetrieveWaitingThreads(1, waiting_threads, condition_variable_addr);
|
|
|
|
RetrieveWaitingThreads(2, waiting_threads, condition_variable_addr);
|
|
|
|
RetrieveWaitingThreads(3, waiting_threads, condition_variable_addr);
|
|
|
|
// Sort them by priority, such that the highest priority ones come first.
|
|
|
|
std::sort(waiting_threads.begin(), waiting_threads.end(),
|
|
|
|
[](const SharedPtr<Thread>& lhs, const SharedPtr<Thread>& rhs) {
|
2018-10-03 22:47:57 +00:00
|
|
|
return lhs->GetPriority() < rhs->GetPriority();
|
2018-05-19 21:58:30 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
// Only process up to 'target' threads, unless 'target' is -1, in which case process
|
|
|
|
// them all.
|
2018-09-15 13:21:06 +00:00
|
|
|
std::size_t last = waiting_threads.size();
|
2018-05-19 21:58:30 +00:00
|
|
|
if (target != -1)
|
|
|
|
last = target;
|
|
|
|
|
|
|
|
// If there are no threads waiting on this condition variable, just exit
|
|
|
|
if (last > waiting_threads.size())
|
|
|
|
return RESULT_SUCCESS;
|
2018-05-06 02:00:34 +00:00
|
|
|
|
2018-09-15 13:21:06 +00:00
|
|
|
for (std::size_t index = 0; index < last; ++index) {
|
2018-05-19 21:58:30 +00:00
|
|
|
auto& thread = waiting_threads[index];
|
2018-05-06 02:00:34 +00:00
|
|
|
|
2018-10-03 22:47:57 +00:00
|
|
|
ASSERT(thread->GetCondVarWaitAddress() == condition_variable_addr);
|
2018-05-06 02:00:34 +00:00
|
|
|
|
2018-09-15 13:21:06 +00:00
|
|
|
std::size_t current_core = Core::System::GetInstance().CurrentCoreIndex();
|
2018-07-22 17:27:24 +00:00
|
|
|
|
|
|
|
auto& monitor = Core::System::GetInstance().Monitor();
|
|
|
|
|
|
|
|
// Atomically read the value of the mutex.
|
|
|
|
u32 mutex_val = 0;
|
|
|
|
do {
|
2018-10-03 22:47:57 +00:00
|
|
|
monitor.SetExclusive(current_core, thread->GetMutexWaitAddress());
|
2018-07-22 17:27:24 +00:00
|
|
|
|
|
|
|
// If the mutex is not yet acquired, acquire it.
|
2018-10-03 22:47:57 +00:00
|
|
|
mutex_val = Memory::Read32(thread->GetMutexWaitAddress());
|
2018-07-22 17:27:24 +00:00
|
|
|
|
|
|
|
if (mutex_val != 0) {
|
|
|
|
monitor.ClearExclusive();
|
|
|
|
break;
|
|
|
|
}
|
2018-10-03 22:47:57 +00:00
|
|
|
} while (!monitor.ExclusiveWrite32(current_core, thread->GetMutexWaitAddress(),
|
|
|
|
thread->GetWaitHandle()));
|
2018-05-19 21:58:30 +00:00
|
|
|
|
|
|
|
if (mutex_val == 0) {
|
|
|
|
// We were able to acquire the mutex, resume this thread.
|
2018-10-03 22:47:57 +00:00
|
|
|
ASSERT(thread->GetStatus() == ThreadStatus::WaitMutex);
|
2018-05-19 21:58:30 +00:00
|
|
|
thread->ResumeFromWait();
|
2018-01-07 21:55:17 +00:00
|
|
|
|
2018-10-03 22:47:57 +00:00
|
|
|
auto* const lock_owner = thread->GetLockOwner();
|
|
|
|
if (lock_owner != nullptr) {
|
2018-05-19 21:58:30 +00:00
|
|
|
lock_owner->RemoveMutexWaiter(thread);
|
2018-10-03 22:47:57 +00:00
|
|
|
}
|
2018-05-19 21:58:30 +00:00
|
|
|
|
2018-10-03 22:47:57 +00:00
|
|
|
thread->SetLockOwner(nullptr);
|
|
|
|
thread->SetMutexWaitAddress(0);
|
|
|
|
thread->SetCondVarWaitAddress(0);
|
|
|
|
thread->SetWaitHandle(0);
|
2018-05-19 21:58:30 +00:00
|
|
|
} else {
|
2018-07-22 17:27:24 +00:00
|
|
|
// Atomically signal that the mutex now has a waiting thread.
|
|
|
|
do {
|
2018-10-03 22:47:57 +00:00
|
|
|
monitor.SetExclusive(current_core, thread->GetMutexWaitAddress());
|
2018-07-22 17:27:24 +00:00
|
|
|
|
|
|
|
// Ensure that the mutex value is still what we expect.
|
2018-10-03 22:47:57 +00:00
|
|
|
u32 value = Memory::Read32(thread->GetMutexWaitAddress());
|
2018-07-22 17:27:24 +00:00
|
|
|
// TODO(Subv): When this happens, the kernel just clears the exclusive state and
|
|
|
|
// retries the initial read for this thread.
|
|
|
|
ASSERT_MSG(mutex_val == value, "Unhandled synchronization primitive case");
|
2018-10-03 22:47:57 +00:00
|
|
|
} while (!monitor.ExclusiveWrite32(current_core, thread->GetMutexWaitAddress(),
|
2018-07-22 17:27:24 +00:00
|
|
|
mutex_val | Mutex::MutexHasWaitersFlag));
|
|
|
|
|
|
|
|
// The mutex is already owned by some other thread, make this thread wait on it.
|
2018-10-20 18:34:41 +00:00
|
|
|
const Handle owner_handle = static_cast<Handle>(mutex_val & Mutex::MutexOwnerMask);
|
|
|
|
const auto& handle_table = Core::CurrentProcess()->GetHandleTable();
|
|
|
|
auto owner = handle_table.Get<Thread>(owner_handle);
|
2018-05-19 21:58:30 +00:00
|
|
|
ASSERT(owner);
|
2018-10-03 22:47:57 +00:00
|
|
|
ASSERT(thread->GetStatus() == ThreadStatus::WaitMutex);
|
|
|
|
thread->InvalidateWakeupCallback();
|
2018-05-19 21:58:30 +00:00
|
|
|
|
|
|
|
owner->AddMutexWaiter(thread);
|
|
|
|
|
2018-10-03 22:47:57 +00:00
|
|
|
Core::System::GetInstance().CpuCore(thread->GetProcessorID()).PrepareReschedule();
|
2018-05-19 21:58:30 +00:00
|
|
|
}
|
|
|
|
}
|
2018-01-07 21:55:17 +00:00
|
|
|
|
2015-01-23 05:36:58 +00:00
|
|
|
return RESULT_SUCCESS;
|
2014-12-09 04:52:27 +00:00
|
|
|
}
|
|
|
|
|
2018-06-21 06:49:43 +00:00
|
|
|
// Wait for an address (via Address Arbiter)
|
|
|
|
static ResultCode WaitForAddress(VAddr address, u32 type, s32 value, s64 timeout) {
|
2018-07-02 16:13:26 +00:00
|
|
|
LOG_WARNING(Kernel_SVC, "called, address=0x{:X}, type=0x{:X}, value=0x{:X}, timeout={}",
|
2018-07-02 16:20:50 +00:00
|
|
|
address, type, value, timeout);
|
2018-06-21 06:49:43 +00:00
|
|
|
// If the passed address is a kernel virtual address, return invalid memory state.
|
2018-06-22 06:47:59 +00:00
|
|
|
if (Memory::IsKernelVirtualAddress(address)) {
|
2018-11-26 08:47:39 +00:00
|
|
|
LOG_ERROR(Kernel_SVC, "Address is a kernel virtual address, address={:016X}", address);
|
2018-06-21 06:49:43 +00:00
|
|
|
return ERR_INVALID_ADDRESS_STATE;
|
|
|
|
}
|
|
|
|
// If the address is not properly aligned to 4 bytes, return invalid address.
|
2018-11-26 08:47:39 +00:00
|
|
|
if (!Common::IsWordAligned(address)) {
|
|
|
|
LOG_ERROR(Kernel_SVC, "Address is not word aligned, address={:016X}", address);
|
2018-06-21 06:49:43 +00:00
|
|
|
return ERR_INVALID_ADDRESS;
|
|
|
|
}
|
|
|
|
|
2018-06-22 06:47:59 +00:00
|
|
|
switch (static_cast<AddressArbiter::ArbitrationType>(type)) {
|
2018-06-22 03:09:51 +00:00
|
|
|
case AddressArbiter::ArbitrationType::WaitIfLessThan:
|
|
|
|
return AddressArbiter::WaitForAddressIfLessThan(address, value, timeout, false);
|
|
|
|
case AddressArbiter::ArbitrationType::DecrementAndWaitIfLessThan:
|
|
|
|
return AddressArbiter::WaitForAddressIfLessThan(address, value, timeout, true);
|
|
|
|
case AddressArbiter::ArbitrationType::WaitIfEqual:
|
|
|
|
return AddressArbiter::WaitForAddressIfEqual(address, value, timeout);
|
|
|
|
default:
|
2018-11-26 08:47:39 +00:00
|
|
|
LOG_ERROR(Kernel_SVC,
|
|
|
|
"Invalid arbitration type, expected WaitIfLessThan, DecrementAndWaitIfLessThan "
|
|
|
|
"or WaitIfEqual but got {}",
|
|
|
|
type);
|
2018-06-22 03:09:51 +00:00
|
|
|
return ERR_INVALID_ENUM_VALUE;
|
2018-06-21 06:49:43 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Signals to an address (via Address Arbiter)
|
|
|
|
static ResultCode SignalToAddress(VAddr address, u32 type, s32 value, s32 num_to_wake) {
|
2018-07-02 16:20:50 +00:00
|
|
|
LOG_WARNING(Kernel_SVC, "called, address=0x{:X}, type=0x{:X}, value=0x{:X}, num_to_wake=0x{:X}",
|
|
|
|
address, type, value, num_to_wake);
|
2018-06-21 06:49:43 +00:00
|
|
|
// If the passed address is a kernel virtual address, return invalid memory state.
|
2018-06-22 06:47:59 +00:00
|
|
|
if (Memory::IsKernelVirtualAddress(address)) {
|
2018-11-26 08:47:39 +00:00
|
|
|
LOG_ERROR(Kernel_SVC, "Address is a kernel virtual address, address={:016X}", address);
|
2018-06-21 06:49:43 +00:00
|
|
|
return ERR_INVALID_ADDRESS_STATE;
|
|
|
|
}
|
|
|
|
// If the address is not properly aligned to 4 bytes, return invalid address.
|
2018-11-26 08:47:39 +00:00
|
|
|
if (!Common::IsWordAligned(address)) {
|
|
|
|
LOG_ERROR(Kernel_SVC, "Address is not word aligned, address={:016X}", address);
|
2018-06-21 06:49:43 +00:00
|
|
|
return ERR_INVALID_ADDRESS;
|
|
|
|
}
|
|
|
|
|
2018-06-22 06:47:59 +00:00
|
|
|
switch (static_cast<AddressArbiter::SignalType>(type)) {
|
2018-06-22 03:09:51 +00:00
|
|
|
case AddressArbiter::SignalType::Signal:
|
|
|
|
return AddressArbiter::SignalToAddress(address, num_to_wake);
|
|
|
|
case AddressArbiter::SignalType::IncrementAndSignalIfEqual:
|
|
|
|
return AddressArbiter::IncrementAndSignalToAddressIfEqual(address, value, num_to_wake);
|
|
|
|
case AddressArbiter::SignalType::ModifyByWaitingCountAndSignalIfEqual:
|
|
|
|
return AddressArbiter::ModifyByWaitingCountAndSignalToAddressIfEqual(address, value,
|
|
|
|
num_to_wake);
|
|
|
|
default:
|
2018-11-26 08:47:39 +00:00
|
|
|
LOG_ERROR(Kernel_SVC,
|
|
|
|
"Invalid signal type, expected Signal, IncrementAndSignalIfEqual "
|
|
|
|
"or ModifyByWaitingCountAndSignalIfEqual but got {}",
|
|
|
|
type);
|
2018-06-22 03:09:51 +00:00
|
|
|
return ERR_INVALID_ENUM_VALUE;
|
2018-06-21 06:49:43 +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() {
|
2018-11-26 06:06:13 +00:00
|
|
|
LOG_TRACE(Kernel_SVC, "called");
|
|
|
|
|
2018-01-12 02:59:31 +00:00
|
|
|
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 21:30:07 +00:00
|
|
|
/// Close a handle
|
2018-01-03 01:40:30 +00:00
|
|
|
static ResultCode CloseHandle(Handle handle) {
|
2018-07-02 16:13:26 +00:00
|
|
|
LOG_TRACE(Kernel_SVC, "Closing handle 0x{:08X}", handle);
|
2018-08-28 16:30:33 +00:00
|
|
|
|
2018-10-20 18:34:41 +00:00
|
|
|
auto& handle_table = Core::CurrentProcess()->GetHandleTable();
|
|
|
|
return handle_table.Close(handle);
|
2015-08-06 00:39:53 +00:00
|
|
|
}
|
|
|
|
|
2018-12-05 00:59:29 +00:00
|
|
|
/// Clears the signaled state of an event or process.
|
2018-01-08 02:24:19 +00:00
|
|
|
static ResultCode ResetSignal(Handle handle) {
|
2018-11-18 20:49:17 +00:00
|
|
|
LOG_DEBUG(Kernel_SVC, "called handle 0x{:08X}", handle);
|
2018-08-28 16:30:33 +00:00
|
|
|
|
2018-10-20 18:34:41 +00:00
|
|
|
const auto& handle_table = Core::CurrentProcess()->GetHandleTable();
|
2018-12-05 00:59:29 +00:00
|
|
|
|
2018-11-26 23:34:07 +00:00
|
|
|
auto event = handle_table.Get<ReadableEvent>(handle);
|
2018-12-05 00:59:29 +00:00
|
|
|
if (event) {
|
|
|
|
return event->Reset();
|
|
|
|
}
|
2018-08-28 16:30:33 +00:00
|
|
|
|
2018-12-05 00:59:29 +00:00
|
|
|
auto process = handle_table.Get<Process>(handle);
|
|
|
|
if (process) {
|
|
|
|
return process->ClearSignalState();
|
|
|
|
}
|
2018-08-28 16:30:33 +00:00
|
|
|
|
2018-12-05 00:59:29 +00:00
|
|
|
LOG_ERROR(Kernel_SVC, "Invalid handle (0x{:08X})", handle);
|
|
|
|
return ERR_INVALID_HANDLE;
|
2018-01-08 02:24:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Creates a TransferMemory object
|
|
|
|
static ResultCode CreateTransferMemory(Handle* handle, VAddr addr, u64 size, u32 permissions) {
|
2018-11-09 22:02:50 +00:00
|
|
|
LOG_DEBUG(Kernel_SVC, "called addr=0x{:X}, size=0x{:X}, perms=0x{:08X}", addr, size,
|
|
|
|
permissions);
|
|
|
|
|
|
|
|
if (!Common::Is4KBAligned(addr)) {
|
|
|
|
LOG_ERROR(Kernel_SVC, "Address ({:016X}) is not page aligned!", addr);
|
|
|
|
return ERR_INVALID_ADDRESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!Common::Is4KBAligned(size) || size == 0) {
|
|
|
|
LOG_ERROR(Kernel_SVC, "Size ({:016X}) is not page aligned or equal to zero!", size);
|
|
|
|
return ERR_INVALID_ADDRESS;
|
|
|
|
}
|
|
|
|
|
2018-11-11 21:39:25 +00:00
|
|
|
if (!IsValidAddressRange(addr, size)) {
|
2018-11-09 22:02:50 +00:00
|
|
|
LOG_ERROR(Kernel_SVC, "Address and size cause overflow! (address={:016X}, size={:016X})",
|
|
|
|
addr, size);
|
|
|
|
return ERR_INVALID_ADDRESS_STATE;
|
|
|
|
}
|
|
|
|
|
2018-11-11 21:39:25 +00:00
|
|
|
const auto perms = static_cast<MemoryPermission>(permissions);
|
|
|
|
if (perms != MemoryPermission::None && perms != MemoryPermission::Read &&
|
|
|
|
perms != MemoryPermission::ReadWrite) {
|
2018-11-09 22:02:50 +00:00
|
|
|
LOG_ERROR(Kernel_SVC, "Invalid memory permissions for transfer memory! (perms={:08X})",
|
|
|
|
permissions);
|
|
|
|
return ERR_INVALID_MEMORY_PERMISSIONS;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto& kernel = Core::System::GetInstance().Kernel();
|
2018-12-10 18:42:01 +00:00
|
|
|
auto process = kernel.CurrentProcess();
|
|
|
|
auto& handle_table = process->GetHandleTable();
|
|
|
|
const auto shared_mem_handle = SharedMemory::Create(kernel, process, size, perms, perms, addr);
|
2018-11-09 22:02:50 +00:00
|
|
|
|
|
|
|
CASCADE_RESULT(*handle, handle_table.Create(shared_mem_handle));
|
2018-01-08 02:24:19 +00:00
|
|
|
return RESULT_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2018-05-06 03:13:15 +00:00
|
|
|
static ResultCode GetThreadCoreMask(Handle thread_handle, u32* core, u64* mask) {
|
2018-07-02 16:13:26 +00:00
|
|
|
LOG_TRACE(Kernel_SVC, "called, handle=0x{:08X}", thread_handle);
|
2018-05-06 03:13:15 +00:00
|
|
|
|
2018-10-20 18:34:41 +00:00
|
|
|
const auto& handle_table = Core::CurrentProcess()->GetHandleTable();
|
|
|
|
const SharedPtr<Thread> thread = handle_table.Get<Thread>(thread_handle);
|
2018-05-06 03:13:15 +00:00
|
|
|
if (!thread) {
|
2018-11-26 08:47:39 +00:00
|
|
|
LOG_ERROR(Kernel_SVC, "Thread handle does not exist, thread_handle=0x{:08X}",
|
|
|
|
thread_handle);
|
2018-05-06 03:13:15 +00:00
|
|
|
return ERR_INVALID_HANDLE;
|
|
|
|
}
|
|
|
|
|
2018-10-03 22:47:57 +00:00
|
|
|
*core = thread->GetIdealCore();
|
|
|
|
*mask = thread->GetAffinityMask();
|
2018-05-06 03:13:15 +00:00
|
|
|
|
2018-03-30 01:07:49 +00:00
|
|
|
return RESULT_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2018-05-06 03:13:15 +00:00
|
|
|
static ResultCode SetThreadCoreMask(Handle thread_handle, u32 core, u64 mask) {
|
2018-11-26 06:06:13 +00:00
|
|
|
LOG_DEBUG(Kernel_SVC, "called, handle=0x{:08X}, mask=0x{:016X}, core=0x{:X}", thread_handle,
|
2018-07-02 16:20:50 +00:00
|
|
|
mask, core);
|
2018-05-06 03:13:15 +00:00
|
|
|
|
2018-10-20 18:34:41 +00:00
|
|
|
const auto& handle_table = Core::CurrentProcess()->GetHandleTable();
|
|
|
|
const SharedPtr<Thread> thread = handle_table.Get<Thread>(thread_handle);
|
2018-05-06 03:13:15 +00:00
|
|
|
if (!thread) {
|
2018-11-26 08:47:39 +00:00
|
|
|
LOG_ERROR(Kernel_SVC, "Thread handle does not exist, thread_handle=0x{:08X}",
|
|
|
|
thread_handle);
|
2018-05-06 03:13:15 +00:00
|
|
|
return ERR_INVALID_HANDLE;
|
|
|
|
}
|
|
|
|
|
svc: Correct always true assertion case in SetThreadCoreMask
The reason this would never be true is that ideal_processor is a u8 and
THREADPROCESSORID_DEFAULT is an s32. In this case, it boils down to how
arithmetic conversions are performed before performing the comparison.
If an unsigned value has a lesser conversion rank (aka smaller size)
than the signed type being compared, then the unsigned value is promoted
to the signed value (i.e. u8 -> s32 happens before the comparison). No
sign-extension occurs here either.
An alternative phrasing:
Say we have a variable named core and it's given a value of -2.
u8 core = -2;
This becomes 254 due to the lack of sign. During integral promotion to
the signed type, this still remains as 254, and therefore the condition
will always be true, because no matter what value the u8 is given it
will never be -2 in terms of 32 bits.
Now, if one type was a s32 and one was a u32, this would be entirely
different, since they have the same bit width (and the signed type would
be converted to unsigned instead of the other way around) but would
still have its representation preserved in terms of bits, allowing the
comparison to be false in some cases, as opposed to being true all the
time.
---
We also get rid of two signed/unsigned comparison warnings while we're
at it.
2018-07-19 18:36:22 +00:00
|
|
|
if (core == static_cast<u32>(THREADPROCESSORID_DEFAULT)) {
|
2018-10-03 22:47:57 +00:00
|
|
|
const u8 default_processor_id = thread->GetOwnerProcess()->GetDefaultProcessorID();
|
|
|
|
|
|
|
|
ASSERT(default_processor_id != static_cast<u8>(THREADPROCESSORID_DEFAULT));
|
|
|
|
|
2018-05-30 17:03:19 +00:00
|
|
|
// Set the target CPU to the one specified in the process' exheader.
|
2018-10-03 22:47:57 +00:00
|
|
|
core = default_processor_id;
|
|
|
|
mask = 1ULL << core;
|
2018-05-30 17:03:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (mask == 0) {
|
2018-11-26 06:06:13 +00:00
|
|
|
LOG_ERROR(Kernel_SVC, "Mask is 0");
|
2018-11-16 19:24:27 +00:00
|
|
|
return ERR_INVALID_COMBINATION;
|
2018-05-30 17:03:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/// This value is used to only change the affinity mask without changing the current ideal core.
|
|
|
|
static constexpr u32 OnlyChangeMask = static_cast<u32>(-3);
|
|
|
|
|
|
|
|
if (core == OnlyChangeMask) {
|
2018-10-03 22:47:57 +00:00
|
|
|
core = thread->GetIdealCore();
|
svc: Correct always true assertion case in SetThreadCoreMask
The reason this would never be true is that ideal_processor is a u8 and
THREADPROCESSORID_DEFAULT is an s32. In this case, it boils down to how
arithmetic conversions are performed before performing the comparison.
If an unsigned value has a lesser conversion rank (aka smaller size)
than the signed type being compared, then the unsigned value is promoted
to the signed value (i.e. u8 -> s32 happens before the comparison). No
sign-extension occurs here either.
An alternative phrasing:
Say we have a variable named core and it's given a value of -2.
u8 core = -2;
This becomes 254 due to the lack of sign. During integral promotion to
the signed type, this still remains as 254, and therefore the condition
will always be true, because no matter what value the u8 is given it
will never be -2 in terms of 32 bits.
Now, if one type was a s32 and one was a u32, this would be entirely
different, since they have the same bit width (and the signed type would
be converted to unsigned instead of the other way around) but would
still have its representation preserved in terms of bits, allowing the
comparison to be false in some cases, as opposed to being true all the
time.
---
We also get rid of two signed/unsigned comparison warnings while we're
at it.
2018-07-19 18:36:22 +00:00
|
|
|
} else if (core >= Core::NUM_CPU_CORES && core != static_cast<u32>(-1)) {
|
2018-11-26 08:47:39 +00:00
|
|
|
LOG_ERROR(Kernel_SVC, "Invalid core specified, got {}", core);
|
2018-11-16 19:24:27 +00:00
|
|
|
return ERR_INVALID_PROCESSOR_ID;
|
2018-05-30 17:03:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Error out if the input core isn't enabled in the input mask.
|
2018-06-20 16:39:10 +00:00
|
|
|
if (core < Core::NUM_CPU_CORES && (mask & (1ull << core)) == 0) {
|
2018-11-26 08:47:39 +00:00
|
|
|
LOG_ERROR(Kernel_SVC, "Core is not enabled for the current mask, core={}, mask={:016X}",
|
|
|
|
core, mask);
|
2018-11-16 19:24:27 +00:00
|
|
|
return ERR_INVALID_COMBINATION;
|
2018-05-30 17:03:19 +00:00
|
|
|
}
|
|
|
|
|
2018-05-06 03:13:15 +00:00
|
|
|
thread->ChangeCore(core, mask);
|
|
|
|
|
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-07-02 16:13:26 +00:00
|
|
|
LOG_TRACE(Kernel_SVC, "called, size=0x{:X}, localPerms=0x{:08X}, remotePerms=0x{:08X}", size,
|
2018-07-02 16:20:50 +00:00
|
|
|
local_permissions, remote_permissions);
|
2018-11-26 08:47:39 +00:00
|
|
|
if (size == 0) {
|
|
|
|
LOG_ERROR(Kernel_SVC, "Size is 0");
|
2018-11-27 01:29:06 +00:00
|
|
|
return ERR_INVALID_SIZE;
|
2018-11-26 08:47:39 +00:00
|
|
|
}
|
2018-11-27 01:29:06 +00:00
|
|
|
if (!Common::Is4KBAligned(size)) {
|
|
|
|
LOG_ERROR(Kernel_SVC, "Size is not aligned to 4KB, 0x{:016X}", size);
|
|
|
|
return ERR_INVALID_SIZE;
|
|
|
|
}
|
2018-08-28 16:30:33 +00:00
|
|
|
|
2018-11-27 01:53:18 +00:00
|
|
|
if (size >= MAIN_MEMORY_SIZE) {
|
2018-11-27 01:29:06 +00:00
|
|
|
LOG_ERROR(Kernel_SVC, "Size is not less than 8GB, 0x{:016X}", size);
|
2018-09-14 01:04:43 +00:00
|
|
|
return ERR_INVALID_SIZE;
|
|
|
|
}
|
|
|
|
|
|
|
|
const auto local_perms = static_cast<MemoryPermission>(local_permissions);
|
|
|
|
if (local_perms != MemoryPermission::Read && local_perms != MemoryPermission::ReadWrite) {
|
2018-11-26 08:47:39 +00:00
|
|
|
LOG_ERROR(Kernel_SVC,
|
|
|
|
"Invalid local memory permissions, expected Read or ReadWrite but got "
|
|
|
|
"local_permissions={}",
|
|
|
|
static_cast<u32>(local_permissions));
|
2018-09-14 01:04:43 +00:00
|
|
|
return ERR_INVALID_MEMORY_PERMISSIONS;
|
|
|
|
}
|
|
|
|
|
|
|
|
const auto remote_perms = static_cast<MemoryPermission>(remote_permissions);
|
|
|
|
if (remote_perms != MemoryPermission::Read && remote_perms != MemoryPermission::ReadWrite &&
|
|
|
|
remote_perms != MemoryPermission::DontCare) {
|
2018-11-26 08:47:39 +00:00
|
|
|
LOG_ERROR(Kernel_SVC,
|
|
|
|
"Invalid remote memory permissions, expected Read, ReadWrite or DontCare but got "
|
|
|
|
"remote_permissions={}",
|
|
|
|
static_cast<u32>(remote_permissions));
|
2018-09-14 01:04:43 +00:00
|
|
|
return ERR_INVALID_MEMORY_PERMISSIONS;
|
|
|
|
}
|
|
|
|
|
2018-08-28 16:30:33 +00:00
|
|
|
auto& kernel = Core::System::GetInstance().Kernel();
|
2018-12-10 18:42:01 +00:00
|
|
|
auto process = kernel.CurrentProcess();
|
|
|
|
auto& handle_table = process->GetHandleTable();
|
|
|
|
auto shared_mem_handle = SharedMemory::Create(kernel, process, size, local_perms, remote_perms);
|
2018-01-20 00:35:25 +00:00
|
|
|
|
2018-08-28 16:30:33 +00:00
|
|
|
CASCADE_RESULT(*handle, handle_table.Create(shared_mem_handle));
|
2018-01-20 00:35:25 +00:00
|
|
|
return RESULT_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2018-12-04 20:11:18 +00:00
|
|
|
static ResultCode CreateEvent(Handle* write_handle, Handle* read_handle) {
|
|
|
|
LOG_DEBUG(Kernel_SVC, "called");
|
|
|
|
|
|
|
|
auto& kernel = Core::System::GetInstance().Kernel();
|
|
|
|
const auto [readable_event, writable_event] =
|
|
|
|
WritableEvent::CreateEventPair(kernel, ResetType::Sticky, "CreateEvent");
|
|
|
|
|
|
|
|
HandleTable& handle_table = kernel.CurrentProcess()->GetHandleTable();
|
|
|
|
|
|
|
|
const auto write_create_result = handle_table.Create(writable_event);
|
|
|
|
if (write_create_result.Failed()) {
|
|
|
|
return write_create_result.Code();
|
|
|
|
}
|
|
|
|
*write_handle = *write_create_result;
|
|
|
|
|
|
|
|
const auto read_create_result = handle_table.Create(readable_event);
|
|
|
|
if (read_create_result.Failed()) {
|
|
|
|
handle_table.Close(*write_create_result);
|
|
|
|
return read_create_result.Code();
|
|
|
|
}
|
|
|
|
*read_handle = *read_create_result;
|
|
|
|
|
|
|
|
LOG_DEBUG(Kernel_SVC,
|
|
|
|
"successful. Writable event handle=0x{:08X}, Readable event handle=0x{:08X}",
|
|
|
|
*write_create_result, *read_create_result);
|
|
|
|
return RESULT_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2018-02-22 14:28:15 +00:00
|
|
|
static ResultCode ClearEvent(Handle handle) {
|
2018-07-02 16:13:26 +00:00
|
|
|
LOG_TRACE(Kernel_SVC, "called, event=0x{:08X}", handle);
|
2018-02-22 14:28:15 +00:00
|
|
|
|
2018-10-20 18:34:41 +00:00
|
|
|
const auto& handle_table = Core::CurrentProcess()->GetHandleTable();
|
2018-12-04 03:50:16 +00:00
|
|
|
|
|
|
|
auto writable_event = handle_table.Get<WritableEvent>(handle);
|
|
|
|
if (writable_event) {
|
|
|
|
writable_event->Clear();
|
|
|
|
return RESULT_SUCCESS;
|
2018-10-20 18:34:41 +00:00
|
|
|
}
|
|
|
|
|
2018-12-04 03:50:16 +00:00
|
|
|
auto readable_event = handle_table.Get<ReadableEvent>(handle);
|
|
|
|
if (readable_event) {
|
|
|
|
readable_event->Clear();
|
|
|
|
return RESULT_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
LOG_ERROR(Kernel_SVC, "Event handle does not exist, handle=0x{:08X}", handle);
|
|
|
|
return ERR_INVALID_HANDLE;
|
2018-02-22 14:28:15 +00:00
|
|
|
}
|
|
|
|
|
2018-12-04 20:39:49 +00:00
|
|
|
static ResultCode SignalEvent(Handle handle) {
|
|
|
|
LOG_DEBUG(Kernel_SVC, "called. Handle=0x{:08X}", handle);
|
|
|
|
|
|
|
|
HandleTable& handle_table = Core::CurrentProcess()->GetHandleTable();
|
|
|
|
auto writable_event = handle_table.Get<WritableEvent>(handle);
|
|
|
|
|
|
|
|
if (!writable_event) {
|
|
|
|
LOG_ERROR(Kernel_SVC, "Non-existent writable event handle used (0x{:08X})", handle);
|
|
|
|
return ERR_INVALID_HANDLE;
|
|
|
|
}
|
|
|
|
|
|
|
|
writable_event->Signal();
|
|
|
|
return RESULT_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2018-10-13 18:31:46 +00:00
|
|
|
static ResultCode GetProcessInfo(u64* out, Handle process_handle, u32 type) {
|
|
|
|
LOG_DEBUG(Kernel_SVC, "called, handle=0x{:08X}, type=0x{:X}", process_handle, type);
|
|
|
|
|
|
|
|
// This function currently only allows retrieving a process' status.
|
|
|
|
enum class InfoType {
|
|
|
|
Status,
|
|
|
|
};
|
|
|
|
|
2018-10-20 18:34:41 +00:00
|
|
|
const auto& handle_table = Core::CurrentProcess()->GetHandleTable();
|
|
|
|
const auto process = handle_table.Get<Process>(process_handle);
|
2018-10-13 18:31:46 +00:00
|
|
|
if (!process) {
|
2018-11-26 08:47:39 +00:00
|
|
|
LOG_ERROR(Kernel_SVC, "Process handle does not exist, process_handle=0x{:08X}",
|
|
|
|
process_handle);
|
2018-10-13 18:31:46 +00:00
|
|
|
return ERR_INVALID_HANDLE;
|
|
|
|
}
|
|
|
|
|
|
|
|
const auto info_type = static_cast<InfoType>(type);
|
|
|
|
if (info_type != InfoType::Status) {
|
2018-11-26 08:47:39 +00:00
|
|
|
LOG_ERROR(Kernel_SVC, "Expected info_type to be Status but got {} instead", type);
|
2018-10-13 18:31:46 +00:00
|
|
|
return ERR_INVALID_ENUM_VALUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
*out = static_cast<u64>(process->GetStatus());
|
|
|
|
return RESULT_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2018-11-26 23:23:12 +00:00
|
|
|
static ResultCode CreateResourceLimit(Handle* out_handle) {
|
|
|
|
LOG_DEBUG(Kernel_SVC, "called");
|
|
|
|
|
|
|
|
auto& kernel = Core::System::GetInstance().Kernel();
|
|
|
|
auto resource_limit = ResourceLimit::Create(kernel);
|
|
|
|
|
|
|
|
auto* const current_process = kernel.CurrentProcess();
|
|
|
|
ASSERT(current_process != nullptr);
|
|
|
|
|
|
|
|
const auto handle = current_process->GetHandleTable().Create(std::move(resource_limit));
|
|
|
|
if (handle.Failed()) {
|
|
|
|
return handle.Code();
|
|
|
|
}
|
|
|
|
|
|
|
|
*out_handle = *handle;
|
|
|
|
return RESULT_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2018-11-26 23:48:07 +00:00
|
|
|
static ResultCode GetResourceLimitLimitValue(u64* out_value, Handle resource_limit,
|
|
|
|
u32 resource_type) {
|
|
|
|
LOG_DEBUG(Kernel_SVC, "called. Handle={:08X}, Resource type={}", resource_limit, resource_type);
|
|
|
|
|
2018-11-27 00:14:29 +00:00
|
|
|
const auto limit_value = RetrieveResourceLimitValue(resource_limit, resource_type,
|
|
|
|
ResourceLimitValueType::LimitValue);
|
|
|
|
if (limit_value.Failed()) {
|
|
|
|
return limit_value.Code();
|
2018-11-26 23:48:07 +00:00
|
|
|
}
|
|
|
|
|
2018-11-27 00:14:29 +00:00
|
|
|
*out_value = static_cast<u64>(*limit_value);
|
|
|
|
return RESULT_SUCCESS;
|
|
|
|
}
|
2018-11-26 23:48:07 +00:00
|
|
|
|
2018-11-27 00:14:29 +00:00
|
|
|
static ResultCode GetResourceLimitCurrentValue(u64* out_value, Handle resource_limit,
|
|
|
|
u32 resource_type) {
|
|
|
|
LOG_DEBUG(Kernel_SVC, "called. Handle={:08X}, Resource type={}", resource_limit, resource_type);
|
|
|
|
|
|
|
|
const auto current_value = RetrieveResourceLimitValue(resource_limit, resource_type,
|
|
|
|
ResourceLimitValueType::CurrentValue);
|
|
|
|
if (current_value.Failed()) {
|
|
|
|
return current_value.Code();
|
2018-11-26 23:48:07 +00:00
|
|
|
}
|
|
|
|
|
2018-11-27 00:14:29 +00:00
|
|
|
*out_value = static_cast<u64>(*current_value);
|
2018-11-26 23:48:07 +00:00
|
|
|
return RESULT_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2018-11-27 00:51:09 +00:00
|
|
|
static ResultCode SetResourceLimitLimitValue(Handle resource_limit, u32 resource_type, u64 value) {
|
|
|
|
LOG_DEBUG(Kernel_SVC, "called. Handle={:08X}, Resource type={}, Value={}", resource_limit,
|
|
|
|
resource_type, value);
|
|
|
|
|
|
|
|
const auto type = static_cast<ResourceType>(resource_type);
|
|
|
|
if (!IsValidResourceType(type)) {
|
|
|
|
LOG_ERROR(Kernel_SVC, "Invalid resource limit type: '{}'", resource_type);
|
|
|
|
return ERR_INVALID_ENUM_VALUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto& kernel = Core::System::GetInstance().Kernel();
|
|
|
|
auto* const current_process = kernel.CurrentProcess();
|
|
|
|
ASSERT(current_process != nullptr);
|
|
|
|
|
|
|
|
auto resource_limit_object =
|
|
|
|
current_process->GetHandleTable().Get<ResourceLimit>(resource_limit);
|
|
|
|
if (!resource_limit_object) {
|
|
|
|
LOG_ERROR(Kernel_SVC, "Handle to non-existent resource limit instance used. Handle={:08X}",
|
|
|
|
resource_limit);
|
|
|
|
return ERR_INVALID_HANDLE;
|
|
|
|
}
|
|
|
|
|
|
|
|
const auto set_result = resource_limit_object->SetLimitValue(type, static_cast<s64>(value));
|
|
|
|
if (set_result.IsError()) {
|
|
|
|
LOG_ERROR(
|
|
|
|
Kernel_SVC,
|
|
|
|
"Attempted to lower resource limit ({}) for category '{}' below its current value ({})",
|
|
|
|
resource_limit_object->GetMaxResourceValue(type), resource_type,
|
|
|
|
resource_limit_object->GetCurrentResourceValue(type));
|
|
|
|
return set_result;
|
|
|
|
}
|
|
|
|
|
|
|
|
return RESULT_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2015-05-06 03:04:25 +00:00
|
|
|
namespace {
|
2016-09-18 00:38:01 +00:00
|
|
|
struct FunctionDef {
|
|
|
|
using Func = void();
|
2015-05-06 03:04:25 +00:00
|
|
|
|
2016-09-18 00:38:01 +00:00
|
|
|
u32 id;
|
|
|
|
Func* func;
|
|
|
|
const char* name;
|
|
|
|
};
|
2017-10-14 21:30:07 +00:00
|
|
|
} // namespace
|
2015-05-06 03:04:25 +00:00
|
|
|
|
|
|
|
static const FunctionDef SVC_Table[] = {
|
2016-09-18 00:38:01 +00:00
|
|
|
{0x00, nullptr, "Unknown"},
|
2018-01-03 01:47:26 +00:00
|
|
|
{0x01, SvcWrap<SetHeapSize>, "SetHeapSize"},
|
2018-11-03 15:01:34 +00:00
|
|
|
{0x02, SvcWrap<SetMemoryPermission>, "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 01:07:49 +00: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"},
|
2018-12-04 20:39:49 +00:00
|
|
|
{0x11, SvcWrap<SignalEvent>, "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"},
|
2018-09-24 00:03:38 +00:00
|
|
|
{0x2E, nullptr, "GetFutureThreadInfo"},
|
2018-01-03 01:47:26 +00:00
|
|
|
{0x2F, nullptr, "GetLastThreadInfo"},
|
2018-11-26 23:48:07 +00:00
|
|
|
{0x30, SvcWrap<GetResourceLimitLimitValue>, "GetResourceLimitLimitValue"},
|
2018-11-27 00:14:29 +00:00
|
|
|
{0x31, SvcWrap<GetResourceLimitCurrentValue>, "GetResourceLimitCurrentValue"},
|
2018-04-03 03:50:17 +00:00
|
|
|
{0x32, SvcWrap<SetThreadActivity>, "SetThreadActivity"},
|
|
|
|
{0x33, SvcWrap<GetThreadContext>, "GetThreadContext"},
|
2018-06-21 06:49:43 +00:00
|
|
|
{0x34, SvcWrap<WaitForAddress>, "WaitForAddress"},
|
|
|
|
{0x35, SvcWrap<SignalToAddress>, "SignalToAddress"},
|
2017-10-14 21:30:07 +00:00
|
|
|
{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"},
|
2018-04-17 15:37:43 +00:00
|
|
|
{0x3D, nullptr, "DumpInfoNew"},
|
2017-10-14 21:30:07 +00:00
|
|
|
{0x3E, nullptr, "Unknown"},
|
2016-09-18 00:38:01 +00: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"},
|
2018-12-04 20:11:18 +00:00
|
|
|
{0x45, SvcWrap<CreateEvent>, "CreateEvent"},
|
2016-09-18 00:38:01 +00:00
|
|
|
{0x46, nullptr, "Unknown"},
|
2017-10-14 21:30:07 +00:00
|
|
|
{0x47, nullptr, "Unknown"},
|
2018-09-24 00:03:38 +00:00
|
|
|
{0x48, nullptr, "MapPhysicalMemoryUnsafe"},
|
|
|
|
{0x49, nullptr, "UnmapPhysicalMemoryUnsafe"},
|
|
|
|
{0x4A, nullptr, "SetUnsafeLimit"},
|
|
|
|
{0x4B, nullptr, "CreateCodeMemory"},
|
|
|
|
{0x4C, nullptr, "ControlCodeMemory"},
|
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 00:38:01 +00:00
|
|
|
{0x6E, nullptr, "Unknown"},
|
2018-09-24 00:03:38 +00:00
|
|
|
{0x6F, nullptr, "GetSystemInfo"},
|
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"},
|
2018-12-12 18:42:21 +00:00
|
|
|
{0x76, SvcWrap<QueryProcessMemory>, "QueryProcessMemory"},
|
2018-01-03 01:47:26 +00:00
|
|
|
{0x77, nullptr, "MapProcessCodeMemory"},
|
|
|
|
{0x78, nullptr, "UnmapProcessCodeMemory"},
|
|
|
|
{0x79, nullptr, "CreateProcess"},
|
|
|
|
{0x7A, nullptr, "StartProcess"},
|
|
|
|
{0x7B, nullptr, "TerminateProcess"},
|
2018-10-13 18:31:46 +00:00
|
|
|
{0x7C, SvcWrap<GetProcessInfo>, "GetProcessInfo"},
|
2018-11-26 23:23:12 +00:00
|
|
|
{0x7D, SvcWrap<CreateResourceLimit>, "CreateResourceLimit"},
|
2018-11-27 00:51:09 +00:00
|
|
|
{0x7E, SvcWrap<SetResourceLimitLimitValue>, "SetResourceLimitLimitValue"},
|
2018-01-03 01:47:26 +00:00
|
|
|
{0x7F, nullptr, "CallSecureMonitor"},
|
2014-04-10 23:58:28 +00:00
|
|
|
};
|
|
|
|
|
2015-07-21 07:51:36 +00:00
|
|
|
static const FunctionDef* GetSVCInfo(u32 func_num) {
|
2018-04-20 02:36:48 +00:00
|
|
|
if (func_num >= std::size(SVC_Table)) {
|
2018-07-02 16:13:26 +00:00
|
|
|
LOG_ERROR(Kernel_SVC, "Unknown svc=0x{:02X}", func_num);
|
2015-05-06 03:04:25 +00:00
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
return &SVC_Table[func_num];
|
|
|
|
}
|
|
|
|
|
2015-08-17 21:25:21 +00:00
|
|
|
MICROPROFILE_DEFINE(Kernel_SVC, "Kernel", "SVC", MP_RGB(70, 200, 70));
|
|
|
|
|
2015-07-21 07:51:36 +00:00
|
|
|
void CallSVC(u32 immediate) {
|
2015-08-17 21:25:21 +00:00
|
|
|
MICROPROFILE_SCOPE(Kernel_SVC);
|
2015-05-06 03:04:25 +00:00
|
|
|
|
2017-10-14 21:30:07 +00: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 07:51:36 +00:00
|
|
|
const FunctionDef* info = GetSVCInfo(immediate);
|
2015-05-06 03:04:25 +00:00
|
|
|
if (info) {
|
|
|
|
if (info->func) {
|
|
|
|
info->func();
|
|
|
|
} else {
|
2018-07-02 16:13:26 +00:00
|
|
|
LOG_CRITICAL(Kernel_SVC, "Unimplemented SVC function {}(..)", info->name);
|
2015-05-06 03:04:25 +00:00
|
|
|
}
|
2017-10-14 21:30:07 +00:00
|
|
|
} else {
|
2018-07-02 16:13:26 +00:00
|
|
|
LOG_CRITICAL(Kernel_SVC, "Unknown SVC function 0x{:X}", immediate);
|
2015-05-06 03:04:25 +00:00
|
|
|
}
|
2014-04-10 23:58:28 +00:00
|
|
|
}
|
2014-04-11 22:44:21 +00:00
|
|
|
|
2018-01-03 01:40:30 +00:00
|
|
|
} // namespace Kernel
|