os/ossrv/ossrv_pub/boost_apis/boost/multi_index_container.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
/* Multiply indexed container.
sl@0
     2
 *
sl@0
     3
 * Copyright 2003-2007 Joaquín M López Muñoz.
sl@0
     4
 * Distributed under the Boost Software License, Version 1.0.
sl@0
     5
 * (See accompanying file LICENSE_1_0.txt or copy at
sl@0
     6
 * http://www.boost.org/LICENSE_1_0.txt)
sl@0
     7
 *
sl@0
     8
 * See http://www.boost.org/libs/multi_index for library home page.
sl@0
     9
 */
sl@0
    10
sl@0
    11
#ifndef BOOST_MULTI_INDEX_HPP
sl@0
    12
#define BOOST_MULTI_INDEX_HPP
sl@0
    13
sl@0
    14
#if defined(_MSC_VER)&&(_MSC_VER>=1200)
sl@0
    15
#pragma once
sl@0
    16
#endif
sl@0
    17
sl@0
    18
#include <boost/config.hpp> /* keep it first to prevent nasty warns in MSVC */
sl@0
    19
#include <algorithm>
sl@0
    20
#include <boost/detail/allocator_utilities.hpp>
sl@0
    21
#include <boost/detail/no_exceptions_support.hpp>
sl@0
    22
#include <boost/detail/workaround.hpp>
sl@0
    23
#include <boost/mpl/at.hpp>
sl@0
    24
#include <boost/mpl/contains.hpp>
sl@0
    25
#include <boost/mpl/find_if.hpp>
sl@0
    26
#include <boost/mpl/identity.hpp>
sl@0
    27
#include <boost/mpl/int.hpp>
sl@0
    28
#include <boost/mpl/size.hpp>
sl@0
    29
#include <boost/mpl/deref.hpp>
sl@0
    30
#include <boost/multi_index_container_fwd.hpp>
sl@0
    31
#include <boost/multi_index/detail/access_specifier.hpp>
sl@0
    32
#include <boost/multi_index/detail/base_type.hpp>
sl@0
    33
#include <boost/multi_index/detail/converter.hpp>
sl@0
    34
#include <boost/multi_index/detail/def_ctor_tuple_cons.hpp>
sl@0
    35
#include <boost/multi_index/detail/header_holder.hpp>
sl@0
    36
#include <boost/multi_index/detail/has_tag.hpp>
sl@0
    37
#include <boost/multi_index/detail/no_duplicate_tags.hpp>
sl@0
    38
#include <boost/multi_index/detail/prevent_eti.hpp>
sl@0
    39
#include <boost/multi_index/detail/safe_mode.hpp>
sl@0
    40
#include <boost/multi_index/detail/scope_guard.hpp>
sl@0
    41
#include <boost/static_assert.hpp>
sl@0
    42
#include <boost/type_traits/is_same.hpp>
sl@0
    43
#include <boost/utility/base_from_member.hpp>
sl@0
    44
sl@0
    45
#if !defined(BOOST_MULTI_INDEX_DISABLE_SERIALIZATION)
sl@0
    46
#include <boost/multi_index/detail/archive_constructed.hpp>
sl@0
    47
#include <boost/serialization/nvp.hpp>
sl@0
    48
#include <boost/serialization/split_member.hpp>
sl@0
    49
#include <boost/throw_exception.hpp> 
sl@0
    50
#endif
sl@0
    51
sl@0
    52
#if defined(BOOST_MULTI_INDEX_ENABLE_INVARIANT_CHECKING)
sl@0
    53
#include <boost/multi_index/detail/invariant_assert.hpp>
sl@0
    54
#define BOOST_MULTI_INDEX_CHECK_INVARIANT                                    \
sl@0
    55
  detail::scope_guard BOOST_JOIN(check_invariant_,__LINE__)=                 \
sl@0
    56
    detail::make_obj_guard(*this,&multi_index_container::check_invariant_);  \
sl@0
    57
  BOOST_JOIN(check_invariant_,__LINE__).touch();
sl@0
    58
#else
sl@0
    59
#define BOOST_MULTI_INDEX_CHECK_INVARIANT
sl@0
    60
#endif
sl@0
    61
sl@0
    62
