os/ossrv/ossrv_pub/boost_apis/boost/foreach.hpp
author sl@SLION-WIN7.fritz.box
Fri, 15 Jun 2012 03:10:57 +0200
changeset 0 bde4ae8d615e
permissions -rw-r--r--
First public contribution.
sl@0
     1
///////////////////////////////////////////////////////////////////////////////
sl@0
     2
// foreach.hpp header file
sl@0
     3
//
sl@0
     4
// Copyright 2004 Eric Niebler.
sl@0
     5
// Distributed under the Boost Software License, Version 1.0. (See
sl@0
     6
// accompanying file LICENSE_1_0.txt or copy at
sl@0
     7
// http://www.boost.org/LICENSE_1_0.txt)
sl@0
     8
//
sl@0
     9
// Credits:
sl@0
    10
//  Anson Tsao        - for the initial inspiration and several good suggestions.
sl@0
    11
//  Thorsten Ottosen  - for Boost.Range, and for suggesting a way to detect
sl@0
    12
//                      const-qualified rvalues at compile time on VC7.1+
sl@0
    13
//  Russell Hind      - For help porting to Borland
sl@0
    14
//  Alisdair Meredith - For help porting to Borland
sl@0
    15
//  Stefan Slapeta    - For help porting to Intel
sl@0
    16
sl@0
    17
#ifndef BOOST_FOREACH
sl@0
    18
sl@0
    19
// MS compatible compilers support #pragma once
sl@0
    20
#if defined(_MSC_VER) && (_MSC_VER >= 1020)
sl@0
    21
# pragma once
sl@0
    22
#endif
sl@0
    23
sl@0
    24
#include <cstddef>
sl@0
    25
#include <utility>  // for std::pair
sl@0
    26
sl@0
    27
#include <boost/config.hpp>
sl@0
    28
#include <boost/detail/workaround.hpp>
sl@0
    29
sl@0
    30
// Some compilers let us detect even const-qualified rvalues at compile-time
sl@0
    31
#if BOOST_WORKAROUND(BOOST_MSVC, >= 1310)                                                       \
sl@0
    32
 || (BOOST_WORKAROUND(__GNUC__, >= 4) && !defined(BOOST_INTEL))                                 \
sl@0
    33
 || (BOOST_WORKAROUND(__GNUC__, == 3) && (__GNUC_MINOR__ >= 4) && !defined(BOOST_INTEL))
sl@0
    34
# define BOOST_FOREACH_COMPILE_TIME_CONST_RVALUE_DETECTION
sl@0
    35
#else
sl@0
    36
// Some compilers allow temporaries to be bound to non-const references.
sl@0
    37
// These compilers make it impossible to for BOOST_FOREACH to detect
sl@0
    38
// temporaries and avoid reevaluation of the collection expression.
sl@0
    39
# if BOOST_WORKAROUND(BOOST_MSVC, <= 1300)                                                      \
sl@0
    40
  || BOOST_WORKAROUND(__BORLANDC__, BOOST_TESTED_AT(0x564))                                     \
sl@0
    41
  || (BOOST_WORKAROUND(BOOST_INTEL_CXX_VERSION, <= 700) && defined(_MSC_VER))                   \
sl@0
    42
  || BOOST_WORKAROUND(__SUNPRO_CC, BOOST_TESTED_AT(0x570))                                      \
sl@0
    43
  || BOOST_WORKAROUND(__DECCXX_VER, BOOST_TESTED_AT(60590042))
sl@0
    44
#  define BOOST_FOREACH_NO_RVALUE_DETECTION
sl@0
    45
# endif
sl@0
    46
// Some compilers do not correctly implement the lvalue/rvalue conversion
sl@0
    47
// rules of the ternary conditional operator.
sl@0
    48
# if defined(BOOST_FOREACH_NO_RVALUE_DETECTION)                                                 \
sl@0
    49
  || defined(BOOST_NO_SFINAE)                                                                   \
sl@0
    50
  || BOOST_WORKAROUND(BOOST_MSVC, BOOST_TESTED_AT(1400))                                        \
sl@0
    51
  || BOOST_WORKAROUND(BOOST_INTEL_WIN, <= 810)                                                  \
sl@0
    52
  || BOOST_WORKAROUND(__GNUC__, < 3)                                                            \
sl@0
    53
  || (BOOST_WORKAROUND(__GNUC__, == 3) && (__GNUC_MINOR__ <= 2))                                \
sl@0
    54
  || (BOOST_WORKAROUND(__GNUC__, == 3) && (__GNUC_MINOR__ <= 3) && defined(__APPLE_CC__))       \
sl@0
    55
  || BOOST_WORKAROUND(__IBMCPP__, BOOST_TESTED_AT(600))                                         \
sl@0
    56
  || BOOST_WORKAROUND(__MWERKS__, BOOST_TESTED_AT(0x3206))
sl@0
    57
#  define BOOST_FOREACH_NO_CONST_RVALUE_DETECTION
sl@0
    58
# else
sl@0
    59
#  define BOOST_FOREACH_RUN_TIME_CONST_RVALUE_DETECTION
sl@0
    60
# endif
sl@0
    61
#endif
sl@0
    62
sl@0
    63
#include <boost/mpl/if.hpp>
sl@0
    64
#include <boost/mpl/logical.hpp>
sl@0
    65
#include <boost/mpl/eval_if.hpp>
sl@0
    66
#include <boost/noncopyable.hpp>
sl@0
    67
#include <boost/range/end.hpp>
sl@0
    68
#include <boost/range/begin.hpp>
sl@0
    69
#include <boost/range/result_iterator.hpp>
sl@0
    70
#include <boost/type_traits/is_array.hpp>
sl@0
    71
#include <boost/type_traits/is_const.hpp>
sl@0
    72
#include <boost/type_traits/is_abstract.hpp>
sl@0
    73
#include <boost/type_traits/is_base_and_derived.hpp>
sl@0
    74
#include <boost/iterator/iterator_traits.hpp>
sl@0
    75
#include <boost/utility/addressof.hpp>
sl@0
    76
