mirror of
https://github.com/yuzu-emu/yuzu-android.git
synced 2025-01-07 20:35:29 +00:00
02bc9e9de1
Removes a few inclusion dependencies from the headers or replaces existing ones with ones that don't indirectly include the required headers. This allows removing an inclusion of core/memory.h, meaning that if the memory header is ever changed in the future, it won't result in rebuilding the entirety of the HLE services (as the IPC headers are used quite ubiquitously throughout the HLE service implementations).
369 lines
10 KiB
C++
369 lines
10 KiB
C++
// Copyright 2014 Citra Emulator Project
|
|
// Licensed under GPLv2 or any later version
|
|
// Refer to the license.txt file included.
|
|
|
|
#pragma once
|
|
|
|
#include <new>
|
|
#include <utility>
|
|
#include "common/assert.h"
|
|
#include "common/bit_field.h"
|
|
#include "common/common_types.h"
|
|
|
|
// All the constants in this file come from http://switchbrew.org/index.php?title=Error_codes
|
|
|
|
/**
|
|
* Detailed description of the error. Code 0 always means success.
|
|
*/
|
|
enum class ErrorDescription : u32 {
|
|
Success = 0,
|
|
RemoteProcessDead = 301,
|
|
};
|
|
|
|
/**
|
|
* Identifies the module which caused the error. Error codes can be propagated through a call
|
|
* chain, meaning that this doesn't always correspond to the module where the API call made is
|
|
* contained.
|
|
*/
|
|
enum class ErrorModule : u32 {
|
|
Common = 0,
|
|
Kernel = 1,
|
|
FS = 2,
|
|
OS = 3, // used for Memory, Thread, Mutex, Nvidia
|
|
HTCS = 4,
|
|
NCM = 5,
|
|
DD = 6,
|
|
LR = 8,
|
|
Loader = 9,
|
|
CMIF = 10,
|
|
HIPC = 11,
|
|
PM = 15,
|
|
NS = 16,
|
|
HTC = 18,
|
|
NCMContent = 20,
|
|
SM = 21,
|
|
RO = 22,
|
|
SDMMC = 24,
|
|
OVLN = 25,
|
|
SPL = 26,
|
|
ETHC = 100,
|
|
I2C = 101,
|
|
GPIO = 102,
|
|
UART = 103,
|
|
Settings = 105,
|
|
WLAN = 107,
|
|
XCD = 108,
|
|
NIFM = 110,
|
|
Hwopus = 111,
|
|
Bluetooth = 113,
|
|
VI = 114,
|
|
NFP = 115,
|
|
Time = 116,
|
|
FGM = 117,
|
|
OE = 118,
|
|
PCIe = 120,
|
|
Friends = 121,
|
|
BCAT = 122,
|
|
SSL = 123,
|
|
Account = 124,
|
|
News = 125,
|
|
Mii = 126,
|
|
NFC = 127,
|
|
AM = 128,
|
|
PlayReport = 129,
|
|
AHID = 130,
|
|
Qlaunch = 132,
|
|
PCV = 133,
|
|
OMM = 134,
|
|
BPC = 135,
|
|
PSM = 136,
|
|
NIM = 137,
|
|
PSC = 138,
|
|
TC = 139,
|
|
USB = 140,
|
|
NSD = 141,
|
|
PCTL = 142,
|
|
BTM = 143,
|
|
ETicket = 145,
|
|
NGC = 146,
|
|
ERPT = 147,
|
|
APM = 148,
|
|
Profiler = 150,
|
|
ErrorUpload = 151,
|
|
Audio = 153,
|
|
NPNS = 154,
|
|
NPNSHTTPSTREAM = 155,
|
|
ARP = 157,
|
|
SWKBD = 158,
|
|
BOOT = 159,
|
|
NFCMifare = 161,
|
|
UserlandAssert = 162,
|
|
Fatal = 163,
|
|
NIMShop = 164,
|
|
SPSM = 165,
|
|
BGTC = 167,
|
|
UserlandCrash = 168,
|
|
SREPO = 180,
|
|
Dauth = 181,
|
|
HID = 202,
|
|
LDN = 203,
|
|
Irsensor = 205,
|
|
Capture = 206,
|
|
Manu = 208,
|
|
ATK = 209,
|
|
GRC = 212,
|
|
Migration = 216,
|
|
MigrationLdcServ = 217,
|
|
GeneralWebApplet = 800,
|
|
WifiWebAuthApplet = 809,
|
|
WhitelistedApplet = 810,
|
|
ShopN = 811,
|
|
};
|
|
|
|
/// Encapsulates a CTR-OS error code, allowing it to be separated into its constituent fields.
|
|
union ResultCode {
|
|
u32 raw;
|
|
|
|
BitField<0, 9, ErrorModule> module;
|
|
BitField<9, 13, u32> description;
|
|
|
|
// The last bit of `level` is checked by apps and the kernel to determine if a result code is an
|
|
// error
|
|
BitField<31, 1, u32> is_error;
|
|
|
|
constexpr explicit ResultCode(u32 raw) : raw(raw) {}
|
|
|
|
constexpr ResultCode(ErrorModule module, ErrorDescription description)
|
|
: ResultCode(module, static_cast<u32>(description)) {}
|
|
|
|
constexpr ResultCode(ErrorModule module_, u32 description_)
|
|
: raw(module.FormatValue(module_) | description.FormatValue(description_)) {}
|
|
|
|
constexpr ResultCode& operator=(const ResultCode& o) {
|
|
raw = o.raw;
|
|
return *this;
|
|
}
|
|
|
|
constexpr bool IsSuccess() const {
|
|
return raw == 0;
|
|
}
|
|
|
|
constexpr bool IsError() const {
|
|
return raw != 0;
|
|
}
|
|
};
|
|
|
|
constexpr bool operator==(const ResultCode& a, const ResultCode& b) {
|
|
return a.raw == b.raw;
|
|
}
|
|
|
|
constexpr bool operator!=(const ResultCode& a, const ResultCode& b) {
|
|
return a.raw != b.raw;
|
|
}
|
|
|
|
// Convenience functions for creating some common kinds of errors:
|
|
|
|
/// The default success `ResultCode`.
|
|
constexpr ResultCode RESULT_SUCCESS(0);
|
|
|
|
/**
|
|
* This is an optional value type. It holds a `ResultCode` and, if that code is a success code,
|
|
* also holds a result of type `T`. If the code is an error code then trying to access the inner
|
|
* value fails, thus ensuring that the ResultCode of functions is always checked properly before
|
|
* their return value is used. It is similar in concept to the `std::optional` type
|
|
* (http://en.cppreference.com/w/cpp/experimental/optional) originally proposed for inclusion in
|
|
* C++14, or the `Result` type in Rust (http://doc.rust-lang.org/std/result/index.html).
|
|
*
|
|
* An example of how it could be used:
|
|
* \code
|
|
* ResultVal<int> Frobnicate(float strength) {
|
|
* if (strength < 0.f || strength > 1.0f) {
|
|
* // Can't frobnicate too weakly or too strongly
|
|
* return ResultCode(ErrorDescription::OutOfRange, ErrorModule::Common,
|
|
* ErrorSummary::InvalidArgument, ErrorLevel::Permanent);
|
|
* } else {
|
|
* // Frobnicated! Give caller a cookie
|
|
* return MakeResult<int>(42);
|
|
* }
|
|
* }
|
|
* \endcode
|
|
*
|
|
* \code
|
|
* ResultVal<int> frob_result = Frobnicate(0.75f);
|
|
* if (frob_result) {
|
|
* // Frobbed ok
|
|
* printf("My cookie is %d\n", *frob_result);
|
|
* } else {
|
|
* printf("Guess I overdid it. :( Error code: %ux\n", frob_result.code().hex);
|
|
* }
|
|
* \endcode
|
|
*/
|
|
template <typename T>
|
|
class ResultVal {
|
|
public:
|
|
/// Constructs an empty `ResultVal` with the given error code. The code must not be a success
|
|
/// code.
|
|
ResultVal(ResultCode error_code = ResultCode(-1)) : result_code(error_code) {
|
|
ASSERT(error_code.IsError());
|
|
}
|
|
|
|
/**
|
|
* Similar to the non-member function `MakeResult`, with the exception that you can manually
|
|
* specify the success code. `success_code` must not be an error code.
|
|
*/
|
|
template <typename... Args>
|
|
static ResultVal WithCode(ResultCode success_code, Args&&... args) {
|
|
ResultVal<T> result;
|
|
result.emplace(success_code, std::forward<Args>(args)...);
|
|
return result;
|
|
}
|
|
|
|
ResultVal(const ResultVal& o) : result_code(o.result_code) {
|
|
if (!o.empty()) {
|
|
new (&object) T(o.object);
|
|
}
|
|
}
|
|
|
|
ResultVal(ResultVal&& o) noexcept : result_code(o.result_code) {
|
|
if (!o.empty()) {
|
|
new (&object) T(std::move(o.object));
|
|
}
|
|
}
|
|
|
|
~ResultVal() {
|
|
if (!empty()) {
|
|
object.~T();
|
|
}
|
|
}
|
|
|
|
ResultVal& operator=(const ResultVal& o) {
|
|
if (this == &o) {
|
|
return *this;
|
|
}
|
|
if (!empty()) {
|
|
if (!o.empty()) {
|
|
object = o.object;
|
|
} else {
|
|
object.~T();
|
|
}
|
|
} else {
|
|
if (!o.empty()) {
|
|
new (&object) T(o.object);
|
|
}
|
|
}
|
|
result_code = o.result_code;
|
|
|
|
return *this;
|
|
}
|
|
|
|
/**
|
|
* Replaces the current result with a new constructed result value in-place. The code must not
|
|
* be an error code.
|
|
*/
|
|
template <typename... Args>
|
|
void emplace(ResultCode success_code, Args&&... args) {
|
|
ASSERT(success_code.IsSuccess());
|
|
if (!empty()) {
|
|
object.~T();
|
|
}
|
|
new (&object) T(std::forward<Args>(args)...);
|
|
result_code = success_code;
|
|
}
|
|
|
|
/// Returns true if the `ResultVal` contains an error code and no value.
|
|
bool empty() const {
|
|
return result_code.IsError();
|
|
}
|
|
|
|
/// Returns true if the `ResultVal` contains a return value.
|
|
bool Succeeded() const {
|
|
return result_code.IsSuccess();
|
|
}
|
|
/// Returns true if the `ResultVal` contains an error code and no value.
|
|
bool Failed() const {
|
|
return empty();
|
|
}
|
|
|
|
ResultCode Code() const {
|
|
return result_code;
|
|
}
|
|
|
|
const T& operator*() const {
|
|
return object;
|
|
}
|
|
T& operator*() {
|
|
return object;
|
|
}
|
|
const T* operator->() const {
|
|
return &object;
|
|
}
|
|
T* operator->() {
|
|
return &object;
|
|
}
|
|
|
|
/// Returns the value contained in this `ResultVal`, or the supplied default if it is missing.
|
|
template <typename U>
|
|
T ValueOr(U&& value) const {
|
|
return !empty() ? object : std::move(value);
|
|
}
|
|
|
|
/// Asserts that the result succeeded and returns a reference to it.
|
|
T& Unwrap() & {
|
|
ASSERT_MSG(Succeeded(), "Tried to Unwrap empty ResultVal");
|
|
return **this;
|
|
}
|
|
|
|
T&& Unwrap() && {
|
|
ASSERT_MSG(Succeeded(), "Tried to Unwrap empty ResultVal");
|
|
return std::move(**this);
|
|
}
|
|
|
|
private:
|
|
// A union is used to allocate the storage for the value, while allowing us to construct and
|
|
// destruct it at will.
|
|
union {
|
|
T object;
|
|
};
|
|
ResultCode result_code;
|
|
};
|
|
|
|
/**
|
|
* This function is a helper used to construct `ResultVal`s. It receives the arguments to construct
|
|
* `T` with and creates a success `ResultVal` contained the constructed value.
|
|
*/
|
|
template <typename T, typename... Args>
|
|
ResultVal<T> MakeResult(Args&&... args) {
|
|
return ResultVal<T>::WithCode(RESULT_SUCCESS, std::forward<Args>(args)...);
|
|
}
|
|
|
|
/**
|
|
* Deducible overload of MakeResult, allowing the template parameter to be ommited if you're just
|
|
* copy or move constructing.
|
|
*/
|
|
template <typename Arg>
|
|
ResultVal<std::remove_reference_t<Arg>> MakeResult(Arg&& arg) {
|
|
return ResultVal<std::remove_reference_t<Arg>>::WithCode(RESULT_SUCCESS,
|
|
std::forward<Arg>(arg));
|
|
}
|
|
|
|
/**
|
|
* Check for the success of `source` (which must evaluate to a ResultVal). If it succeeds, unwraps
|
|
* the contained value and assigns it to `target`, which can be either an l-value expression or a
|
|
* variable declaration. If it fails the return code is returned from the current function. Thus it
|
|
* can be used to cascade errors out, achieving something akin to exception handling.
|
|
*/
|
|
#define CASCADE_RESULT(target, source) \
|
|
auto CONCAT2(check_result_L, __LINE__) = source; \
|
|
if (CONCAT2(check_result_L, __LINE__).Failed()) \
|
|
return CONCAT2(check_result_L, __LINE__).Code(); \
|
|
target = std::move(*CONCAT2(check_result_L, __LINE__))
|
|
|
|
/**
|
|
* Analogous to CASCADE_RESULT, but for a bare ResultCode. The code will be propagated if
|
|
* non-success, or discarded otherwise.
|
|
*/
|
|
#define CASCADE_CODE(source) \
|
|
auto CONCAT2(check_result_L, __LINE__) = source; \
|
|
if (CONCAT2(check_result_L, __LINE__).IsError()) \
|
|
return CONCAT2(check_result_L, __LINE__);
|