437 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
	
	
			
		
		
	
	
			437 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
	
	
| // Copyright Daniel Wallin, David Abrahams 2005. Use, modification and
 | |
| // distribution is subject to 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 ARG_LIST_050329_HPP
 | |
| #define ARG_LIST_050329_HPP
 | |
| 
 | |
| #include <boost/parameter/aux_/void.hpp>
 | |
| #include <boost/parameter/aux_/result_of0.hpp>
 | |
| #include <boost/parameter/aux_/default.hpp>
 | |
| #include <boost/parameter/aux_/parameter_requirements.hpp>
 | |
| #include <boost/parameter/aux_/yesno.hpp>
 | |
| #include <boost/parameter/aux_/is_maybe.hpp>
 | |
| #include <boost/parameter/config.hpp>
 | |
| 
 | |
| #include <boost/mpl/apply.hpp>
 | |
| #include <boost/mpl/assert.hpp>
 | |
| #include <boost/mpl/begin.hpp>
 | |
| #include <boost/mpl/end.hpp>
 | |
| #include <boost/mpl/iterator_tags.hpp>
 | |
| 
 | |
| #include <boost/type_traits/add_reference.hpp>
 | |
| #include <boost/type_traits/is_same.hpp>
 | |
| #include <boost/preprocessor/repetition/enum_params.hpp>
 | |
| #include <boost/preprocessor/repetition/enum_binary_params.hpp>
 | |
| #include <boost/preprocessor/facilities/intercept.hpp>
 | |
| 
 | |