sl@0
    77
#ifdef BOOST_FOREACH_RUN_TIME_CONST_RVALUE_DETECTION
sl@0
    78
# include <new>
sl@0
    79
# include <boost/aligned_storage.hpp>
sl@0
    80
# include <boost/utility/enable_if.hpp>
sl@0
    81
# include <boost/type_traits/remove_const.hpp>
sl@0
    82
#endif
sl@0
    83
sl@0
    84
// This must be at global scope, hence the uglified name
sl@0
    85
enum boost_foreach_argument_dependent_lookup_hack
sl@0
    86
{
sl@0
    87
    boost_foreach_argument_dependent_lookup_hack_value
sl@0
    88
};
sl@0
    89
sl@0
    90
namespace boost
sl@0
    91
{
sl@0
    92
sl@0
    93
// forward declarations for iterator_range
sl@0
    94
template<typename T>
sl@0
    95
class iterator_range;
sl@0
    96
sl@0
    97
// forward declarations for sub_range
sl@0
    98
template<typename T>
sl@0
    99
class sub_range;
sl@0
   100
sl@0
   101
namespace foreach
sl@0
   102
{
sl@0
   103
    ///////////////////////////////////////////////////////////////////////////////
sl@0
   104
    // in_range
sl@0
   105
    //
sl@0
   106
    template<typename T>
sl@0
   107
    inline std::pair<T, T> in_range(T begin, T end)
sl@0
   108
    {
sl@0
   109
        return std::make_pair(begin, end);
sl@0
   110
    }
sl@0
   111
sl@0
   112
    ///////////////////////////////////////////////////////////////////////////////
sl@0
   113
    // boost::foreach::tag
sl@0
   114
    //
sl@0
   115
    typedef boost_foreach_argument_dependent_lookup_hack tag;
sl@0
   116
sl@0
   117
    ///////////////////////////////////////////////////////////////////////////////
sl@0
   118
    // boost::foreach::is_lightweight_proxy
sl@0
   119
    //   Specialize this for user-defined collection types if they are inexpensive to copy.
sl@0
   120
    //   This tells BOOST_FOREACH it can avoid the rvalue/lvalue detection stuff.
sl@0
   121
    template<typename T>
sl@0
   122
    struct is_lightweight_proxy
sl@0
   123
      : boost::mpl::false_
sl@0
   124
    {
sl@0
   125
    };
sl@0
   126
sl@0
   127
    ///////////////////////////////////////////////////////////////////////////////
sl@0
   128
    // boost::foreach::is_noncopyable
sl@0
   129
    //   Specialize this for user-defined collection types if they cannot be copied.
sl@0
   130
    //   This also tells BOOST_FOREACH to avoid the rvalue/lvalue detection stuff.
sl@0
   131
    template<typename T>
sl@0
   132
    struct is_noncopyable
sl@0
   133
    #if !defined(BOOST_BROKEN_IS_BASE_AND_DERIVED) && !defined(BOOST_NO_IS_ABSTRACT)
sl@0
   134
      : boost::mpl::or_<
sl@0
   135
            boost::is_abstract<T>
sl@0
   136
          , boost::is_base_and_derived<boost::noncopyable, T>
sl@0
   137
        >
sl@0
   138
    #elif !defined(BOOST_BROKEN_IS_BASE_AND_DERIVED)
sl@0
   139
      : boost::is_base_and_derived<boost::noncopyable, T>
sl@0
   140
    #elif !defined(BOOST_NO_IS_ABSTRACT)
sl@0
   141
      : boost::is_abstract<T>
sl@0
   142
    #else
sl@0
   143
      : boost::mpl::false_
sl@0
   144
    #endif
sl@0
   145
    {
sl@0
   146
    };
sl@0
   147
sl@0
   148
} // namespace foreach
sl@0
   149
sl@0
   150
} // namespace boost
sl@0
   151
sl@0
   152
// vc6/7 needs help ordering the following overloads
sl@0
   153
#ifdef BOOST_NO_FUNCTION_TEMPLATE_ORDERING
sl@0
   154
# define BOOST_FOREACH_TAG_DEFAULT ...
sl@0
   155
#else
sl@0
   156
# define BOOST_FOREACH_TAG_DEFAULT boost::foreach::tag
sl@0
   157
#endif
sl@0
   158
sl@0
   159
///////////////////////////////////////////////////////////////////////////////
sl@0
   160
// boost_foreach_is_lightweight_proxy
sl@0
   161
//   Another customization point for the is_lightweight_proxy optimization,
sl@0
   162
//   this one works on legacy compilers. Overload boost_foreach_is_lightweight_proxy
sl@0
   163
//   at the global namespace for your type.
sl@0
   164
template<typename T>
sl@0
   165
inline boost::foreach::is_lightweight_proxy<T> *
sl@0
   166
boost_foreach_is_lightweight_proxy(T *&, BOOST_FOREACH_TAG_DEFAULT) { return 0; }
sl@0
   167
sl@0
   168
template<typename T>
sl@0
   169
inline boost::mpl::true_ *
sl@0
   170
boost_foreach_is_lightweight_proxy(std::pair<T, T> *&, boost::foreach::tag) { return 0; }
sl@0
   171
sl@0
   172
template<typename T>
sl@0
   173
inline boost::mpl::true_ *
sl@0
   174
boost_foreach_is_lightweight_proxy(boost::iterator_range<T> *&, boost::foreach::tag) { return 0; }
sl@0
   175
sl@0
   176
template<typename T>
sl@0
   177
inline boost::mpl::true_ *
sl@0
   178
boost_foreach_is_lightweight_proxy(boost::sub_range<T> *&, boost::foreach::tag) { return 0; }
sl@0
   179
sl@0
   180
template<typename T>
sl@0
   181
inline boost::mpl::true_ *
sl@0
   182
boost_foreach_is_lightweight_proxy(T **&, boost::foreach::tag) { return 0; }
sl@0
   183
sl@0
   184
///////////////////////////////////////////////////////////////////////////////
sl@0
   185
