// Copyright 2020 The Tint Authors. // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. #ifndef SRC_TINT_TRAITS_H_ #define SRC_TINT_TRAITS_H_ #include #include #include #include namespace tint::traits { /// Convience type definition for std::decay::type template using Decay = typename std::decay::type; /// NthTypeOf returns the `N`th type in `Types` template using NthTypeOf = typename std::tuple_element>::type; /// Signature describes the signature of a function. template struct Signature { /// The return type of the function signature using ret = RETURN; /// The parameters of the function signature held in a std::tuple using parameters = std::tuple; /// The type of the Nth parameter of function signature template using parameter = NthTypeOf; /// The total number of parameters static constexpr std::size_t parameter_count = sizeof...(PARAMETERS); }; /// SignatureOf is a traits helper that infers the signature of the function, /// method, static method, lambda, or function-like object `F`. template struct SignatureOf { /// The signature of the function-like object `F` using type = typename SignatureOf::type; }; /// SignatureOf specialization for a regular function or static method. template struct SignatureOf { /// The signature of the function-like object `F` using type = Signature::type, typename std::decay::type...>; }; /// SignatureOf specialization for a non-static method. template struct SignatureOf { /// The signature of the function-like object `F` using type = Signature::type, typename std::decay::type...>; }; /// SignatureOf specialization for a non-static, const method. template struct SignatureOf { /// The signature of the function-like object `F` using type = Signature::type, typename std::decay::type...>; }; /// SignatureOfT is an alias to `typename SignatureOf::type`. template using SignatureOfT = typename SignatureOf>::type; /// ParameterType is an alias to `typename SignatureOf::type::parameter`. template using ParameterType = typename SignatureOfT>::template parameter; /// LastParameterType returns the type of the last parameter of `F`. `F` must have at least one /// parameter. template using LastParameterType = ParameterType>::parameter_count - 1>; /// ReturnType is an alias to `typename SignatureOf::type::ret`. template using ReturnType = typename SignatureOfT>::ret; /// Returns true iff decayed T and decayed U are the same. template static constexpr bool IsType = std::is_same, Decay>::value; /// IsTypeOrDerived is true iff `T` is of type `BASE`, or derives from /// `BASE`. template static constexpr bool IsTypeOrDerived = std::is_base_of>::value || std::is_same>::value; /// If `CONDITION` is true then EnableIf resolves to type T, otherwise an /// invalid type. template using EnableIf = typename std::enable_if::type; /// If `T` is of type `BASE`, or derives from `BASE`, then EnableIfIsType /// resolves to type `T`, otherwise an invalid type. template using EnableIfIsType = EnableIf, T>; /// If `T` is not of type `BASE`, or does not derive from `BASE`, then /// EnableIfIsNotType resolves to type `T`, otherwise an invalid type. template using EnableIfIsNotType = EnableIf, T>; /// @returns the std::index_sequence with all the indices shifted by OFFSET. template constexpr auto Shift(std::index_sequence) { return std::integer_sequence{}; } /// @returns a std::integer_sequence with the integers `[OFFSET..OFFSET+COUNT)` template constexpr auto Range() { return Shift(std::make_index_sequence{}); } namespace detail { /// @returns the tuple `t` swizzled by `INDICES` template constexpr auto Swizzle(TUPLE&& t, std::index_sequence) -> std::tuple>...> { return {std::forward>>( std::get(std::forward(t)))...}; } /// @returns a nullptr of the tuple type `TUPLE` swizzled by `INDICES`. /// @note: This function is intended to be used in a `decltype()` expression, /// and returns a pointer-to-tuple as the tuple may hold non-constructable /// types. template constexpr auto* SwizzlePtrTy(std::index_sequence) { using Swizzled = std::tuple...>; return static_cast(nullptr); } } // namespace detail /// @returns the slice of the tuple `t` with the tuple elements /// `[OFFSET..OFFSET+COUNT)` template constexpr auto Slice(TUPLE&& t) { return detail::Swizzle(std::forward(t), Range()); } /// Resolves to the slice of the tuple `t` with the tuple elements /// `[OFFSET..OFFSET+COUNT)` template using SliceTuple = std::remove_pointer_t(Range()))>; namespace detail { /// Base template for IsTypeIn template struct IsTypeIn; /// Specialization for IsTypeIn template class TypeContainer, class... Ts> struct IsTypeIn> : std::disjunction...> {}; } // namespace detail /// Evaluates to true if T is one of the types in the TypeContainer's template arguments. /// Works for std::variant, std::tuple, std::pair, or any class template where all parameters are /// types. template static constexpr bool IsTypeIn = detail::IsTypeIn::value; /// Evaluates to the decayed pointer element type, or the decayed type T if T is not a pointer. template using PtrElTy = Decay>>; /// Evaluates to true if `T` decayed is a `std::string`, `std::string_view` or `const char*` template static constexpr bool IsStringLike = std::is_same_v, std::string> || std::is_same_v, std::string_view> || std::is_same_v, const char*>; } // namespace tint::traits #endif // SRC_TINT_TRAITS_H_