namespace boost{
sl@0
    63
sl@0
    64
namespace multi_index{
sl@0
    65
sl@0
    66
template<typename Value,typename IndexSpecifierList,typename Allocator>
sl@0
    67
class multi_index_container:
sl@0
    68
  private ::boost::base_from_member<
sl@0
    69
    typename boost::detail::allocator::rebind_to<
sl@0
    70
      Allocator,
sl@0
    71
      typename detail::multi_index_node_type<
sl@0
    72
        Value,IndexSpecifierList,Allocator>::type
sl@0
    73
    >::type>,
sl@0
    74
  BOOST_MULTI_INDEX_PRIVATE_IF_MEMBER_TEMPLATE_FRIENDS detail::header_holder<
sl@0
    75
    typename detail::multi_index_node_type<
sl@0
    76
      Value,IndexSpecifierList,Allocator>::type,
sl@0
    77
    multi_index_container<Value,IndexSpecifierList,Allocator> >,
sl@0
    78
  public detail::multi_index_base_type<
sl@0
    79
    Value,IndexSpecifierList,Allocator>::type
sl@0
    80
{
sl@0
    81
#if defined(BOOST_MULTI_INDEX_ENABLE_INVARIANT_CHECKING)&&\
sl@0
    82
    BOOST_WORKAROUND(__MWERKS__,<=0x3003)
sl@0
    83
/* The "ISO C++ Template Parser" option in CW8.3 has a problem with the
sl@0
    84
 * lifetime of const references bound to temporaries --precisely what
sl@0
    85
 * scopeguards are.
sl@0
    86
 */
sl@0
    87
sl@0
    88
#pragma parse_mfunc_templ off
sl@0
    89
#endif
sl@0
    90
sl@0
    91
private:
sl@0
    92
#if !defined(BOOST_NO_MEMBER_TEMPLATE_FRIENDS)
sl@0
    93
  template <typename,typename,typename> friend class  detail::index_base;
sl@0
    94
  template <typename,typename>          friend class  detail::header_holder;
sl@0
    95
  template <typename,typename>          friend class  detail::converter;
sl@0
    96
#endif
sl@0
    97
sl@0
    98
  typedef typename detail::multi_index_base_type<
sl@0
    99
      Value,IndexSpecifierList,Allocator>::type   super;
sl@0
   100
  typedef ::boost::base_from_member<
sl@0
   101
    typename boost::detail::allocator::rebind_to<
sl@0
   102
      Allocator,
sl@0
   103
      typename super::node_type
sl@0
   104
    >::type>                                      bfm_allocator;
sl@0
   105
  typedef detail::header_holder<
sl@0
   106
    typename super::node_type,
sl@0
   107
    multi_index_container>                        bfm_header;
sl@0
   108
sl@0
   109
#if BOOST_WORKAROUND(BOOST_MSVC,<1300)
sl@0
   110
  /* see definition of index_type_list below */
sl@0
   111
  typedef typename super::index_type_list         super_index_type_list;
sl@0
   112
#endif
sl@0
   113
sl@0
   114
public:
sl@0
   115
  /* All types are inherited from super, a few are explicitly
sl@0
   116
   * brought forward here to save us some typename's.
sl@0
   117
   */
sl@0
   118
sl@0
   119
#if defined(BOOST_MSVC)
sl@0
   120
  typedef 
sl@0
   121
    detail::default_constructible_tuple_cons<
sl@0
   122
      typename super::ctor_args_list>              ctor_args_list;
sl@0
   123
#else
sl@0
   124
  typedef typename super::ctor_args_list           ctor_args_list;
sl@0
   125
#endif
sl@0
   126
sl@0
   127
  typedef IndexSpecifierList                       index_specifier_type_list;
sl@0
   128
 
sl@0
   129
#if BOOST_WORKAROUND(BOOST_MSVC,<1300)
sl@0
   130
  /* MSVC++ 6.0 chokes on moderately long index lists (around 6 indices
sl@0
   131
   * or more), with errors ranging from corrupt exes to duplicate
sl@0
   132
   * comdats. The following type hiding hack alleviates this condition;
sl@0
   133
   * best results combined with type hiding of the indexed_by construct
sl@0
   134
   * itself, as explained in the "Compiler specifics" section of
sl@0
   135
   * the documentation.
sl@0
   136
   */
sl@0
   137
sl@0
   138
  struct index_type_list:super_index_type_list
sl@0
   139
  {
sl@0
   140
    typedef index_type_list                      type;
sl@0
   141
    typedef typename super_index_type_list::back back;
sl@0
   142
    typedef mpl::v_iter<type,0>                  begin;
sl@0
   143
    typedef mpl::v_iter<
sl@0
   144
      type,
sl@0
   145
      mpl::size<super_index_type_list>::value>   end;
sl@0
   146
  };
sl@0
   147
#else
sl@0
   148
  typedef typename super::index_type_list          index_type_list;
sl@0
   149
#endif
sl@0
   150
sl@0
   151
  typedef typename super::iterator_type_list       iterator_type_list;
sl@0
   152
  typedef typename super::const_iterator_type_list const_iterator_type_list;
sl@0
   153
  typedef typename super::value_type               value_type;
sl@0
   154
  typedef typename super::final_allocator_type     allocator_type;
sl@0
   155
  typedef typename super::iterator                 iterator;
sl@0
   156
  typedef typename super::const_iterator           const_iterator;
sl@0
   157
sl@0
   158
  BOOST_STATIC_ASSERT(
sl@0
   159
    detail::no_duplicate_tags_in_index_list<index_type_list>::value);
sl@0
   160
sl@0
   161
  /* global project() needs to see this publicly */
sl@0
   162
sl@0
   163
  typedef typename super::node_type node_type;
sl@0
   164
sl@0
   165
  /* construct/copy/destroy */
sl@0
   166
sl@0
   167
  explicit multi_index_container(
sl@0
   168
sl@0
   169
#if BOOST_WORKAROUND(__IBMCPP__,<=600)
sl@0
   170
    /* VisualAge seems to have an ETI issue with the default values
sl@0
   171
     * for arguments args_list and al.
sl@0
   172
     */
sl@0
   173
sl@0
   174
    const ctor_args_list& args_list=
sl@0
   175
      typename mpl::identity<multi_index_container>::type::
sl@0
   176
        ctor_args_list(),
sl@0
   177
    const allocator_type& al=
sl@0
   178
      typename mpl::identity<multi_index_container>::type::
sl@0
   179
        allocator_type()):
sl@0
   180
#else
sl@0
   181
    const ctor_args_list& args_list=ctor_args_list(),
sl@0
   182
    const allocator_type& al=allocator_type()):
sl@0
   183
#endif
sl@0
   184
sl@0
   185
    bfm_allocator(al),
sl@0
   186
    super(args_list,bfm_allocator::member),
sl@0
   187
    node_count(0)
sl@0
   188
  {
sl@0
   189
    BOOST_MULTI_INDEX_CHECK_INVARIANT;
sl@0
   190
  }    
sl@0
   191
sl@0
   192
  template<typename InputIterator>
sl@0
   193
  multi_index_container(
sl@0
   194
    InputIterator first,InputIterator last,
sl@0
   195
sl@0
   196
#if BOOST_WORKAROUND(__IBMCPP__,<=600)
sl@0
   197
    /* VisualAge seems to have an ETI issue with the default values
sl@0
   198
     * for arguments args_list and al.
sl@0
   199
     */
sl@0
   200
sl@0
   201
    const ctor_args_list& args_list=
sl@0
   202
      typename mpl::identity<multi_index_container>::type::
sl@0
   203
        ctor_args_list(),
sl@0
   204
    const allocator_type& al=
sl@0
   205
      typename mpl::identity<multi_index_container>::type::
sl@0
   206
        allocator_type()):
sl@0
   207
#else
sl@0
   208
    const ctor_args_list& args_list=ctor_args_list(),
sl@0
   209
    const allocator_type& al=allocator_type()):
sl@0
   210
#endif
sl@0
   211
sl@0
   212
    bfm_allocator(al),
sl@0
   213
    super(args_list,bfm_allocator::member),
sl@0
   214
    node_count(0)
sl@0
   215
  {
sl@0
   216
    BOOST_MULTI_INDEX_CHECK_INVARIANT;
sl@0
   217
    BOOST_TRY{
sl@0
   218
      iterator hint=super::end();
sl@0
   219
      for(;first!=last;++first){
sl@0
   220
        hint=super::make_iterator(insert_(*first,hint.get_node()).first);
sl@0
   221
      }
sl@0
   222
    }
sl@0
   223
    BOOST_CATCH(...){
sl@0
   224
      clear_();
sl@0
   225
      BOOST_RETHROW;
sl@0
   226
    }
sl@0
   227
    BOOST_CATCH_END
sl@0
   228
  }
sl@0
   229
sl@0
   230
  multi_index_container(
sl@0
   231
    const multi_index_container<Value,IndexSpecifierList,Allocator>& x):
sl@0
   232
    bfm_allocator(x.bfm_allocator::member),
sl@0
   233
    bfm_header(),
sl@0
   234
    super(x),
sl@0
   235
    node_count(0)
sl@0
   236
  {
sl@0
   237
    copy_map_type map(bfm_allocator::member,x.size(),x.header(),header());
sl@0
   238
    for(const_iterator it=x.begin(),it_end=x.end();it!=it_end;++it){
sl@0
   239
      map.clone(it.get_node());
sl@0
   240
    }
sl@0
   241
    super::copy_(x,map);
sl@0
   242
    map.release();
sl@0
   243
    node_count=x.size();
sl@0
   244
sl@0
   245
    /* Not until this point are the indices required to be consistent,
sl@0
   246
     * hence the position of the invariant checker.
sl@0
   247
     */
sl@0
   248
sl@0
   249
    BOOST_MULTI_INDEX_CHECK_INVARIANT;
sl@0
   250
  }
sl@0
   251
sl@0
   252
  ~multi_index_container()
sl@0
   253
  {
sl@0
   254
    delete_all_nodes_();
sl@0
   255
  }
sl@0
   256
sl@0
   257
  multi_index_container<Value,IndexSpecifierList,Allocator>& operator=(
sl@0
   258
    const multi_index_container<Value,IndexSpecifierList,Allocator>& x)
sl@0
   259
  {
sl@0
   260
    BOOST_MULTI_INDEX_CHECK_INVARIANT;
sl@0
   261
    multi_index_container<Value,IndexSpecifierList,Allocator> tmp(x);
sl@0
   262
    this->swap(tmp);
sl@0
   263
    return *this;
sl@0
   264
  }
sl@0
   265
sl@0
   266
  allocator_type get_allocator()const
sl@0
   267
  {
sl@0
   268
    return allocator_type(bfm_allocator::member);
sl@0
   269
  }
sl@0
   270
sl@0
   271
  /* retrieval of indices by number */
sl@0
   272
sl@0
   273
#if !defined(BOOST_NO_MEMBER_TEMPLATES)
sl@0
   274
  template<int N>
sl@0
   275
  struct nth_index
sl@0
   276
  {
sl@0
   277
    BOOST_STATIC_ASSERT(N>=0&&N<mpl::size<index_type_list>::type::value);
sl@0
   278
    typedef typename mpl::at_c<index_type_list,N>::type type;
sl@0
   279
  };
sl@0
   280
sl@0
   281
  template<int N>
sl@0
   282
  typename nth_index<N>::type& get(BOOST_EXPLICIT_TEMPLATE_NON_TYPE(int,N))
sl@0
   283
  {
sl@0
   284
    BOOST_STATIC_ASSERT(N>=0&&N<mpl::size<index_type_list>::type::value);
sl@0
   285
    return *this;
sl@0
   286
  }
sl@0
   287
sl@0
   288
  template<int N>
sl@0
   289
  const typename nth_index<N>::type& get(
sl@0
   290
    BOOST_EXPLICIT_TEMPLATE_NON_TYPE(int,N))const
sl@0
   291
  {
sl@0
   292
    BOOST_STATIC_ASSERT(N>=0&&N<mpl::size<index_type_list>::type::value);
sl@0
   293
    return *this;
sl@0
   294
  }
sl@0
   295
#endif
sl@0
   296
sl@0
   297
  /* retrieval of indices by tag */
sl@0
   298
sl@0
   299
#if !defined(BOOST_NO_MEMBER_TEMPLATES)
sl@0
   300
  template<typename Tag>
sl@0
   301
  struct index
sl@0
   302
  {
sl@0
   303
    typedef typename mpl::find_if<
sl@0
   304
      index_type_list,
sl@0
   305
      detail::has_tag<Tag>
sl@0
   306
    >::type                                    iter;
sl@0
   307
sl@0
   308
    BOOST_STATIC_CONSTANT(
sl@0
   309
      bool,index_found=!(is_same<iter,typename mpl::end<index_type_list>::type >::value));
sl@0
   310
    BOOST_STATIC_ASSERT(index_found);
sl@0
   311
sl@0
   312
    typedef typename mpl::deref<iter>::type    type;
sl@0
   313
  };
sl@0
   314
sl@0
   315
  template<typename Tag>
sl@0
   316
  typename index<Tag>::type& get(BOOST_EXPLICIT_TEMPLATE_TYPE(Tag))
sl@0
   317
  {
sl@0
   318
    return *this;
sl@0
   319
  }
sl@0
   320
sl@0
   321
  template<typename Tag>
sl@0
   322
  const typename index<Tag>::type& get(
sl@0
   323
    BOOST_EXPLICIT_TEMPLATE_TYPE(Tag))const
sl@0
   324
  {
sl@0
   325
    return *this;
sl@0
   326
  }
sl@0
   327
#endif
sl@0
   328
sl@0
   329
  /* projection of iterators by number */
sl@0
   330
sl@0
   331
#if !defined(BOOST_NO_MEMBER_TEMPLATES)
sl@0
   332
  template<int N>
sl@0
   333
  struct nth_index_iterator
sl@0
   334
  {
sl@0
   335
    typedef typename nth_index<N>::type::iterator type;
sl@0
   336
  };
sl@0
   337
sl@0
   338
  template<int N>
sl@0
   339
  struct nth_index_const_iterator
sl@0
   340
  {
sl@0
   341
    typedef typename nth_index<N>::type::const_iterator type;
sl@0
   342
  };
sl@0
   343
sl@0
   344
  template<int N,typename IteratorType>
sl@0
   345
  typename nth_index_iterator<N>::type project(
sl@0
   346
    IteratorType it
sl@0
   347
    BOOST_APPEND_EXPLICIT_TEMPLATE_NON_TYPE(int,N))
sl@0
   348
  {
sl@0
   349
    typedef typename nth_index<N>::type index;
sl@0
   350
sl@0
   351
    BOOST_STATIC_ASSERT(
sl@0
   352
      (mpl::contains<iterator_type_list,IteratorType>::value));
sl@0
   353
sl@0
   354
    BOOST_MULTI_INDEX_CHECK_VALID_ITERATOR(it);
sl@0
   355
    BOOST_MULTI_INDEX_CHECK_IS_OWNER(
sl@0
   356
      it,static_cast<typename IteratorType::container_type&>(*this));
sl@0
   357
sl@0
   358
    return index::make_iterator(static_cast<node_type*>(it.get_node()));
sl@0
   359
  }
sl@0
   360
sl@0
   361
  template<int N,typename IteratorType>
sl@0
   362
  typename nth_index_const_iterator<N>::type project(
sl@0
   363
    IteratorType it
sl@0
   364
    BOOST_APPEND_EXPLICIT_TEMPLATE_NON_TYPE(int,N))const
sl@0
   365
  {
sl@0
   366
    typedef typename nth_index<N>::type index;
sl@0
   367
sl@0
   368
    BOOST_STATIC_ASSERT((
sl@0
   369
      mpl::contains<iterator_type_list,IteratorType>::value||
sl@0
   370
      mpl::contains<const_iterator_type_list,IteratorType>::value));
sl@0
   371
sl@0
   372
    BOOST_MULTI_INDEX_CHECK_VALID_ITERATOR(it);
sl@0
   373
    BOOST_MULTI_INDEX_CHECK_IS_OWNER(
sl@0
   374
      it,static_cast<const typename IteratorType::container_type&>(*this));
sl@0
   375
    return index::make_iterator(static_cast<node_type*>(it.get_node()));
sl@0
   376
  }
sl@0
   377
#endif
sl@0
   378
sl@0
   379
  /* projection of iterators by tag */
sl@0
   380
sl@0
   381
#if !defined(BOOST_NO_MEMBER_TEMPLATES)
sl@0
   382
  template<typename Tag>
sl@0
   383
  struct index_iterator
sl@0
   384
  {
sl@0
   385
    typedef typename index<Tag>::type::iterator type;
sl@0
   386
  };
sl@0
   387
sl@0
   388
  template<typename Tag>
sl@0
   389
  struct index_const_iterator
sl@0
   390
  {
sl@0
   391
    typedef typename index<Tag>::type::const_iterator type;
sl@0
   392
  };
sl@0
   393
sl@0
   394
  template<typename Tag,typename IteratorType>
sl@0
   395
  typename index_iterator<Tag>::type project(
sl@0
   396
    IteratorType it
sl@0
   397
    BOOST_APPEND_EXPLICIT_TEMPLATE_TYPE(Tag))
sl@0
   398
  {
sl@0
   399
    typedef typename index<Tag>::type index;
sl@0
   400
sl@0
   401
    BOOST_STATIC_ASSERT(
sl@0
   402
      (mpl::contains<iterator_type_list,IteratorType>::value));
sl@0
   403
sl@0
   404
    BOOST_MULTI_INDEX_CHECK_VALID_ITERATOR(it);
sl@0
   405
    BOOST_MULTI_INDEX_CHECK_IS_OWNER(
sl@0
   406
      it,static_cast<typename IteratorType::container_type&>(*this));
sl@0
   407
    return index::make_iterator(static_cast<node_type*>(it.get_node()));
sl@0
   408
  }
sl@0
   409
sl@0
   410
  template<typename Tag,typename IteratorType>
sl@0
   411
  typename index_const_iterator<Tag>::type project(
sl@0
   412
    IteratorType it
sl@0
   413
    BOOST_APPEND_EXPLICIT_TEMPLATE_TYPE(Tag))const
sl@0
   414
  {
sl@0
   415
    typedef typename index<Tag>::type index;
sl@0
   416
sl@0
   417
    BOOST_STATIC_ASSERT((
sl@0
   418
      mpl::contains<iterator_type_list,IteratorType>::value||
sl@0
   419
      mpl::contains<const_iterator_type_list,IteratorType>::value));
sl@0
   420
sl@0
   421
    BOOST_MULTI_INDEX_CHECK_VALID_ITERATOR(it);
sl@0
   422
    BOOST_MULTI_INDEX_CHECK_IS_OWNER(
sl@0
   423
      it,static_cast<const typename IteratorType::container_type&>(*this));
sl@0
   424
    return index::make_iterator(static_cast<node_type*>(it.get_node()));
sl@0
   425
  }
sl@0
   426
#endif
sl@0
   427
sl@0
   428
BOOST_MULTI_INDEX_PROTECTED_IF_MEMBER_TEMPLATE_FRIENDS:
sl@0
   429
  typedef typename super::copy_map_type copy_map_type;
sl@0
   430
sl@0
   431
  node_type* header()const
sl@0
   432
  {
sl@0
   433
    return bfm_header::member;
sl@0
   434
  }
sl@0
   435
sl@0
   436
  node_type* allocate_node()
sl@0
   437
  {
sl@0
   438
    return bfm_allocator::member.allocate(1);
sl@0
   439
  }
sl@0
   440
sl@0
   441
  void deallocate_node(node_type* x)
sl@0
   442
  {
sl@0
   443
    bfm_allocator::member.deallocate(x,1);
sl@0
   444
  }
sl@0
   445
sl@0
   446
  bool empty_()const
sl@0
   447
  {
sl@0
   448
    return node_count==0;
sl@0
   449
  }
sl@0
   450
sl@0
   451
  std::size_t size_()const
sl@0
   452
  {
sl@0
   453
    return node_count;
sl@0
   454
  }
sl@0
   455
sl@0
   456
  std::size_t max_size_()const
sl@0
   457
  {
sl@0
   458
    return static_cast<std::size_t >(-1);
sl@0
   459
  }
sl@0
   460
sl@0
   461
  std::pair<node_type*,bool> insert_(const Value& v)
sl@0
   462
  {
sl@0
   463
    node_type* x=allocate_node();
sl@0
   464
    BOOST_TRY{
sl@0
   465
      node_type* res=super::insert_(v,x);
sl@0
   466
      if(res==x){
sl@0
   467
        ++node_count;
sl@0
   468
        return std::pair<node_type*,bool>(res,true);
sl@0
   469
      }
sl@0
   470
      else{
sl@0
   471
        deallocate_node(x);
sl@0
   472
        return std::pair<node_type*,bool>(res,false);
sl@0
   473
      }
sl@0
   474
    }
sl@0
   475
    BOOST_CATCH(...){
sl@0
   476
      deallocate_node(x);
sl@0
   477
      BOOST_RETHROW;
sl@0
   478
    }
sl@0
   479
    BOOST_CATCH_END
sl@0
   480
  }
sl@0
   481
sl@0
   482
  std::pair<node_type*,bool> insert_(const Value& v,node_type* position)
sl@0
   483
  {
sl@0
   484
    node_type* x=allocate_node();
sl@0
   485
    BOOST_TRY{
sl@0
   486
      node_type* res=super::insert_(v,position,x);
sl@0
   487
      if(res==x){
sl@0
   488
        ++node_count;
sl@0
   489
        return std::pair<node_type*,bool>(res,true);
sl@0
   490
      }
sl@0
   491
      else{
sl@0
   492
        deallocate_node(x);
sl@0
   493
        return std::pair<node_type*,bool>(res,false);
sl@0
   494
      }
sl@0
   495
    }
sl@0
   496
    BOOST_CATCH(...){
sl@0
   497
      deallocate_node(x);
sl@0
   498
      BOOST_RETHROW;
sl@0
   499
    }
sl@0
   500
    BOOST_CATCH_END
sl@0
   501
  }
sl@0
   502
sl@0
   503
  void erase_(node_type* x)
sl@0
   504
  {
sl@0
   505
    super::erase_(x);
sl@0
   506
    deallocate_node(x);
sl@0
   507
    --node_count;
sl@0
   508
  }
sl@0
   509
sl@0
   510
  void delete_node_(node_type* x)
sl@0
   511
  {
sl@0
   512
    super::delete_node_(x);
sl@0
   513
    deallocate_node(x);
sl@0
   514
  }
sl@0
   515
sl@0
   516
  void delete_all_nodes_()
sl@0
   517
  {
sl@0
   518
    super::delete_all_nodes_();
sl@0
   519
  }
sl@0
   520
sl@0
   521
  void clear_()
sl@0
   522
  {
sl@0
   523
    delete_all_nodes_();
sl@0
   524
    super::clear_();
sl@0
   525
    node_count=0;
sl@0
   526
  }
sl@0
   527
sl@0
   528
  void swap_(multi_index_container<Value,IndexSpecifierList,Allocator>& x)
sl@0
   529
  {
sl@0
   530
    std::swap(bfm_header::member,x.bfm_header::member);
sl@0
   531
    super::swap_(x);
sl@0
   532
    std::swap(node_count,x.node_count);
sl@0
   533
  }
sl@0
   534
sl@0
   535
  bool replace_(const Value& k,node_type* x)
sl@0
   536
  {
sl@0
   537
    return super::replace_(k,x);
sl@0
   538
  }
sl@0
   539
sl@0
   540
  template<typename Modifier>
sl@0
   541
  bool modify_(Modifier mod,node_type* x)
sl@0
   542
  {
sl@0
   543
    mod(const_cast<value_type&>(x->value()));
sl@0
   544
sl@0
   545
    BOOST_TRY{
sl@0
   546
      if(!super::modify_(x)){
sl@0
   547
        deallocate_node(x);
sl@0
   548
        --node_count;
sl@0
   549
        return false;
sl@0
   550
      }
sl@0
   551
      else return true;
sl@0
   552
    }
sl@0
   553
    BOOST_CATCH(...){
sl@0
   554
      deallocate_node(x);
sl@0
   555
      --node_count;
sl@0
   556
      BOOST_RETHROW;
sl@0
   557
    }
sl@0
   558
    BOOST_CATCH_END
sl@0
   559
  }
sl@0
   560
sl@0
   561
#if !defined(BOOST_MULTI_INDEX_DISABLE_SERIALIZATION)
sl@0
   562
  /* serialization */
sl@0
   563
sl@0
   564
  friend class boost::serialization::access;
sl@0
   565
sl@0
   566
  BOOST_SERIALIZATION_SPLIT_MEMBER()
sl@0
   567
sl@0
   568
  typedef typename super::index_saver_type        index_saver_type;
sl@0
   569
  typedef typename super::index_loader_type       index_loader_type;
sl@0
   570
sl@0
   571
  template<class Archive>
sl@0
   572
  void save(Archive& ar,const unsigned int version)const
sl@0
   573
  {
sl@0
   574
    const std::size_t s=size_();
sl@0
   575
    ar<<serialization::make_nvp("count",s);
sl@0
   576
    index_saver_type sm(bfm_allocator::member,s);
sl@0
   577
sl@0
   578
    for(iterator it=super::begin(),it_end=super::end();it!=it_end;++it){
sl@0
   579
      ar<<serialization::make_nvp("item",*it);
sl@0
   580
      sm.add(it.get_node(),ar,version);
sl@0
   581
    }
sl@0
   582
    sm.add_track(header(),ar,version);
sl@0
   583
sl@0
   584
    super::save_(ar,version,sm);
sl@0
   585
  }
sl@0
   586
sl@0
   587
  template<class Archive>
sl@0
   588
  void load(Archive& ar,const unsigned int version)
sl@0
   589
  {
sl@0
   590
    BOOST_MULTI_INDEX_CHECK_INVARIANT;
sl@0
   591
sl@0
   592
    clear_(); 
sl@0
   593
sl@0
   594
    std::size_t s;
sl@0
   595
    ar>>serialization::make_nvp("count",s);
sl@0
   596
    index_loader_type lm(bfm_allocator::member,s);
sl@0
   597
sl@0
   598
    for(std::size_t n=0;n<s;++n){
sl@0
   599
      detail::archive_constructed<Value> value("item",ar,version);
sl@0
   600
      std::pair<node_type*,bool> p=insert_(
sl@0
   601
        value.get(),super::end().get_node());
sl@0
   602
      if(!p.second)throw_exception(
sl@0
   603
        archive::archive_exception(
sl@0
   604
          archive::archive_exception::other_exception));
sl@0
   605
      ar.reset_object_address(&p.first->value(),&value.get());
sl@0
   606
      lm.add(p.first,ar,version);
sl@0
   607
    }
sl@0
   608
    lm.add_track(header(),ar,version);
sl@0
   609
sl@0
   610
    super::load_(ar,version,lm);
sl@0
   611
  }
sl@0
   612
#endif
sl@0
   613
sl@0
   614
#if defined(BOOST_MULTI_INDEX_ENABLE_INVARIANT_CHECKING)
sl@0
   615
  /* invariant stuff */
sl@0
   616
sl@0
   617
  bool invariant_()const
sl@0
   618
  {
sl@0
   619
    return super::invariant_();
sl@0
   620
  }
sl@0
   621
sl@0
   622
  void check_invariant_()const
sl@0
   623
  {
sl@0
   624
    BOOST_MULTI_INDEX_INVARIANT_ASSERT(invariant_());
sl@0
   625
  }
sl@0
   626
#endif
sl@0
   627
sl@0
   628
private:
sl@0
   629
  std::size_t node_count;
sl@0
   630
sl@0
   631
#if defined(BOOST_MULTI_INDEX_ENABLE_INVARIANT_CHECKING)&&\
sl@0
   632
    BOOST_WORKAROUND(__MWERKS__,<=0x3003)
sl@0
   633
#pragma parse_mfunc_templ reset
sl@0
   634
#endif
sl@0
   635
};
sl@0
   636
