| // Copyright 2023 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_SWITCH_H_ |
| #define SRC_TINT_UTILS_RTTI_SWITCH_H_ |
| |
| #include <tuple> |
| #include <utility> |
| |
| #include "src/tint/utils/ice/ice.h" |
| #include "src/tint/utils/macros/defer.h" |
| #include "src/tint/utils/memory/aligned_storage.h" |
| #include "src/tint/utils/rtti/castable.h" |
| #include "src/tint/utils/rtti/ignore.h" |
| |
| namespace tint { |
| |
| /// Default can be used as the default case for a Switch(), when all previous cases failed to match. |
| /// |
| /// Example: |
| /// ``` |
| /// Switch(object, |
| /// [&](TypeA*) { /* ... */ }, |
| /// [&](TypeB*) { /* ... */ }, |
| /// [&](Default) { /* If not TypeA or TypeB */ }); |
| /// ``` |
| struct Default {}; |
| |
| /// SwitchMustMatchCase is a flag that can be passed as the last argument to Switch() which will |
| /// trigger an ICE if none of the cases matched. Cannot be used with Default. |
| /// See TINT_ICE_ON_NO_MATCH |
| struct SwitchMustMatchCase { |
| /// The source file that holds the TINT_ICE_ON_NO_MATCH |
| const char* file = "<unknown>"; |
| /// The source line that holds the TINT_ICE_ON_NO_MATCH |
| unsigned int line = 0; |
| }; |
| |
| /// SwitchMustMatchCase is a flag that can be passed as the last argument to Switch() which will |
| /// trigger an ICE if none of the cases matched. Cannot be used with Default. |
| /// |
| /// Example: |
| /// ``` |
| /// Switch(object, |
| /// [&](TypeA*) { /* ... */ }, |
| /// [&](TypeB*) { /* ... */ }, |
| /// TINT_ICE_ON_NO_MATCH); |
| /// ``` |
| #define TINT_ICE_ON_NO_MATCH \ |
| tint::SwitchMustMatchCase { \ |
| __FILE__, __LINE__ \ |
| } |
| |
| } // namespace tint |
| |
| namespace tint::detail { |
| |
| /// Evaluates to the Switch case type being matched by the switch case function `FN`. |
| /// @note does not handle the Default case |
| /// @see Switch(). |
| template <typename FN> |
| using SwitchCaseType = |
| std::remove_pointer_t<tint::traits::ParameterType<std::remove_reference_t<FN>, 0>>; |
| |
| /// Searches the list of Switch cases for a Default case, returning the index of the Default case. |
| /// If the a Default case is not found in the tuple, then -1 is returned. |
| template <typename TUPLE, std::size_t START_IDX = 0> |
| constexpr int IndexOfDefaultCase() { |
| if constexpr (START_IDX < std::tuple_size_v<TUPLE>) { |
| using T = std::decay_t<std::tuple_element_t<START_IDX, TUPLE>>; |
| if constexpr (std::is_same_v<T, SwitchMustMatchCase>) { |
| return -1; |
| } else if constexpr (std::is_same_v<tint::traits::ParameterType<T, 0>, Default>) { |
| return static_cast<int>(START_IDX); |
| } else { |
| return IndexOfDefaultCase<TUPLE, START_IDX + 1>(); |
| } |
| } else { |
| return -1; |
| } |
| } |
| |
| /// Searches the list of Switch cases for a SwitchMustMatchCase flag, returning the index of the |
| /// SwitchMustMatchCase case. If the a SwitchMustMatchCase case is not found in the tuple, then -1 |
| /// is returned. |
| template <typename TUPLE, std::size_t START_IDX = 0> |
| constexpr int IndexOfSwitchMustMatchCase() { |
| if constexpr (START_IDX < std::tuple_size_v<TUPLE>) { |
| using T = std::decay_t<std::tuple_element_t<START_IDX, TUPLE>>; |
| return std::is_same_v<T, SwitchMustMatchCase> |
| ? static_cast<int>(START_IDX) |
| : IndexOfSwitchMustMatchCase<TUPLE, START_IDX + 1>(); |
| } else { |
| return -1; |
| } |
| } |
| /// Resolves to T if T is not nullptr_t, otherwise resolves to Ignore. |
| template <typename T> |
| using NullptrToIgnore = std::conditional_t<std::is_same_v<T, std::nullptr_t>, tint::Ignore, T>; |
| |
| /// Resolves to `const TYPE` if any of `CASE_RETURN_TYPES` are const or pointer-to-const, otherwise |
| /// resolves to TYPE. |
| template <typename TYPE, typename... CASE_RETURN_TYPES> |
| using PropagateReturnConst = std::conditional_t< |
| // Are any of the pointer-stripped types const? |
| (std::is_const_v<std::remove_pointer_t<CASE_RETURN_TYPES>> || ...), |
| const TYPE, // Yes: Apply const to TYPE |
| TYPE>; // No: Passthrough |
| |
| /// SwitchReturnTypeImpl is the implementation of SwitchReturnType |
| template <bool IS_CASTABLE, typename REQUESTED_TYPE, typename... CASE_RETURN_TYPES> |
| struct SwitchReturnTypeImpl; |
| |
| /// SwitchReturnTypeImpl specialization for non-castable case types and an explicitly specified |
| /// return type. |
| template <typename REQUESTED_TYPE, typename... CASE_RETURN_TYPES> |
| struct SwitchReturnTypeImpl</*IS_CASTABLE*/ false, REQUESTED_TYPE, CASE_RETURN_TYPES...> { |
| /// Resolves to `REQUESTED_TYPE` |
| using type = REQUESTED_TYPE; |
| }; |
| |
| /// SwitchReturnTypeImpl specialization for non-castable case types and an inferred return type. |
| template <typename... CASE_RETURN_TYPES> |
| struct SwitchReturnTypeImpl</*IS_CASTABLE*/ false, tint::detail::Infer, CASE_RETURN_TYPES...> { |
| /// Resolves to the common type for all the cases return types. |
| using type = std::common_type_t<CASE_RETURN_TYPES...>; |
| }; |
| |
| /// SwitchReturnTypeImpl specialization for castable case types and an explicitly specified return |
| /// type. |
| template <typename REQUESTED_TYPE, typename... CASE_RETURN_TYPES> |
| struct SwitchReturnTypeImpl</*IS_CASTABLE*/ true, REQUESTED_TYPE, CASE_RETURN_TYPES...> { |
| public: |
| /// Resolves to `const REQUESTED_TYPE*` or `REQUESTED_TYPE*` |
| using type = PropagateReturnConst<std::remove_pointer_t<REQUESTED_TYPE>, CASE_RETURN_TYPES...>*; |
| }; |
| |
| /// SwitchReturnTypeImpl specialization for castable case types and an inferred return type. |
| template <typename... CASE_RETURN_TYPES> |
| struct SwitchReturnTypeImpl</*IS_CASTABLE*/ true, tint::detail::Infer, CASE_RETURN_TYPES...> { |
| private: |
| using InferredType = |
| CastableCommonBase<NullptrToIgnore<std::remove_pointer_t<CASE_RETURN_TYPES>>...>; |
| |
| public: |
| /// `const T*` or `T*`, where T is the common base type for all the castable case types. |
| using type = PropagateReturnConst<InferredType, CASE_RETURN_TYPES...>*; |
| }; |
| |
| /// Resolves to the return type for a Switch() with the requested return type `REQUESTED_TYPE` and |
| /// case statement return types. If `REQUESTED_TYPE` is Infer then the return type will be inferred |
| /// from the case return types. |
| template <typename REQUESTED_TYPE, typename... CASE_RETURN_TYPES> |
| using SwitchReturnType = typename SwitchReturnTypeImpl< |
| tint::IsCastable<NullptrToIgnore<std::remove_pointer_t<CASE_RETURN_TYPES>>...>, |
| REQUESTED_TYPE, |
| CASE_RETURN_TYPES...>::type; |
| |
| /// SwitchCaseReturnTypeImpl is the implementation of SwitchCaseReturnType |
| template <typename CASE, bool is_flag> |
| struct SwitchCaseReturnTypeImpl; |
| |
| /// SwitchCaseReturnTypeImpl specialization for non-flags. |
| template <typename CASE> |
| struct SwitchCaseReturnTypeImpl<CASE, /* is_flag */ false> { |
| /// The case function's return type. |
| using type = tint::traits::ReturnType<CASE>; |
| }; |
| |
| /// SwitchCaseReturnTypeImpl specialization for flags. |
| template <typename CASE> |
| struct SwitchCaseReturnTypeImpl<CASE, /* is_flag */ true> { |
| /// These are not functions, they have no return type. |
| using type = tint::Ignore; |
| }; |
| |
| /// Resolves to the return type for a Switch() case. |
| /// If CASE is a flag like SwitchMustMatchCase, then resolves to tint::Ignore |
| template <typename CASE> |
| using SwitchCaseReturnType = typename SwitchCaseReturnTypeImpl< |
| CASE, |
| std::is_same_v<std::decay_t<CASE>, SwitchMustMatchCase>>::type; |
| |
| } // namespace tint::detail |
| |
| namespace tint { |
| |
| /// Switch is used to dispatch one of the provided callback case handler functions based on the type |
| /// of `object` and the parameter type of the case handlers. Switch will sequentially check the type |
| /// of `object` against each of the switch case handler functions, and will invoke the first case |
| /// handler function which has a parameter type that matches the object type. When a case handler is |
| /// matched, it will be called with the single argument of `object` cast to the case handler's |
| /// parameter type. Switch will invoke at most one case handler. Each of the case functions must |
| /// have the signature `R(T*)` or `R(const T*)`, where `T` is the type matched by that case and `R` |
| /// is the return type, consistent across all case handlers. |
| /// |
| /// An optional default case function with the signature `R(Default)` can be used as the last case. |
| /// This default case will be called if all previous cases failed to match. |
| /// |
| /// The last argument may be SwitchMustMatchCase, in which case the Switch will trigger an ICE if |
| /// none of the cases matched. SwitchMustMatchCase cannot be used with a default case. |
| /// |
| /// If `object` is nullptr and a default case is provided, then the default case will be called. If |
| /// `object` is nullptr and no default case is provided, then no cases will be called. |
| /// |
| /// Example: |
| /// ``` |
| /// Switch(object, |
| /// [&](TypeA*) { /* ... */ }, |
| /// [&](TypeB*) { /* ... */ }); |
| /// |
| /// Switch(object, |
| /// [&](TypeA*) { /* ... */ }, |
| /// [&](TypeB*) { /* ... */ }, |
| /// [&](Default) { /* Called if object is not TypeA or TypeB */ }); |
| /// |
| /// Switch(object, |
| /// [&](TypeA*) { /* ... */ }, |
| /// [&](TypeB*) { /* ... */ }, |
| /// SwitchMustMatchCase); /* ICE if object is not TypeA or TypeB */ |
| /// ``` |
| /// |
| /// @param object the object who's type is used to |
| /// @param args the switch cases followed by an optional TINT_ICE_ON_NO_MATCH |
| /// @return the value returned by the called case. If no cases matched, then the zero value for the |
| /// consistent case type. |
| template <typename RETURN_TYPE = tint::detail::Infer, typename T = CastableBase, typename... ARGS> |
| inline auto Switch(T* object, ARGS&&... args) { |
| TINT_BEGIN_DISABLE_WARNING(UNUSED_VALUE); |
| |
| using ArgsTuple = std::tuple<ARGS...>; |
| static constexpr int kMustMatchCaseIndex = |
| tint::detail::IndexOfSwitchMustMatchCase<ArgsTuple>(); |
| static constexpr bool kHasMustMatchCase = kMustMatchCaseIndex >= 0; |
| static constexpr int kDefaultIndex = tint::detail::IndexOfDefaultCase<ArgsTuple>(); |
| static constexpr bool kHasDefaultCase = kDefaultIndex >= 0; |
| using ReturnType = |
| tint::detail::SwitchReturnType<RETURN_TYPE, tint::detail::SwitchCaseReturnType<ARGS>...>; |
| static constexpr bool kHasReturnType = !std::is_same_v<ReturnType, void>; |
| |
| // Static assertions |
| static constexpr bool kDefaultIsOK = |
| kDefaultIndex == -1 || kDefaultIndex == static_cast<int>(sizeof...(ARGS) - 1); |
| static constexpr bool kMustMatchCaseIsOK = |
| kMustMatchCaseIndex == -1 || kMustMatchCaseIndex == static_cast<int>(sizeof...(ARGS) - 1); |
| static constexpr bool kReturnIsOK = |
| kHasDefaultCase || !kHasReturnType || std::is_constructible_v<ReturnType>; |
| static_assert(kDefaultIsOK, "Default case must be last in Switch()"); |
| static_assert(kMustMatchCaseIsOK, "SwitchMustMatchCase must be last argument in Switch()"); |
| static_assert(!kHasDefaultCase || !kHasMustMatchCase, |
| "SwitchMustMatchCase cannot be used with a Default case"); |
| static_assert(kReturnIsOK, |
| "Switch() requires either a Default case or a return type that is either void or " |
| "default-constructable"); |
| |
| if (!object) { // Object is nullptr, so no cases can match |
| if constexpr (kHasMustMatchCase) { |
| const SwitchMustMatchCase& info = (args, ...); |
| tint::InternalCompilerError(info.file, info.line) << "Switch() passed nullptr"; |
| if constexpr (kHasReturnType) { |
| return ReturnType{}; |
| } else { |
| return; |
| } |
| } else if constexpr (kHasDefaultCase) { |
| // Evaluate default case. |
| const auto& default_case = (args, ...); |
| return static_cast<ReturnType>(default_case(Default{})); |
| } else { |
| // No default case, no case can match. |
| if constexpr (kHasReturnType) { |
| return ReturnType{}; |
| } else { |
| return; |
| } |
| } |
| } |
| |
| AlignedStorage<std::conditional_t<kHasReturnType, ReturnType, uint8_t>> return_storage; |
| auto* result = &return_storage.Get(); |
| |
| const tint::TypeInfo& type_info = object->TypeInfo(); |
| |
| // Examines the parameter type of the case function. |
| // If the parameter is a pointer type that `object` is of, or derives from, then that case |
| // function is called with `object` cast to that type, and `try_case` returns true. |
| // If the parameter is of type `Default`, then that case function is called and `try_case` |
| // returns true. |
| // Otherwise `try_case` returns false. |
| // If the case function is called and it returns a value, then this is copy constructed to the |
| // `result` pointer. |
| auto try_case = [&](auto&& case_fn) { |
| using CaseFunc = std::decay_t<decltype(case_fn)>; |
| bool success = false; |
| if constexpr (std::is_same_v<CaseFunc, SwitchMustMatchCase>) { |
| tint::InternalCompilerError(case_fn.file, case_fn.line) |
| << "Switch() matched no cases. Type: " << type_info.name; |
| } else { |
| using CaseType = tint::detail::SwitchCaseType<CaseFunc>; |
| if constexpr (std::is_same_v<CaseType, Default>) { |
| if constexpr (kHasReturnType) { |
| new (result) ReturnType(static_cast<ReturnType>(case_fn(Default{}))); |
| } else { |
| case_fn(Default{}); |
| } |
| success = true; |
| } else { |
| if (type_info.Is<CaseType>()) { |
| auto* v = static_cast<CaseType*>(object); |
| if constexpr (kHasReturnType) { |
| new (result) ReturnType(static_cast<ReturnType>(case_fn(v))); |
| } else { |
| case_fn(v); |
| } |
| success = true; |
| } |
| } |
| } |
| return success; |
| }; |
| |
| // Use a logical-or fold expression to try each of the cases in turn, until one matches the |
| // object type or a Default is reached. `handled` is true if a case function was called. |
| bool handled = ((try_case(std::forward<ARGS>(args)) || ...)); |
| |
| if constexpr (kHasReturnType) { |
| if constexpr (kHasDefaultCase) { |
| // Default case means there must be a returned value. |
| // No need to check handled, no requirement for a zero-initializer of ReturnType. |
| TINT_DEFER(result->~ReturnType()); |
| return *result; |
| } else { |
| if (handled) { |
| TINT_DEFER(result->~ReturnType()); |
| return *result; |
| } |
| return ReturnType{}; |
| } |
| } |
| |
| TINT_END_DISABLE_WARNING(UNUSED_VALUE); |
| } |
| |
| } // namespace tint |
| |
| #endif // SRC_TINT_UTILS_RTTI_SWITCH_H_ |