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 | ||
|  | 
 |