/*============================================================================= Copyright (c) 2014 Paul Fultz II function.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_FUNCTION_H #define FIT_GUARD_FUNCTION_FUNCTION_H /// FIT_STATIC_FUNCTION /// =================== /// /// Description /// ----------- /// /// The `FIT_STATIC_FUNCTION` macro allows initializing a function object from /// a `constexpr` expression. It also ensures that the function object will /// have a unique address across translation units. This helps to avoid ODR /// violations. As such, the object that is deduced is default constructed. /// /// Example /// ------- /// /// struct sum_f /// { /// template /// T operator()(T x, U y) const /// { /// return x+y; /// } /// }; /// /// FIT_STATIC_FUNCTION(sum) = sum_f(); /// FIT_STATIC_FUNCTION(partial_sum) = fit::partial(sum_f()); /// assert(sum(1, 2) == partial_sum(1)(2)); /// #include #include #include #include namespace fit { namespace detail { struct reveal_static_const_factory { constexpr reveal_static_const_factory() {} #if FIT_NO_UNIQUE_STATIC_VAR template constexpr reveal_adaptor operator=(const F& f) const { static_assert(is_default_constructible::value, "Static functions must be default constructible"); return reveal_adaptor(f); } #else template constexpr const reveal_adaptor& operator=(const F&) const { static_assert(is_default_constructible::value, "Static functions must be default constructible"); return static_const_var>(); } #endif }; }} // namespace fit #if FIT_NO_UNIQUE_STATIC_VAR #define FIT_STATIC_FUNCTION(name) FIT_STATIC_CONSTEXPR auto name = FIT_DETAIL_MSVC_CONSTEXPR_DEDUCE fit::detail::reveal_static_const_factory() #else #define FIT_STATIC_FUNCTION(name) FIT_STATIC_AUTO_REF name = fit::detail::reveal_static_const_factory() #endif #endif