sl@0
   637
/* retrieval of indices by number */
sl@0
   638
sl@0
   639
template<typename MultiIndexContainer,int N>
sl@0
   640
struct nth_index
sl@0
   641
{
sl@0
   642
  BOOST_STATIC_CONSTANT(
sl@0
   643
    int,
sl@0
   644
    M=mpl::size<typename MultiIndexContainer::index_type_list>::type::value);
sl@0
   645
  BOOST_STATIC_ASSERT(N>=0&&N<M);
sl@0
   646
  typedef typename mpl::at_c<
sl@0
   647
    typename MultiIndexContainer::index_type_list,N>::type type;
sl@0
   648
};
sl@0
   649
sl@0
   650
template<int N,typename Value,typename IndexSpecifierList,typename Allocator>
sl@0
   651
typename nth_index<
sl@0
   652
  multi_index_container<Value,IndexSpecifierList,Allocator>,N>::type&
sl@0
   653
get(
sl@0
   654
  multi_index_container<Value,IndexSpecifierList,Allocator>& m
sl@0
   655
  BOOST_APPEND_EXPLICIT_TEMPLATE_NON_TYPE(int,N))
sl@0
   656
{
sl@0
   657
  typedef multi_index_container<
sl@0
   658
    Value,IndexSpecifierList,Allocator>    multi_index_type;
sl@0
   659
  typedef typename nth_index<
sl@0
   660
    multi_index_container<
sl@0
   661
      Value,IndexSpecifierList,Allocator>,
sl@0
   662
    N
sl@0
   663
  >::type                                  index;
sl@0
   664
sl@0
   665
  BOOST_STATIC_ASSERT(N>=0&&
sl@0
   666
    N<
sl@0
   667
    mpl::size<
sl@0
   668
      BOOST_DEDUCED_TYPENAME multi_index_type::index_type_list
sl@0
   669
    >::type::value);
sl@0
   670
sl@0
   671
  return detail::converter<multi_index_type,index>::index(m);
sl@0
   672
}
sl@0
   673
