210 lines
		
	
	
		
			6.1 KiB
		
	
	
	
		
			C++
		
	
	
	
		
		
			
		
	
	
			210 lines
		
	
	
		
			6.1 KiB
		
	
	
	
		
			C++
		
	
	
	
|  | //-----------------------------------------------------------------------------
 | ||
|  | // boost variant/recursive_variant.hpp header file
 | ||
|  | // See http://www.boost.org for updates, documentation, and revision history.
 | ||
|  | //-----------------------------------------------------------------------------
 | ||
|  | //
 | ||
|  | // Copyright (c) 2003 Eric Friedman
 | ||
|  | // Copyright (c) 2013 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_RECURSIVE_VARIANT_HPP
 | ||
|  | #define BOOST_VARIANT_RECURSIVE_VARIANT_HPP
 | ||
|  | 
 | ||
|  | #include <boost/variant/variant_fwd.hpp>
 | ||
|  | #include <boost/variant/detail/enable_recursive.hpp>
 | ||
|  | #include <boost/variant/detail/substitute_fwd.hpp>
 | ||
|  | #include <boost/variant/detail/make_variant_list.hpp>
 | ||
|  | #include <boost/variant/detail/over_sequence.hpp>
 | ||
|  | 
 | ||
|  | #include <boost/mpl/aux_/lambda_arity_param.hpp>
 | ||
|  | 
 | ||
|  | #include <boost/mpl/equal.hpp>
 | ||
|  | #include <boost/mpl/eval_if.hpp>
 | ||
|  | #include <boost/mpl/identity.hpp>
 | ||
|  | #include <boost/mpl/if.hpp>
 | ||
|  | #include <boost/mpl/protect.hpp>
 | ||
|  | #include <boost/mpl/transform.hpp>
 | ||
|  | #include <boost/type_traits/is_same.hpp>
 | ||
|  | #include <boost/preprocessor/cat.hpp>
 | ||
|  | #include <boost/preprocessor/repeat.hpp>
 | ||
|  | 
 | ||
|  | #include <boost/mpl/bool.hpp>
 | ||
|  | #include <boost/mpl/is_sequence.hpp>
 | ||
|  | #include <boost/variant/variant.hpp>
 | ||
|  | 
 | ||
