//----------------------------------------------------------------------------- // boost variant/detail/apply_visitor_unary.hpp header file // See http://www.boost.org for updates, documentation, and revision history. //----------------------------------------------------------------------------- // // Copyright (c) 2002-2003 Eric Friedman // Copyright (c) 2014 Antony Polukhin // // 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 BOOST_VARIANT_DETAIL_APPLY_VISITOR_UNARY_HPP #define BOOST_VARIANT_DETAIL_APPLY_VISITOR_UNARY_HPP #include #include #include #include #if BOOST_WORKAROUND(__EDG__, BOOST_TESTED_AT(302)) #include #include #include #include #endif #if !defined(BOOST_NO_CXX14_DECLTYPE_AUTO) && !defined(BOOST_NO_CXX11_DECLTYPE_N3276) # include # include # include # include # include # include # include #endif namespace boost { ////////////////////////////////////////////////////////////////////////// // function template apply_visitor(visitor, visitable) // // Visits visitable with visitor. // // // nonconst-visitor version: // #if !BOOST_WORKAROUND(__EDG__, BOOST_TESTED_AT(302)) # define BOOST_VARIANT_AUX_APPLY_VISITOR_NON_CONST_RESULT_TYPE(V) \ BOOST_VARIANT_AUX_GENERIC_RESULT_TYPE(typename V::result_type) \ /**/ #else // EDG-based compilers # define BOOST_VARIANT_AUX_APPLY_VISITOR_NON_CONST_RESULT_TYPE(V) \ typename enable_if< \ mpl::not_< is_const< V > > \ , BOOST_VARIANT_AUX_GENERIC_RESULT_TYPE(typename V::result_type) \ >::type \ /**/ #endif // EDG-based compilers workaround #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES template inline BOOST_VARIANT_AUX_APPLY_VISITOR_NON_CONST_RESULT_TYPE(Visitor) apply_visitor(Visitor& visitor, Visitable&& visitable) { return ::boost::forward(visitable).apply_visitor(visitor); } #else template inline BOOST_VARIANT_AUX_APPLY_VISITOR_NON_CONST_RESULT_TYPE(Visitor) apply_visitor(Visitor& visitor, Visitable& visitable) { return visitable.apply_visitor(visitor); } #endif #undef BOOST_VARIANT_AUX_APPLY_VISITOR_NON_CONST_RESULT_TYPE // // const-visitor version: // #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES template inline BOOST_VARIANT_AUX_GENERIC_RESULT_TYPE(typename Visitor::result_type) apply_visitor(const Visitor& visitor, Visitable&& visitable) { return ::boost::forward(visitable).apply_visitor(visitor); } #else template inline BOOST_VARIANT_AUX_GENERIC_RESULT_TYPE(typename Visitor::result_type) apply_visitor(const Visitor& visitor, Visitable& visitable) { return visitable.apply_visitor(visitor); } #endif #if !defined(BOOST_NO_CXX14_DECLTYPE_AUTO) && !defined(BOOST_NO_CXX11_DECLTYPE_N3276) // C++14 namespace detail { namespace variant { // This class serves only metaprogramming purposes. none of its methods must be called at runtime! template struct result_multideduce1 { typedef typename Variant::types types; typedef typename boost::mpl::begin::type begin_it; typedef typename boost::mpl::advance< begin_it, boost::mpl::int_::type::value - 1> >::type last_it; // For metaprogramming purposes ONLY! Do not use this method (and class) at runtime! static Visitor& vis() BOOST_NOEXCEPT { // Functions that work with lambdas must be defined in same translation unit. // Because of that, we can not use `boost::decval()` here. Visitor&(*f)() = 0; // pointer to function return f(); } static decltype(auto) deduce_impl(last_it, unsigned /*helper*/) { typedef typename boost::mpl::deref::type value_t; return vis()( boost::declval< value_t& >() ); } template static decltype(auto) deduce_impl(It, unsigned helper) { typedef typename boost::mpl::next::type next_t; typedef typename boost::mpl::deref::type value_t; if (helper == boost::mpl::distance::type::value) { return deduce_impl(next_t(), ++helper); } return vis()( boost::declval< value_t& >() ); } static decltype(auto) deduce() { return deduce_impl(begin_it(), 0); } }; template struct result_wrapper1 { typedef decltype(result_multideduce1::deduce()) result_type; Visitor&& visitor_; explicit result_wrapper1(Visitor&& visitor) BOOST_NOEXCEPT : visitor_(::boost::forward(visitor)) {} template result_type operator()(T&& val) const { return visitor_(::boost::forward(val)); } }; }} // namespace detail::variant template inline decltype(auto) apply_visitor(Visitor&& visitor, Visitable&& visitable, typename boost::disable_if< boost::detail::variant::has_result_type >::type* = 0) { boost::detail::variant::result_wrapper1::type> cpp14_vis(::boost::forward(visitor)); return ::boost::forward(visitable).apply_visitor(cpp14_vis); } #endif // !defined(BOOST_NO_CXX14_DECLTYPE_AUTO) && !defined(BOOST_NO_CXX11_DECLTYPE_N3276) } // namespace boost #endif // BOOST_VARIANT_DETAIL_APPLY_VISITOR_UNARY_HPP