sl@0
   674
template<int N,typename Value,typename IndexSpecifierList,typename Allocator>
sl@0
   675
const typename nth_index<
sl@0
   676
  multi_index_container<Value,IndexSpecifierList,Allocator>,N>::type&
sl@0
   677
get(
sl@0
   678
  const multi_index_container<Value,IndexSpecifierList,Allocator>& m
sl@0
   679
  BOOST_APPEND_EXPLICIT_TEMPLATE_NON_TYPE(int,N))
sl@0
   680
{
sl@0
   681
  typedef multi_index_container<
sl@0
   682
    Value,IndexSpecifierList,Allocator>    multi_index_type;
sl@0
   683
  typedef typename nth_index<
sl@0
   684
    multi_index_container<
sl@0
   685
      Value,IndexSpecifierList,Allocator>,
sl@0
   686
    N
sl@0
   687
  >::type                                  index;
sl@0
   688
sl@0
   689
  BOOST_STATIC_ASSERT(N>=0&&
sl@0
   690
    N<
sl@0
   691
    mpl::size<
sl@0
   692
      BOOST_DEDUCED_TYPENAME multi_index_type::index_type_list
sl@0
   693
    >::type::value);
sl@0
   694
sl@0
   695
  return detail::converter<multi_index_type,index>::index(m);
sl@0
   696
}
sl@0
   697