// boost_foreach_is_noncopyable
sl@0
   186
//   Another customization point for the is_noncopyable trait,
sl@0
   187
//   this one works on legacy compilers. Overload boost_foreach_is_noncopyable
sl@0
   188
//   at the global namespace for your type.
sl@0
   189
template<typename T>
sl@0
   190
inline boost::foreach::is_noncopyable<T> *
sl@0
   191
boost_foreach_is_noncopyable(T *&, BOOST_FOREACH_TAG_DEFAULT) { return 0; }
sl@0
   192
sl@0
   193
namespace boost
sl@0
   194
{
sl@0
   195
sl@0
   196
namespace foreach_detail_
sl@0
   197
{
sl@0
   198
sl@0
   199
///////////////////////////////////////////////////////////////////////////////
sl@0
   200
// Define some utilities for assessing the properties of expressions
sl@0
   201
//
sl@0
   202
typedef char yes_type;
sl@0
   203
typedef char (&no_type)[2];
sl@0
   204
yes_type is_true(boost::mpl::true_ *);
sl@0
   205
no_type is_true(boost::mpl::false_ *);
sl@0
   206
sl@0
   207
// Extracts the desired property from the expression without evaluating it
sl@0
   208
#define BOOST_FOREACH_PROTECT(expr)                                                             \
sl@0
   209
    (static_cast<boost::mpl::bool_<1 == sizeof(boost::foreach_detail_::is_true(expr))> *>(0))
sl@0
   210
sl@0
   211
template<typename Bool1, typename Bool2>
sl@0
   212
inline boost::mpl::and_<Bool1, Bool2> *and_(Bool1 *, Bool2 *) { return 0; }
sl@0
   213
sl@0
   214
template<typename Bool1, typename Bool2, typename Bool3>
sl@0
   215
inline boost::mpl::and_<Bool1, Bool2, Bool3> *and_(Bool1 *, Bool2 *, Bool3 *) { return 0; }
sl@0
   216
sl@0
   217
template<typename Bool1, typename Bool2>
sl@0
   218
inline boost::mpl::or_<Bool1, Bool2> *or_(Bool1 *, Bool2 *) { return 0; }
sl@0
   219
sl@0
   220
template<typename Bool1, typename Bool2, typename Bool3>
sl@0
   221
inline boost::mpl::or_<Bool1, Bool2, Bool3> *or_(Bool1 *, Bool2 *, Bool3 *) { return 0; }
sl@0
   222
sl@0
   223
template<typename Bool>
sl@0
   224
inline boost::mpl::not_<Bool> *not_(Bool *) { return 0; }
sl@0
   225
sl@0
   226
template<typename T>
sl@0
   227
inline boost::mpl::false_ *is_rvalue_(T &, int) { return 0; }
sl@0
   228
sl@0
   229
template<typename T>
sl@0
   230
inline boost::mpl::true_ *is_rvalue_(T const &, ...) { return 0; }
sl@0
   231
sl@0
   232
template<typename T>
sl@0
   233
inline boost::is_array<T> *is_array_(T const &) { return 0; }
sl@0
   234
sl@0
   235
template<typename T>
sl@0
   236
inline boost::is_const<T> *is_const_(T &) { return 0; }
sl@0
   237
sl@0
   238
#ifndef BOOST_FOREACH_NO_RVALUE_DETECTION
sl@0
   239
template<typename T>
sl@0
   240
inline boost::mpl::true_ *is_const_(T const &) { return 0; }
sl@0
   241
#endif
sl@0
   242
sl@0
   243
///////////////////////////////////////////////////////////////////////////////
sl@0
   244
// auto_any_t/auto_any
sl@0
   245
//  General utility for putting an object of any type into automatic storage
sl@0
   246
struct auto_any_base
sl@0
   247
{
sl@0
   248
    // auto_any_base must evaluate to false in boolean context so that
sl@0
   249
    // they can be declared in if() statements.
sl@0
   250
    operator bool() const
sl@0
   251
    {
sl@0
   252
        return false;
sl@0
   253
    }
sl@0
   254
};
sl@0
   255
sl@0
   256
template<typename T>
sl@0
   257
struct auto_any : auto_any_base
sl@0
   258
{
sl@0
   259
    auto_any(T const &t)
sl@0
   260
      : item(t)
sl@0
   261
    {
sl@0
   262
    }
sl@0
   263
sl@0
   264
    // temporaries of type auto_any will be bound to const auto_any_base
sl@0
   265
    // references, but we still want to be able to mutate the stored
sl@0
   266
    // data, so declare it as mutable.
sl@0
   267
    mutable T item;
sl@0
   268
};
sl@0
   269
sl@0
   270
typedef auto_any_base const &auto_any_t;
sl@0
   271
sl@0
   272
template<typename T, typename C>
sl@0
   273
inline BOOST_DEDUCED_TYPENAME boost::mpl::if_<C, T const, T>::type &auto_any_cast(auto_any_t a)
sl@0
   274
{
sl@0
   275
    return static_cast<auto_any<T> const &>(a).item;
sl@0
   276
}
sl@0
   277
sl@0
   278
typedef boost::mpl::true_ const_;
sl@0
   279
sl@0
   280
///////////////////////////////////////////////////////////////////////////////
sl@0
   281
// type2type
sl@0
   282
//
sl@0
   283
template<typename T, typename C = boost::mpl::false_>
sl@0
   284
struct type2type
sl@0
   285
  : boost::mpl::if_<C, T const, T>
sl@0
   286
{
sl@0
   287
};
sl@0
   288
sl@0
   289
template<typename T, typename C = boost::mpl::false_>
sl@0
   290
struct foreach_iterator
sl@0
   291
{
sl@0
   292
    typedef BOOST_DEDUCED_TYPENAME boost::mpl::eval_if<
sl@0
   293
        C
sl@0
   294
      , range_const_iterator<T>
sl@0
   295
      , range_iterator<T>
sl@0
   296
    >::type type;
sl@0
   297
};
sl@0
   298
sl@0
   299
template<typename T, typename C = boost::mpl::false_>
sl@0
   300
struct foreach_reference
sl@0
   301
  : iterator_reference<BOOST_DEDUCED_TYPENAME foreach_iterator<T, C>::type>
sl@0
   302
{
sl@0
   303
};
sl@0
   304
sl@0
   305
///////////////////////////////////////////////////////////////////////////////
sl@0
   306
// encode_type
sl@0
   307
//
sl@0
   308
template<typename T>
sl@0
   309
inline type2type<T> *encode_type(T &, boost::mpl::false_ *) { return 0; }
sl@0
   310
sl@0
   311
template<typename T>
sl@0
   312
inline type2type<T, const_> *encode_type(T const &, boost::mpl::true_ *) { return 0; }
sl@0
   313
sl@0
   314
///////////////////////////////////////////////////////////////////////////////
sl@0
   315
// set_false
sl@0
   316
//
sl@0
   317
inline bool set_false(bool &b) { return b = false; }
sl@0
   318
sl@0
   319
///////////////////////////////////////////////////////////////////////////////
sl@0
   320
// to_ptr
sl@0
   321
//
sl@0
   322
template<typename T>
sl@0
   323
inline T *&to_ptr(T const &)
sl@0
   324
{
sl@0
   325
    static T *t = 0;
sl@0
   326
    return t;
sl@0
   327
}
sl@0
   328
sl@0
   329
// Borland needs a little extra help with arrays
sl@0
   330
#if BOOST_WORKAROUND(__BORLANDC__, BOOST_TESTED_AT(0x564))
sl@0
   331
template<typename T,std::size_t N>
sl@0
   332
inline T (*&to_ptr(T (&)[N]))[N]
sl@0
   333
{
sl@0
   334
    static T (*t)[N] = 0;
sl@0
   335
    return t;
sl@0
   336
}
sl@0
   337
#endif
sl@0
   338
sl@0
   339
///////////////////////////////////////////////////////////////////////////////
sl@0
   340
// derefof
sl@0
   341
//
sl@0
   342
template<typename T>
sl@0
   343
inline T &derefof(T *t)
sl@0
   344
{
sl@0
   345
    // This is a work-around for a compiler bug in Borland. If T* is a pointer to array type U(*)[N],
sl@0
   346
    // then dereferencing it results in a U* instead of U(&)[N]. The cast forces the issue.
sl@0
   347
    return reinterpret_cast<T &>(
sl@0
   348
        *const_cast<char *>(
sl@0
   349
            reinterpret_cast<char const volatile *>(t)
sl@0
   350
        )
sl@0
   351
    );
sl@0
   352
}
sl@0
   353
sl@0
   354
#ifdef BOOST_FOREACH_COMPILE_TIME_CONST_RVALUE_DETECTION
sl@0
   355
///////////////////////////////////////////////////////////////////////////////
sl@0
   356
// Detect at compile-time whether an expression yields an rvalue or
sl@0
   357
// an lvalue. This is rather non-standard, but some popular compilers
sl@0
   358
// accept it.
sl@0
   359
///////////////////////////////////////////////////////////////////////////////
sl@0
   360
sl@0
   361
///////////////////////////////////////////////////////////////////////////////
sl@0
   362
// rvalue_probe
sl@0
   363
//
sl@0
   364
template<typename T>
sl@0
   365
struct rvalue_probe
sl@0
   366
{
sl@0
   367
    struct private_type_ {};
sl@0
   368
    // can't ever return an array by value
sl@0
   369
    typedef BOOST_DEDUCED_TYPENAME boost::mpl::if_<
sl@0
   370
        boost::mpl::or_<boost::is_abstract<T>, boost::is_array<T> >, private_type_, T
sl@0
   371
    >::type value_type;
sl@0
   372
    operator value_type();
sl@0
   373
    operator T &() const;
sl@0
   374
};
sl@0
   375
sl@0
   376
template<typename T>
sl@0
   377
rvalue_probe<T> const make_probe(T const &t);
sl@0
   378
sl@0
   379
# define BOOST_FOREACH_IS_RVALUE(COL)                                                           \
sl@0
   380
    boost::foreach_detail_::and_(                                                               \
sl@0
   381
        boost::foreach_detail_::not_(boost::foreach_detail_::is_array_(COL))                    \
sl@0
   382
      , BOOST_FOREACH_PROTECT(boost::foreach_detail_::is_rvalue_(                               \
sl@0
   383
            (true ? boost::foreach_detail_::make_probe(COL) : (COL)), 0)))
sl@0
   384
sl@0
   385
#elif defined(BOOST_FOREACH_RUN_TIME_CONST_RVALUE_DETECTION)
sl@0
   386
///////////////////////////////////////////////////////////////////////////////
sl@0
   387
// Detect at run-time whether an expression yields an rvalue
sl@0
   388
// or an lvalue. This is 100% standard C++, but not all compilers
sl@0
   389
// accept it. Also, it causes FOREACH to break when used with non-
sl@0
   390
// copyable collection types.
sl@0
   391
///////////////////////////////////////////////////////////////////////////////
sl@0
   392
sl@0
   393
///////////////////////////////////////////////////////////////////////////////
sl@0
   394
// rvalue_probe
sl@0
   395
//
sl@0
   396
template<typename T>
sl@0
   397
struct rvalue_probe
sl@0
   398
{
sl@0
   399
    rvalue_probe(T &t, bool &b)
sl@0
   400
      : value(t)
sl@0
   401
      , is_rvalue(b)
sl@0
   402
    {
sl@0
   403
    }
sl@0
   404
sl@0
   405
    struct private_type_ {};
sl@0
   406
    // can't ever return an array or an abstract type by value
sl@0
   407
    #ifdef BOOST_NO_IS_ABSTRACT
sl@0
   408
    typedef BOOST_DEDUCED_TYPENAME boost::mpl::if_<
sl@0
   409
        boost::is_array<T>, private_type_, T
sl@0
   410
    >::type value_type;
sl@0
   411
    #else
sl@0
   412
    typedef BOOST_DEDUCED_TYPENAME boost::mpl::if_<
sl@0
   413
        boost::mpl::or_<boost::is_abstract<T>, boost::is_array<T> >, private_type_, T
sl@0
   414
    >::type value_type;
sl@0
   415
    #endif
sl@0
   416
    
sl@0
   417
    operator value_type()
sl@0
   418
    {
sl@0
   419
        this->is_rvalue = true;
sl@0
   420
        return this->value;
sl@0
   421
    }
sl@0
   422
sl@0
   423
    operator T &() const
sl@0
   424
    {
sl@0
   425
        return this->value;
sl@0
   426
    }
sl@0
   427
sl@0
   428
private:
sl@0
   429
    T &value;
sl@0
   430
    bool &is_rvalue;
sl@0
   431
};
sl@0
   432
sl@0
   433
template<typename T>
sl@0
   434
rvalue_probe<T> make_probe(T &t, bool &b) { return rvalue_probe<T>(t, b); }
sl@0
   435
sl@0
   436
template<typename T>
sl@0
   437
rvalue_probe<T const> make_probe(T const &t, bool &b)  { return rvalue_probe<T const>(t, b); }
sl@0
   438
sl@0
   439
///////////////////////////////////////////////////////////////////////////////
sl@0
   440
// simple_variant
sl@0
   441
//  holds either a T or a T const*
sl@0
   442
template<typename T>
sl@0
   443
struct simple_variant
sl@0
   444
{
sl@0
   445
    simple_variant(T const *t)
sl@0
   446
      : is_rvalue(false)
sl@0
   447
    {
sl@0
   448
        *static_cast<T const **>(this->data.address()) = t;
sl@0
   449
    }
sl@0
   450
sl@0
   451
    simple_variant(T const &t)
sl@0
   452
      : is_rvalue(true)
sl@0
   453
    {
sl@0
   454
        ::new(this->data.address()) T(t);
sl@0
   455
    }
sl@0
   456
sl@0
   457
    simple_variant(simple_variant const &that)
sl@0
   458
      : is_rvalue(that.is_rvalue)
sl@0
   459
    {
sl@0
   460
        if(this->is_rvalue)
sl@0
   461
            ::new(this->data.address()) T(*that.get());
sl@0
   462
        else
sl@0
   463
            *static_cast<T const **>(this->data.address()) = that.get();
sl@0
   464
    }
sl@0
   465
sl@0
   466
    ~simple_variant()
sl@0
   467
    {
sl@0
   468
        if(this->is_rvalue)
sl@0
   469
            this->get()->~T();
sl@0
   470
    }
sl@0
   471
sl@0
   472
    T const *get() const
sl@0
   473
    {
sl@0
   474
        if(this->is_rvalue)
sl@0
   475
            return static_cast<T const *>(this->data.address());
sl@0
   476
        else
sl@0
   477
            return *static_cast<T const * const *>(this->data.address());
sl@0
   478
    }
sl@0
   479
sl@0
   480
private:
sl@0
   481
    enum size_type { size = sizeof(T) > sizeof(T*) ? sizeof(T) : sizeof(T*) };
sl@0
   482
    simple_variant &operator =(simple_variant const &); 
sl@0
   483
    bool const is_rvalue;
sl@0
   484
    aligned_storage<size> data;
sl@0
   485
};
sl@0
   486
sl@0
   487
// If the collection is an array or is noncopyable, it must be an lvalue.
sl@0
   488
// If the collection is a lightweight proxy, treat it as an rvalue
sl@0
   489
// BUGBUG what about a noncopyable proxy?
sl@0
   490
template<typename LValue, typename IsProxy>
sl@0
   491
inline BOOST_DEDUCED_TYPENAME boost::enable_if<boost::mpl::or_<LValue, IsProxy>, IsProxy>::type *
sl@0
   492
should_copy_impl(LValue *, IsProxy *, bool *)
sl@0
   493
{
sl@0
   494
    return 0;
sl@0
   495
}
sl@0
   496
sl@0
   497
// Otherwise, we must determine at runtime whether it's an lvalue or rvalue
sl@0
   498
inline bool *
sl@0
   499
should_copy_impl(boost::mpl::false_ *, boost::mpl::false_ *, bool *is_rvalue)
sl@0
   500
{
sl@0
   501
    return is_rvalue;
sl@0
   502
}
sl@0
   503
sl@0
   504
#endif
sl@0
   505
sl@0
   506
///////////////////////////////////////////////////////////////////////////////
sl@0
   507
// contain
sl@0
   508
//
sl@0
   509
template<typename T>
sl@0
   510
inline auto_any<T> contain(T const &t, boost::mpl::true_ *) // rvalue
sl@0
   511
{
sl@0
   512
    return t;
sl@0
   513
}
sl@0
   514
sl@0
   515
template<typename T>
sl@0
   516
inline auto_any<T *> contain(T &t, boost::mpl::false_ *) // lvalue
sl@0
   517
{
sl@0
   518
    // Cannot seem to get sunpro to handle addressof() with array types.
sl@0
   519
    #if BOOST_WORKAROUND(__SUNPRO_CC, BOOST_TESTED_AT(0x570))
sl@0
   520
    return &t;
sl@0
   521
    #else
sl@0
   522
    return boost::addressof(t);
sl@0
   523
    #endif
sl@0
   524
}
sl@0
   525
sl@0
   526
#ifdef BOOST_FOREACH_RUN_TIME_CONST_RVALUE_DETECTION
sl@0
   527
template<typename T>
sl@0
   528
auto_any<simple_variant<T> >
sl@0
   529
contain(T const &t, bool *rvalue)
sl@0
   530
{
sl@0
   531
    return *rvalue ? simple_variant<T>(t) : simple_variant<T>(&t);
sl@0
   532
}
sl@0
   533
#endif
sl@0
   534
sl@0
   535
/////////////////////////////////////////////////////////////////////////////
sl@0
   536
// begin
sl@0
   537
//
sl@0
   538
template<typename T, typename C>
sl@0
   539
inline auto_any<BOOST_DEDUCED_TYPENAME foreach_iterator<T, C>::type>
sl@0
   540
begin(auto_any_t col, type2type<T, C> *, boost::mpl::true_ *) // rvalue
sl@0
   541
{
sl@0
   542
    return boost::begin(auto_any_cast<T, C>(col));
sl@0
   543
}
sl@0
   544
sl@0
   545
template<typename T, typename C>
sl@0
   546
inline auto_any<BOOST_DEDUCED_TYPENAME foreach_iterator<T, C>::type>
sl@0
   547
begin(auto_any_t col, type2type<T, C> *, boost::mpl::false_ *) // lvalue
sl@0
   548
{
sl@0
   549
    typedef BOOST_DEDUCED_TYPENAME type2type<T, C>::type type;
sl@0
   550
    typedef BOOST_DEDUCED_TYPENAME foreach_iterator<T, C>::type iterator;
sl@0
   551
    return iterator(boost::begin(derefof(auto_any_cast<type *, boost::mpl::false_>(col))));
sl@0
   552
}
sl@0
   553
sl@0
   554
#ifdef BOOST_FOREACH_RUN_TIME_CONST_RVALUE_DETECTION
sl@0
   555
template<typename T>
sl@0
   556
auto_any<BOOST_DEDUCED_TYPENAME foreach_iterator<T, const_>::type>
sl@0
   557
begin(auto_any_t col, type2type<T, const_> *, bool *)
sl@0
   558
{
sl@0
   559
    return boost::begin(*auto_any_cast<simple_variant<T>, boost::mpl::false_>(col).get());
sl@0
   560
}
sl@0
   561
#endif
sl@0
   562
sl@0
   563
///////////////////////////////////////////////////////////////////////////////
sl@0
   564
// end
sl@0
   565
//
sl@0
   566
template<typename T, typename C>
sl@0
   567
inline auto_any<BOOST_DEDUCED_TYPENAME foreach_iterator<T, C>::type>
sl@0
   568
end(auto_any_t col, type2type<T, C> *, boost::mpl::true_ *) // rvalue
sl@0
   569
{
sl@0
   570
    return boost::end(auto_any_cast<T, C>(col));
sl@0
   571
}
sl@0
   572
sl@0
   573
template<typename T, typename C>
sl@0
   574
inline auto_any<BOOST_DEDUCED_TYPENAME foreach_iterator<T, C>::type>
sl@0
   575
end(auto_any_t col, type2type<T, C> *, boost::mpl::false_ *) // lvalue
sl@0
   576
{
sl@0
   577
    typedef BOOST_DEDUCED_TYPENAME type2type<T, C>::type type;
sl@0
   578
    typedef BOOST_DEDUCED_TYPENAME foreach_iterator<T, C>::type iterator;
sl@0
   579
    return iterator(boost::end(derefof(auto_any_cast<type *, boost::mpl::false_>(col))));
sl@0
   580
}
sl@0
   581
sl@0
   582
#ifdef BOOST_FOREACH_RUN_TIME_CONST_RVALUE_DETECTION
sl@0
   583
template<typename T>
sl@0
   584
auto_any<BOOST_DEDUCED_TYPENAME foreach_iterator<T, const_>::type>
sl@0
   585
end(auto_any_t col, type2type<T, const_> *, bool *)
sl@0
   586
{
sl@0
   587
    return boost::end(*auto_any_cast<simple_variant<T>, boost::mpl::false_>(col).get());
sl@0
   588
}
sl@0
   589
#endif
sl@0
   590
sl@0
   591
#ifndef BOOST_NO_FUNCTION_TEMPLATE_ORDERING
sl@0
   592
template<typename T, typename C>
sl@0
   593
inline auto_any<int>
sl@0
   594
end(auto_any_t col, type2type<T *, C> *, boost::mpl::true_ *) // null-terminated C-style strings
sl@0
   595
{
sl@0
   596
    return 0; // not used
sl@0
   597
}
sl@0
   598
#endif
sl@0
   599
sl@0
   600
///////////////////////////////////////////////////////////////////////////////
sl@0
   601
// done
sl@0
   602
//
sl@0
   603
template<typename T, typename C>
sl@0
   604
inline bool done(auto_any_t cur, auto_any_t end, type2type<T, C> *)
sl@0
   605
{
sl@0
   606
    typedef BOOST_DEDUCED_TYPENAME foreach_iterator<T, C>::type iter_t;
sl@0
   607
    return auto_any_cast<iter_t, boost::mpl::false_>(cur) == auto_any_cast<iter_t, boost::mpl::false_>(end);
sl@0
   608
}
sl@0
   609
sl@0
   610
#ifndef BOOST_NO_FUNCTION_TEMPLATE_ORDERING
sl@0
   611
template<typename T, typename C>
sl@0
   612
inline bool done(auto_any_t cur, auto_any_t, type2type<T *, C> *) // null-terminated C-style strings
sl@0
   613
{
sl@0
   614
    return ! *auto_any_cast<T *, boost::mpl::false_>(cur);
sl@0
   615
}
sl@0
   616
#endif
sl@0
   617
sl@0
   618
///////////////////////////////////////////////////////////////////////////////
sl@0
   619
// next
sl@0
   620
//
sl@0
   621
template<typename T, typename C>
sl@0
   622
inline void next(auto_any_t cur, type2type<T, C> *)
sl@0
   623
{
sl@0
   624
    typedef BOOST_DEDUCED_TYPENAME foreach_iterator<T, C>::type iter_t;
sl@0
   625
    ++auto_any_cast<iter_t, boost::mpl::false_>(cur);
sl@0
   626
}
sl@0
   627
sl@0
   628
///////////////////////////////////////////////////////////////////////////////
sl@0
   629
// deref
sl@0
   630
//
sl@0
   631
template<typename T, typename C>
sl@0
   632
inline BOOST_DEDUCED_TYPENAME foreach_reference<T, C>::type
sl@0
   633
deref(auto_any_t cur, type2type<T, C> *)
sl@0
   634
{
sl@0
   635
    typedef BOOST_DEDUCED_TYPENAME foreach_iterator<T, C>::type iter_t;
sl@0
   636
    return *auto_any_cast<iter_t, boost::mpl::false_>(cur);
sl@0
   637
}
sl@0
   638
sl@0
   639
} // namespace foreach_detail_
sl@0
   640
} // namespace boost
sl@0
   641