| namespace boost { namespace parameter {
 | |
| 
 | |
| // Forward declaration for aux::arg_list, below.
 | |
| template<class T> struct keyword;
 | |
| 
 | |
| namespace aux {
 | |
| 
 | |
| // Tag type passed to MPL lambda.
 | |
| struct lambda_tag;
 | |
| 
 | |
| //
 | |
| // Structures used to build the tuple of actual arguments.  The
 | |
| // tuple is a nested cons-style list of arg_list specializations
 | |
| // terminated by an empty_arg_list.
 | |
| //
 | |
| // Each specialization of arg_list is derived from its successor in
 | |
| // the list type.  This feature is used along with using
 | |
| // declarations to build member function overload sets that can
 | |
| // match against keywords.
 | |
| //
 | |
| 
 | |
| // MPL sequence support
 | |
| struct arg_list_tag;
 | |
| 
 | |
| // Terminates arg_list<> and represents an empty list.  Since this
 | |
| // is just the terminating case you might want to look at arg_list
 | |
| // first, to get a feel for what's really happening here.
 | |
| 
 | |
| struct empty_arg_list
 | |
| {
 | |
|     empty_arg_list() {}
 | |
| 
 | |
|     // Constructor taking BOOST_PARAMETER_MAX_ARITY empty_arg_list
 | |
|     // arguments; this makes initialization
 | |
|     empty_arg_list(
 | |
|         BOOST_PP_ENUM_PARAMS(
 | |
|             BOOST_PARAMETER_MAX_ARITY, void_ BOOST_PP_INTERCEPT
 | |
|         ))
 | |
|     {}
 | |
| 
 | |
|     // A metafunction class that, given a keyword and a default
 | |
|     // type, returns the appropriate result type for a keyword
 | |
|     // lookup given that default
 | |
|     struct binding
 | |
|     {
 | |
|         template<class KW, class Default, class Reference>
 | |
|         struct apply
 | |
|         {
 | |
|             typedef Default type;
 | |
|         };
 | |
|     };
 | |
| 
 | |
|     // Terminator for has_key, indicating that the keyword is unique
 | |
|     template <class KW>
 | |
|     static no_tag has_key(KW*);
 | |
| 
 | |
| #if BOOST_WORKAROUND(__BORLANDC__, BOOST_TESTED_AT(0x564))
 | |
| 
 | |
|     // The overload set technique doesn't work with these older
 | |
|     // compilers, so they need some explicit handholding.
 | |
| 
 | |
|     // A metafunction class that, given a keyword, returns the type
 | |
|     // of the base sublist whose get() function can produce the
 | |
|     // value for that key
 | |
|     struct key_owner
 | |
|     {
 | |
|         template<class KW>
 | |
|         struct apply
 | |
|         {
 | |
|             typedef empty_arg_list type;
 | |
|         };
 | |
|     };
 | |
| 
 | |
|     template <class K, class T>
 | |
|     T& get(default_<K,T> x) const
 | |
|     {
 | |
|         return x.value;
 | |
|     }
 | |
| 
 | |
|     template <class K, class F>
 | |
|     typename result_of0<F>::type
 | |
|     get(lazy_default<K,F> x) const
 | |
|     {
 | |
|         return x.compute_default();
 | |
|     }
 | |
| #endif
 | |
| 
 | |
|     // If this function is called, it means there is no argument
 | |
|     // in the list that matches the supplied keyword. Just return
 | |
|     // the default value.
 | |
|     template <class K, class Default>
 | |
|     Default& operator[](default_<K, Default> x) const
 | |
|     {
 | |
|         return x.value;
 | |
|     }
 | |
| 
 | |
|     // If this function is called, it means there is no argument
 | |
|     // in the list that matches the supplied keyword. Just evaluate
 | |
|     // and return the default value.
 | |
|     template <class K, class F>
 | |
|     typename result_of0<F>::type
 | |
|     operator[](
 | |
|         BOOST_PARAMETER_lazy_default_fallback<K,F> x) const
 | |
|     {
 | |
|         return x.compute_default();
 | |
|     }
 | |
| 
 | |
|     // No argument corresponding to ParameterRequirements::key_type
 | |
|     // was found if we match this overload, so unless that parameter
 | |
|     // has a default, we indicate that the actual arguments don't
 | |
|     // match the function's requirements.
 | |
|     template <class ParameterRequirements, class ArgPack>
 | |
|     static typename ParameterRequirements::has_default
 | |
|     satisfies(ParameterRequirements*, ArgPack*);
 | |
| 
 | |
|     // MPL sequence support
 | |
|     typedef empty_arg_list type;   // convenience
 | |
|     typedef arg_list_tag tag; // For dispatching to sequence intrinsics
 | |
| };
 | |
| 
 | |
| // Forward declaration for arg_list::operator,
 | |
| template <class KW, class T>
 | |
| struct tagged_argument;
 | |
| 
 | |
| template <class T>
 | |
| struct get_reference
 | |
| {
 | |
|     typedef typename T::reference type;
 | |
| };
 | |
| 
 | |
| // A tuple of tagged arguments, terminated with empty_arg_list.
 | |
| // Every TaggedArg is an instance of tagged_argument<>.
 | |
| template <class TaggedArg, class Next = empty_arg_list>
 | |
| struct arg_list : Next
 | |
| {
 | |
|     typedef arg_list<TaggedArg,Next> self;
 | |
|     typedef typename TaggedArg::key_type key_type;
 | |
| 
 | |
|     typedef typename is_maybe<typename TaggedArg::value_type>::type holds_maybe;
 | |
| 
 | |
|     typedef typename mpl::eval_if<
 | |
|         holds_maybe
 | |
|       , get_reference<typename TaggedArg::value_type>
 | |
|       , get_reference<TaggedArg>
 | |
|     >::type reference;
 | |
| 
 | |
|     typedef typename mpl::if_<
 | |
|         holds_maybe
 | |
|       , reference
 | |
|       , typename TaggedArg::value_type
 | |
|     >::type value_type;
 | |
| 
 | |
|     TaggedArg arg;      // Stores the argument
 | |
| 
 | |
|     // Store the arguments in successive nodes of this list
 | |
|     template< // class A0, class A1, ...
 | |
|         BOOST_PP_ENUM_PARAMS(BOOST_PARAMETER_MAX_ARITY, class A)
 | |
|     >
 | |
|     arg_list( // A0& a0, A1& a1, ...
 | |
|         BOOST_PP_ENUM_BINARY_PARAMS(BOOST_PARAMETER_MAX_ARITY, A, & a)
 | |
|     )
 | |
|       : Next( // a1, a2, ...
 | |
|             BOOST_PP_ENUM_SHIFTED_PARAMS(BOOST_PARAMETER_MAX_ARITY, a)
 | |
|           , void_reference()
 | |
|         )
 | |
|       , arg(a0)
 | |
|     {}
 | |
| 
 | |
|     // Create a new list by prepending arg to a copy of tail.  Used
 | |
|     // when incrementally building this structure with the comma
 | |
|     // operator.
 | |
|     arg_list(TaggedArg head, Next const& tail)
 | |
|       : Next(tail)
 | |
|       , arg(head)
 | |
|     {}
 | |
| 
 | |
|     // A metafunction class that, given a keyword and a default
 | |
|     // type, returns the appropriate result type for a keyword
 | |
|     // lookup given that default
 | |
|     struct binding
 | |
|     {
 | |
|         template <class KW, class Default, class Reference>
 | |
|         struct apply
 | |
|         {
 | |
|           typedef typename mpl::eval_if<
 | |
|                 boost::is_same<KW, key_type>
 | |
|               , mpl::if_<Reference, reference, value_type>
 | |
|               , mpl::apply_wrap3<typename Next::binding, KW, Default, Reference>
 | |
|           >::type type;
 | |
|         };
 | |
|     };
 | |
| 
 | |
| #if !BOOST_WORKAROUND(__BORLANDC__, BOOST_TESTED_AT(0x564))
 | |
|     // Overload for key_type, so the assert below will fire if the
 | |
|     // same keyword is used again
 | |
|     static yes_tag has_key(key_type*);
 | |
|     using Next::has_key;
 | |
| 
 | |
|     BOOST_MPL_ASSERT_MSG(
 | |
|         sizeof(Next::has_key((key_type*)0)) == sizeof(no_tag)
 | |
|       , duplicate_keyword, (key_type)
 | |
|     );
 | |
| 
 | |
| #endif
 | |
|     //
 | |
|     // Begin implementation of indexing operators for looking up
 | |
|     // specific arguments by name
 | |
|     //
 | |
| 
 | |
|     // Helpers that handle the case when TaggedArg is
 | |
|     // empty<T>.
 | |
|     template <class D>
 | |
|     reference get_default(D const&, mpl::false_) const
 | |
|     {
 | |
|         return arg.value;
 | |
|     }
 | |
| 
 | |
|     template <class D>
 | |
|     reference get_default(D const& d, mpl::true_) const
 | |
|     {
 | |
|         return arg.value ? arg.value.get() : arg.value.construct(d.value);
 | |
|     }
 | |
| 
 | |
| #if BOOST_WORKAROUND(__BORLANDC__, BOOST_TESTED_AT(0x564))
 | |
|     // These older compilers don't support the overload set creation
 | |
|     // idiom well, so we need to do all the return type calculation
 | |
|     // for the compiler and dispatch through an outer function template
 | |
| 
 | |
|     // A metafunction class that, given a keyword, returns the base
 | |
|     // sublist whose get() function can produce the value for that
 | |
|     // key.
 | |
|     struct key_owner
 | |
|     {
 | |
|         template<class KW>
 | |
|         struct apply
 | |
|         {
 | |
|           typedef typename mpl::eval_if<
 | |
|                 boost::is_same<KW, key_type>
 | |
|               , mpl::identity<arg_list<TaggedArg,Next> >
 | |
|               , mpl::apply_wrap1<typename Next::key_owner,KW>
 | |
|           >::type type;
 | |
|         };
 | |
|     };
 | |
| 
 | |
|     // Outer indexing operators that dispatch to the right node's
 | |
|     // get() function.
 | |
|     template <class KW>
 | |
|     typename mpl::apply_wrap3<binding, KW, void_, mpl::true_>::type
 | |
|     operator[](keyword<KW> const& x) const
 | |
|     {
 | |
|         typename mpl::apply_wrap1<key_owner, KW>::type const& sublist = *this;
 | |
|         return sublist.get(x);
 | |
|     }
 | |
| 
 | |
|     template <class KW, class Default>
 | |
|     typename mpl::apply_wrap3<binding, KW, Default&, mpl::true_>::type
 | |
|     operator[](default_<KW, Default> x) const
 | |
|     {
 | |
|         typename mpl::apply_wrap1<key_owner, KW>::type const& sublist = *this;
 | |
|         return sublist.get(x);
 | |
|     }
 | |
| 
 | |
|     template <class KW, class F>
 | |
|     typename mpl::apply_wrap3<
 | |
|         binding,KW
 | |
|       , typename result_of0<F>::type
 | |
|       , mpl::true_
 | |
|     >::type
 | |
|     operator[](lazy_default<KW,F> x) const
 | |
|     {
 | |
|         typename mpl::apply_wrap1<key_owner, KW>::type const& sublist = *this;
 | |
|         return sublist.get(x);
 | |
|     }
 | |
| 
 | |
|     // These just return the stored value; when empty_arg_list is
 | |
|     // reached, indicating no matching argument was passed, the
 | |
|     // default is returned, or if no default_ or lazy_default was
 | |
|     // passed, compilation fails.
 | |
|     reference get(keyword<key_type> const&) const
 | |
|     {
 | |
|         BOOST_MPL_ASSERT_NOT((holds_maybe));
 | |
|         return arg.value;
 | |
|     }
 | |
| 
 | |
|     template <class Default>
 | |
|     reference get(default_<key_type,Default> const& d) const
 | |
|     {
 | |
|         return get_default(d, holds_maybe());
 | |
|     }
 | |
| 
 | |
|     template <class Default>
 | |
|     reference get(lazy_default<key_type, Default>) const
 | |
|     {
 | |
|         return arg.value;
 | |
|     }
 | |
| 
 | |
| #else
 | |
| 
 | |
|     reference operator[](keyword<key_type> const&) const
 | |
|     {
 | |
|         BOOST_MPL_ASSERT_NOT((holds_maybe));
 | |
|         return arg.value;
 | |
|     }
 | |
| 
 | |
|     template <class Default>
 | |
|     reference operator[](default_<key_type, Default> const& d) const
 | |
|     {
 | |
|         return get_default(d, holds_maybe());
 | |
|     }
 | |
| 
 | |
|     template <class Default>
 | |
|     reference operator[](lazy_default<key_type, Default>) const
 | |
|     {
 | |
|         return arg.value;
 | |
|     }
 | |
| 
 | |
|     // Builds an overload set including operator[]s defined in base
 | |
|     // classes.
 | |
|     using Next::operator[];
 | |
| 
 | |
|     //
 | |
|     // End of indexing support
 | |
|     //
 | |
| 
 | |
| 
 | |
|     //
 | |
|     // For parameter_requirements matching this node's key_type,
 | |
|     // return a bool constant wrapper indicating whether the
 | |
|     // requirements are satisfied by TaggedArg.  Used only for
 | |
|     // compile-time computation and never really called, so a
 | |
|     // declaration is enough.
 | |
|     //
 | |
|     template <class HasDefault, class Predicate, class ArgPack>
 | |
|     static typename mpl::apply_wrap2<
 | |
|         typename mpl::lambda<Predicate, lambda_tag>::type
 | |
|       , value_type, ArgPack
 | |
|     >::type
 | |
|     satisfies(
 | |
|         parameter_requirements<key_type,Predicate,HasDefault>*
 | |
|       , ArgPack*
 | |
|     );
 | |
| 
 | |
|     // Builds an overload set including satisfies functions defined
 | |
|     // in base classes.
 | |
|     using Next::satisfies;
 | |
| #endif
 | |
| 
 | |
|     // Comma operator to compose argument list without using parameters<>.
 | |
|     // Useful for argument lists with undetermined length.
 | |
|     template <class KW, class T2>
 | |
|     arg_list<tagged_argument<KW, T2>, self>
 | |
|     operator,(tagged_argument<KW,T2> x) const
 | |
|     {
 | |
|         return arg_list<tagged_argument<KW,T2>, self>(x, *this);
 | |
|     }
 | |
| 
 | |
|     // MPL sequence support
 | |
|     typedef self type;             // Convenience for users
 | |
|     typedef Next tail_type;        // For the benefit of iterators
 | |
|     typedef arg_list_tag tag; // For dispatching to sequence intrinsics
 | |
| };
 | |
| 
 | |
| // MPL sequence support
 | |
| template <class ArgumentPack>
 | |
| struct arg_list_iterator
 | |
| {
 | |
|     typedef mpl::forward_iterator_tag category;
 | |
| 
 | |
|     // The incremented iterator
 | |
|     typedef arg_list_iterator<typename ArgumentPack::tail_type> next;
 | |
| 
 | |
|     // dereferencing yields the key type
 | |
|     typedef typename ArgumentPack::key_type type;
 | |
| };
 | |
| 
 | |
| template <>
 | |
| struct arg_list_iterator<empty_arg_list> {};
 | |
| 
 | |
| }} // namespace parameter::aux
 | |
| 
 | |
| // MPL sequence support
 | |
| namespace mpl
 | |
| {
 | |
|   template <>
 | |
|   struct begin_impl<parameter::aux::arg_list_tag>
 | |
|   {
 | |
|       template <class S>
 | |
|       struct apply
 | |
|       {
 | |
|           typedef parameter::aux::arg_list_iterator<S> type;
 | |
|       };
 | |
|   };
 | |
| 
 | |
|   template <>
 | |
|   struct end_impl<parameter::aux::arg_list_tag>
 | |
|   {
 | |
|       template <class>
 | |
|       struct apply
 | |
|       {
 | |
|           typedef parameter::aux::arg_list_iterator<parameter::aux::empty_arg_list> type;
 | |
|       };
 | |
|   };
 | |
| }
 | |
| 
 | |
| } // namespace boost
 | |
| 
 | |
| #endif // ARG_LIST_050329_HPP
 | |
| 
 | 