sl@0
   698
/* retrieval of indices by tag */
sl@0
   699
sl@0
   700
template<typename MultiIndexContainer,typename Tag>
sl@0
   701
struct index
sl@0
   702
{
sl@0
   703
  typedef typename MultiIndexContainer::index_type_list index_type_list;
sl@0
   704
sl@0
   705
  typedef typename mpl::find_if<
sl@0
   706
    index_type_list,
sl@0
   707
    detail::has_tag<Tag>
sl@0
   708
  >::type                                      iter;
sl@0
   709
sl@0
   710
  BOOST_STATIC_CONSTANT(
sl@0
   711
    bool,index_found=!(is_same<iter,typename mpl::end<index_type_list>::type >::value));
sl@0
   712
  BOOST_STATIC_ASSERT(index_found);
sl@0
   713
sl@0
   714
  typedef typename mpl::deref<iter>::type       type;
sl@0
   715
};
sl@0
   716
sl@0
   717
template<
sl@0
   718
  typename Tag,typename Value,typename IndexSpecifierList,typename Allocator
sl@0
   719
>
sl@0
   720
typename ::boost::multi_index::index<
sl@0
   721
  multi_index_container<Value,IndexSpecifierList,Allocator>,Tag>::type&
sl@0
   722
get(
sl@0
   723
  multi_index_container<Value,IndexSpecifierList,Allocator>& m
sl@0
   724
  BOOST_APPEND_EXPLICIT_TEMPLATE_TYPE(Tag))
