2019-06-21 01:23:40 +00:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2018-2019 Atmosphère-NX
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or modify it
|
|
|
|
* under the terms and conditions of the GNU General Public License,
|
|
|
|
* version 2, as published by the Free Software Foundation.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope it will be useful, but WITHOUT
|
|
|
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
|
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
|
|
|
* more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
#include "sm_service_manager.hpp"
|
|
|
|
|
2019-10-24 09:30:10 +00:00
|
|
|
namespace ams::sm::impl {
|
2019-06-21 01:23:40 +00:00
|
|
|
|
|
|
|
/* Anonymous namespace for implementation details. */
|
|
|
|
namespace {
|
|
|
|
/* Constexpr definitions. */
|
2019-07-12 05:23:23 +00:00
|
|
|
static constexpr size_t ProcessCountMax = 0x40;
|
|
|
|
static constexpr size_t ServiceCountMax = 0x100;
|
|
|
|
static constexpr size_t FutureMitmCountMax = 0x20;
|
2019-06-21 01:23:40 +00:00
|
|
|
static constexpr size_t AccessControlSizeMax = 0x200;
|
|
|
|
|
|
|
|
/* Types. */
|
|
|
|
struct ProcessInfo {
|
2019-10-11 09:15:14 +00:00
|
|
|
os::ProcessId pid;
|
2019-07-04 05:57:49 +00:00
|
|
|
ncm::TitleId tid;
|
2019-06-21 01:23:40 +00:00
|
|
|
size_t access_control_size;
|
|
|
|
u8 access_control[AccessControlSizeMax];
|
2019-06-21 06:34:59 +00:00
|
|
|
|
|
|
|
ProcessInfo() {
|
|
|
|
this->Free();
|
|
|
|
}
|
|
|
|
|
|
|
|
void Free() {
|
2019-10-11 09:15:14 +00:00
|
|
|
this->pid = os::InvalidProcessId;
|
2019-07-04 05:57:49 +00:00
|
|
|
this->tid = ncm::TitleId::Invalid;
|
2019-06-21 06:34:59 +00:00
|
|
|
this->access_control_size = 0;
|
|
|
|
std::memset(this->access_control, 0, sizeof(this->access_control));
|
|
|
|
}
|
2019-06-21 01:23:40 +00:00
|
|
|
};
|
|
|
|
|
2019-07-04 05:57:49 +00:00
|
|
|
/* Forward declaration, for use in ServiceInfo. */
|
2019-10-11 09:15:14 +00:00
|
|
|
ncm::TitleId GetTitleIdForMitm(os::ProcessId pid);
|
2019-07-04 05:57:49 +00:00
|
|
|
|
2019-06-21 01:23:40 +00:00
|
|
|
struct ServiceInfo {
|
|
|
|
ServiceName name;
|
2019-10-11 09:15:14 +00:00
|
|
|
os::ProcessId owner_pid;
|
2019-09-28 01:04:58 +00:00
|
|
|
os::ManagedHandle port_h;
|
2019-06-21 01:23:40 +00:00
|
|
|
|
|
|
|
/* Debug. */
|
|
|
|
u64 max_sessions;
|
|
|
|
bool is_light;
|
|
|
|
|
|
|
|
/* Mitm Extension. */
|
2019-10-11 09:15:14 +00:00
|
|
|
os::ProcessId mitm_pid;
|
2019-09-28 01:04:58 +00:00
|
|
|
os::ManagedHandle mitm_port_h;
|
|
|
|
os::ManagedHandle mitm_query_h;
|
2019-06-21 01:23:40 +00:00
|
|
|
|
|
|
|
/* Acknowledgement members. */
|
|
|
|
bool mitm_waiting_ack;
|
2019-10-11 09:15:14 +00:00
|
|
|
os::ProcessId mitm_waiting_ack_pid;
|
2019-09-28 01:04:58 +00:00
|
|
|
os::ManagedHandle mitm_fwd_sess_h;
|
2019-06-21 06:34:59 +00:00
|
|
|
|
|
|
|
ServiceInfo() {
|
|
|
|
this->Free();
|
|
|
|
}
|
|
|
|
|
|
|
|
void Free() {
|
|
|
|
/* Close any open handles. */
|
2019-06-21 06:51:15 +00:00
|
|
|
this->port_h.Clear();
|
|
|
|
this->mitm_port_h.Clear();
|
|
|
|
this->mitm_query_h.Clear();
|
|
|
|
this->mitm_fwd_sess_h.Clear();
|
2019-06-21 06:34:59 +00:00
|
|
|
|
2019-06-21 06:51:15 +00:00
|
|
|
/* Reset all other members. */
|
2019-06-21 06:34:59 +00:00
|
|
|
this->name = InvalidServiceName;
|
2019-10-11 09:15:14 +00:00
|
|
|
this->owner_pid = os::InvalidProcessId;
|
2019-06-21 06:34:59 +00:00
|
|
|
this->max_sessions = 0;
|
|
|
|
this->is_light = false;
|
2019-10-11 09:15:14 +00:00
|
|
|
this->mitm_pid = os::InvalidProcessId;
|
2019-06-21 06:34:59 +00:00
|
|
|
this->mitm_waiting_ack = false;
|
2019-10-11 09:15:14 +00:00
|
|
|
this->mitm_waiting_ack_pid = os::InvalidProcessId;
|
2019-06-21 06:34:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void FreeMitm() {
|
|
|
|
/* Close mitm handles. */
|
2019-06-21 06:51:15 +00:00
|
|
|
this->mitm_port_h.Clear();
|
|
|
|
this->mitm_query_h.Clear();
|
2019-06-21 06:34:59 +00:00
|
|
|
|
|
|
|
/* Reset mitm members. */
|
2019-10-11 09:15:14 +00:00
|
|
|
this->mitm_pid = os::InvalidProcessId;
|
2019-06-21 06:34:59 +00:00
|
|
|
}
|
|
|
|
|
2019-10-11 09:15:14 +00:00
|
|
|
void AcknowledgeMitmSession(os::ProcessId *out_pid, ncm::TitleId *out_tid, Handle *out_hnd) {
|
2019-06-21 06:34:59 +00:00
|
|
|
/* Copy to output. */
|
|
|
|
*out_pid = this->mitm_waiting_ack_pid;
|
2019-07-04 05:57:49 +00:00
|
|
|
*out_tid = GetTitleIdForMitm(this->mitm_waiting_ack_pid);
|
2019-06-21 06:51:15 +00:00
|
|
|
*out_hnd = this->mitm_fwd_sess_h.Move();
|
2019-06-21 06:34:59 +00:00
|
|
|
this->mitm_waiting_ack = false;
|
2019-10-11 09:15:14 +00:00
|
|
|
this->mitm_waiting_ack_pid = os::InvalidProcessId;
|
2019-06-21 06:34:59 +00:00
|
|
|
}
|
2019-06-21 01:23:40 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
class AccessControlEntry {
|
|
|
|
private:
|
|
|
|
const u8 *entry;
|
2019-06-21 06:34:59 +00:00
|
|
|
size_t capacity;
|
2019-06-21 01:23:40 +00:00
|
|
|
public:
|
2019-06-21 06:34:59 +00:00
|
|
|
AccessControlEntry(const void *e, size_t c) : entry(reinterpret_cast<const u8 *>(e)), capacity(c) {
|
2019-06-21 01:23:40 +00:00
|
|
|
/* ... */
|
|
|
|
}
|
|
|
|
|
|
|
|
AccessControlEntry GetNextEntry() const {
|
2019-06-21 06:34:59 +00:00
|
|
|
return AccessControlEntry(this->entry + this->GetSize(), this->capacity - this->GetSize());
|
2019-06-21 01:23:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
size_t GetSize() const {
|
|
|
|
return this->GetServiceNameSize() + 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t GetServiceNameSize() const {
|
|
|
|
return (this->entry[0] & 7) + 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
ServiceName GetServiceName() const {
|
|
|
|
return ServiceName::Encode(reinterpret_cast<const char *>(this->entry + 1), this->GetServiceNameSize());
|
|
|
|
}
|
|
|
|
|
|
|
|
bool IsHost() const {
|
|
|
|
return (this->entry[0] & 0x80) != 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool IsWildcard() const {
|
|
|
|
return this->entry[this->GetServiceNameSize()] == '*';
|
|
|
|
}
|
|
|
|
|
|
|
|
bool IsValid() const {
|
|
|
|
/* Validate that we can access data. */
|
2019-06-21 06:34:59 +00:00
|
|
|
if (this->entry == nullptr || this->capacity == 0) {
|
2019-06-21 01:23:40 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Validate that the size is correct. */
|
2019-06-21 06:34:59 +00:00
|
|
|
return this->GetSize() <= this->capacity;
|
2019-06-21 01:23:40 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
class InitialProcessIdLimits {
|
|
|
|
private:
|
2019-10-11 09:15:14 +00:00
|
|
|
os::ProcessId min;
|
|
|
|
os::ProcessId max;
|
2019-06-21 01:23:40 +00:00
|
|
|
public:
|
|
|
|
InitialProcessIdLimits() {
|
2019-07-04 05:57:49 +00:00
|
|
|
/* Retrieve process limits. */
|
|
|
|
cfg::GetInitialProcessRange(&this->min, &this->max);
|
2019-06-21 01:23:40 +00:00
|
|
|
|
|
|
|
/* Ensure range is sane. */
|
2019-10-24 09:30:10 +00:00
|
|
|
AMS_ASSERT(this->min <= this->max);
|
2019-06-21 01:23:40 +00:00
|
|
|
}
|
|
|
|
|
2019-10-11 09:15:14 +00:00
|
|
|
bool IsInitialProcess(os::ProcessId pid) const {
|
2019-10-24 09:30:10 +00:00
|
|
|
AMS_ASSERT(pid != os::InvalidProcessId);
|
2019-06-21 01:23:40 +00:00
|
|
|
return this->min <= pid && pid <= this->max;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
/* Static members. */
|
|
|
|
ProcessInfo g_process_list[ProcessCountMax];
|
|
|
|
ServiceInfo g_service_list[ServiceCountMax];
|
2019-07-12 05:23:23 +00:00
|
|
|
ServiceName g_future_mitm_list[FutureMitmCountMax];
|
2019-06-21 01:23:40 +00:00
|
|
|
InitialProcessIdLimits g_initial_process_id_limits;
|
|
|
|
bool g_ended_initial_defers;
|
|
|
|
|
|
|
|
/* Helper functions for interacting with processes/services. */
|
2019-10-11 09:15:14 +00:00
|
|
|
ProcessInfo *GetProcessInfo(os::ProcessId pid) {
|
2019-06-21 01:23:40 +00:00
|
|
|
for (size_t i = 0; i < ProcessCountMax; i++) {
|
|
|
|
if (g_process_list[i].pid == pid) {
|
|
|
|
return &g_process_list[i];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
ProcessInfo *GetFreeProcessInfo() {
|
2019-10-11 09:15:14 +00:00
|
|
|
return GetProcessInfo(os::InvalidProcessId);
|
2019-06-21 01:23:40 +00:00
|
|
|
}
|
|
|
|
|
2019-10-11 09:15:14 +00:00
|
|
|
bool HasProcessInfo(os::ProcessId pid) {
|
2019-06-21 01:23:40 +00:00
|
|
|
return GetProcessInfo(pid) != nullptr;
|
|
|
|
}
|
|
|
|
|
2019-10-11 09:15:14 +00:00
|
|
|
bool IsInitialProcess(os::ProcessId pid) {
|
2019-07-12 05:23:23 +00:00
|
|
|
return g_initial_process_id_limits.IsInitialProcess(pid);
|
|
|
|
}
|
|
|
|
|
2019-10-11 09:15:14 +00:00
|
|
|
constexpr inline bool IsValidProcessId(os::ProcessId pid) {
|
|
|
|
return pid != os::InvalidProcessId;
|
2019-07-12 05:23:23 +00:00
|
|
|
}
|
|
|
|
|
2019-06-21 01:23:40 +00:00
|
|
|
ServiceInfo *GetServiceInfo(ServiceName service_name) {
|
|
|
|
for (size_t i = 0; i < ServiceCountMax; i++) {
|
|
|
|
if (g_service_list[i].name == service_name) {
|
|
|
|
return &g_service_list[i];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
ServiceInfo *GetFreeServiceInfo() {
|
|
|
|
return GetServiceInfo(InvalidServiceName);
|
|
|
|
}
|
|
|
|
|
2019-06-21 06:34:59 +00:00
|
|
|
bool HasServiceInfo(ServiceName service) {
|
|
|
|
return GetServiceInfo(service) != nullptr;
|
|
|
|
}
|
|
|
|
|
2019-07-12 05:23:23 +00:00
|
|
|
bool HasMitm(ServiceName service) {
|
|
|
|
const ServiceInfo *service_info = GetServiceInfo(service);
|
|
|
|
return service_info != nullptr && IsValidProcessId(service_info->mitm_pid);
|
|
|
|
}
|
|
|
|
|
2019-10-11 09:15:14 +00:00
|
|
|
ncm::TitleId GetTitleIdForMitm(os::ProcessId pid) {
|
2019-07-12 05:23:23 +00:00
|
|
|
/* Anything that can request a mitm session must have a process info. */
|
|
|
|
const auto process_info = GetProcessInfo(pid);
|
2019-10-24 09:30:10 +00:00
|
|
|
AMS_ASSERT(process_info != nullptr);
|
2019-07-12 05:23:23 +00:00
|
|
|
return process_info->tid;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool IsMitmDisallowed(ncm::TitleId title_id) {
|
|
|
|
/* Mitm used on certain titles can prevent the boot process from completing. */
|
|
|
|
/* TODO: Is there a way to do this that's less hardcoded? Needs design thought. */
|
2019-07-12 12:31:00 +00:00
|
|
|
return title_id == ncm::TitleId::Loader ||
|
|
|
|
title_id == ncm::TitleId::Boot ||
|
|
|
|
title_id == ncm::TitleId::AtmosphereMitm ||
|
|
|
|
title_id == ncm::TitleId::Creport;
|
2019-07-12 05:23:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Result AddFutureMitmDeclaration(ServiceName service) {
|
|
|
|
for (size_t i = 0; i < FutureMitmCountMax; i++) {
|
|
|
|
if (g_future_mitm_list[i] == InvalidServiceName) {
|
|
|
|
g_future_mitm_list[i] = service;
|
2019-10-24 08:40:44 +00:00
|
|
|
return ResultSuccess();
|
2019-07-12 05:23:23 +00:00
|
|
|
}
|
|
|
|
}
|
2019-10-24 08:40:44 +00:00
|
|
|
return sm::ResultOutOfServices();
|
2019-07-12 05:23:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool HasFutureMitmDeclaration(ServiceName service) {
|
|
|
|
for (size_t i = 0; i < FutureMitmCountMax; i++) {
|
|
|
|
if (g_future_mitm_list[i] == service) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ClearFutureMitmDeclaration(ServiceName service) {
|
|
|
|
for (size_t i = 0; i < FutureMitmCountMax; i++) {
|
|
|
|
if (g_future_mitm_list[i] == service) {
|
|
|
|
g_future_mitm_list[i] = InvalidServiceName;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-06-21 01:23:40 +00:00
|
|
|
void GetServiceInfoRecord(ServiceRecord *out_record, const ServiceInfo *service_info) {
|
|
|
|
out_record->service = service_info->name;
|
|
|
|
out_record->owner_pid = service_info->owner_pid;
|
|
|
|
out_record->max_sessions = service_info->max_sessions;
|
|
|
|
out_record->mitm_pid = service_info->mitm_pid;
|
|
|
|
out_record->mitm_waiting_ack_pid = service_info->mitm_waiting_ack_pid;
|
|
|
|
out_record->is_light = service_info->is_light;
|
|
|
|
out_record->mitm_waiting_ack = service_info->mitm_waiting_ack;
|
|
|
|
}
|
|
|
|
|
|
|
|
Result ValidateAccessControl(AccessControlEntry access_control, ServiceName service, bool is_host, bool is_wildcard) {
|
|
|
|
/* Iterate over all entries in the access control, checking to see if we have a match. */
|
|
|
|
while (access_control.IsValid()) {
|
|
|
|
if (access_control.IsHost() == is_host) {
|
2019-10-24 08:40:44 +00:00
|
|
|
bool is_valid = true;
|
|
|
|
|
2019-06-21 01:23:40 +00:00
|
|
|
if (access_control.IsWildcard() == is_wildcard) {
|
|
|
|
/* Check for exact match. */
|
2019-10-24 08:40:44 +00:00
|
|
|
is_valid &= access_control.GetServiceName() == service;
|
2019-06-21 01:23:40 +00:00
|
|
|
} else if (access_control.IsWildcard()) {
|
|
|
|
/* Also allow fuzzy match for wildcard. */
|
|
|
|
ServiceName ac_service = access_control.GetServiceName();
|
2019-10-24 08:40:44 +00:00
|
|
|
is_valid &= std::memcmp(&ac_service, &service, access_control.GetServiceNameSize() - 1) == 0;
|
2019-06-21 01:23:40 +00:00
|
|
|
}
|
2019-10-24 08:40:44 +00:00
|
|
|
|
|
|
|
R_UNLESS(!is_valid, ResultSuccess());
|
2019-06-21 01:23:40 +00:00
|
|
|
}
|
|
|
|
access_control = access_control.GetNextEntry();
|
|
|
|
}
|
|
|
|
|
2019-10-24 08:40:44 +00:00
|
|
|
return sm::ResultNotAllowed();
|
2019-06-21 01:23:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Result ValidateAccessControl(AccessControlEntry restriction, AccessControlEntry access) {
|
|
|
|
/* Ensure that every entry in the access control is allowed by the restriction control. */
|
|
|
|
while (access.IsValid()) {
|
|
|
|
R_TRY(ValidateAccessControl(restriction, access.GetServiceName(), access.IsHost(), access.IsWildcard()));
|
|
|
|
access = access.GetNextEntry();
|
|
|
|
}
|
|
|
|
|
2019-10-24 08:40:44 +00:00
|
|
|
return ResultSuccess();
|
2019-06-21 01:23:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Result ValidateServiceName(ServiceName service) {
|
|
|
|
/* Service names must be non-empty. */
|
2019-10-24 08:40:44 +00:00
|
|
|
R_UNLESS(service.name[0] != 0, sm::ResultInvalidServiceName());
|
2019-06-21 01:23:40 +00:00
|
|
|
|
|
|
|
/* Get name length. */
|
2019-06-21 06:34:59 +00:00
|
|
|
size_t name_len;
|
|
|
|
for (name_len = 1; name_len < sizeof(service); name_len++) {
|
2019-06-21 01:23:40 +00:00
|
|
|
if (service.name[name_len] == 0) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Names must be all-zero after they end. */
|
|
|
|
while (name_len < sizeof(service)) {
|
2019-10-24 08:40:44 +00:00
|
|
|
R_UNLESS(service.name[name_len++] == 0, sm::ResultInvalidServiceName());
|
2019-06-21 01:23:40 +00:00
|
|
|
}
|
|
|
|
|
2019-10-24 08:40:44 +00:00
|
|
|
return ResultSuccess();
|
2019-06-21 01:23:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool ShouldDeferForInit(ServiceName service) {
|
|
|
|
/* Once end has been called, we're done. */
|
|
|
|
if (g_ended_initial_defers) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* This is a mechanism by which certain services will always be deferred until sm:m receives a special command. */
|
|
|
|
/* This can be extended with more services as needed at a later date. */
|
|
|
|
return service == ServiceName::Encode("fsp-srv");
|
|
|
|
}
|
|
|
|
|
2019-10-11 09:15:14 +00:00
|
|
|
Result GetMitmServiceHandleImpl(Handle *out, ServiceInfo *service_info, os::ProcessId process_id, ncm::TitleId title_id) {
|
2019-06-21 01:23:40 +00:00
|
|
|
/* Send command to query if we should mitm. */
|
|
|
|
bool should_mitm;
|
|
|
|
{
|
2019-10-11 09:15:14 +00:00
|
|
|
Service srv { .session = service_info->mitm_query_h.Get() };
|
|
|
|
const struct {
|
|
|
|
os::ProcessId process_id;
|
|
|
|
ncm::TitleId title_id;
|
|
|
|
} in = { process_id, title_id };
|
|
|
|
R_TRY(serviceDispatchInOut(&srv, 65000, in, should_mitm));
|
2019-06-21 01:23:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* If we shouldn't mitm, give normal session. */
|
2019-10-24 08:40:44 +00:00
|
|
|
R_UNLESS(should_mitm, svcConnectToPort(out, service_info->port_h.Get()));
|
2019-06-21 01:23:40 +00:00
|
|
|
|
2019-06-21 06:51:15 +00:00
|
|
|
/* Create both handles. */
|
|
|
|
{
|
2019-09-28 01:04:58 +00:00
|
|
|
os::ManagedHandle fwd_hnd, hnd;
|
2019-06-21 06:51:15 +00:00
|
|
|
R_TRY(svcConnectToPort(fwd_hnd.GetPointer(), service_info->port_h.Get()));
|
|
|
|
R_TRY(svcConnectToPort(hnd.GetPointer(), service_info->mitm_port_h.Get()));
|
|
|
|
service_info->mitm_fwd_sess_h = std::move(fwd_hnd);
|
|
|
|
*out = hnd.Move();
|
|
|
|
}
|
2019-06-21 01:23:40 +00:00
|
|
|
|
2019-10-11 09:15:14 +00:00
|
|
|
service_info->mitm_waiting_ack_pid = process_id;
|
2019-06-21 01:23:40 +00:00
|
|
|
service_info->mitm_waiting_ack = true;
|
|
|
|
|
2019-10-24 08:40:44 +00:00
|
|
|
return ResultSuccess();
|
2019-06-21 01:23:40 +00:00
|
|
|
}
|
|
|
|
|
2019-10-11 09:15:14 +00:00
|
|
|
Result GetServiceHandleImpl(Handle *out, ServiceInfo *service_info, os::ProcessId pid) {
|
2019-06-21 01:23:40 +00:00
|
|
|
/* Clear handle output. */
|
2019-06-21 06:34:59 +00:00
|
|
|
*out = INVALID_HANDLE;
|
2019-06-21 01:23:40 +00:00
|
|
|
|
2019-07-12 05:23:23 +00:00
|
|
|
/* Check if we should return a mitm handle. */
|
|
|
|
if (IsValidProcessId(service_info->mitm_pid) && service_info->mitm_pid != pid) {
|
|
|
|
/* Get title id, ensure that we're allowed to mitm the given title. */
|
|
|
|
const ncm::TitleId title_id = GetTitleIdForMitm(pid);
|
|
|
|
if (!IsMitmDisallowed(title_id)) {
|
|
|
|
/* We're mitm'd. Assert, because mitm service host dead is an error state. */
|
|
|
|
R_ASSERT(GetMitmServiceHandleImpl(out, service_info, pid, title_id));
|
2019-10-24 08:40:44 +00:00
|
|
|
return ResultSuccess();
|
2019-07-12 05:23:23 +00:00
|
|
|
}
|
2019-06-21 01:23:40 +00:00
|
|
|
}
|
|
|
|
|
2019-07-12 05:23:23 +00:00
|
|
|
/* We're not returning a mitm handle, so just return a normal port handle. */
|
|
|
|
return svcConnectToPort(out, service_info->port_h.Get());
|
2019-06-21 01:23:40 +00:00
|
|
|
}
|
|
|
|
|
2019-10-11 09:15:14 +00:00
|
|
|
Result RegisterServiceImpl(Handle *out, os::ProcessId pid, ServiceName service, size_t max_sessions, bool is_light) {
|
2019-06-21 01:23:40 +00:00
|
|
|
/* Validate service name. */
|
|
|
|
R_TRY(ValidateServiceName(service));
|
|
|
|
|
|
|
|
/* Don't try to register something already registered. */
|
2019-10-24 08:40:44 +00:00
|
|
|
R_UNLESS(!HasServiceInfo(service), sm::ResultAlreadyRegistered());
|
2019-06-21 01:23:40 +00:00
|
|
|
|
|
|
|
/* Adjust session limit, if compile flags tell us to. */
|
|
|
|
#ifdef SM_MINIMUM_SESSION_LIMIT
|
|
|
|
if (max_sessions < SM_MINIMUM_SESSION_LIMIT) {
|
|
|
|
max_sessions = SM_MINIMUM_SESSION_LIMIT;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* Get free service. */
|
|
|
|
ServiceInfo *free_service = GetFreeServiceInfo();
|
2019-10-24 08:40:44 +00:00
|
|
|
R_UNLESS(free_service != nullptr, sm::ResultOutOfServices());
|
2019-06-21 01:23:40 +00:00
|
|
|
|
|
|
|
/* Create the new service. */
|
2019-06-21 06:51:15 +00:00
|
|
|
*out = INVALID_HANDLE;
|
|
|
|
R_TRY(svcCreatePort(out, free_service->port_h.GetPointerAndClear(), max_sessions, is_light, free_service->name.name));
|
2019-06-21 01:23:40 +00:00
|
|
|
|
|
|
|
/* Save info. */
|
|
|
|
free_service->name = service;
|
|
|
|
free_service->owner_pid = pid;
|
|
|
|
free_service->max_sessions = max_sessions;
|
|
|
|
free_service->is_light = is_light;
|
|
|
|
|
2019-10-24 08:40:44 +00:00
|
|
|
return ResultSuccess();
|
2019-06-21 01:23:40 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Process management. */
|
2019-10-11 09:15:14 +00:00
|
|
|
Result RegisterProcess(os::ProcessId pid, ncm::TitleId tid, const void *acid_sac, size_t acid_sac_size, const void *aci_sac, size_t aci_sac_size) {
|
2019-06-21 01:23:40 +00:00
|
|
|
/* Check that access control will fit in the ServiceInfo. */
|
2019-10-24 08:40:44 +00:00
|
|
|
R_UNLESS(aci_sac_size <= AccessControlSizeMax, sm::ResultTooLargeAccessControl());
|
2019-06-21 01:23:40 +00:00
|
|
|
|
|
|
|
/* Get free process. */
|
|
|
|
ProcessInfo *proc = GetFreeProcessInfo();
|
2019-10-24 08:40:44 +00:00
|
|
|
R_UNLESS(proc != nullptr, sm::ResultOutOfProcesses());
|
2019-06-21 01:23:40 +00:00
|
|
|
|
|
|
|
/* Validate restrictions. */
|
2019-10-24 08:40:44 +00:00
|
|
|
R_UNLESS(aci_sac_size != 0, sm::ResultNotAllowed());
|
2019-07-04 05:57:49 +00:00
|
|
|
R_TRY(ValidateAccessControl(AccessControlEntry(acid_sac, acid_sac_size), AccessControlEntry(aci_sac, aci_sac_size)));
|
2019-06-21 01:23:40 +00:00
|
|
|
|
|
|
|
/* Save info. */
|
|
|
|
proc->pid = pid;
|
2019-07-04 05:57:49 +00:00
|
|
|
proc->tid = tid;
|
|
|
|
proc->access_control_size = aci_sac_size;
|
|
|
|
std::memcpy(proc->access_control, aci_sac, proc->access_control_size);
|
2019-10-24 08:40:44 +00:00
|
|
|
return ResultSuccess();
|
2019-06-21 01:23:40 +00:00
|
|
|
}
|
|
|
|
|
2019-10-11 09:15:14 +00:00
|
|
|
Result UnregisterProcess(os::ProcessId pid) {
|
2019-06-21 01:23:40 +00:00
|
|
|
/* Find the process. */
|
|
|
|
ProcessInfo *proc = GetProcessInfo(pid);
|
2019-10-24 08:40:44 +00:00
|
|
|
R_UNLESS(proc != nullptr, sm::ResultInvalidClient());
|
2019-06-21 01:23:40 +00:00
|
|
|
|
2019-06-21 06:34:59 +00:00
|
|
|
proc->Free();
|
2019-10-24 08:40:44 +00:00
|
|
|
return ResultSuccess();
|
2019-06-21 01:23:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Service management. */
|
|
|
|
Result HasService(bool *out, ServiceName service) {
|
2019-06-25 00:57:49 +00:00
|
|
|
/* Validate service name. */
|
|
|
|
R_TRY(ValidateServiceName(service));
|
|
|
|
|
2019-06-21 01:23:40 +00:00
|
|
|
*out = HasServiceInfo(service);
|
2019-10-24 08:40:44 +00:00
|
|
|
return ResultSuccess();
|
2019-06-21 01:23:40 +00:00
|
|
|
}
|
|
|
|
|
2019-07-03 05:21:47 +00:00
|
|
|
Result WaitService(ServiceName service) {
|
|
|
|
bool has_service = false;
|
|
|
|
R_TRY(impl::HasService(&has_service, service));
|
|
|
|
|
|
|
|
/* Wait until we have the service. */
|
2019-10-24 08:40:44 +00:00
|
|
|
R_UNLESS(has_service, sf::ResultRequestDeferredByUser());
|
|
|
|
return ResultSuccess();
|
2019-07-03 05:21:47 +00:00
|
|
|
}
|
|
|
|
|
2019-10-11 09:15:14 +00:00
|
|
|
Result GetServiceHandle(Handle *out, os::ProcessId pid, ServiceName service) {
|
2019-06-21 01:23:40 +00:00
|
|
|
/* Validate service name. */
|
|
|
|
R_TRY(ValidateServiceName(service));
|
|
|
|
|
|
|
|
/* In 8.0.0, Nintendo removed the service apm:p -- however, all homebrew attempts to get */
|
|
|
|
/* a handle to this when calling appletInitialize(). Because hbl has access to all services, */
|
|
|
|
/* This would return true, and homebrew would *wait forever* trying to get a handle to a service */
|
|
|
|
/* that will never register. Thus, in the interest of not breaking every single piece of homebrew */
|
|
|
|
/* we will provide a little first class help. */
|
2019-10-24 08:40:44 +00:00
|
|
|
constexpr ServiceName ApmP = ServiceName::Encode("apm:p");
|
|
|
|
R_UNLESS((hos::GetVersion() < hos::Version_800) || (service != ApmP), sm::ResultNotAllowed());
|
2019-06-21 01:23:40 +00:00
|
|
|
|
2019-06-21 06:51:15 +00:00
|
|
|
/* Check that the process is registered and allowed to get the service. */
|
2019-06-21 01:23:40 +00:00
|
|
|
if (!IsInitialProcess(pid)) {
|
|
|
|
ProcessInfo *proc = GetProcessInfo(pid);
|
2019-10-24 08:40:44 +00:00
|
|
|
R_UNLESS(proc != nullptr, sm::ResultInvalidClient());
|
2019-06-21 01:23:40 +00:00
|
|
|
R_TRY(ValidateAccessControl(AccessControlEntry(proc->access_control, proc->access_control_size), service, false, false));
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Get service info. Check to see if we need to defer this until later. */
|
|
|
|
ServiceInfo *service_info = GetServiceInfo(service);
|
2019-10-24 08:40:44 +00:00
|
|
|
R_UNLESS(service_info != nullptr, sf::ResultRequestDeferredByUser());
|
|
|
|
R_UNLESS(!ShouldDeferForInit(service), sf::ResultRequestDeferredByUser());
|
|
|
|
R_UNLESS(!HasFutureMitmDeclaration(service), sf::ResultRequestDeferredByUser());
|
|
|
|
R_UNLESS(!service_info->mitm_waiting_ack, sf::ResultRequestDeferredByUser());
|
2019-06-21 01:23:40 +00:00
|
|
|
|
|
|
|
/* Get a handle from the service info. */
|
|
|
|
R_TRY_CATCH(GetServiceHandleImpl(out, service_info, pid)) {
|
2019-10-24 08:40:44 +00:00
|
|
|
R_CONVERT(svc::ResultOutOfSessions, sm::ResultOutOfSessions())
|
2019-06-21 01:23:40 +00:00
|
|
|
} R_END_TRY_CATCH;
|
|
|
|
|
2019-10-24 08:40:44 +00:00
|
|
|
return ResultSuccess();
|
2019-06-21 01:23:40 +00:00
|
|
|
}
|
|
|
|
|
2019-10-11 09:15:14 +00:00
|
|
|
Result RegisterService(Handle *out, os::ProcessId pid, ServiceName service, size_t max_sessions, bool is_light) {
|
2019-06-21 01:23:40 +00:00
|
|
|
/* Validate service name. */
|
|
|
|
R_TRY(ValidateServiceName(service));
|
|
|
|
|
|
|
|
/* Check that the process is registered and allowed to register the service. */
|
|
|
|
if (!IsInitialProcess(pid)) {
|
|
|
|
ProcessInfo *proc = GetProcessInfo(pid);
|
2019-10-24 08:40:44 +00:00
|
|
|
R_UNLESS(proc != nullptr, sm::ResultInvalidClient());
|
2019-06-21 01:23:40 +00:00
|
|
|
|
|
|
|
R_TRY(ValidateAccessControl(AccessControlEntry(proc->access_control, proc->access_control_size), service, true, false));
|
|
|
|
}
|
|
|
|
|
2019-10-24 08:40:44 +00:00
|
|
|
R_UNLESS(!HasServiceInfo(service), sm::ResultAlreadyRegistered());
|
2019-06-21 01:23:40 +00:00
|
|
|
return RegisterServiceImpl(out, pid, service, max_sessions, is_light);
|
|
|
|
}
|
|
|
|
|
|
|
|
Result RegisterServiceForSelf(Handle *out, ServiceName service, size_t max_sessions) {
|
2019-10-11 09:15:14 +00:00
|
|
|
return RegisterServiceImpl(out, os::GetCurrentProcessId(), service, max_sessions, false);
|
2019-06-21 01:23:40 +00:00
|
|
|
}
|
|
|
|
|
2019-10-11 09:15:14 +00:00
|
|
|
Result UnregisterService(os::ProcessId pid, ServiceName service) {
|
2019-06-21 01:23:40 +00:00
|
|
|
/* Validate service name. */
|
|
|
|
R_TRY(ValidateServiceName(service));
|
|
|
|
|
|
|
|
/* Check that the process is registered. */
|
|
|
|
if (!IsInitialProcess(pid)) {
|
2019-10-24 08:40:44 +00:00
|
|
|
R_UNLESS(HasProcessInfo(pid), sm::ResultInvalidClient());
|
2019-06-21 01:23:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Ensure that the service is actually registered. */
|
|
|
|
ServiceInfo *service_info = GetServiceInfo(service);
|
2019-10-24 08:40:44 +00:00
|
|
|
R_UNLESS(service_info != nullptr, sm::ResultNotRegistered());
|
2019-06-21 01:23:40 +00:00
|
|
|
|
|
|
|
/* Check if we have permission to do this. */
|
2019-10-24 08:40:44 +00:00
|
|
|
R_UNLESS(service_info->owner_pid == pid, sm::ResultNotAllowed());
|
2019-06-21 01:23:40 +00:00
|
|
|
|
|
|
|
/* Unregister the service. */
|
2019-06-21 06:34:59 +00:00
|
|
|
service_info->Free();
|
2019-10-24 08:40:44 +00:00
|
|
|
return ResultSuccess();
|
2019-06-21 01:23:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Mitm extensions. */
|
|
|
|
Result HasMitm(bool *out, ServiceName service) {
|
|
|
|
/* Validate service name. */
|
|
|
|
R_TRY(ValidateServiceName(service));
|
|
|
|
|
|
|
|
const ServiceInfo *service_info = GetServiceInfo(service);
|
|
|
|
*out = service_info != nullptr && IsValidProcessId(service_info->mitm_pid);
|
2019-10-24 08:40:44 +00:00
|
|
|
return ResultSuccess();
|
2019-06-21 01:23:40 +00:00
|
|
|
}
|
|
|
|
|
2019-07-03 05:21:47 +00:00
|
|
|
Result WaitMitm(ServiceName service) {
|
|
|
|
bool has_mitm = false;
|
|
|
|
R_TRY(impl::HasMitm(&has_mitm, service));
|
|
|
|
|
|
|
|
/* Wait until we have the mitm. */
|
2019-10-24 08:40:44 +00:00
|
|
|
R_UNLESS(has_mitm, sf::ResultRequestDeferredByUser());
|
|
|
|
return ResultSuccess();
|
2019-07-03 05:21:47 +00:00
|
|
|
}
|
|
|
|
|
2019-10-11 09:15:14 +00:00
|
|
|
Result InstallMitm(Handle *out, Handle *out_query, os::ProcessId pid, ServiceName service) {
|
2019-06-21 01:23:40 +00:00
|
|
|
/* Validate service name. */
|
|
|
|
R_TRY(ValidateServiceName(service));
|
|
|
|
|
|
|
|
/* Check that the process is registered and allowed to register the service. */
|
|
|
|
if (!IsInitialProcess(pid)) {
|
|
|
|
ProcessInfo *proc = GetProcessInfo(pid);
|
2019-10-24 08:40:44 +00:00
|
|
|
R_UNLESS(proc != nullptr, sm::ResultInvalidClient());
|
2019-06-21 01:23:40 +00:00
|
|
|
R_TRY(ValidateAccessControl(AccessControlEntry(proc->access_control, proc->access_control_size), service, true, false));
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Validate that the service exists. */
|
|
|
|
ServiceInfo *service_info = GetServiceInfo(service);
|
2019-10-24 08:40:44 +00:00
|
|
|
|
|
|
|
/* If it doesn't exist, defer until it does. */
|
|
|
|
R_UNLESS(service_info != nullptr, sf::ResultRequestDeferredByUser());
|
2019-06-21 01:23:40 +00:00
|
|
|
|
|
|
|
/* Validate that the service isn't already being mitm'd. */
|
2019-10-24 08:40:44 +00:00
|
|
|
R_UNLESS(!IsValidProcessId(service_info->mitm_pid), sm::ResultAlreadyRegistered());
|
2019-06-21 01:23:40 +00:00
|
|
|
|
|
|
|
/* Always clear output. */
|
2019-06-21 06:34:59 +00:00
|
|
|
*out = INVALID_HANDLE;
|
|
|
|
*out_query = INVALID_HANDLE;
|
2019-06-21 01:23:40 +00:00
|
|
|
|
|
|
|
/* Create mitm handles. */
|
2019-06-21 06:51:15 +00:00
|
|
|
{
|
2019-09-28 01:04:58 +00:00
|
|
|
os::ManagedHandle hnd, port_hnd, qry_hnd, mitm_qry_hnd;
|
2019-06-21 06:51:15 +00:00
|
|
|
u64 x = 0;
|
|
|
|
R_TRY(svcCreatePort(hnd.GetPointer(), port_hnd.GetPointer(), service_info->max_sessions, service_info->is_light, reinterpret_cast<char *>(&x)));
|
|
|
|
R_TRY(svcCreateSession(qry_hnd.GetPointer(), mitm_qry_hnd.GetPointer(), 0, 0));
|
|
|
|
|
|
|
|
/* Copy to output. */
|
|
|
|
service_info->mitm_pid = pid;
|
|
|
|
service_info->mitm_port_h = std::move(port_hnd);
|
|
|
|
service_info->mitm_query_h = std::move(mitm_qry_hnd);
|
|
|
|
*out = hnd.Move();
|
|
|
|
*out_query = qry_hnd.Move();
|
|
|
|
}
|
2019-06-21 01:23:40 +00:00
|
|
|
|
2019-07-12 05:23:23 +00:00
|
|
|
/* Clear the future declaration, if one exists. */
|
|
|
|
ClearFutureMitmDeclaration(service);
|
|
|
|
|
2019-10-24 08:40:44 +00:00
|
|
|
return ResultSuccess();
|
2019-06-21 01:23:40 +00:00
|
|
|
}
|
|
|
|
|
2019-10-11 09:15:14 +00:00
|
|
|
Result UninstallMitm(os::ProcessId pid, ServiceName service) {
|
2019-06-21 01:23:40 +00:00
|
|
|
/* Validate service name. */
|
|
|
|
R_TRY(ValidateServiceName(service));
|
|
|
|
|
|
|
|
/* Check that the process is registered. */
|
|
|
|
if (!IsInitialProcess(pid)) {
|
|
|
|
ProcessInfo *proc = GetProcessInfo(pid);
|
2019-10-24 08:40:44 +00:00
|
|
|
R_UNLESS(proc != nullptr, sm::ResultInvalidClient());
|
2019-06-21 01:23:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Validate that the service exists. */
|
|
|
|
ServiceInfo *service_info = GetServiceInfo(service);
|
2019-10-24 08:40:44 +00:00
|
|
|
R_UNLESS(service_info != nullptr, sm::ResultNotRegistered());
|
2019-06-21 01:23:40 +00:00
|
|
|
|
|
|
|
/* Validate that the client pid is the mitm process. */
|
2019-10-24 08:40:44 +00:00
|
|
|
R_UNLESS(service_info->mitm_pid == pid, sm::ResultNotAllowed());
|
2019-06-21 01:23:40 +00:00
|
|
|
|
|
|
|
/* Free Mitm session info. */
|
2019-06-21 06:34:59 +00:00
|
|
|
service_info->FreeMitm();
|
2019-10-24 08:40:44 +00:00
|
|
|
return ResultSuccess();
|
2019-06-21 01:23:40 +00:00
|
|
|
}
|
|
|
|
|
2019-10-11 09:15:14 +00:00
|
|
|
Result DeclareFutureMitm(os::ProcessId pid, ServiceName service) {
|
2019-07-12 05:23:23 +00:00
|
|
|
/* Validate service name. */
|
|
|
|
R_TRY(ValidateServiceName(service));
|
|
|
|
|
|
|
|
/* Check that the process is registered and allowed to register the service. */
|
|
|
|
if (!IsInitialProcess(pid)) {
|
|
|
|
ProcessInfo *proc = GetProcessInfo(pid);
|
2019-10-24 08:40:44 +00:00
|
|
|
R_UNLESS(proc != nullptr, sm::ResultInvalidClient());
|
2019-07-12 05:23:23 +00:00
|
|
|
R_TRY(ValidateAccessControl(AccessControlEntry(proc->access_control, proc->access_control_size), service, true, false));
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Check that mitm hasn't already been registered or declared. */
|
2019-10-24 08:40:44 +00:00
|
|
|
R_UNLESS(!HasMitm(service), sm::ResultAlreadyRegistered());
|
|
|
|
R_UNLESS(!HasFutureMitmDeclaration(service), sm::ResultAlreadyRegistered());
|
2019-07-12 05:23:23 +00:00
|
|
|
|
|
|
|
/* Try to forward declare it. */
|
|
|
|
R_TRY(AddFutureMitmDeclaration(service));
|
2019-10-24 08:40:44 +00:00
|
|
|
return ResultSuccess();
|
2019-07-12 05:23:23 +00:00
|
|
|
}
|
|
|
|
|
2019-10-11 09:15:14 +00:00
|
|
|
Result AcknowledgeMitmSession(os::ProcessId *out_pid, ncm::TitleId *out_tid, Handle *out_hnd, os::ProcessId pid, ServiceName service) {
|
2019-06-21 01:23:40 +00:00
|
|
|
/* Validate service name. */
|
|
|
|
R_TRY(ValidateServiceName(service));
|
|
|
|
|
|
|
|
/* Check that the process is registered. */
|
|
|
|
if (!IsInitialProcess(pid)) {
|
|
|
|
ProcessInfo *proc = GetProcessInfo(pid);
|
2019-10-24 08:40:44 +00:00
|
|
|
R_UNLESS(proc != nullptr, sm::ResultInvalidClient());
|
2019-06-21 01:23:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Validate that the service exists. */
|
|
|
|
ServiceInfo *service_info = GetServiceInfo(service);
|
2019-10-24 08:40:44 +00:00
|
|
|
R_UNLESS(service_info != nullptr, sm::ResultNotRegistered());
|
2019-06-21 01:23:40 +00:00
|
|
|
|
|
|
|
/* Validate that the client pid is the mitm process, and that an acknowledgement is waiting. */
|
2019-10-24 08:40:44 +00:00
|
|
|
R_UNLESS(service_info->mitm_pid == pid, sm::ResultNotAllowed());
|
|
|
|
R_UNLESS(service_info->mitm_waiting_ack, sm::ResultNotAllowed());
|
2019-06-21 01:23:40 +00:00
|
|
|
|
2019-06-21 06:34:59 +00:00
|
|
|
/* Acknowledge. */
|
2019-07-04 05:57:49 +00:00
|
|
|
service_info->AcknowledgeMitmSession(out_pid, out_tid, out_hnd);
|
2019-10-24 08:40:44 +00:00
|
|
|
return ResultSuccess();
|
2019-06-21 01:23:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Dmnt record extensions. */
|
|
|
|
Result GetServiceRecord(ServiceRecord *out, ServiceName service) {
|
|
|
|
/* Validate service name. */
|
|
|
|
R_TRY(ValidateServiceName(service));
|
|
|
|
|
|
|
|
/* Validate that the service exists. */
|
|
|
|
const ServiceInfo *service_info = GetServiceInfo(service);
|
2019-10-24 08:40:44 +00:00
|
|
|
R_UNLESS(service_info != nullptr, sm::ResultNotRegistered());
|
2019-06-21 01:23:40 +00:00
|
|
|
|
|
|
|
GetServiceInfoRecord(out, service_info);
|
2019-10-24 08:40:44 +00:00
|
|
|
return ResultSuccess();
|
2019-06-21 01:23:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Result ListServiceRecords(ServiceRecord *out, u64 *out_count, u64 offset, u64 max_count) {
|
|
|
|
u64 count = 0;
|
|
|
|
|
|
|
|
for (size_t i = 0; i < ServiceCountMax && count < max_count; i++) {
|
|
|
|
const ServiceInfo *service_info = &g_service_list[i];
|
|
|
|
if (service_info->name != InvalidServiceName) {
|
|
|
|
if (offset == 0) {
|
|
|
|
GetServiceInfoRecord(out++, service_info);
|
|
|
|
count++;
|
|
|
|
} else {
|
|
|
|
offset--;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
*out_count = 0;
|
2019-10-24 08:40:44 +00:00
|
|
|
return ResultSuccess();
|
2019-06-21 01:23:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Deferral extension (works around FS bug). */
|
|
|
|
Result EndInitialDefers() {
|
|
|
|
g_ended_initial_defers = true;
|
2019-10-24 08:40:44 +00:00
|
|
|
return ResultSuccess();
|
2019-06-21 01:23:40 +00:00
|
|
|
}
|
|
|
|
|
2019-06-21 01:32:00 +00:00
|
|
|
}
|