|  | // Copyright 2020 The Dawn & Tint Authors | 
|  | // | 
|  | // Redistribution and use in source and binary forms, with or without | 
|  | // modification, are permitted provided that the following conditions are met: | 
|  | // | 
|  | // 1. Redistributions of source code must retain the above copyright notice, this | 
|  | //    list of conditions and the following disclaimer. | 
|  | // | 
|  | // 2. Redistributions in binary form must reproduce the above copyright notice, | 
|  | //    this list of conditions and the following disclaimer in the documentation | 
|  | //    and/or other materials provided with the distribution. | 
|  | // | 
|  | // 3. Neither the name of the copyright holder nor the names of its | 
|  | //    contributors may be used to endorse or promote products derived from | 
|  | //    this software without specific prior written permission. | 
|  | // | 
|  | // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" | 
|  | // AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE | 
|  | // IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | 
|  | // DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE | 
|  | // FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL | 
|  | // DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR | 
|  | // SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER | 
|  | // CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, | 
|  | // OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE | 
|  | // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | 
|  |  | 
|  | #ifndef SRC_TINT_UTILS_RTTI_CASTABLE_H_ | 
|  | #define SRC_TINT_UTILS_RTTI_CASTABLE_H_ | 
|  |  | 
|  | #include <stdint.h> | 
|  | #include <functional> | 
|  | #include <tuple> | 
|  | #include <type_traits> | 
|  | #include <utility> | 
|  |  | 
|  | #include "src/tint/utils/macros/compiler.h" | 
|  | #include "src/tint/utils/math/crc32.h" | 
|  | #include "src/tint/utils/math/hash.h" | 
|  | #include "src/tint/utils/rtti/ignore.h" | 
|  | #include "src/tint/utils/traits/traits.h" | 
|  |  | 
|  | #if defined(__clang__) | 
|  | /// Temporarily disable certain warnings when using Castable API | 
|  | #define TINT_CASTABLE_PUSH_DISABLE_WARNINGS()                                 \ | 
|  | _Pragma("clang diagnostic push")                                     /**/ \ | 
|  | _Pragma("clang diagnostic ignored \"-Wundefined-var-template\"") /**/ \ | 
|  | TINT_REQUIRE_SEMICOLON | 
|  |  | 
|  | /// Restore disabled warnings | 
|  | #define TINT_CASTABLE_POP_DISABLE_WARNINGS() \ | 
|  | _Pragma("clang diagnostic pop") /**/     \ | 
|  | TINT_REQUIRE_SEMICOLON | 
|  | #else | 
|  | #define TINT_CASTABLE_PUSH_DISABLE_WARNINGS() TINT_REQUIRE_SEMICOLON | 
|  | #define TINT_CASTABLE_POP_DISABLE_WARNINGS() TINT_REQUIRE_SEMICOLON | 
|  | #endif | 
|  |  | 
|  | TINT_CASTABLE_PUSH_DISABLE_WARNINGS(); | 
|  |  | 
|  | // Forward declarations | 
|  | namespace tint { | 
|  | class CastableBase; | 
|  | }  // namespace tint | 
|  |  | 
|  | namespace tint::detail { | 
|  | template <typename T> | 
|  | struct TypeInfoOf; | 
|  | }  // namespace tint::detail | 
|  |  | 
|  | namespace tint { | 
|  |  | 
|  | /// True if all template types that are not Ignore derive from CastableBase | 
|  | template <typename... TYPES> | 
|  | static constexpr bool IsCastable = | 
|  | ((tint::traits::IsTypeOrDerived<TYPES, CastableBase> || std::is_same_v<TYPES, Ignore>)&&...) && | 
|  | !(std::is_same_v<TYPES, Ignore> && ...); | 
|  |  | 
|  | /// Helper macro to instantiate the TypeInfo<T> template for `CLASS`. | 
|  | #define TINT_INSTANTIATE_TYPEINFO(CLASS)                        \ | 
|  | TINT_CASTABLE_PUSH_DISABLE_WARNINGS();                      \ | 
|  | template <>                                                 \ | 
|  | const tint::TypeInfo tint::detail::TypeInfoOf<CLASS>::info{ \ | 
|  | &tint::detail::TypeInfoOf<CLASS::TrueBase>::info,       \ | 
|  | #CLASS,                                                 \ | 
|  | tint::TypeCode::Of<CLASS>(),                            \ | 
|  | tint::TypeCodeSet::OfHierarchy<CLASS>(),                \ | 
|  | };                                                          \ | 
|  | TINT_CASTABLE_POP_DISABLE_WARNINGS();                       \ | 
|  | static_assert(std::is_same_v<CLASS, CLASS::Base::Class>,    \ | 
|  | #CLASS " does not derive from Castable<" #CLASS "[, BASE]>") | 
|  |  | 
|  | /// Bit flags that can be passed to the template parameter `FLAGS` of Is() and As(). | 
|  | enum CastFlags { | 
|  | /// Disables the static_assert() inside Is(), that compile-time-verifies that the cast is | 
|  | /// possible. This flag may be useful for highly-generic template | 
|  | /// code that needs to compile for template permutations that generate | 
|  | /// impossible casts. | 
|  | kDontErrorOnImpossibleCast = 1, | 
|  | }; | 
|  |  | 
|  | /// TypeCode is a bit pattern used by Tint's RTTI system to determine whether two types are related | 
|  | /// by inheritance. | 
|  | /// Each TypeCode has exactly two bits set. | 
|  | struct TypeCode { | 
|  | /// @returns a compile-time TypeCode for the type `T`. | 
|  | template <typename T> | 
|  | static constexpr inline TypeCode Of() { | 
|  | static_assert(IsCastable<T>, "T is not Castable"); | 
|  | static_assert(std::is_same_v<T, std::remove_cv_t<T>>, | 
|  | "Strip const / volatile decorations before calling Of"); | 
|  | /// Use the compiler's "pretty" function name, which includes the template type, to obtain a | 
|  | /// unique hash value. | 
|  | #ifdef _MSC_VER | 
|  | constexpr Bits crc = tint::CRC32(__FUNCSIG__); | 
|  | #else | 
|  | constexpr Bits crc = tint::CRC32(__PRETTY_FUNCTION__); | 
|  | #endif | 
|  | constexpr Bits bit_a = (crc & 63); | 
|  | constexpr Bits bit_b = ((crc >> 6) & 63); | 
|  | constexpr Bits bit_c = (bit_a == bit_b) ? ((bit_a + 1) & 63) : bit_b; | 
|  | return {(static_cast<Bits>(1) << bit_a) | (static_cast<Bits>(1) << bit_c)}; | 
|  | } | 
|  |  | 
|  | /// The unsigned integer type that holds the bits of the TypeCode | 
|  | using Bits = uint64_t; | 
|  |  | 
|  | /// The bits pattern of the TypeCode | 
|  | const Bits bits; | 
|  | }; | 
|  |  | 
|  | /// TypeCodeSet is a set of TypeCodes, and internally uses a single integer to represent its | 
|  | /// contents. TypeCodeSet acts as a bloom-filter, exposing methods to query whether the set _may_ | 
|  | /// contain one or more TypeCodes. If these methods return `false` then the set definitely does | 
|  | /// contain the TypeCode(s), however returning `true` means the *set has a possibility* of | 
|  | /// containing the TypeCodes(s). | 
|  | /// @see https://en.wikipedia.org/wiki/Bloom_filter | 
|  | struct TypeCodeSet { | 
|  | /// @returns the TypeCodeSet that contains the TypeCode of `T` and all its ancestor types. | 
|  | template <typename T> | 
|  | static constexpr inline TypeCodeSet OfHierarchy() { | 
|  | if constexpr (std::is_same_v<T, CastableBase>) { | 
|  | return {TypeCode::Of<T>().bits}; | 
|  | } else { | 
|  | return {TypeCode::Of<T>().bits | TypeCodeSet::OfHierarchy<typename T::TrueBase>().bits}; | 
|  | } | 
|  | } | 
|  |  | 
|  | /// @returns the TypeCodeSet of all the types in the tuple `TUPLE`. | 
|  | template <typename TUPLE> | 
|  | static constexpr inline TypeCodeSet OfTuple() { | 
|  | constexpr auto kCount = std::tuple_size_v<TUPLE>; | 
|  | if constexpr (kCount == 0) { | 
|  | return {0}; | 
|  | } else if constexpr (kCount == 1) { | 
|  | return {TypeCode::Of<std::remove_cv_t<std::tuple_element_t<0, TUPLE>>>().bits}; | 
|  | } else { | 
|  | constexpr auto kMid = kCount / 2; | 
|  | return {OfTuple<tint::traits::SliceTuple<0, kMid, TUPLE>>().bits | | 
|  | OfTuple<tint::traits::SliceTuple<kMid, kCount - kMid, TUPLE>>().bits}; | 
|  | } | 
|  | } | 
|  |  | 
|  | /// @returns true if this TypeCodeSet may contain the TypeCode @p type_code. | 
|  | inline bool MayContain(TypeCode type_code) const { | 
|  | return (bits & type_code.bits) == type_code.bits; | 
|  | } | 
|  |  | 
|  | /// @returns true if TypeCodeSet may contain any of the TypeCodes in @p type_codes. | 
|  | inline bool MayContainAnyOf(TypeCodeSet type_codes) const { | 
|  | // TypeCode always has exactly two bits set, so if the intersection of this TypeCodeSet and | 
|  | // type_codes contains fewer than two bits set, we know there's no possibility of the same | 
|  | // type being part of both sets. | 
|  | Bits mask = bits & type_codes.bits; | 
|  | bool has_at_least_two_bits = (mask & (mask - 1)) != 0; | 
|  | return has_at_least_two_bits; | 
|  | } | 
|  |  | 
|  | /// The unsigned integer type that holds the bits of the TypeCode | 
|  | using Bits = typename TypeCode::Bits; | 
|  |  | 
|  | /// The bits pattern of the TypeCode | 
|  | const Bits bits; | 
|  | }; | 
|  |  | 
|  | /// TypeInfo holds type information for a Castable type. | 
|  | struct TypeInfo { | 
|  | /// The base class of this type | 
|  | const TypeInfo* base; | 
|  | /// The type name | 
|  | const char* name; | 
|  | /// The type's TypeCode | 
|  | const TypeCode type_code; | 
|  | /// The set of this type's TypeCode and all ancestor's TypeCodes | 
|  | const TypeCodeSet full_type_code; | 
|  |  | 
|  | /// @returns true if `type` derives from the class `TO` | 
|  | /// @param object the object type to test from, which must be, or derive from type `FROM`. | 
|  | /// @see CastFlags | 
|  | template <typename TO, typename FROM, int FLAGS = 0> | 
|  | static inline bool Is(const tint::TypeInfo* object) { | 
|  | constexpr const bool downcast = std::is_base_of<FROM, TO>::value; | 
|  | constexpr const bool upcast = std::is_base_of<TO, FROM>::value; | 
|  | constexpr const bool nocast = std::is_same<FROM, TO>::value; | 
|  | constexpr const bool assert_is_castable = (FLAGS & kDontErrorOnImpossibleCast) == 0; | 
|  |  | 
|  | static_assert(upcast || downcast || nocast || !assert_is_castable, "impossible cast"); | 
|  |  | 
|  | return upcast || nocast || object->Is<TO>(); | 
|  | } | 
|  |  | 
|  | /// @returns true if this type derives from the class `T` | 
|  | template <typename T> | 
|  | inline bool Is() const { | 
|  | auto* type = &Of<std::remove_cv_t<T>>(); | 
|  |  | 
|  | if constexpr (std::is_final_v<T>) { | 
|  | // T is final, so nothing can derive from T. | 
|  | // We do not need to check ancestors, only whether this type is equal to the type T. | 
|  | return type == this; | 
|  | } else { | 
|  | return Is(type); | 
|  | } | 
|  | } | 
|  |  | 
|  | /// @param type the test type info | 
|  | /// @returns true if the class with this TypeInfo is of, or derives from the | 
|  | /// class with the given TypeInfo. | 
|  | inline bool Is(const tint::TypeInfo* type) const { | 
|  | if (!full_type_code.MayContain(type->type_code)) { | 
|  | return false; | 
|  | } | 
|  |  | 
|  | // Walk the base types, starting with this TypeInfo, to see if any of the pointers match | 
|  | // `type`. | 
|  | for (auto* ti = this; ti != nullptr; ti = ti->base) { | 
|  | if (ti == type) { | 
|  | return true; | 
|  | } | 
|  | } | 
|  | return false; | 
|  | } | 
|  |  | 
|  | /// @returns the static TypeInfo for the type T | 
|  | template <typename T> | 
|  | static const TypeInfo& Of() { | 
|  | return tint::detail::TypeInfoOf<std::remove_cv_t<T>>::info; | 
|  | } | 
|  |  | 
|  | /// @returns true if this TypeInfo is of, or derives from any of the types in `TUPLE`. | 
|  | template <typename TUPLE> | 
|  | inline bool IsAnyOfTuple() const { | 
|  | constexpr auto kCount = std::tuple_size_v<TUPLE>; | 
|  | if constexpr (kCount == 0) { | 
|  | return false; | 
|  | } else if constexpr (kCount == 1) { | 
|  | return Is(&Of<std::tuple_element_t<0, TUPLE>>()); | 
|  | } else { | 
|  | if (full_type_code.MayContainAnyOf(TypeCodeSet::OfTuple<TUPLE>())) { | 
|  | // Possibly one of the types in `TUPLE`. | 
|  | // Split the search in two, and scan each block. | 
|  | static constexpr auto kMid = kCount / 2; | 
|  | return IsAnyOfTuple<tint::traits::SliceTuple<0, kMid, TUPLE>>() || | 
|  | IsAnyOfTuple<tint::traits::SliceTuple<kMid, kCount - kMid, TUPLE>>(); | 
|  | } | 
|  | return false; | 
|  | } | 
|  | } | 
|  |  | 
|  | /// @returns true if this TypeInfo is of, or derives from any of the types in `TYPES`. | 
|  | template <typename... TYPES> | 
|  | inline bool IsAnyOf() const { | 
|  | return IsAnyOfTuple<std::tuple<TYPES...>>(); | 
|  | } | 
|  | }; | 
|  |  | 
|  | namespace detail { | 
|  |  | 
|  | /// TypeInfoOf contains a single TypeInfo field for the type T. | 
|  | /// TINT_INSTANTIATE_TYPEINFO() must be defined in a .cpp file for each type `T`. | 
|  | template <typename T> | 
|  | struct TypeInfoOf { | 
|  | /// The unique TypeInfo for the type T. | 
|  | static const TypeInfo info; | 
|  | }; | 
|  |  | 
|  | /// A placeholder structure used for template parameters that need a default type, but can always be | 
|  | /// automatically inferred. | 
|  | struct Infer; | 
|  |  | 
|  | }  // namespace detail | 
|  |  | 
|  | /// @returns true if `obj` is a valid pointer, and is of, or derives from the class `TO` | 
|  | /// @param obj the object to test from | 
|  | /// @see CastFlags | 
|  | template <typename TO, int FLAGS = 0, typename FROM = tint::detail::Infer> | 
|  | inline bool Is(FROM* obj) { | 
|  | if (obj == nullptr) { | 
|  | return false; | 
|  | } | 
|  | return TypeInfo::Is<TO, FROM, FLAGS>(&obj->TypeInfo()); | 
|  | } | 
|  |  | 
|  | /// @returns true if `obj` is a valid pointer, and is of, or derives from the type `TYPE`, and | 
|  | /// pred(const TYPE*) returns true | 
|  | /// @param obj the object to test from | 
|  | /// @param pred predicate function with signature `bool(const TYPE*)` called iff object is of, or | 
|  | /// derives from the class `TYPE`. | 
|  | /// @see CastFlags | 
|  | template <typename TYPE, | 
|  | int FLAGS = 0, | 
|  | typename OBJ = tint::detail::Infer, | 
|  | typename Pred = tint::detail::Infer> | 
|  | inline bool Is(OBJ* obj, Pred&& pred) { | 
|  | return Is<TYPE, FLAGS, OBJ>(obj) && pred(static_cast<std::add_const_t<TYPE>*>(obj)); | 
|  | } | 
|  |  | 
|  | /// @returns true if `obj` is a valid pointer, and is of, or derives from any of the types in | 
|  | /// `TYPES`. | 
|  | /// @param obj the object to query. | 
|  | template <typename... TYPES, typename OBJ> | 
|  | inline bool IsAnyOf(OBJ* obj) { | 
|  | if (!obj) { | 
|  | return false; | 
|  | } | 
|  | return obj->TypeInfo().template IsAnyOf<TYPES...>(); | 
|  | } | 
|  |  | 
|  | /// @returns obj dynamically cast to the type `TO` or `nullptr` if this object does not derive from | 
|  | /// `TO`. | 
|  | /// @param obj the object to cast from | 
|  | /// @see CastFlags | 
|  | template <typename TO, int FLAGS = 0, typename FROM = tint::detail::Infer> | 
|  | inline TO* As(FROM* obj) { | 
|  | auto* as_castable = static_cast<CastableBase*>(obj); | 
|  | return Is<TO, FLAGS>(obj) ? static_cast<TO*>(as_castable) : nullptr; | 
|  | } | 
|  |  | 
|  | /// @returns obj dynamically cast to the type `TO` or `nullptr` if this object does not derive from | 
|  | /// `TO`. | 
|  | /// @param obj the object to cast from | 
|  | /// @see CastFlags | 
|  | template <typename TO, int FLAGS = 0, typename FROM = tint::detail::Infer> | 
|  | inline const TO* As(const FROM* obj) { | 
|  | auto* as_castable = static_cast<const CastableBase*>(obj); | 
|  | return Is<TO, FLAGS>(obj) ? static_cast<const TO*>(as_castable) : nullptr; | 
|  | } | 
|  |  | 
|  | /// CastableBase is the base class for all Castable objects. | 
|  | /// It is not encouraged to directly derive from CastableBase without using the Castable helper | 
|  | /// template. | 
|  | /// @see Castable | 
|  | class CastableBase { | 
|  | public: | 
|  | /// Copy constructor | 
|  | CastableBase(const CastableBase&); | 
|  |  | 
|  | /// Destructor | 
|  | virtual ~CastableBase(); | 
|  |  | 
|  | /// Copy assignment | 
|  | /// @param other the CastableBase to copy | 
|  | /// @returns the new CastableBase | 
|  | CastableBase& operator=(const CastableBase& other) = default; | 
|  |  | 
|  | /// @returns the TypeInfo of the object | 
|  | inline const tint::TypeInfo& TypeInfo() const { return *type_info_; } | 
|  |  | 
|  | /// @returns true if this object is of, or derives from the class `TO` | 
|  | template <typename TO> | 
|  | inline bool Is() const { | 
|  | return tint::Is<TO>(this); | 
|  | } | 
|  |  | 
|  | /// @returns true if this object is of, or derives from the class `TO` and pred(const TO*) | 
|  | /// returns true | 
|  | /// @param pred predicate function with signature `bool(const TO*)` called iff object is of, or | 
|  | /// derives from the class `TO`. | 
|  | template <typename TO, int FLAGS = 0, typename Pred = tint::detail::Infer> | 
|  | inline bool Is(Pred&& pred) const { | 
|  | return tint::Is<TO, FLAGS>(this, std::forward<Pred>(pred)); | 
|  | } | 
|  |  | 
|  | /// @returns true if this object is of, or derives from any of the `TO` classes. | 
|  | template <typename... TO> | 
|  | inline bool IsAnyOf() const { | 
|  | return tint::IsAnyOf<TO...>(this); | 
|  | } | 
|  |  | 
|  | /// @returns this object dynamically cast to the type `TO` or `nullptr` if this object does not | 
|  | /// derive from `TO`. | 
|  | /// @see CastFlags | 
|  | template <typename TO, int FLAGS = 0> | 
|  | inline TO* As() { | 
|  | return tint::As<TO, FLAGS>(this); | 
|  | } | 
|  |  | 
|  | /// @returns this object dynamically cast to the type `TO` or `nullptr` if this object does not | 
|  | /// derive from `TO`. | 
|  | /// @see CastFlags | 
|  | template <typename TO, int FLAGS = 0> | 
|  | inline const TO* As() const { | 
|  | return tint::As<const TO, FLAGS>(this); | 
|  | } | 
|  |  | 
|  | protected: | 
|  | CastableBase() = default; | 
|  |  | 
|  | /// The type information for the object | 
|  | const tint::TypeInfo* type_info_ = nullptr; | 
|  | }; | 
|  |  | 
|  | /// Castable is a helper to derive `CLASS` from `BASE`, automatically implementing the Is() and As() | 
|  | /// methods, along with a #Base type alias. | 
|  | /// | 
|  | /// Example usage: | 
|  | /// | 
|  | /// ``` | 
|  | /// class Animal : public Castable<Animal> {}; | 
|  | /// | 
|  | /// class Sheep : public Castable<Sheep, Animal> {}; | 
|  | /// | 
|  | /// Sheep* cast_to_sheep(Animal* animal) { | 
|  | ///    // You can query whether a Castable is of the given type with Is<T>(): | 
|  | ///    printf("animal is a sheep? %s", animal->Is<Sheep>() ? "yes" : "no"); | 
|  | /// | 
|  | ///    // You can always just try the cast with As<T>(). | 
|  | ///    // If the object is not of the correct type, As<T>() will return nullptr: | 
|  | ///    return animal->As<Sheep>(); | 
|  | /// } | 
|  | /// ``` | 
|  | template <typename CLASS, typename BASE = CastableBase> | 
|  | class Castable : public BASE { | 
|  | public: | 
|  | /// A type alias to this Castable. Commonly used in derived type constructors to forward | 
|  | /// constructor arguments to BASE. | 
|  | using Base = Castable; | 
|  |  | 
|  | /// A type alias for `BASE`. | 
|  | using TrueBase = BASE; | 
|  |  | 
|  | /// A type alias for `CLASS`. | 
|  | using Class = CLASS; | 
|  |  | 
|  | /// Constructor | 
|  | /// @param arguments the arguments to forward to the base class. | 
|  | template <typename... ARGS> | 
|  | inline explicit Castable(ARGS&&... arguments) : TrueBase(std::forward<ARGS>(arguments)...) { | 
|  | this->type_info_ = &TypeInfo::Of<CLASS>(); | 
|  | } | 
|  |  | 
|  | /// @returns true if this object is of, or derives from the class `TO` | 
|  | /// @see CastFlags | 
|  | template <typename TO, int FLAGS = 0> | 
|  | inline bool Is() const { | 
|  | return tint::Is<TO, FLAGS>(static_cast<const CLASS*>(this)); | 
|  | } | 
|  |  | 
|  | /// @returns true if this object is of, or derives from the class `TO` and | 
|  | /// pred(const TO*) returns true | 
|  | /// @param pred predicate function with signature `bool(const TO*)` called iff | 
|  | /// object is of, or derives from the class `TO`. | 
|  | template <int FLAGS = 0, typename Pred = tint::detail::Infer> | 
|  | inline bool Is(Pred&& pred) const { | 
|  | using TO = typename std::remove_pointer<tint::traits::ParameterType<Pred, 0>>::type; | 
|  | return tint::Is<TO, FLAGS>(static_cast<const CLASS*>(this), std::forward<Pred>(pred)); | 
|  | } | 
|  |  | 
|  | /// @returns true if this object is of, or derives from any of the `TO` | 
|  | /// classes. | 
|  | template <typename... TO> | 
|  | inline bool IsAnyOf() const { | 
|  | return tint::IsAnyOf<TO...>(static_cast<const CLASS*>(this)); | 
|  | } | 
|  |  | 
|  | /// @returns this object dynamically cast to the type `TO` or `nullptr` if | 
|  | /// this object does not derive from `TO`. | 
|  | /// @see CastFlags | 
|  | template <typename TO, int FLAGS = 0> | 
|  | inline TO* As() { | 
|  | return tint::As<TO, FLAGS>(this); | 
|  | } | 
|  |  | 
|  | /// @returns this object dynamically cast to the type `TO` or `nullptr` if | 
|  | /// this object does not derive from `TO`. | 
|  | /// @see CastFlags | 
|  | template <typename TO, int FLAGS = 0> | 
|  | inline const TO* As() const { | 
|  | return tint::As<const TO, FLAGS>(this); | 
|  | } | 
|  | }; | 
|  |  | 
|  | namespace detail { | 
|  | /// <code>typename CastableCommonBaseImpl<TYPES>::type</code> resolves to the common base class for | 
|  | /// all of TYPES. | 
|  | template <typename... TYPES> | 
|  | struct CastableCommonBaseImpl {}; | 
|  |  | 
|  | /// Alias to typename CastableCommonBaseImpl<TYPES>::type | 
|  | template <typename... TYPES> | 
|  | using CastableCommonBase = typename CastableCommonBaseImpl<TYPES...>::type; | 
|  |  | 
|  | /// CastableCommonBaseImpl template specialization for a single type | 
|  | template <typename T> | 
|  | struct CastableCommonBaseImpl<T> { | 
|  | /// Common base class of a single type is itself | 
|  | using type = T; | 
|  | }; | 
|  |  | 
|  | /// CastableCommonBaseImpl A <-> CastableBase specialization | 
|  | template <typename A> | 
|  | struct CastableCommonBaseImpl<A, CastableBase> { | 
|  | /// Common base class for A and CastableBase is CastableBase | 
|  | using type = CastableBase; | 
|  | }; | 
|  |  | 
|  | /// CastableCommonBaseImpl T <-> Ignore specialization | 
|  | template <typename T> | 
|  | struct CastableCommonBaseImpl<T, Ignore> { | 
|  | /// Resolves to T as the other type is ignored | 
|  | using type = T; | 
|  | }; | 
|  |  | 
|  | /// CastableCommonBaseImpl Ignore <-> T specialization | 
|  | template <typename T> | 
|  | struct CastableCommonBaseImpl<Ignore, T> { | 
|  | /// Resolves to T as the other type is ignored | 
|  | using type = T; | 
|  | }; | 
|  |  | 
|  | /// CastableCommonBaseImpl A <-> B specialization | 
|  | template <typename A, typename B> | 
|  | struct CastableCommonBaseImpl<A, B> { | 
|  | /// The common base class for A, B and OTHERS | 
|  | using type = std::conditional_t<tint::traits::IsTypeOrDerived<A, B>, | 
|  | B,  // A derives from B | 
|  | CastableCommonBase<A, typename B::TrueBase>>; | 
|  | }; | 
|  |  | 
|  | /// CastableCommonBaseImpl 3+ types specialization | 
|  | template <typename A, typename B, typename... OTHERS> | 
|  | struct CastableCommonBaseImpl<A, B, OTHERS...> { | 
|  | /// The common base class for A, B and OTHERS | 
|  | using type = CastableCommonBase<CastableCommonBase<A, B>, OTHERS...>; | 
|  | }; | 
|  |  | 
|  | }  // namespace detail | 
|  |  | 
|  | /// Resolves to the common most derived type that each of the types in `TYPES` derives from. | 
|  | template <typename... TYPES> | 
|  | using CastableCommonBase = tint::detail::CastableCommonBase<TYPES...>; | 
|  |  | 
|  | }  // namespace tint | 
|  |  | 
|  | namespace tint { | 
|  |  | 
|  | using tint::As; | 
|  | using tint::Is; | 
|  |  | 
|  | }  // namespace tint | 
|  |  | 
|  | TINT_CASTABLE_POP_DISABLE_WARNINGS(); | 
|  |  | 
|  | #endif  // SRC_TINT_UTILS_RTTI_CASTABLE_H_ |