sl@0
   725
{
sl@0
   726
  typedef multi_index_container<
sl@0
   727
    Value,IndexSpecifierList,Allocator>         multi_index_type;
sl@0
   728
  typedef typename ::boost::multi_index::index<
sl@0
   729
    multi_index_container<
sl@0
   730
      Value,IndexSpecifierList,Allocator>,
sl@0
   731
    Tag
sl@0
   732
  >::type                                       index;
sl@0
   733
sl@0
   734
  return detail::converter<multi_index_type,index>::index(m);
sl@0
   735
}
sl@0
   736
sl@0
   737
template<
sl@0
   738
  typename Tag,typename Value,typename IndexSpecifierList,typename Allocator
sl@0
   739
>
sl@0
   740
const typename ::boost::multi_index::index<
sl@0
   741
  multi_index_container<Value,IndexSpecifierList,Allocator>,Tag>::type&
sl@0
   742
get(
sl@0
   743
  const multi_index_container<Value,IndexSpecifierList,Allocator>& m
sl@0
   744
  BOOST_APPEND_EXPLICIT_TEMPLATE_TYPE(Tag))
sl@0
   745
{
sl@0
   746
  typedef multi_index_container<
sl@0
   747
    Value,IndexSpecifierList,Allocator>         multi_index_type;
sl@0
   748
  typedef typename ::boost::multi_index::index<
sl@0
   749
    multi_index_container<
sl@0
   750
      Value,IndexSpecifierList,Allocator>,
sl@0
   751
    Tag
sl@0
   752
  >::type                                       index;
sl@0
   753
sl@0
   754
  return detail::converter<multi_index_type,index>::index(m);
sl@0
   755
}
sl@0
   756
sl@0
   757
/* projection of iterators by number */
sl@0
   758
sl@0
   759
template<typename MultiIndexContainer,int N>
sl@0
   760
struct nth_index_iterator
sl@0
   761
{
sl@0
   762
  typedef typename detail::prevent_eti<
sl@0
   763
    nth_index<MultiIndexContainer,N>,
sl@0
   764
    typename nth_index<MultiIndexContainer,N>::type>::type::iterator type;
sl@0
   765
};
sl@0
   766
sl@0
   767
template<typename MultiIndexContainer,int N>
sl@0
   768
struct nth_index_const_iterator
sl@0
   769
{
sl@0
   770
  typedef typename detail::prevent_eti<
sl@0
   771
    nth_index<MultiIndexContainer,N>,
sl@0
   772
    typename nth_index<MultiIndexContainer,N>::type
sl@0
   773
  >::type::const_iterator type;
sl@0
   774
};
sl@0
   775
sl@0
   776
template<
sl@0
   777
  int N,typename IteratorType,
sl@0
   778
  typename Value,typename IndexSpecifierList,typename Allocator>
sl@0
   779
typename nth_index_iterator<
sl@0
   780
  multi_index_container<Value,IndexSpecifierList,Allocator>,N>::type
sl@0
   781
project(
sl@0
   782
  multi_index_container<Value,IndexSpecifierList,Allocator>& m,
sl@0
   783
  IteratorType it
sl@0
   784
  BOOST_APPEND_EXPLICIT_TEMPLATE_NON_TYPE(int,N))
