mirror of
https://github.com/Atmosphere-NX/Atmosphere
synced 2025-01-25 00:14:27 +00:00
96f95b9f95
* result: try out some experimental shenanigans * result: sketch out some more shenanigans * result: see what it looks like to convert kernel to use result conds instead of guards * make rest of kernel use experimental new macro-ing
150 lines
6.6 KiB
C++
150 lines
6.6 KiB
C++
/*
|
|
* Copyright (c) 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 <mesosphere.hpp>
|
|
|
|
namespace ams::kern::svc {
|
|
|
|
/* ============================= Common ============================= */
|
|
|
|
namespace {
|
|
|
|
constexpr bool IsValidLimitableResource(ams::svc::LimitableResource which) {
|
|
return which < ams::svc::LimitableResource_Count;
|
|
}
|
|
|
|
Result GetResourceLimitLimitValue(int64_t *out_limit_value, ams::svc::Handle resource_limit_handle, ams::svc::LimitableResource which) {
|
|
/* Validate the resource. */
|
|
R_UNLESS(IsValidLimitableResource(which), svc::ResultInvalidEnumValue());
|
|
|
|
/* Get the resource limit. */
|
|
KScopedAutoObject resource_limit = GetCurrentProcess().GetHandleTable().GetObject<KResourceLimit>(resource_limit_handle);
|
|
R_UNLESS(resource_limit.IsNotNull(), svc::ResultInvalidHandle());
|
|
|
|
/* Get the limit value. */
|
|
*out_limit_value = resource_limit->GetLimitValue(which);
|
|
|
|
R_SUCCEED();
|
|
}
|
|
|
|
Result GetResourceLimitCurrentValue(int64_t *out_current_value, ams::svc::Handle resource_limit_handle, ams::svc::LimitableResource which) {
|
|
/* Validate the resource. */
|
|
R_UNLESS(IsValidLimitableResource(which), svc::ResultInvalidEnumValue());
|
|
|
|
/* Get the resource limit. */
|
|
KScopedAutoObject resource_limit = GetCurrentProcess().GetHandleTable().GetObject<KResourceLimit>(resource_limit_handle);
|
|
R_UNLESS(resource_limit.IsNotNull(), svc::ResultInvalidHandle());
|
|
|
|
/* Get the current value. */
|
|
*out_current_value = resource_limit->GetCurrentValue(which);
|
|
|
|
R_SUCCEED();
|
|
}
|
|
|
|
Result GetResourceLimitPeakValue(int64_t *out_peak_value, ams::svc::Handle resource_limit_handle, ams::svc::LimitableResource which) {
|
|
/* Validate the resource. */
|
|
R_UNLESS(IsValidLimitableResource(which), svc::ResultInvalidEnumValue());
|
|
|
|
/* Get the resource limit. */
|
|
KScopedAutoObject resource_limit = GetCurrentProcess().GetHandleTable().GetObject<KResourceLimit>(resource_limit_handle);
|
|
R_UNLESS(resource_limit.IsNotNull(), svc::ResultInvalidHandle());
|
|
|
|
/* Get the peak value. */
|
|
*out_peak_value = resource_limit->GetPeakValue(which);
|
|
|
|
R_SUCCEED();
|
|
}
|
|
|
|
Result CreateResourceLimit(ams::svc::Handle *out_handle) {
|
|
/* Create a new resource limit. */
|
|
KResourceLimit *resource_limit = KResourceLimit::Create();
|
|
R_UNLESS(resource_limit != nullptr, svc::ResultOutOfResource());
|
|
|
|
/* Ensure we don't leak a reference to the limit. */
|
|
ON_SCOPE_EXIT { resource_limit->Close(); };
|
|
|
|
/* Initialize the resource limit. */
|
|
resource_limit->Initialize();
|
|
|
|
/* Register the limit. */
|
|
KResourceLimit::Register(resource_limit);
|
|
|
|
/* Add the limit to the handle table. */
|
|
R_TRY(GetCurrentProcess().GetHandleTable().Add(out_handle, resource_limit));
|
|
|
|
R_SUCCEED();
|
|
}
|
|
|
|
Result SetResourceLimitLimitValue(ams::svc::Handle resource_limit_handle, ams::svc::LimitableResource which, int64_t limit_value) {
|
|
/* Validate the resource. */
|
|
R_UNLESS(IsValidLimitableResource(which), svc::ResultInvalidEnumValue());
|
|
|
|
/* Get the resource limit. */
|
|
KScopedAutoObject resource_limit = GetCurrentProcess().GetHandleTable().GetObject<KResourceLimit>(resource_limit_handle);
|
|
R_UNLESS(resource_limit.IsNotNull(), svc::ResultInvalidHandle());
|
|
|
|
/* Set the limit value. */
|
|
R_TRY(resource_limit->SetLimitValue(which, limit_value));
|
|
|
|
R_SUCCEED();
|
|
}
|
|
|
|
}
|
|
|
|
/* ============================= 64 ABI ============================= */
|
|
|
|
Result GetResourceLimitLimitValue64(int64_t *out_limit_value, ams::svc::Handle resource_limit_handle, ams::svc::LimitableResource which) {
|
|
R_RETURN(GetResourceLimitLimitValue(out_limit_value, resource_limit_handle, which));
|
|
}
|
|
|
|
Result GetResourceLimitCurrentValue64(int64_t *out_current_value, ams::svc::Handle resource_limit_handle, ams::svc::LimitableResource which) {
|
|
R_RETURN(GetResourceLimitCurrentValue(out_current_value, resource_limit_handle, which));
|
|
}
|
|
|
|
Result GetResourceLimitPeakValue64(int64_t *out_peak_value, ams::svc::Handle resource_limit_handle, ams::svc::LimitableResource which) {
|
|
R_RETURN(GetResourceLimitPeakValue(out_peak_value, resource_limit_handle, which));
|
|
}
|
|
|
|
Result CreateResourceLimit64(ams::svc::Handle *out_handle) {
|
|
R_RETURN(CreateResourceLimit(out_handle));
|
|
}
|
|
|
|
Result SetResourceLimitLimitValue64(ams::svc::Handle resource_limit_handle, ams::svc::LimitableResource which, int64_t limit_value) {
|
|
R_RETURN(SetResourceLimitLimitValue(resource_limit_handle, which, limit_value));
|
|
}
|
|
|
|
/* ============================= 64From32 ABI ============================= */
|
|
|
|
Result GetResourceLimitLimitValue64From32(int64_t *out_limit_value, ams::svc::Handle resource_limit_handle, ams::svc::LimitableResource which) {
|
|
R_RETURN(GetResourceLimitLimitValue(out_limit_value, resource_limit_handle, which));
|
|
}
|
|
|
|
Result GetResourceLimitCurrentValue64From32(int64_t *out_current_value, ams::svc::Handle resource_limit_handle, ams::svc::LimitableResource which) {
|
|
R_RETURN(GetResourceLimitCurrentValue(out_current_value, resource_limit_handle, which));
|
|
}
|
|
|
|
Result GetResourceLimitPeakValue64From32(int64_t *out_peak_value, ams::svc::Handle resource_limit_handle, ams::svc::LimitableResource which) {
|
|
R_RETURN(GetResourceLimitPeakValue(out_peak_value, resource_limit_handle, which));
|
|
}
|
|
|
|
Result CreateResourceLimit64From32(ams::svc::Handle *out_handle) {
|
|
R_RETURN(CreateResourceLimit(out_handle));
|
|
}
|
|
|
|
Result SetResourceLimitLimitValue64From32(ams::svc::Handle resource_limit_handle, ams::svc::LimitableResource which, int64_t limit_value) {
|
|
R_RETURN(SetResourceLimitLimitValue(resource_limit_handle, which, limit_value));
|
|
}
|
|
|
|
}
|