/*============================================================================= Copyright (c) 2014 Paul Fultz II fix.h Distributed under the Boost Software License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt) ==============================================================================*/ #ifndef FIT_GUARD_FUNCTION_FIX_H #define FIT_GUARD_FUNCTION_FIX_H /// fix /// === /// /// Description /// ----------- /// /// The `fix` function adaptor implements a fixed-point combinator. This can be /// used to write recursive functions. /// /// When using `constexpr`, a function can recurse to a depth that is defined by /// `FIT_RECURSIVE_CONSTEXPR_DEPTH`(default is 16). There is no limitiation on /// recursion depth for non-constexpr functions. In addition, due to the /// eagerness of `constexpr` to instantiation templates, in some cases, an /// explicit return type must be specified in order to avoid reaching the /// recursion limits of the compiler. This can be accomplished using /// [`fit::result`](result.md): /// /// int r = fit::result(factorial)(5); /// /// Synopsis /// -------- /// /// template /// constexpr fix_adaptor fix(F f); /// /// Semantics /// --------- /// /// assert(fix(f)(xs...) == f(fix(f), xs...)); /// /// Requirements /// ------------ /// /// F must be: /// /// * [FunctionObject](concepts.md#functionobject) /// * MoveConstructible /// /// Example /// ------- /// /// auto factorial = fit::fix( /// [](auto recurse, auto x) -> decltype(x) { /// return x == 0 ? 1 : x * recurse(x-1); /// } /// ); /// int r = fit::result(factorial)(5); /// assert(r == 5*4*3*2*1); /// #include #include #include #include #include #include #include #include #include #include namespace fit { namespace detail{ template struct compute_indirect_ref { typedef indirect_adaptor type; }; template struct compute_indirect_ref> { typedef indirect_adaptor type; }; template constexpr indirect_adaptor make_indirect_ref(const F& f) { return indirect_adaptor(&f); } template constexpr indirect_adaptor make_indirect_ref(const indirect_adaptor& f) { return f; } template struct fix_result { template struct apply { typedef decltype(std::declval()(std::declval()...)) type; }; }; template struct fix_result::type> { template struct apply { typedef typename F::result_type type; }; }; template struct fix_adaptor_base : F { FIT_INHERIT_CONSTRUCTOR(fix_adaptor_base, F); typedef typename compute_indirect_ref::type base_ref_type; typedef fix_adaptor_base derived; template constexpr const F& base_function(Ts&&... xs) const { return always_ref(*this)(xs...); } template constexpr derived derived_function(Ts&&... xs) const { return derived(make_indirect_ref(this->base_function(xs...))); } struct fix_failure { template struct apply { template struct of : Failure::template of {}; }; }; struct failure : failure_map {}; FIT_RETURNS_CLASS(fix_adaptor_base); template constexpr FIT_SFINAE_RESULT(const F&, id_, id_...) operator()(Ts&&... xs) const FIT_SFINAE_RETURNS ( FIT_MANGLE_CAST(const F&)(FIT_CONST_THIS->base_function(xs...)) ( FIT_MANGLE_CAST(derived)(FIT_CONST_THIS->derived_function(xs...)), FIT_FORWARD(Ts)(xs)... ) ); }; template struct fix_adaptor_base : F { FIT_INHERIT_CONSTRUCTOR(fix_adaptor_base, F); template const F& base_function(Ts&&...) const { return *this; } struct fix_failure { template struct apply { template struct of : Failure::template of {}; }; }; struct failure : failure_map {}; FIT_RETURNS_CLASS(fix_adaptor_base); template typename Result::template apply::type operator()(Ts&&... xs) const { return this->base_function(xs...)(*this, FIT_FORWARD(Ts)(xs)...); } }; } template struct fix_adaptor : detail::fix_adaptor_base, FIT_RECURSIVE_CONSTEXPR_DEPTH> { typedef fix_adaptor fit_rewritable1_tag; typedef detail::fix_adaptor_base, FIT_RECURSIVE_CONSTEXPR_DEPTH> base; FIT_INHERIT_CONSTRUCTOR(fix_adaptor, base); }; template struct result_adaptor> : fix_adaptor> { typedef fix_adaptor> base; FIT_INHERIT_CONSTRUCTOR(result_adaptor, base) }; FIT_DECLARE_STATIC_VAR(fix, detail::make); } // namespace fit #endif