sl@0
   642
// A sneaky way to get the type of the collection without evaluating the expression
sl@0
   643
#define BOOST_FOREACH_TYPEOF(COL)                                                               \
sl@0
   644
    (true ? 0 : boost::foreach_detail_::encode_type(COL, boost::foreach_detail_::is_const_(COL)))
sl@0
   645
sl@0
   646
// returns true_* if the type is noncopyable
sl@0
   647
#define BOOST_FOREACH_IS_NONCOPYABLE(COL)                                                       \
sl@0
   648
    boost_foreach_is_noncopyable(                                                               \
sl@0
   649
        boost::foreach_detail_::to_ptr(COL)                                                     \
sl@0
   650
      , boost_foreach_argument_dependent_lookup_hack_value)
sl@0
   651
sl@0
   652
// returns true_* if the type is a lightweight proxy (and is not noncopyable)
sl@0
   653
#define BOOST_FOREACH_IS_LIGHTWEIGHT_PROXY(COL)                                                 \
sl@0
   654
    boost::foreach_detail_::and_(                                                               \
sl@0
   655
        boost::foreach_detail_::not_(BOOST_FOREACH_IS_NONCOPYABLE(COL))                         \
sl@0
   656
      , boost_foreach_is_lightweight_proxy(                                                     \
sl@0
   657
            boost::foreach_detail_::to_ptr(COL)                                                 \
sl@0
   658
          , boost_foreach_argument_dependent_lookup_hack_value))
