mirror of
https://github.com/Atmosphere-NX/Atmosphere.git
synced 2025-01-07 10:06:10 +00:00
204 lines
8.6 KiB
C++
204 lines
8.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/>.
|
|
*/
|
|
#pragma once
|
|
#include <mesosphere/kern_common.hpp>
|
|
#include <mesosphere/kern_k_typed_address.hpp>
|
|
|
|
namespace ams::kern {
|
|
|
|
class KAutoObject;
|
|
|
|
class KSystemResource;
|
|
|
|
#define FOR_EACH_K_CLASS_TOKEN_OBJECT_TYPE(HANDLER) \
|
|
HANDLER(KAutoObject) \
|
|
\
|
|
HANDLER(KSynchronizationObject) \
|
|
HANDLER(KReadableEvent) \
|
|
\
|
|
HANDLER(KInterruptEvent) \
|
|
HANDLER(KDebug) \
|
|
HANDLER(KThread) \
|
|
HANDLER(KServerPort) \
|
|
HANDLER(KServerSession) \
|
|
HANDLER(KClientPort) \
|
|
HANDLER(KClientSession) \
|
|
HANDLER(KProcess) \
|
|
HANDLER(KResourceLimit) \
|
|
HANDLER(KLightSession) \
|
|
HANDLER(KPort) \
|
|
HANDLER(KSession) \
|
|
HANDLER(KSharedMemory) \
|
|
HANDLER(KEvent) \
|
|
HANDLER(KLightClientSession) \
|
|
HANDLER(KLightServerSession) \
|
|
HANDLER(KTransferMemory) \
|
|
HANDLER(KDeviceAddressSpace) \
|
|
HANDLER(KSessionRequest) \
|
|
HANDLER(KCodeMemory) \
|
|
HANDLER(KIoPool) \
|
|
HANDLER(KIoRegion) \
|
|
HANDLER(KSystemResource)
|
|
|
|
class KClassTokenGenerator {
|
|
public:
|
|
using TokenBaseType = u16;
|
|
public:
|
|
static constexpr size_t BaseClassBits = 8;
|
|
static constexpr size_t FinalClassBits = (sizeof(TokenBaseType) * CHAR_BIT) - BaseClassBits;
|
|
/* One bit per base class. */
|
|
static constexpr size_t NumBaseClasses = BaseClassBits;
|
|
/* Final classes are permutations of three bits. */
|
|
static constexpr size_t NumFinalClasses = [] {
|
|
TokenBaseType index = 0;
|
|
for (size_t i = 0; i < FinalClassBits; i++) {
|
|
for (size_t j = i + 1; j < FinalClassBits; j++) {
|
|
for (size_t k = j + 1; k < FinalClassBits; k++) {
|
|
index++;
|
|
}
|
|
}
|
|
}
|
|
return index;
|
|
}();
|
|
private:
|
|
template<TokenBaseType Index>
|
|
static constexpr inline TokenBaseType BaseClassToken = BIT(Index);
|
|
|
|
template<TokenBaseType Index>
|
|
static constexpr inline TokenBaseType FinalClassToken = [] {
|
|
TokenBaseType index = 0;
|
|
for (size_t i = 0; i < FinalClassBits; i++) {
|
|
for (size_t j = i + 1; j < FinalClassBits; j++) {
|
|
for (size_t k = j + 1; k < FinalClassBits; k++) {
|
|
if ((index++) == Index) {
|
|
return ((1ul << i) | (1ul << j) | (1ul << k)) << BaseClassBits;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
__builtin_unreachable();
|
|
}();
|
|
|
|
template<typename T>
|
|
static constexpr inline TokenBaseType GetClassToken() {
|
|
static_assert(std::is_base_of<KAutoObject, T>::value);
|
|
if constexpr (std::is_same<T, KAutoObject>::value) {
|
|
static_assert(T::ObjectType == ObjectType::KAutoObject);
|
|
return 0;
|
|
} else if constexpr (!std::is_final<T>::value && !std::same_as<T, KSystemResource>) {
|
|
static_assert(ObjectType::BaseClassesStart <= T::ObjectType && T::ObjectType < ObjectType::BaseClassesEnd);
|
|
constexpr auto ClassIndex = static_cast<TokenBaseType>(T::ObjectType) - static_cast<TokenBaseType>(ObjectType::BaseClassesStart);
|
|
return BaseClassToken<ClassIndex> | GetClassToken<typename T::BaseClass>();
|
|
} else if constexpr (ObjectType::FinalClassesStart <= T::ObjectType && T::ObjectType < ObjectType::FinalClassesEnd) {
|
|
constexpr auto ClassIndex = static_cast<TokenBaseType>(T::ObjectType) - static_cast<TokenBaseType>(ObjectType::FinalClassesStart);
|
|
return FinalClassToken<ClassIndex> | GetClassToken<typename T::BaseClass>();
|
|
} else {
|
|
static_assert(!std::is_same<T, T>::value, "GetClassToken: Invalid Type");
|
|
}
|
|
};
|
|
public:
|
|
enum class ObjectType {
|
|
KAutoObject,
|
|
|
|
BaseClassesStart,
|
|
|
|
KSynchronizationObject = BaseClassesStart,
|
|
KReadableEvent,
|
|
|
|
BaseClassesEnd,
|
|
|
|
FinalClassesStart = BaseClassesEnd,
|
|
|
|
KInterruptEvent = FinalClassesStart,
|
|
KDebug,
|
|
KThread,
|
|
KServerPort,
|
|
KServerSession,
|
|
KClientPort,
|
|
KClientSession,
|
|
KProcess,
|
|
KResourceLimit,
|
|
KLightSession,
|
|
KPort,
|
|
KSession,
|
|
KSharedMemory,
|
|
KEvent,
|
|
KLightClientSession,
|
|
KLightServerSession,
|
|
KTransferMemory,
|
|
KDeviceAddressSpace,
|
|
KSessionRequest,
|
|
KCodeMemory,
|
|
KIoPool,
|
|
KIoRegion,
|
|
|
|
/* NOTE: What occupies these gaps? They can be inferred, but they don't make sense. */
|
|
KAlpha,
|
|
KBeta,
|
|
|
|
KSystemResource,
|
|
|
|
FinalClassesLast,
|
|
|
|
FinalClassesEnd = FinalClassesStart + NumFinalClasses,
|
|
};
|
|
static_assert(ObjectType::FinalClassesLast <= ObjectType::FinalClassesEnd);
|
|
|
|
template<typename T>
|
|
static constexpr inline TokenBaseType ClassToken = GetClassToken<T>();
|
|
};
|
|
|
|
using ClassTokenType = KClassTokenGenerator::TokenBaseType;
|
|
|
|
template<typename T>
|
|
static constexpr inline ClassTokenType ClassToken = KClassTokenGenerator::ClassToken<T>;
|
|
|
|
namespace impl {
|
|
|
|
consteval bool IsKClassTokenGeneratorForEachMacroValid() {
|
|
auto IsObjectTypeIncludedByMacro = [](KClassTokenGenerator::ObjectType object_type) -> bool {
|
|
#define CLASS_TOKEN_HANDLER(CLASSNAME) if (object_type == KClassTokenGenerator::ObjectType::CLASSNAME) { return true; }
|
|
FOR_EACH_K_CLASS_TOKEN_OBJECT_TYPE(CLASS_TOKEN_HANDLER)
|
|
#undef CLASS_TOKEN_HANDLER
|
|
return false;
|
|
};
|
|
|
|
if (!IsObjectTypeIncludedByMacro(KClassTokenGenerator::ObjectType::KAutoObject)) {
|
|
return false;
|
|
}
|
|
|
|
for (auto base = util::ToUnderlying(KClassTokenGenerator::ObjectType::BaseClassesStart); base < util::ToUnderlying(KClassTokenGenerator::ObjectType::BaseClassesEnd); ++base) {
|
|
if (!IsObjectTypeIncludedByMacro(static_cast<KClassTokenGenerator::ObjectType>(base))) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
for (auto fin = util::ToUnderlying(KClassTokenGenerator::ObjectType::FinalClassesStart); fin < util::ToUnderlying(KClassTokenGenerator::ObjectType::FinalClassesLast); ++fin) {
|
|
if (const auto o = static_cast<KClassTokenGenerator::ObjectType>(fin); !IsObjectTypeIncludedByMacro(o)) {
|
|
if (o != KClassTokenGenerator::ObjectType::KAlpha && o != KClassTokenGenerator::ObjectType::KBeta) {
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
static_assert(IsKClassTokenGeneratorForEachMacroValid());
|
|
|
|
}
|
|
|
|
}
|