|  | namespace boost { | ||
|  | 
 | ||
|  | namespace detail { namespace variant { | ||
|  | 
 | ||
|  | ///////////////////////////////////////////////////////////////////////////////
 | ||
|  | // (detail) metafunction specialization substitute
 | ||
|  | //
 | ||
|  | // Handles embedded variant types when substituting for recursive_variant_.
 | ||
|  | //
 | ||
|  | 
 | ||
|  | #if !defined(BOOST_VARIANT_DETAIL_NO_SUBSTITUTE)
 | ||
|  | 
 | ||
|  | template < | ||
|  |       BOOST_VARIANT_ENUM_PARAMS(typename T) | ||
|  |     , typename RecursiveVariant | ||
|  |       BOOST_MPL_AUX_LAMBDA_ARITY_PARAM(typename Arity) | ||
|  |     > | ||
|  | struct substitute< | ||
|  |       ::boost::variant< | ||
|  |           recursive_flag< T0 > | ||
|  |         , BOOST_VARIANT_ENUM_SHIFTED_PARAMS(T) | ||
|  |         > | ||
|  |     , RecursiveVariant | ||
|  |     , ::boost::recursive_variant_ | ||
|  |       BOOST_MPL_AUX_LAMBDA_ARITY_PARAM(Arity) | ||
|  |     > | ||
|  | { | ||
|  |     typedef ::boost::variant< | ||
|  |           recursive_flag< T0 > | ||
|  |         , BOOST_VARIANT_ENUM_SHIFTED_PARAMS(T) | ||
|  |         > type; | ||
|  | }; | ||
|  | 
 | ||
|  | template < | ||
|  |       BOOST_VARIANT_ENUM_PARAMS(typename T) | ||
|  |     , typename RecursiveVariant | ||
|  |       BOOST_MPL_AUX_LAMBDA_ARITY_PARAM(typename Arity) | ||
|  |     > | ||
|  | struct substitute< | ||
|  |       ::boost::variant< | ||
|  |           ::boost::detail::variant::over_sequence< T0 > | ||
|  |         , BOOST_VARIANT_ENUM_SHIFTED_PARAMS(T) | ||
|  |         > | ||
|  |     , RecursiveVariant | ||
|  |     , ::boost::recursive_variant_ | ||
|  |       BOOST_MPL_AUX_LAMBDA_ARITY_PARAM(Arity) | ||
|  |     > | ||
|  | { | ||
|  | private: | ||
|  | 
 | ||
|  |     typedef T0 initial_types; | ||
|  | 
 | ||
|  |     typedef typename mpl::transform< | ||
|  |           initial_types | ||
|  |         , mpl::protect< quoted_enable_recursive<RecursiveVariant,mpl::true_> > | ||
|  |         >::type types; | ||
|  | 
 | ||
|  | public: | ||
|  | 
 | ||
|  |     typedef typename mpl::if_< | ||
|  |           mpl::equal<initial_types, types, ::boost::is_same<mpl::_1, mpl::_2> > | ||
|  |         , ::boost::variant< | ||
|  |               ::boost::detail::variant::over_sequence< T0 > | ||
|  |             , BOOST_VARIANT_ENUM_SHIFTED_PARAMS(T) | ||
|  |             > | ||
|  |         , ::boost::variant< over_sequence<types> > | ||
|  |         >::type type; | ||
|  | }; | ||
|  | 
 | ||
|  | template < | ||
|  |       BOOST_VARIANT_ENUM_PARAMS(typename T) | ||
|  |     , typename RecursiveVariant | ||
|  |       BOOST_MPL_AUX_LAMBDA_ARITY_PARAM(typename Arity) | ||
|  |     > | ||
|  | struct substitute< | ||
|  |       ::boost::variant< BOOST_VARIANT_ENUM_PARAMS(T) > | ||
|  |     , RecursiveVariant | ||
|  |     , ::boost::recursive_variant_ | ||
|  |       BOOST_MPL_AUX_LAMBDA_ARITY_PARAM(Arity) | ||
|  |     > | ||
|  | { | ||
|  | #if !defined(BOOST_VARIANT_DO_NOT_USE_VARIADIC_TEMPLATES)
 | ||
|  |      | ||
|  |     typedef ::boost::variant< | ||
|  |         typename enable_recursive<    | ||
|  |               T0               | ||
|  |             , RecursiveVariant                | ||
|  |             , mpl::true_                      | ||
|  |         >::type, | ||
|  |         typename enable_recursive<    | ||
|  |               TN               | ||
|  |             , RecursiveVariant                | ||
|  |             , mpl::true_                      | ||
|  |         >::type...   | ||
|  |     > type; | ||
|  | 
 | ||
|  | #else // defined(BOOST_VARIANT_DO_NOT_USE_VARIADIC_TEMPLATES)
 | ||
|  | 
 | ||
|  | private: // helpers, for metafunction result (below)
 | ||
|  | 
 | ||
|  |     #define BOOST_VARIANT_AUX_ENABLE_RECURSIVE_TYPEDEFS(z,N,_)  \
 | ||
|  |         typedef typename enable_recursive<   \ | ||
|  |               BOOST_PP_CAT(T,N)              \ | ||
|  |             , RecursiveVariant               \ | ||
|  |             , mpl::true_                     \ | ||
|  |             >::type BOOST_PP_CAT(wknd_T,N);  \ | ||
|  |         /**/ | ||
|  | 
 | ||
|  |     BOOST_PP_REPEAT( | ||
|  |           BOOST_VARIANT_LIMIT_TYPES | ||
|  |         , BOOST_VARIANT_AUX_ENABLE_RECURSIVE_TYPEDEFS | ||
|  |         , _ | ||
|  |         ) | ||
|  | 
 | ||
|  |     #undef BOOST_VARIANT_AUX_ENABLE_RECURSIVE_TYPEDEFS
 | ||
|  | 
 | ||
|  | public: // metafunction result
 | ||
|  | 
 | ||
|  |     typedef ::boost::variant< BOOST_VARIANT_ENUM_PARAMS(wknd_T) > type; | ||
|  | #endif // BOOST_VARIANT_DO_NOT_USE_VARIADIC_TEMPLATES workaround
 | ||
|  | }; | ||
|  | 
 | ||
|  | #else // defined(BOOST_VARIANT_DETAIL_NO_SUBSTITUTE)
 | ||
|  | 
 | ||
|  | //
 | ||
|  | // no specializations: embedded variants unsupported on these compilers!
 | ||
|  | //
 | ||
|  | 
 | ||
|  | #endif // !defined(BOOST_VARIANT_DETAIL_NO_SUBSTITUTE)
 | ||
|  | 
 | ||
|  | }} // namespace detail::variant
 | ||
|  | 
 | ||
|  | ///////////////////////////////////////////////////////////////////////////////
 | ||
|  | // metafunction make_recursive_variant
 | ||
|  | //
 | ||
|  | // See docs and boost/variant/variant_fwd.hpp for more information.
 | ||
|  | //
 | ||
|  | template < BOOST_VARIANT_ENUM_PARAMS(typename T) > | ||
|  | struct make_recursive_variant | ||
|  | { | ||
|  | public: // metafunction result
 | ||
|  | 
 | ||
|  |     typedef boost::variant< | ||
|  |           detail::variant::recursive_flag< T0 > | ||
|  |         , BOOST_VARIANT_ENUM_SHIFTED_PARAMS(T) | ||
|  |         > type; | ||
|  | 
 | ||
|  | }; | ||
|  | 
 | ||
|  | ///////////////////////////////////////////////////////////////////////////////
 | ||
|  | // metafunction make_recursive_variant_over
 | ||
|  | //
 | ||
|  | // See docs and boost/variant/variant_fwd.hpp for more information.
 | ||
|  | //
 | ||
|  | template <typename Types> | ||
|  | struct make_recursive_variant_over | ||
|  | { | ||
|  | private: // precondition assertions
 | ||
|  | 
 | ||
|  |     BOOST_STATIC_ASSERT(( ::boost::mpl::is_sequence<Types>::value )); | ||
|  | 
 | ||
|  | public: // metafunction result
 | ||
|  | 
 | ||
|  |     typedef typename make_recursive_variant< | ||
|  |           detail::variant::over_sequence< Types > | ||
|  |         >::type type; | ||
|  | 
 | ||
|  | }; | ||
|  | 
 | ||
|  | } // namespace boost
 | ||
|  | 
 | ||
|  | #endif // BOOST_VARIANT_RECURSIVE_VARIANT_HPP
 |