sl@0
   659
sl@0
   660
#ifdef BOOST_FOREACH_COMPILE_TIME_CONST_RVALUE_DETECTION
sl@0
   661
///////////////////////////////////////////////////////////////////////////////
sl@0
   662
// R-values and const R-values supported here with zero runtime overhead
sl@0
   663
///////////////////////////////////////////////////////////////////////////////
sl@0
   664
sl@0
   665
// No variable is needed to track the rvalue-ness of the collection expression
sl@0
   666
# define BOOST_FOREACH_PREAMBLE()                                                               \
sl@0
   667
    /**/
sl@0
   668
sl@0
   669
// Evaluate the collection expression
sl@0
   670
# define BOOST_FOREACH_EVALUATE(COL)                                                            \
sl@0
   671
    (COL)
sl@0
   672
sl@0
   673
# define BOOST_FOREACH_SHOULD_COPY(COL)                                                         \
sl@0
   674
    (true ? 0 : boost::foreach_detail_::or_(                                                    \
sl@0
   675
        BOOST_FOREACH_IS_RVALUE(COL)                                                            \
sl@0
   676
      , BOOST_FOREACH_IS_LIGHTWEIGHT_PROXY(COL)))
sl@0
   677
sl@0
   678
#elif defined(BOOST_FOREACH_RUN_TIME_CONST_RVALUE_DETECTION)
sl@0
   679