sl@0
   785
{
sl@0
   786
  typedef multi_index_container<
sl@0
   787
    Value,IndexSpecifierList,Allocator>                multi_index_type;
sl@0
   788
  typedef typename nth_index<multi_index_type,N>::type index;
sl@0
   789
sl@0
   790
#if !defined(BOOST_MSVC)||!(BOOST_MSVC<1310) /* ain't work in MSVC++ 6.0/7.0 */
sl@0
   791
  BOOST_STATIC_ASSERT((
sl@0
   792
    mpl::contains<
sl@0
   793
      BOOST_DEDUCED_TYPENAME multi_index_type::iterator_type_list,
sl@0
   794
      IteratorType>::value));
sl@0
   795
#endif
sl@0
   796
sl@0
   797
  BOOST_MULTI_INDEX_CHECK_VALID_ITERATOR(it);
sl@0
   798
sl@0
   799
#if defined(BOOST_MULTI_INDEX_ENABLE_SAFE_MODE)
sl@0
   800
  typedef detail::converter<
sl@0
   801
    multi_index_type,
sl@0
   802
    BOOST_DEDUCED_TYPENAME IteratorType::container_type> converter;
sl@0
   803
  BOOST_MULTI_INDEX_CHECK_IS_OWNER(it,converter::index(m));
sl@0
   804
#endif
sl@0
   805
sl@0
   806
  return detail::converter<multi_index_type,index>::iterator(
sl@0
   807
    m,static_cast<typename multi_index_type::node_type*>(it.get_node()));
sl@0
   808
}
sl@0
   809
sl@0
   810
template<
sl@0
   811
  int N,typename IteratorType,
sl@0
   812
  typename Value,typename IndexSpecifierList,typename Allocator>
sl@0
   813
typename nth_index_const_iterator<
sl@0
   814
  multi_index_container<Value,IndexSpecifierList,Allocator>,N>::type
sl@0
   815
project(
sl@0
   816
  const multi_index_container<Value,IndexSpecifierList,Allocator>& m,
sl@0
   817
  IteratorType it
sl@0
   818
  BOOST_APPEND_EXPLICIT_TEMPLATE_NON_TYPE(int,N))
sl@0
   819
{
sl@0
   820
  typedef multi_index_container<
sl@0
   821
    Value,IndexSpecifierList,Allocator>                multi_index_type;
sl@0
   822
  typedef typename nth_index<multi_index_type,N>::type index;
sl@0
   823
sl@0
   824
#if !defined(BOOST_MSVC)||!(BOOST_MSVC<1310) /* ain't work in MSVC++ 6.0/7.0 */
sl@0
   825
  BOOST_STATIC_ASSERT((
sl@0
   826
    mpl::contains<
sl@0
   827
      BOOST_DEDUCED_TYPENAME multi_index_type::iterator_type_list,
sl@0
   828
      IteratorType>::value||
sl@0
   829
    mpl::contains<
sl@0
   830
      BOOST_DEDUCED_TYPENAME multi_index_type::const_iterator_type_list,
sl@0
   831
      IteratorType>::value));
sl@0
   832
#endif
sl@0
   833
sl@0
   834
  BOOST_MULTI_INDEX_CHECK_VALID_ITERATOR(it);
sl@0
   835
sl@0
   836
#if defined(BOOST_MULTI_INDEX_ENABLE_SAFE_MODE)
sl@0
   837
  typedef detail::converter<
sl@0
   838
    multi_index_type,
sl@0
   839
    BOOST_DEDUCED_TYPENAME IteratorType::container_type> converter;
sl@0
   840
  BOOST_MULTI_INDEX_CHECK_IS_OWNER(it,converter::index(m));
sl@0
   841
#endif
sl@0
   842
sl@0
   843
  return detail::converter<multi_index_type,index>::const_iterator(
sl@0
   844
    m,static_cast<typename multi_index_type::node_type*>(it.get_node()));
sl@0
   845
}
sl@0
   846
sl@0
   847
/* projection of iterators by tag */
sl@0
   848
sl@0
   849
template<typename MultiIndexContainer,typename Tag>
sl@0
   850
struct index_iterator
sl@0
   851
{
sl@0
   852
  typedef typename ::boost::multi_index::index<
sl@0
   853
    MultiIndexContainer,Tag>::type::iterator    type;
sl@0
   854
};
sl@0
   855
sl@0
   856
template<typename MultiIndexContainer,typename Tag>
sl@0
   857
struct index_const_iterator
sl@0
   858
{
sl@0
   859
  typedef typename ::boost::multi_index::index<
sl@0
   860
    MultiIndexContainer,Tag>::type::const_iterator type;
sl@0
   861
};
sl@0
   862
sl@0
   863
template<
sl@0
   864
  typename Tag,typename IteratorType,
sl@0
   865
  typename Value,typename IndexSpecifierList,typename Allocator>
sl@0
   866
typename index_iterator<
sl@0
   867
  multi_index_container<Value,IndexSpecifierList,Allocator>,Tag>::type
sl@0
   868
project(
sl@0
   869
  multi_index_container<Value,IndexSpecifierList,Allocator>& m,
sl@0
   870
  IteratorType it
sl@0
   871
  BOOST_APPEND_EXPLICIT_TEMPLATE_TYPE(Tag))
sl@0
   872
{
sl@0
   873
  typedef multi_index_container<
sl@0
   874
    Value,IndexSpecifierList,Allocator>         multi_index_type;
sl@0
   875
  typedef typename ::boost::multi_index::index<
sl@0
   876
    multi_index_type,Tag>::type                 index;
sl@0
   877
sl@0
   878
#if !defined(BOOST_MSVC)||!(BOOST_MSVC<1310) /* ain't work in MSVC++ 6.0/7.0 */
sl@0
   879
  BOOST_STATIC_ASSERT((
sl@0
   880
    mpl::contains<
sl@0
   881
      BOOST_DEDUCED_TYPENAME multi_index_type::iterator_type_list,
sl@0
   882
      IteratorType>::value));
sl@0
   883
#endif
sl@0
   884
sl@0
   885
  BOOST_MULTI_INDEX_CHECK_VALID_ITERATOR(it);
sl@0
   886
sl@0
   887
#if defined(BOOST_MULTI_INDEX_ENABLE_SAFE_MODE)
sl@0
   888
  typedef detail::converter<
sl@0
   889
    multi_index_type,
sl@0
   890
    BOOST_DEDUCED_TYPENAME IteratorType::container_type> converter;
sl@0
   891
  BOOST_MULTI_INDEX_CHECK_IS_OWNER(it,converter::index(m));
sl@0
   892
#endif
sl@0
   893
sl@0
   894
  return detail::converter<multi_index_type,index>::iterator(
sl@0
   895
    m,static_cast<typename multi_index_type::node_type*>(it.get_node()));
sl@0
   896
}
sl@0
   897
sl@0
   898
template<
sl@0
   899
  typename Tag,typename IteratorType,
sl@0
   900
  typename Value,typename IndexSpecifierList,typename Allocator>
sl@0
   901
typename index_const_iterator<
sl@0
   902
  multi_index_container<Value,IndexSpecifierList,Allocator>,Tag>::type
sl@0
   903
