/* * 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 #include #include #include #include #include #include namespace fruit { namespace impl { namespace meta { struct IsConstructible { template struct apply; template struct apply, Type...> { using type = Bool::value>; }; }; struct IsConstructibleWithVector { template struct apply; template struct apply, Vector...>> { using type = Bool::value>; }; }; struct AddPointer { template struct apply; template struct apply> { using type = Type; }; }; struct IsCallable { template struct apply; template struct apply> { template static Bool test(decltype(&C1::operator())); template static Bool test(...); using type = decltype(test(nullptr)); }; }; struct GetCallOperatorSignature { template struct apply; template struct apply> { using type = Type; }; }; struct AddPointerToVector { template struct apply; template struct apply...>> { using type = Vector...>; }; }; struct GetNthTypeHelper { template struct apply; template struct apply, T, Ts...> { using type = T; }; template struct apply, T, Ts...> { using type = GetNthTypeHelper(Int, Ts...); }; }; struct GetNthType { template struct apply; template struct apply> { using type = GetNthTypeHelper(N, Ts...); }; }; struct FunctorResultHelper { template struct apply; template struct apply> { using type = Type; }; }; struct FunctorResult { template struct apply; template struct apply> { using type = FunctorResultHelper(Type); }; }; struct FunctionSignatureHelper { template struct apply; template struct apply> { using type = Type; }; }; // Function is either a plain function type of the form T(*)(Args...) or a lambda. struct FunctionSignature { template struct apply; template struct apply> { using CandidateSignature = FunctionSignatureHelper(GetCallOperatorSignature(Type)); using type = If(Not(IsCallable(Type)), ConstructError(NotALambdaErrorTag, Type), If(Not(IsConstructible(AddPointer(CandidateSignature), Type)), ConstructError(FunctorUsedAsProviderErrorTag, Type), CandidateSignature)); }; template struct apply> { using type = Type; }; template struct apply> { using type = Type; }; }; } // namespace meta } // namespace impl } // namespace fruit #endif // FRUIT_META_METAPROGRAMMING_H