///////////////////////////////////////////////////////////////////////////////
sl@0
   680
// R-values and const R-values supported here
sl@0
   681
///////////////////////////////////////////////////////////////////////////////
sl@0
   682
sl@0
   683
// Declare a variable to track the rvalue-ness of the collection expression
sl@0
   684
# define BOOST_FOREACH_PREAMBLE()                                                               \
sl@0
   685
    if (bool _foreach_is_rvalue = false) {} else
sl@0
   686
sl@0
   687
// Evaluate the collection expression, and detect if it is an lvalue or and rvalue
sl@0
   688
# define BOOST_FOREACH_EVALUATE(COL)                                                            \
sl@0
   689
    (true ? boost::foreach_detail_::make_probe((COL), _foreach_is_rvalue) : (COL))
sl@0
   690
sl@0
   691
// The rvalue/lvalue-ness of the collection expression is determined dynamically, unless
sl@0
   692
// type type is an array or is noncopyable or is non-const, in which case we know it's an lvalue.
sl@0
   693
// If the type happens to be a lightweight proxy, always make a copy.
sl@0
   694
# define BOOST_FOREACH_SHOULD_COPY(COL)                                                         \
sl@0
   695
    (boost::foreach_detail_::should_copy_impl(                                                  \
sl@0
   696
        true ? 0 : boost::foreach_detail_::or_(                                                 \
sl@0
   697
            boost::foreach_detail_::is_array_(COL)                                              \
sl@0
   698
          , BOOST_FOREACH_IS_NONCOPYABLE(COL)                                                   \
sl@0
   699
          , boost::foreach_detail_::not_(boost::foreach_detail_::is_const_(COL)))               \
sl@0
   700
      , true ? 0 : BOOST_FOREACH_IS_LIGHTWEIGHT_PROXY(COL)                                      \
sl@0
   701
      , &_foreach_is_rvalue))
