parameters.hpp 26 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931
  1. // Copyright David Abrahams, Daniel Wallin 2003. Use, modification and
  2. // distribution is subject to the Boost Software License, Version 1.0.
  3. // (See accompanying file LICENSE_1_0.txt or copy at
  4. // http://www.boost.org/LICENSE_1_0.txt)
  5. #ifndef BOOST_PARAMETERS_031014_HPP
  6. #define BOOST_PARAMETERS_031014_HPP
  7. #include <boost/detail/is_xxx.hpp>
  8. #include <boost/type_traits/is_const.hpp>
  9. #include <boost/mpl/lambda.hpp>
  10. #include <boost/mpl/apply.hpp>
  11. #include <boost/mpl/always.hpp>
  12. #include <boost/mpl/and.hpp>
  13. #include <boost/mpl/or.hpp>
  14. #include <boost/mpl/if.hpp>
  15. #include <boost/mpl/identity.hpp>
  16. #include <boost/mpl/not.hpp>
  17. #include <boost/mpl/eval_if.hpp>
  18. #include <boost/mpl/pair.hpp>
  19. #include <boost/type_traits/is_same.hpp>
  20. #include <boost/type_traits/remove_reference.hpp>
  21. #include <boost/preprocessor/repetition/enum.hpp>
  22. #include <boost/preprocessor/repetition/enum_params.hpp>
  23. #include <boost/preprocessor/repetition/enum_trailing_params.hpp>
  24. #include <boost/preprocessor/arithmetic/sub.hpp>
  25. #include <boost/preprocessor/repetition/repeat.hpp>
  26. #include <boost/preprocessor/repetition/enum_shifted.hpp>
  27. #include <boost/preprocessor/repetition/enum_binary_params.hpp>
  28. #include <boost/preprocessor/repetition/enum_shifted_params.hpp>
  29. #include <boost/preprocessor/seq/elem.hpp>
  30. #include <boost/preprocessor/iteration/iterate.hpp>
  31. #include <boost/preprocessor/facilities/intercept.hpp>
  32. #include <boost/preprocessor/cat.hpp>
  33. #include <boost/parameter/aux_/arg_list.hpp>
  34. #include <boost/parameter/aux_/yesno.hpp>
  35. #include <boost/parameter/aux_/void.hpp>
  36. #include <boost/parameter/aux_/default.hpp>
  37. #include <boost/parameter/aux_/unwrap_cv_reference.hpp>
  38. #include <boost/parameter/aux_/tagged_argument.hpp>
  39. #include <boost/parameter/aux_/tag.hpp>
  40. #include <boost/parameter/aux_/template_keyword.hpp>
  41. #include <boost/parameter/aux_/set.hpp>
  42. #include <boost/parameter/config.hpp>
  43. namespace parameter_
  44. {
  45. template <class T>
  46. struct unmatched_argument
  47. {
  48. BOOST_MPL_ASSERT((boost::is_same<T,void>));
  49. typedef int type;
  50. };
  51. } // namespace parameter_
  52. namespace boost {
  53. template<class T> class reference_wrapper;
  54. namespace parameter {
  55. namespace aux { struct use_default {}; }
  56. // These templates can be used to describe the treatment of particular
  57. // named parameters for the purposes of overload elimination with
  58. // SFINAE, by placing specializations in the parameters<...> list. In
  59. // order for a treated function to participate in overload resolution:
  60. //
  61. // - all keyword tags wrapped in required<...> must have a matching
  62. // actual argument
  63. //
  64. // - The actual argument type matched by every keyword tag
  65. // associated with a predicate must satisfy that predicate
  66. //
  67. // If a keyword k is specified without an optional<...> or
  68. // required<...>, wrapper, it is treated as though optional<k> were
  69. // specified.
  70. //
  71. // If a keyword k is specified with deduced<...>, that keyword
  72. // will be automatically deduced from the argument list.
  73. //
  74. template <class Tag, class Predicate = aux::use_default>
  75. struct required
  76. {
  77. typedef Tag key_type;
  78. typedef Predicate predicate;
  79. };
  80. template <class Tag, class Predicate = aux::use_default>
  81. struct optional
  82. {
  83. typedef Tag key_type;
  84. typedef Predicate predicate;
  85. };
  86. template <class Tag>
  87. struct deduced
  88. {
  89. typedef Tag key_type;
  90. };
  91. namespace aux
  92. {
  93. // Defines metafunctions, is_required and is_optional, that
  94. // identify required<...>, optional<...> and deduced<...> specializations.
  95. BOOST_DETAIL_IS_XXX_DEF(required, required, 2)
  96. BOOST_DETAIL_IS_XXX_DEF(optional, optional, 2)
  97. BOOST_DETAIL_IS_XXX_DEF(deduced_aux, deduced, 1)
  98. template <class S>
  99. struct is_deduced0
  100. : is_deduced_aux<
  101. typename S::key_type
  102. >::type
  103. {};
  104. template <class S>
  105. struct is_deduced
  106. : mpl::eval_if<
  107. mpl::or_<
  108. is_optional<S>, is_required<S>
  109. >
  110. , is_deduced0<S>
  111. , mpl::false_
  112. >::type
  113. {};
  114. //
  115. // key_type, has_default, and predicate --
  116. //
  117. // These metafunctions accept a ParameterSpec and extract the
  118. // keyword tag, whether or not a default is supplied for the
  119. // parameter, and the predicate that the corresponding actual
  120. // argument type is required match.
  121. //
  122. // a ParameterSpec is a specialization of either keyword<...>,
  123. // required<...>, optional<...>
  124. //
  125. // helper for key_type<...>, below.
  126. template <class T>
  127. struct get_tag_type0
  128. {
  129. typedef typename T::key_type type;
  130. };
  131. template <class T>
  132. struct get_tag_type
  133. : mpl::eval_if<
  134. is_deduced_aux<typename T::key_type>
  135. , get_tag_type0<typename T::key_type>
  136. , mpl::identity<typename T::key_type>
  137. >
  138. {};
  139. template <class T>
  140. struct tag_type
  141. : mpl::eval_if<
  142. mpl::or_<
  143. is_optional<T>
  144. , is_required<T>
  145. >
  146. , get_tag_type<T>
  147. , mpl::identity<T>
  148. >
  149. {};
  150. template <class T>
  151. struct has_default
  152. : mpl::not_<is_required<T> >
  153. {};
  154. // helper for get_predicate<...>, below
  155. template <class T>
  156. struct get_predicate_or_default
  157. {
  158. typedef T type;
  159. };
  160. template <>
  161. struct get_predicate_or_default<use_default>
  162. {
  163. typedef mpl::always<mpl::true_> type;
  164. };
  165. // helper for predicate<...>, below
  166. template <class T>
  167. struct get_predicate
  168. {
  169. typedef typename
  170. get_predicate_or_default<typename T::predicate>::type
  171. type;
  172. };
  173. template <class T>
  174. struct predicate
  175. : mpl::eval_if<
  176. mpl::or_<
  177. is_optional<T>
  178. , is_required<T>
  179. >
  180. , get_predicate<T>
  181. , mpl::identity<mpl::always<mpl::true_> >
  182. >
  183. {
  184. };
  185. // Converts a ParameterSpec into a specialization of
  186. // parameter_requirements. We need to do this in order to get the
  187. // tag_type into the type in a way that can be conveniently matched
  188. // by a satisfies(...) member function in arg_list.
  189. template <class ParameterSpec>
  190. struct as_parameter_requirements
  191. {
  192. typedef parameter_requirements<
  193. typename tag_type<ParameterSpec>::type
  194. , typename predicate<ParameterSpec>::type
  195. , typename has_default<ParameterSpec>::type
  196. > type;
  197. };
  198. template <class T>
  199. struct is_named_argument
  200. : mpl::or_<
  201. is_template_keyword<T>
  202. , is_tagged_argument<T>
  203. >
  204. {};
  205. // Returns mpl::true_ iff the given ParameterRequirements are
  206. // satisfied by ArgList.
  207. template <class ArgList, class ParameterRequirements>
  208. struct satisfies
  209. {
  210. #if BOOST_WORKAROUND(BOOST_MSVC, == 1310)
  211. // VC7.1 can't handle the sizeof() implementation below,
  212. // so we use this instead.
  213. typedef typename mpl::apply_wrap3<
  214. typename ArgList::binding
  215. , typename ParameterRequirements::keyword
  216. , void_
  217. , mpl::false_
  218. >::type bound;
  219. typedef typename mpl::eval_if<
  220. is_same<bound, void_>
  221. , typename ParameterRequirements::has_default
  222. , mpl::apply_wrap2<
  223. typename mpl::lambda<
  224. typename ParameterRequirements::predicate, lambda_tag
  225. >::type
  226. , bound
  227. , ArgList
  228. >
  229. >::type type;
  230. #else
  231. BOOST_STATIC_CONSTANT(
  232. bool, value = (
  233. sizeof(
  234. aux::to_yesno(
  235. ArgList::satisfies((ParameterRequirements*)0, (ArgList*)0)
  236. )
  237. ) == sizeof(yes_tag)
  238. )
  239. );
  240. typedef mpl::bool_<satisfies::value> type;
  241. #endif
  242. };
  243. // Returns mpl::true_ if the requirements of the given ParameterSpec
  244. // are satisfied by ArgList.
  245. template <class ArgList, class ParameterSpec>
  246. struct satisfies_requirements_of
  247. : satisfies<
  248. ArgList
  249. , typename as_parameter_requirements<ParameterSpec>::type
  250. >
  251. {};
  252. // Tags a deduced argument Arg with the keyword tag of Spec using TagFn.
  253. // Returns the tagged argument and the mpl::set<> UsedArgs with the
  254. // tag of Spec inserted.
  255. template <class UsedArgs, class Spec, class Arg, class TagFn>
  256. struct tag_deduced
  257. {
  258. typedef mpl::pair<
  259. typename mpl::apply_wrap2<TagFn, typename tag_type<Spec>::type, Arg>::type
  260. , typename aux::insert_<UsedArgs, typename tag_type<Spec>::type>::type
  261. > type;
  262. };
  263. template <
  264. class Argument
  265. , class ArgumentPack
  266. , class DeducedArgs
  267. , class UsedArgs
  268. , class TagFn
  269. >
  270. struct deduce_tag;
  271. // Tag type passed to MPL lambda.
  272. struct lambda_tag;
  273. // Helper for deduce_tag<> below.
  274. template <
  275. class Argument
  276. , class ArgumentPack
  277. , class DeducedArgs
  278. , class UsedArgs
  279. , class TagFn
  280. >
  281. struct deduce_tag0
  282. {
  283. typedef typename DeducedArgs::spec spec;
  284. typedef typename mpl::apply_wrap2<
  285. typename mpl::lambda<
  286. typename spec::predicate, lambda_tag
  287. >::type
  288. , Argument
  289. , ArgumentPack
  290. >::type condition;
  291. // Deduced parameter matches several arguments.
  292. BOOST_MPL_ASSERT((
  293. mpl::not_<mpl::and_<
  294. condition
  295. , aux::has_key_<UsedArgs, typename tag_type<spec>::type>
  296. > >
  297. ));
  298. typedef typename mpl::eval_if<
  299. condition
  300. , tag_deduced<UsedArgs, spec, Argument, TagFn>
  301. , deduce_tag<Argument, ArgumentPack, typename DeducedArgs::tail, UsedArgs, TagFn>
  302. >::type type;
  303. };
  304. // Tries to deduced a keyword tag for a given Argument.
  305. // Returns an mpl::pair<> consisting of the tagged_argument<>,
  306. // and an mpl::set<> where the new tag has been inserted.
  307. //
  308. // Argument: The argument type to be tagged.
  309. //
  310. // ArgumentPack: The ArgumentPack built so far.
  311. //
  312. // DeducedArgs: A specialization of deduced_item<> (see below).
  313. // A list containing only the deduced ParameterSpecs.
  314. //
  315. // UsedArgs: An mpl::set<> containing the keyword tags used so far.
  316. //
  317. // TagFn: A metafunction class used to tag positional or deduced
  318. // arguments with a keyword tag.
  319. template <
  320. class Argument
  321. , class ArgumentPack
  322. , class DeducedArgs
  323. , class UsedArgs
  324. , class TagFn
  325. >
  326. struct deduce_tag
  327. {
  328. typedef typename mpl::eval_if<
  329. is_same<DeducedArgs, void_>
  330. , mpl::pair<void_, UsedArgs>
  331. , deduce_tag0<Argument, ArgumentPack, DeducedArgs, UsedArgs, TagFn>
  332. >::type type;
  333. };
  334. template <
  335. class List
  336. , class DeducedArgs
  337. , class TagFn
  338. , class Positional
  339. , class UsedArgs
  340. , class ArgumentPack
  341. , class Error
  342. >
  343. struct make_arg_list_aux;
  344. // Inserts Tagged::key_type into the UserArgs set.
  345. // Extra indirection to lazily evaluate Tagged::key_type.
  346. template <class UsedArgs, class Tagged>
  347. struct insert_tagged
  348. {
  349. typedef typename aux::insert_<
  350. UsedArgs, typename Tagged::key_type
  351. >::type type;
  352. };
  353. // Borland needs the insane extra-indirection workaround below
  354. // so that it doesn't magically drop the const qualifier from
  355. // the argument type.
  356. template <
  357. class List
  358. , class DeducedArgs
  359. , class TagFn
  360. , class Positional
  361. , class UsedArgs
  362. , class ArgumentPack
  363. #if BOOST_WORKAROUND(__BORLANDC__, BOOST_TESTED_AT(0x564))
  364. , class argument
  365. #endif
  366. , class Error
  367. >
  368. #if BOOST_WORKAROUND(__BORLANDC__, BOOST_TESTED_AT(0x564))
  369. struct make_arg_list00
  370. #else
  371. struct make_arg_list0
  372. #endif
  373. {
  374. #if !BOOST_WORKAROUND(__BORLANDC__, BOOST_TESTED_AT(0x564))
  375. typedef typename List::arg argument;
  376. #endif
  377. typedef typename List::spec parameter_spec;
  378. typedef typename tag_type<parameter_spec>::type tag_;
  379. typedef is_named_argument<argument> is_tagged;
  380. // If this argument is either explicitly tagged or a deduced
  381. // parameter, we turn off positional matching.
  382. typedef mpl::and_<
  383. mpl::not_<
  384. mpl::or_<is_deduced<parameter_spec>, is_tagged>
  385. >
  386. , Positional
  387. > positional;
  388. // If this parameter is explicitly tagged we add it to the
  389. // used-parmeters set. We only really need to add parameters
  390. // that are deduced, but we would need a way to check if
  391. // a given tag corresponds to a deduced parameter spec.
  392. typedef typename mpl::eval_if<
  393. is_tagged
  394. , insert_tagged<UsedArgs, argument>
  395. , mpl::identity<UsedArgs>
  396. >::type used_args;
  397. // If this parameter is neither explicitly tagged, nor
  398. // positionally matched; deduce the tag from the deduced
  399. // parameter specs.
  400. typedef typename mpl::eval_if<
  401. mpl::or_<is_tagged, positional>
  402. , mpl::pair<void_, used_args>
  403. , deduce_tag<argument, ArgumentPack, DeducedArgs, used_args, TagFn>
  404. >::type deduced_data;
  405. // If this parameter is explicitly tagged..
  406. typedef typename mpl::eval_if<
  407. is_tagged
  408. , mpl::identity<argument> // .. just use it
  409. , mpl::eval_if< // .. else, if positional matching is turned on..
  410. positional
  411. , mpl::apply_wrap2<TagFn, tag_, argument> // .. tag it positionally
  412. , mpl::first<deduced_data> // .. else, use the deduced tag
  413. >
  414. >::type tagged;
  415. // We build the arg_list incrementally as we go, prepending new
  416. // nodes.
  417. typedef typename mpl::if_<
  418. mpl::and_<
  419. is_same<Error, void_>
  420. , is_same<tagged, void_>
  421. >
  422. , parameter_::unmatched_argument<argument>
  423. , void_
  424. >::type error;
  425. typedef typename mpl::if_<
  426. is_same<tagged, void_>
  427. , ArgumentPack
  428. , arg_list<tagged, ArgumentPack>
  429. >::type argument_pack;
  430. typedef typename make_arg_list_aux<
  431. typename List::tail
  432. , DeducedArgs
  433. , TagFn
  434. , positional
  435. , typename deduced_data::second
  436. , argument_pack
  437. , error
  438. >::type type;
  439. };
  440. #if BOOST_WORKAROUND(__BORLANDC__, BOOST_TESTED_AT(0x564))
  441. template <
  442. class List
  443. , class DeducedArgs
  444. , class TagFn
  445. , class Positional
  446. , class UsedArgs
  447. , class ArgumentPack
  448. , class Error
  449. >
  450. struct make_arg_list0
  451. {
  452. typedef typename mpl::eval_if<
  453. typename List::is_arg_const
  454. , make_arg_list00<
  455. List
  456. , DeducedArgs
  457. , TagFn
  458. , Positional
  459. , UsedArgs
  460. , ArgumentPack
  461. , typename List::arg const
  462. , Error
  463. >
  464. , make_arg_list00<
  465. List
  466. , DeducedArgs
  467. , TagFn
  468. , Positional
  469. , UsedArgs
  470. , ArgumentPack
  471. , typename List::arg
  472. , Error
  473. >
  474. >::type type;
  475. };
  476. #endif
  477. // Returns an ArgumentPack where the list of arguments has
  478. // been tagged with keyword tags.
  479. //
  480. // List: A specialization of item<> (see below). Contains
  481. // both the ordered ParameterSpecs, and the given arguments.
  482. //
  483. // DeducedArgs: A specialization of deduced_item<> (see below).
  484. // A list containing only the deduced ParameterSpecs.
  485. //
  486. // TagFn: A metafunction class used to tag positional or deduced
  487. // arguments with a keyword tag.
  488. //
  489. // Position: An mpl::bool_<> specialization indicating if positional
  490. // matching is to be performed.
  491. //
  492. // DeducedSet: An mpl::set<> containing the keyword tags used so far.
  493. //
  494. // ArgumentPack: The ArgumentPack built so far. This is initially an
  495. // empty_arg_list and is built incrementally.
  496. //
  497. template <
  498. class List
  499. , class DeducedArgs
  500. , class TagFn
  501. , class Positional
  502. , class DeducedSet
  503. , class ArgumentPack
  504. , class Error
  505. >
  506. struct make_arg_list_aux
  507. {
  508. typedef typename mpl::eval_if<
  509. is_same<List, void_>
  510. , mpl::identity<mpl::pair<ArgumentPack, Error> >
  511. , make_arg_list0<List, DeducedArgs, TagFn, Positional, DeducedSet, ArgumentPack, Error>
  512. >::type type;
  513. };
  514. // VC6.5 was choking on the default parameters for make_arg_list_aux, so
  515. // this just forwards to that adding in the defaults.
  516. template <
  517. class List
  518. , class DeducedArgs
  519. , class TagFn
  520. , class EmitErrors = mpl::true_
  521. >
  522. struct make_arg_list
  523. {
  524. typedef typename make_arg_list_aux<
  525. List, DeducedArgs, TagFn, mpl::true_, aux::set0, empty_arg_list, void_
  526. >::type type;
  527. };
  528. // A parameter spec item typelist.
  529. template <class Spec, class Arg, class Tail = void_>
  530. struct item
  531. {
  532. typedef Spec spec;
  533. #if BOOST_WORKAROUND(__BORLANDC__, BOOST_TESTED_AT(0x564))
  534. typedef is_const<Arg> is_arg_const;
  535. #endif
  536. typedef Arg arg;
  537. typedef Tail tail;
  538. };
  539. template <class Spec, class Arg, class Tail>
  540. struct make_item
  541. {
  542. typedef item<Spec, Arg, typename Tail::type> type;
  543. };
  544. // Creates a item typelist.
  545. template <class Spec, class Arg, class Tail>
  546. struct make_items
  547. {
  548. typedef typename mpl::eval_if<
  549. is_same<Arg, void_>
  550. , mpl::identity<void_>
  551. , make_item<Spec, Arg, Tail>
  552. >::type type;
  553. };
  554. // A typelist that stored deduced parameter specs.
  555. template <class ParameterSpec, class Tail = void_>
  556. struct deduced_item
  557. {
  558. typedef ParameterSpec spec;
  559. typedef Tail tail;
  560. };
  561. // Evaluate Tail and construct deduced_item list.
  562. template <class Spec, class Tail>
  563. struct make_deduced_item
  564. {
  565. typedef deduced_item<Spec, typename Tail::type> type;
  566. };
  567. template <class Spec, class Tail>
  568. struct make_deduced_items
  569. {
  570. typedef typename mpl::eval_if<
  571. is_same<Spec, void_>
  572. , mpl::identity<void_>
  573. , mpl::eval_if<
  574. is_deduced<Spec>
  575. , make_deduced_item<Spec, Tail>
  576. , Tail
  577. >
  578. >::type type;
  579. };
  580. // Generates:
  581. //
  582. // make<
  583. // parameter_spec#0, argument_type#0
  584. // , make<
  585. // parameter_spec#1, argument_type#1
  586. // , ... mpl::identity<aux::empty_arg_list>
  587. // ...>
  588. // >
  589. #define BOOST_PARAMETER_make_arg_list(z, n, names) \
  590. BOOST_PP_SEQ_ELEM(0,names)< \
  591. BOOST_PP_CAT(BOOST_PP_SEQ_ELEM(1,names), n), \
  592. BOOST_PP_CAT(BOOST_PP_SEQ_ELEM(2,names), n),
  593. #define BOOST_PARAMETER_right_angle(z, n, text) >
  594. #define BOOST_PARAMETER_build_arg_list(n, make, parameter_spec, argument_type) \
  595. BOOST_PP_REPEAT( \
  596. n, BOOST_PARAMETER_make_arg_list, (make)(parameter_spec)(argument_type)) \
  597. mpl::identity<void_> \
  598. BOOST_PP_REPEAT(n, BOOST_PARAMETER_right_angle, _)
  599. #define BOOST_PARAMETER_make_deduced_list(z, n, names) \
  600. BOOST_PP_SEQ_ELEM(0,names)< \
  601. BOOST_PP_CAT(BOOST_PP_SEQ_ELEM(1,names), n),
  602. #define BOOST_PARAMETER_build_deduced_list(n, make, parameter_spec) \
  603. BOOST_PP_REPEAT( \
  604. n, BOOST_PARAMETER_make_deduced_list, (make)(parameter_spec)) \
  605. mpl::identity<void_> \
  606. BOOST_PP_REPEAT(n, BOOST_PARAMETER_right_angle, _)
  607. struct tag_keyword_arg
  608. {
  609. template <class K, class T>
  610. struct apply
  611. : tag<K,T>
  612. {};
  613. };
  614. struct tag_template_keyword_arg
  615. {
  616. template <class K, class T>
  617. struct apply
  618. {
  619. typedef template_keyword<K,T> type;
  620. };
  621. };
  622. } // namespace aux
  623. #define BOOST_PARAMETER_FORWARD_TYPEDEF(z, i, names) \
  624. typedef BOOST_PP_CAT(BOOST_PP_SEQ_ELEM(0,names),i) BOOST_PP_CAT(BOOST_PP_SEQ_ELEM(1,names),i);
  625. #define BOOST_PARAMETER_FORWARD_TYPEDEFS(n, src, dest) \
  626. BOOST_PP_REPEAT(n, BOOST_PARAMETER_FORWARD_TYPEDEF, (src)(dest))
  627. #define BOOST_PARAMETER_TEMPLATE_ARGS(z, n, text) class BOOST_PP_CAT(PS, n) = void_
  628. template<
  629. class PS0
  630. , BOOST_PP_ENUM_SHIFTED(BOOST_PARAMETER_MAX_ARITY, BOOST_PARAMETER_TEMPLATE_ARGS, _)
  631. >
  632. struct parameters
  633. {
  634. #undef BOOST_PARAMETER_TEMPLATE_ARGS
  635. typedef typename BOOST_PARAMETER_build_deduced_list(
  636. BOOST_PARAMETER_MAX_ARITY, aux::make_deduced_items, PS
  637. )::type deduced_list;
  638. // if the elements of NamedList match the criteria of overload
  639. // resolution, returns a type which can be constructed from
  640. // parameters. Otherwise, this is not a valid metafunction (no nested
  641. // ::type).
  642. #if ! defined(BOOST_NO_SFINAE) && ! BOOST_WORKAROUND(__BORLANDC__, BOOST_TESTED_AT(0x592))
  643. // If NamedList satisfies the PS0, PS1, ..., this is a
  644. // metafunction returning parameters. Otherwise it
  645. // has no nested ::type.
  646. template <class ArgumentPackAndError>
  647. struct match_base
  648. : mpl::if_<
  649. // mpl::and_<
  650. // aux::satisfies_requirements_of<NamedList,PS0>
  651. // , mpl::and_<
  652. // aux::satisfies_requirements_of<NamedList,PS1>...
  653. // ..., mpl::true_
  654. // ...> >
  655. # define BOOST_PARAMETER_satisfies(z, n, text) \
  656. mpl::and_< \
  657. aux::satisfies_requirements_of< \
  658. typename mpl::first<ArgumentPackAndError>::type \
  659. , BOOST_PP_CAT(PS, n)> \
  660. ,
  661. mpl::and_<
  662. is_same<typename mpl::second<ArgumentPackAndError>::type, void_>
  663. , BOOST_PP_REPEAT(BOOST_PARAMETER_MAX_ARITY, BOOST_PARAMETER_satisfies, _)
  664. mpl::true_
  665. BOOST_PP_REPEAT(BOOST_PARAMETER_MAX_ARITY, BOOST_PARAMETER_right_angle, _)
  666. >
  667. # undef BOOST_PARAMETER_satisfies
  668. , mpl::identity<parameters>
  669. , void_
  670. >
  671. {};
  672. #endif
  673. // Specializations are to be used as an optional argument to
  674. // eliminate overloads via SFINAE
  675. template<
  676. #if BOOST_WORKAROUND(__BORLANDC__, BOOST_TESTED_AT(0x564))
  677. // Borland simply can't handle default arguments in member
  678. // class templates. People wishing to write portable code can
  679. // explicitly specify BOOST_PARAMETER_MAX_ARITY arguments
  680. BOOST_PP_ENUM_PARAMS(BOOST_PARAMETER_MAX_ARITY, class A)
  681. #else
  682. BOOST_PP_ENUM_BINARY_PARAMS(
  683. BOOST_PARAMETER_MAX_ARITY, class A, = void_ BOOST_PP_INTERCEPT
  684. )
  685. #endif
  686. >
  687. struct match
  688. # if ! defined(BOOST_NO_SFINAE) && ! BOOST_WORKAROUND(__BORLANDC__, BOOST_TESTED_AT(0x592))
  689. : match_base<
  690. typename aux::make_arg_list<
  691. typename BOOST_PARAMETER_build_arg_list(
  692. BOOST_PARAMETER_MAX_ARITY, aux::make_items, PS, A
  693. )::type
  694. , deduced_list
  695. , aux::tag_keyword_arg
  696. , mpl::false_ // Don't emit errors when doing SFINAE
  697. >::type
  698. >::type
  699. {};
  700. # else
  701. {
  702. typedef parameters<
  703. BOOST_PP_ENUM_PARAMS(BOOST_PARAMETER_MAX_ARITY, PS)
  704. > type;
  705. };
  706. # endif
  707. // Metafunction that returns an ArgumentPack.
  708. // TODO, bind has to instantiate the error type in the result
  709. // of make_arg_list.
  710. template <
  711. #if BOOST_WORKAROUND(__BORLANDC__, BOOST_TESTED_AT(0x564))
  712. // Borland simply can't handle default arguments in member
  713. // class templates. People wishing to write portable code can
  714. // explicitly specify BOOST_PARAMETER_MAX_ARITY arguments
  715. BOOST_PP_ENUM_PARAMS(BOOST_PARAMETER_MAX_ARITY, class A)
  716. #else
  717. BOOST_PP_ENUM_BINARY_PARAMS(
  718. BOOST_PARAMETER_MAX_ARITY, class A, = void_ BOOST_PP_INTERCEPT
  719. )
  720. #endif
  721. >
  722. struct bind
  723. {
  724. typedef typename aux::make_arg_list<
  725. typename BOOST_PARAMETER_build_arg_list(
  726. BOOST_PARAMETER_MAX_ARITY, aux::make_items, PS, A
  727. )::type
  728. , deduced_list
  729. , aux::tag_template_keyword_arg
  730. >::type result;
  731. typedef typename mpl::first<result>::type type;
  732. };
  733. BOOST_PARAMETER_FORWARD_TYPEDEFS(BOOST_PARAMETER_MAX_ARITY, PS, parameter_spec)
  734. //
  735. // The function call operator is used to build an arg_list that
  736. // labels the positional parameters and maintains whatever other
  737. // tags may have been specified by the caller.
  738. //
  739. // !!!NOTE!!!
  740. //
  741. // The make_arg_list<> produces a reversed arg_list, so
  742. // we need to pass the arguments to its constructor
  743. // reversed.
  744. //
  745. aux::empty_arg_list operator()() const
  746. {
  747. return aux::empty_arg_list();
  748. }
  749. template<class A0>
  750. typename mpl::first<
  751. typename aux::make_arg_list<
  752. aux::item<
  753. PS0,A0
  754. >
  755. , deduced_list
  756. , aux::tag_keyword_arg
  757. >::type
  758. >::type
  759. operator()(A0& a0) const
  760. {
  761. typedef typename aux::make_arg_list<
  762. aux::item<
  763. PS0,A0
  764. >
  765. , deduced_list
  766. , aux::tag_keyword_arg
  767. >::type result;
  768. typedef typename mpl::first<result>::type result_type;
  769. typedef typename mpl::second<result>::type error;
  770. error();
  771. return result_type(
  772. a0
  773. // , void_(), void_(), void_() ...
  774. BOOST_PP_ENUM_TRAILING_PARAMS(
  775. BOOST_PP_SUB(BOOST_PARAMETER_MAX_ARITY, 1)
  776. , aux::void_reference() BOOST_PP_INTERCEPT)
  777. );
  778. }
  779. template<class A0, class A1>
  780. typename mpl::first<
  781. typename aux::make_arg_list<
  782. aux::item<
  783. PS0,A0
  784. , aux::item<
  785. PS1,A1
  786. >
  787. >
  788. , deduced_list
  789. , aux::tag_keyword_arg
  790. >::type
  791. >::type
  792. operator()(A0& a0, A1& a1) const
  793. {
  794. typedef typename aux::make_arg_list<
  795. aux::item<
  796. PS0,A0
  797. , aux::item<
  798. PS1,A1
  799. >
  800. >
  801. , deduced_list
  802. , aux::tag_keyword_arg
  803. >::type result;
  804. typedef typename mpl::first<result>::type result_type;
  805. typedef typename mpl::second<result>::type error;
  806. error();
  807. return result_type(
  808. a1,a0
  809. // , void_(), void_() ...
  810. BOOST_PP_ENUM_TRAILING_PARAMS(
  811. BOOST_PP_SUB(BOOST_PARAMETER_MAX_ARITY, 2)
  812. , aux::void_reference() BOOST_PP_INTERCEPT)
  813. );
  814. }
  815. // Higher arities are handled by the preprocessor
  816. #define BOOST_PP_ITERATION_PARAMS_1 (3,( \
  817. 3,BOOST_PARAMETER_MAX_ARITY,<boost/parameter/aux_/overloads.hpp> \
  818. ))
  819. #include BOOST_PP_ITERATE()
  820. };
  821. } // namespace parameter
  822. } // namespace boost
  823. #endif // BOOST_PARAMETERS_031014_HPP