/* * 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_FOLD_H #define FRUIT_META_FOLD_H #include <functional> namespace fruit { namespace impl { namespace meta { struct Fold { template <typename F, typename InitialValue, typename... Types> struct apply; template <typename F, typename InitialValue> struct apply<F, InitialValue> { using type = InitialValue; }; template <typename F, typename InitialValue, typename T0> struct apply<F, InitialValue, T0> { using type = typename F::template apply<InitialValue, T0>::type; }; template <typename F, typename InitialValue, typename T0, typename T1> struct apply<F, InitialValue, T0, T1> { using type = typename F::template apply<typename DoEval<typename F::template apply<InitialValue, T0>::type>::type, T1>::type; }; template <typename F, typename InitialValue, typename T0, typename T1, typename T2> struct apply<F, InitialValue, T0, T1, T2> { using type = typename F::template apply< typename DoEval<typename F::template apply< typename DoEval<typename F::template apply<InitialValue, T0>::type>::type, T1>::type>::type, T2>::type; }; template <typename F, typename InitialValue, typename T0, typename T1, typename T2, typename T3> struct apply<F, InitialValue, T0, T1, T2, T3> { using type = typename F::template apply< typename DoEval<typename F::template apply< typename DoEval<typename F::template apply< typename DoEval<typename F::template apply<InitialValue, T0>::type>::type, T1>::type>::type, T2>::type>::type, T3>::type; }; template <typename F, typename InitialValue, typename T0, typename T1, typename T2, typename T3, typename T4> struct apply<F, InitialValue, T0, T1, T2, T3, T4> { using type = typename F::template apply< typename DoEval<typename F::template apply< typename DoEval<typename F::template apply< typename DoEval<typename F::template apply< typename DoEval<typename F::template apply<InitialValue, T0>::type>::type, T1>::type>::type, T2>::type>::type, T3>::type>::type, T4>::type; }; template <typename F, typename InitialValue, typename T0, typename T1, typename T2, typename T3, typename T4, typename... Types> struct apply<F, InitialValue, T0, T1, T2, T3, T4, Types...> { using type = Fold( F, typename F::template apply< typename DoEval<typename F::template apply< typename DoEval<typename F::template apply< typename DoEval<typename F::template apply< typename DoEval<typename F::template apply<InitialValue, T0>::type>::type, T1>::type>::type, T2>::type>::type, T3>::type>::type, T4>::type, Types...); }; // Optimized specialization, processing 10 values at a time. template <typename F, typename InitialValue, typename T0, typename T1, typename T2, typename T3, typename T4, typename T5, typename T6, typename T7, typename T8, typename T9, typename... Types> struct apply<F, InitialValue, T0, T1, T2, T3, T4, T5, T6, T7, T8, T9, Types...> { using type = Fold( F, typename F::template apply< typename DoEval<typename F::template apply< typename DoEval<typename F::template apply< typename DoEval<typename F::template apply< typename DoEval<typename F::template apply< typename DoEval<typename F::template apply< typename DoEval<typename F::template apply< typename DoEval<typename F::template apply< typename DoEval<typename F::template apply< typename DoEval<typename F::template apply<InitialValue, T0>::type>::type, T1>::type>::type, T2>::type>::type, T3>::type>::type, T4>::type>::type, T5>::type>::type, T6>::type>::type, T7>::type>::type, T8>::type>::type, T9>::type, Types...); }; }; } // namespace meta } // namespace impl } // namespace fruit #endif // FRUIT_META_FOLD_H