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