// 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 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; /// ReturnType is an alias to `typename SignatureOf::type::ret`. template using ReturnType = typename SignatureOfT::ret; /// 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< std::tuple_element_t>...> { return {std::forward< std::tuple_element_t>>( 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 tint::traits #endif // SRC_TINT_TRAITS_H_