boost/parameter/aux_/arg_list.hpp
// 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/config.hpp> #include <boost/mpl/apply.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/facilities/intercept.hpp> namespace boost { namespace parameter { // Forward declaration for aux::arg_list, below. template<class T> struct keyword; namespace aux { // // 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. // // 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> struct apply { typedef Default type; }; }; #if BOOST_WORKAROUND(BOOST_MSVC, <= 1300) \ || (BOOST_WORKAROUND(__GNUC__, < 3)) \ || 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> static typename ParameterRequirements::has_default satisfies(ParameterRequirements*); }; // Forward declaration for arg_list::operator, template <class KW, class T> struct tagged_argument; // 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 TaggedArg::value_type value_type; typedef typename TaggedArg::reference reference; 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 const& a0, A1 const& a1, ... BOOST_PP_ENUM_BINARY_PARAMS(BOOST_PARAMETER_MAX_ARITY, A, const & a) ) : Next( // a1, a2, ... BOOST_PP_ENUM_SHIFTED_PARAMS(BOOST_PARAMETER_MAX_ARITY, a) , void_() ) , 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 arg, Next const& tail) : Next(tail) , arg(arg) {} // 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> struct apply { typedef typename mpl::eval_if< boost::is_same<KW, key_type> , mpl::identity<reference> , mpl::apply_wrap2<typename Next::binding, KW, Default> >::type type; }; }; // // Begin implementation of indexing operators for looking up // specific arguments by name // #if BOOST_WORKAROUND(BOOST_MSVC, <= 1300) \ || BOOST_WORKAROUND(__GNUC__, < 3) \ || 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_wrap2<binding, KW, void_>::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_wrap2<binding, KW, Default&>::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_wrap2< binding,KW , typename result_of0<F>::type >::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 { return arg.value; } template <class Default> reference get(default_<key_type,Default>) const { return arg.value; } template <class Default> reference get(lazy_default<key_type, Default>) const { return arg.value; } #else reference operator[](keyword<key_type> const&) const { return arg.value; } template <class Default> reference operator[](default_<key_type, Default>) const { return arg.value; } 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> static typename mpl::apply1<Predicate, value_type>::type satisfies( parameter_requirements<key_type,Predicate,HasDefault>* ); // 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) { return arg_list<tagged_argument<KW,T2>, self>(x, *this); } }; #if BOOST_WORKAROUND(BOOST_MSVC, <= 1300) // ETI workaround template <> struct arg_list<int,int> {}; #endif }}} // namespace boost::parameter::aux #endif // ARG_LIST_050329_HPP