sl@0
   702
sl@0
   703
#elif !defined(BOOST_FOREACH_NO_RVALUE_DETECTION)
sl@0
   704
///////////////////////////////////////////////////////////////////////////////
sl@0
   705
// R-values supported here, const R-values NOT supported here
sl@0
   706
///////////////////////////////////////////////////////////////////////////////
sl@0
   707
sl@0
   708
// No variable is needed to track the rvalue-ness of the collection expression
sl@0
   709
# define BOOST_FOREACH_PREAMBLE()                                                               \
sl@0
   710
    /**/
sl@0
   711
sl@0
   712
// Evaluate the collection expression
sl@0
   713
# define BOOST_FOREACH_EVALUATE(COL)                                                            \
sl@0
   714
    (COL)
sl@0
   715
sl@0
   716
// Determine whether the collection expression is an lvalue or an rvalue.
sl@0
   717
// NOTE: this gets the answer wrong for const rvalues.
sl@0
   718
# define BOOST_FOREACH_SHOULD_COPY(COL)                                                         \
sl@0
   719
    (true ? 0 : boost::foreach_detail_::or_(                                                    \
sl@0
   720
        boost::foreach_detail_::is_rvalue_((COL), 0)                                            \
sl@0
   721
      , BOOST_FOREACH_IS_LIGHTWEIGHT_PROXY(COL)))