project(
sl@0
   904
  const multi_index_container<Value,IndexSpecifierList,Allocator>& m,
sl@0
   905
  IteratorType it
sl@0
   906
  BOOST_APPEND_EXPLICIT_TEMPLATE_TYPE(Tag))
sl@0
   907
{
sl@0
   908
  typedef multi_index_container<
sl@0
   909
    Value,IndexSpecifierList,Allocator>         multi_index_type;
sl@0
   910
  typedef typename ::boost::multi_index::index<
sl@0
   911
    multi_index_type,Tag>::type                 index;
sl@0
   912
sl@0
   913
#if !defined(BOOST_MSVC)||!(BOOST_MSVC<1310) /* ain't work in MSVC++ 6.0/7.0 */
sl@0
   914
  BOOST_STATIC_ASSERT((
sl@0
   915
    mpl::contains<
sl@0
   916
      BOOST_DEDUCED_TYPENAME multi_index_type::iterator_type_list,
sl@0
   917
      IteratorType>::value||
sl@0
   918
    mpl::contains<
sl@0
   919
      BOOST_DEDUCED_TYPENAME multi_index_type::const_iterator_type_list,
sl@0
   920
      IteratorType>::value));
sl@0
   921
#endif
sl@0
   922
sl@0
   923
  BOOST_MULTI_INDEX_CHECK_VALID_ITERATOR(it);
sl@0
   924
sl@0
   925
#if defined(BOOST_MULTI_INDEX_ENABLE_SAFE_MODE)
sl@0
   926
  typedef detail::converter<
sl@0
   927
    multi_index_type,
sl@0
   928
    BOOST_DEDUCED_TYPENAME IteratorType::container_type> converter;
sl@0
   929
  BOOST_MULTI_INDEX_CHECK_IS_OWNER(it,converter::index(m));
sl@0
   930
#endif
sl@0
   931
sl@0
   932
  return detail::converter<multi_index_type,index>::const_iterator(
sl@0
   933
    m,static_cast<typename multi_index_type::node_type*>(it.get_node()));
sl@0
   934
}
sl@0
   935
sl@0
   936
/* Comparison. Simple forward to first index. */
sl@0
   937
sl@0
   938
template<
sl@0
   939
  typename Value1,typename IndexSpecifierList1,typename Allocator1,
sl@0
   940
  typename Value2,typename IndexSpecifierList2,typename Allocator2
sl@0
   941
>
sl@0
   942
bool operator==(
sl@0
   943
  const multi_index_container<Value1,IndexSpecifierList1,Allocator1>& x,
sl@0
   944
  const multi_index_container<Value2,IndexSpecifierList2,Allocator2>& y)
sl@0
   945
{
sl@0
   946
  return get<0>(x)==get<0>(y);
sl@0
   947
}
sl@0
   948
sl@0
   949
template<
sl@0
   950
  typename Value1,typename IndexSpecifierList1,typename Allocator1,
sl@0
   951
  typename Value2,typename IndexSpecifierList2,typename Allocator2
sl@0
   952
>
sl@0
   953
bool operator<(
sl@0
   954
  const multi_index_container<Value1,IndexSpecifierList1,Allocator1>& x,
sl@0
   955
  const multi_index_container<Value2,IndexSpecifierList2,Allocator2>& y)
sl@0
   956
{
sl@0
   957
  return get<0>(x)<get<0>(y);
sl@0
   958
}
sl@0
   959
sl@0
   960
template<
sl@0
   961
  typename Value1,typename IndexSpecifierList1,typename Allocator1,
sl@0
   962
  typename Value2,typename IndexSpecifierList2,typename Allocator2
sl@0
   963
>
sl@0
   964
bool operator!=(
sl@0
   965
  const multi_index_container<Value1,IndexSpecifierList1,Allocator1>& x,
sl@0
   966
  const multi_index_container<Value2,IndexSpecifierList2,Allocator2>& y)
sl@0
   967
{
sl@0
   968
  return get<0>(x)!=get<0>(y);
sl@0
   969
}
sl@0
   970
sl@0
   971
template<
sl@0
   972
  typename Value1,typename IndexSpecifierList1,typename Allocator1,
sl@0
   973
  typename Value2,typename IndexSpecifierList2,typename Allocator2
sl@0
   974
>
sl@0
   975
bool operator>(
sl@0
   976
  const multi_index_container<Value1,IndexSpecifierList1,Allocator1>& x,
sl@0
   977
  const multi_index_container<Value2,IndexSpecifierList2,Allocator2>& y)
sl@0
   978
{
sl@0
   979
  return get<0>(x)>get<0>(y);
sl@0
   980
}
sl@0
   981
sl@0
   982
template<
sl@0
   983
  typename Value1,typename IndexSpecifierList1,typename Allocator1,
sl@0
   984
  typename Value2,typename IndexSpecifierList2,typename Allocator2
sl@0
   985
>
sl@0
   986
bool operator>=(
sl@0
   987
  const multi_index_container<Value1,IndexSpecifierList1,Allocator1>& x,
sl@0
   988
  const multi_index_container<Value2,IndexSpecifierList2,Allocator2>& y)
sl@0
   989
{
sl@0
   990
  return get<0>(x)>=get<0>(y);
sl@0
   991
}
sl@0
   992
sl@0
   993
template<
sl@0
   994
  typename Value1,typename IndexSpecifierList1,typename Allocator1,
sl@0
   995
  typename Value2,typename IndexSpecifierList2,typename Allocator2
sl@0
   996
>
sl@0
   997
bool operator<=(
sl@0
   998
  const multi_index_container<Value1,IndexSpecifierList1,Allocator1>& x,
sl@0
   999
  const multi_index_container<Value2,IndexSpecifierList2,Allocator2>& y)
sl@0
  1000
{
sl@0
  1001
  return get<0>(x)<=get<0>(y);
sl@0
  1002
}
sl@0
  1003
sl@0
  1004
/*  specialized algorithms */
sl@0
  1005
sl@0
  1006
template<typename Value,typename IndexSpecifierList,typename Allocator>
sl@0
  1007
void swap(
sl@0
  1008
  multi_index_container<Value,IndexSpecifierList,Allocator>& x,
sl@0
  1009
  multi_index_container<Value,IndexSpecifierList,Allocator>& y)
sl@0
  1010
{
sl@0
  1011
  x.swap(y);
sl@0
  1012
}
sl@0
  1013
sl@0
  1014
} /* namespace multi_index */
sl@0
  1015
sl@0
  1016
/* Associated global functions are promoted to namespace boost, except
sl@0
  1017
 * comparison operators and swap, which are meant to be Koenig looked-up.
sl@0
  1018
 */
sl@0
  1019
sl@0
  1020
using multi_index::get;
sl@0
  1021
using multi_index::project;
sl@0
  1022
sl@0
  1023
} /* namespace boost */
sl@0
  1024
sl@0
  1025
#undef BOOST_MULTI_INDEX_CHECK_INVARIANT
sl@0
  1026
sl@0
  1027
#endif