/* * Copyright 2014 Google Inc. All rights reserved. * * 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 FRUIT_META_METAPROGRAMMING_H #define FRUIT_META_METAPROGRAMMING_H #include <fruit/impl/meta/basics.h> #include <fruit/impl/meta/vector.h> #include <fruit/impl/fruit_assert.h> #include <fruit/impl/fruit_internal_forward_decls.h> #include <fruit/impl/injection_errors.h> #include <fruit/impl/meta/errors.h> #include <memory> namespace fruit { namespace impl { namespace meta { struct IsConstructible { template <typename C, typename... Args> struct apply; template <typename C, typename... Args> struct apply<Type<C>, Type<Args>...> { using type = Bool<std::is_constructible<C, Args...>::value>; }; }; struct IsConstructibleWithVector { template <typename C, typename V> struct apply; template <typename C, typename... Types> struct apply<Type<C>, Vector<Type<Types>...>> { using type = Bool<std::is_constructible<C, Types...>::value>; }; }; struct AddPointer { template <typename T> struct apply; template <typename T> struct apply<Type<T>> { using type = Type<T*>; }; }; struct IsCallable { template <typename T> struct apply; template <typename C> struct apply<Type<C>> { template <typename C1> static Bool<true> test(decltype(&C1::operator())); template <typename> static Bool<false> test(...); using type = decltype(test<C>(nullptr)); }; }; struct GetCallOperatorSignature { template <typename T> struct apply; template <typename C> struct apply<Type<C>> { using type = Type<decltype(&C::operator())>; }; }; struct AddPointerToVector { template <typename V> struct apply; template <typename... Ts> struct apply<Vector<Type<Ts>...>> { using type = Vector<Type<Ts*>...>; }; }; struct GetNthTypeHelper { template <typename N, typename... Ts> struct apply; template <typename T, typename... Ts> struct apply<Int<0>, T, Ts...> { using type = T; }; template <int n, typename T, typename... Ts> struct apply<Int<n>, T, Ts...> { using type = GetNthTypeHelper(Int<n - 1>, Ts...); }; }; struct GetNthType { template <typename N, typename V> struct apply; template <typename N, typename... Ts> struct apply<N, Vector<Ts...>> { using type = GetNthTypeHelper(N, Ts...); }; }; struct FunctorResultHelper { template <typename MethodSignature> struct apply; template <typename Result, typename Functor, typename... Args> struct apply<Type<Result (Functor::*)(Args...)>> { using type = Type<Result>; }; }; struct FunctorResult { template <typename F> struct apply; template <typename F> struct apply<Type<F>> { using type = FunctorResultHelper(Type<decltype(&F::operator())>); }; }; struct FunctionSignatureHelper { template <typename LambdaMethod> struct apply; template <typename Result, typename LambdaObject, typename... Args> struct apply<Type<Result (LambdaObject::*)(Args...) const>> { using type = Type<Result(Args...)>; }; }; // Function is either a plain function type of the form T(*)(Args...) or a lambda. struct FunctionSignature { template <typename Function> struct apply; template <typename Function> struct apply<Type<Function>> { using CandidateSignature = FunctionSignatureHelper(GetCallOperatorSignature(Type<Function>)); using type = If(Not(IsCallable(Type<Function>)), ConstructError(NotALambdaErrorTag, Type<Function>), If(Not(IsConstructible(AddPointer(CandidateSignature), Type<Function>)), ConstructError(FunctorUsedAsProviderErrorTag, Type<Function>), CandidateSignature)); }; template <typename Result, typename... Args> struct apply<Type<Result(Args...)>> { using type = Type<Result(Args...)>; }; template <typename Result, typename... Args> struct apply<Type<Result (*)(Args...)>> { using type = Type<Result(Args...)>; }; }; } // namespace meta } // namespace impl } // namespace fruit #endif // FRUIT_META_METAPROGRAMMING_H