sl@0
   722
sl@0
   723
#else
sl@0
   724
///////////////////////////////////////////////////////////////////////////////
sl@0
   725
// R-values NOT supported here
sl@0
   726
///////////////////////////////////////////////////////////////////////////////
sl@0
   727
sl@0
   728
// No variable is needed to track the rvalue-ness of the collection expression
sl@0
   729
# define BOOST_FOREACH_PREAMBLE()                                                               \
sl@0
   730
    /**/
sl@0
   731
sl@0
   732
// Evaluate the collection expression
sl@0
   733
# define BOOST_FOREACH_EVALUATE(COL)                                                            \
sl@0
   734
    (COL)
sl@0
   735
sl@0
   736
// Can't use rvalues with BOOST_FOREACH (unless they are lightweight proxies)
sl@0
   737
# define BOOST_FOREACH_SHOULD_COPY(COL)                                                         \
sl@0
   738
    (true ? 0 : BOOST_FOREACH_IS_LIGHTWEIGHT_PROXY(COL))
sl@0
   739
sl@0
   740
#endif
sl@0
   741
sl@0
   742
#define BOOST_FOREACH_CONTAIN(COL)                                                              \
sl@0
   743
    boost::foreach_detail_::contain(                                                            \
sl@0
   744
        BOOST_FOREACH_EVALUATE(COL)                                                             \
sl@0
   745
      , BOOST_FOREACH_SHOULD_COPY(COL))
sl@0
   746
sl@0
   747
#define BOOST_FOREACH_BEGIN(COL)                                                                \
sl@0
   748
    boost::foreach_detail_::begin(                                                              \
sl@0
   749
        _foreach_col                                                                            \
sl@0
   750
      , BOOST_FOREACH_TYPEOF(COL)                                                               \
sl@0
   751
      , BOOST_FOREACH_SHOULD_COPY(COL))
sl@0
   752
sl@0
   753
#define BOOST_FOREACH_END(COL)                                                                  \
sl@0
   754
    boost::foreach_detail_::end(                                                                \
sl@0
   755
        _foreach_col                                                                            \
sl@0
   756
      , BOOST_FOREACH_TYPEOF(COL)                                                               \
sl@0
   757
      , BOOST_FOREACH_SHOULD_COPY(COL))
sl@0
   758
sl@0
   759
#define BOOST_FOREACH_DONE(COL)                                                                 \
sl@0
   760
    boost::foreach_detail_::done(                                                               \
sl@0
   761
        _foreach_cur                                                                            \
sl@0
   762
      , _foreach_end                                                                            \
sl@0
   763
      , BOOST_FOREACH_TYPEOF(COL))
sl@0
   764
sl@0
   765
#define BOOST_FOREACH_NEXT(COL)                                                                 \
sl@0
   766
    boost::foreach_detail_::next(                                                               \
sl@0
   767
        _foreach_cur                                                                            \
sl@0
   768
      , BOOST_FOREACH_TYPEOF(COL))
sl@0
   769
sl@0
   770
#define BOOST_FOREACH_DEREF(COL)                                                                \
sl@0
   771
    boost::foreach_detail_::deref(                                                              \
sl@0
   772
        _foreach_cur                                                                            \
sl@0
   773
      , BOOST_FOREACH_TYPEOF(COL))
sl@0
   774
sl@0
   775
///////////////////////////////////////////////////////////////////////////////
sl@0
   776
// BOOST_FOREACH
sl@0
   777
//
sl@0
   778
//   For iterating over collections. Collections can be
sl@0
   779
//   arrays, null-terminated strings, or STL containers.
sl@0
   780
//   The loop variable can be a value or reference. For
sl@0
   781
//   example:
sl@0
   782
//
sl@0
   783
//   std::list<int> int_list(/*stuff*/);
sl@0
   784
//   BOOST_FOREACH(int &i, int_list)
sl@0
   785
//   {
sl@0
   786
//       /* 
sl@0
   787
//        * loop body goes here.
sl@0
   788
//        * i is a reference to the int in int_list.
sl@0
   789
//        */
sl@0
   790
//   }
sl@0
   791
//
sl@0
   792
//   Alternately, you can declare the loop variable first,
sl@0
   793
//   so you can access it after the loop finishes. Obviously,
sl@0
   794
//   if you do it this way, then the loop variable cannot be
sl@0
   795
//   a reference.
sl@0
   796
//
sl@0
   797
//   int i;
sl@0
   798
//   BOOST_FOREACH(i, int_list)
sl@0
   799
//       { ... }
sl@0
   800
//
sl@0
   801
#define BOOST_FOREACH(VAR, COL)                                                                 \
sl@0
   802
    BOOST_FOREACH_PREAMBLE()                                                                    \
sl@0
   803
    if (boost::foreach_detail_::auto_any_t _foreach_col = BOOST_FOREACH_CONTAIN(COL)) {} else   \
sl@0
   804
    if (boost::foreach_detail_::auto_any_t _foreach_cur = BOOST_FOREACH_BEGIN(COL)) {} else     \
sl@0
   805
    if (boost::foreach_detail_::auto_any_t _foreach_end = BOOST_FOREACH_END(COL)) {} else       \
sl@0
   806
    for (bool _foreach_continue = true;                                                         \
sl@0
   807
              _foreach_continue && !BOOST_FOREACH_DONE(COL);                                    \
sl@0
   808
              _foreach_continue ? BOOST_FOREACH_NEXT(COL) : (void)0)                            \
sl@0
   809
        if  (boost::foreach_detail_::set_false(_foreach_continue)) {} else                      \
sl@0
   810
        for (VAR = BOOST_FOREACH_DEREF(COL); !_foreach_continue; _foreach_continue = true)
sl@0
   811
sl@0
   812
#endif