os/ossrv/ossrv_pub/boost_apis/boost/multi_index/member.hpp
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
sl@0
     1
/* Copyright 2003-2006 Joaquín M López Muñoz.
sl@0
     2
 * Distributed under 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
 * See http://www.boost.org/libs/multi_index for library home page.
sl@0
     7
 */
sl@0
     8
sl@0
     9
#ifndef BOOST_MULTI_INDEX_MEMBER_HPP
sl@0
    10
#define BOOST_MULTI_INDEX_MEMBER_HPP
sl@0
    11
sl@0
    12
#if defined(_MSC_VER)&&(_MSC_VER>=1200)
sl@0
    13
#pragma once
sl@0
    14
#endif
sl@0
    15
sl@0
    16
#include <boost/config.hpp> /* keep it first to prevent nasty warns in MSVC */
sl@0
    17
#include <boost/mpl/if.hpp>
sl@0
    18
#include <boost/type_traits/is_const.hpp>
sl@0
    19
#include <boost/utility/enable_if.hpp>
sl@0
    20
#include <cstddef>
sl@0
    21
sl@0
    22
#if !defined(BOOST_NO_SFINAE)
sl@0
    23
#include <boost/type_traits/is_convertible.hpp>
sl@0
    24
#endif
sl@0
    25
sl@0
    26
namespace boost{
sl@0
    27
sl@0
    28
template<class T> class reference_wrapper; /* fwd decl. */
sl@0
    29
sl@0
    30
namespace multi_index{
sl@0
    31
sl@0
    32
namespace detail{
sl@0
    33
sl@0
    34
/* member is a read/write key extractor for accessing a given
sl@0
    35
 * member of a class.
sl@0
    36
 * Additionally, member is overloaded to support referece_wrappers
sl@0
    37
 * of T and "chained pointers" to T's. By chained pointer to T we mean
sl@0
    38
 * a type P  such that, given a p of Type P
sl@0
    39
 *   *...n...*x is convertible to T&, for some n>=1.
sl@0
    40
 * Examples of chained pointers are raw and smart pointers, iterators and
sl@0
    41
 * arbitrary combinations of these (vg. T** or auto_ptr<T*>.)
sl@0
    42
 */
sl@0
    43
sl@0
    44
/* NB. Some overloads of operator() have an extra dummy parameter int=0.
sl@0
    45
 * This disambiguator serves several purposes:
sl@0
    46
 *  - Without it, MSVC++ 6.0 incorrectly regards some overloads as
sl@0
    47
 *    specializations of a previous member function template.
sl@0
    48
 *  - MSVC++ 6.0/7.0 seem to incorrectly treat some different memfuns
sl@0
    49
 *    as if they have the same signature.
sl@0
    50
 *  - If remove_const is broken due to lack of PTS, int=0 avoids the
sl@0
    51
 *    declaration of memfuns with identical signature.
sl@0
    52
 */
sl@0
    53
sl@0
    54
template<class Class,typename Type,Type Class::*PtrToMember>
sl@0
    55
struct const_member_base
sl@0
    56
{
sl@0
    57
  typedef Type result_type;
sl@0
    58
sl@0
    59
  template<typename ChainedPtr>
sl@0
    60
sl@0
    61
#if !defined(BOOST_NO_SFINAE)
sl@0
    62
  typename disable_if<
sl@0
    63
    is_convertible<const ChainedPtr&,const Class&>,Type&>::type
sl@0
    64
#else
sl@0
    65
  Type&
sl@0
    66
#endif
sl@0
    67
  
sl@0
    68
  operator()(const ChainedPtr& x)const
sl@0
    69
  {
sl@0
    70
    return operator()(*x);
sl@0
    71
  }
sl@0
    72
sl@0
    73
  Type& operator()(const Class& x)const
sl@0
    74
  {
sl@0
    75
    return x.*PtrToMember;
sl@0
    76
  }
sl@0
    77
sl@0
    78
  Type& operator()(const reference_wrapper<const Class>& x)const
sl@0
    79
  {
sl@0
    80
    return operator()(x.get());
sl@0
    81
  }
sl@0
    82
sl@0
    83
  Type& operator()(const reference_wrapper<Class>& x,int=0)const
sl@0
    84
  { 
sl@0
    85
    return operator()(x.get());
sl@0
    86
  }
sl@0
    87
};
sl@0
    88
sl@0
    89
template<class Class,typename Type,Type Class::*PtrToMember>
sl@0
    90
struct non_const_member_base
sl@0
    91
{
sl@0
    92
  typedef Type result_type;
sl@0
    93
sl@0
    94
  template<typename ChainedPtr>
sl@0
    95
sl@0
    96
#if !defined(BOOST_NO_SFINAE)
sl@0
    97
  typename disable_if<
sl@0
    98
    is_convertible<const ChainedPtr&,const Class&>,Type&>::type
sl@0
    99
#else
sl@0
   100
  Type&
sl@0
   101
#endif
sl@0
   102
sl@0
   103
  operator()(const ChainedPtr& x)const
sl@0
   104
  {
sl@0
   105
    return operator()(*x);
sl@0
   106
  }
sl@0
   107
sl@0
   108
  const Type& operator()(const Class& x,int=0)const
sl@0
   109
  {
sl@0
   110
    return x.*PtrToMember;
sl@0
   111
  }
sl@0
   112
sl@0
   113
  Type& operator()(Class& x)const
sl@0
   114
  { 
sl@0
   115
    return x.*PtrToMember;
sl@0
   116
  }
sl@0
   117
sl@0
   118
  const Type& operator()(const reference_wrapper<const Class>& x,int=0)const
sl@0
   119
  {
sl@0
   120
    return operator()(x.get());
sl@0
   121
  }
sl@0
   122
sl@0
   123
  Type& operator()(const reference_wrapper<Class>& x)const
sl@0
   124
  { 
sl@0
   125
    return operator()(x.get());
sl@0
   126
  }
sl@0
   127
};
sl@0
   128
sl@0
   129
} /* namespace multi_index::detail */
sl@0
   130
sl@0
   131
template<class Class,typename Type,Type Class::*PtrToMember>
sl@0
   132
struct member:
sl@0
   133
  mpl::if_c<
sl@0
   134
    is_const<Type>::value,
sl@0
   135
    detail::const_member_base<Class,Type,PtrToMember>,
sl@0
   136
    detail::non_const_member_base<Class,Type,PtrToMember>
sl@0
   137
  >::type
sl@0
   138
{
sl@0
   139
};
sl@0
   140
sl@0
   141
namespace detail{
sl@0
   142
sl@0
   143
/* MSVC++ 6.0 does not support properly pointers to members as
sl@0
   144
 * non-type template arguments, as reported in
sl@0
   145
 *   http://support.microsoft.com/default.aspx?scid=kb;EN-US;249045
sl@0
   146
 * A similar problem (though not identical) is shown by MSVC++ 7.0.
sl@0
   147
 * We provide an alternative to member<> accepting offsets instead
sl@0
   148
 * of pointers to members. This happens to work even for non-POD
sl@0
   149
 * types (although the standard forbids use of offsetof on these),
sl@0
   150
 * so it serves as a workaround in this compiler for all practical
sl@0
   151
 * purposes.
sl@0
   152
 * Surprisingly enough, other compilers, like Intel C++ 7.0/7.1 and
sl@0
   153
 * Visual Age 6.0, have similar bugs. This replacement of member<>
sl@0
   154
 * can be used for them too.
sl@0
   155
 */
sl@0
   156
sl@0
   157
template<class Class,typename Type,std::size_t OffsetOfMember>
sl@0
   158
struct const_member_offset_base
sl@0
   159
{
sl@0
   160
  typedef Type result_type;
sl@0
   161
sl@0
   162
  template<typename ChainedPtr>
sl@0
   163
sl@0
   164
#if !defined(BOOST_NO_SFINAE)
sl@0
   165
  typename disable_if<
sl@0
   166
    is_convertible<const ChainedPtr&,const Class&>,Type&>::type
sl@0
   167
#else
sl@0
   168
  Type&
sl@0
   169
#endif 
sl@0
   170
    
sl@0
   171
  operator()(const ChainedPtr& x)const
sl@0
   172
  {
sl@0
   173
    return operator()(*x);
sl@0
   174
  }
sl@0
   175
sl@0
   176
  Type& operator()(const Class& x)const
sl@0
   177
  {
sl@0
   178
    return *static_cast<const Type*>(
sl@0
   179
      static_cast<const void*>(
sl@0
   180
        static_cast<const char*>(
sl@0
   181
          static_cast<const void *>(&x))+OffsetOfMember));
sl@0
   182
  }
sl@0
   183
sl@0
   184
  Type& operator()(const reference_wrapper<const Class>& x)const
sl@0
   185
  {
sl@0
   186
    return operator()(x.get());
sl@0
   187
  }
sl@0
   188
sl@0
   189
  Type& operator()(const reference_wrapper<Class>& x,int=0)const
sl@0
   190
  {
sl@0
   191
    return operator()(x.get());
sl@0
   192
  }
sl@0
   193
};
sl@0
   194
sl@0
   195
template<class Class,typename Type,std::size_t OffsetOfMember>
sl@0
   196
struct non_const_member_offset_base
sl@0
   197
{
sl@0
   198
  typedef Type result_type;
sl@0
   199
sl@0
   200
  template<typename ChainedPtr>
sl@0
   201
sl@0
   202
#if !defined(BOOST_NO_SFINAE)
sl@0
   203
  typename disable_if<
sl@0
   204
    is_convertible<const ChainedPtr&,const Class&>,Type&>::type
sl@0
   205
#else
sl@0
   206
  Type&
sl@0
   207
#endif 
sl@0
   208
  
sl@0
   209
  operator()(const ChainedPtr& x)const
sl@0
   210
  {
sl@0
   211
    return operator()(*x);
sl@0
   212
  }
sl@0
   213
sl@0
   214
  const Type& operator()(const Class& x,int=0)const
sl@0
   215
  {
sl@0
   216
    return *static_cast<const Type*>(
sl@0
   217
      static_cast<const void*>(
sl@0
   218
        static_cast<const char*>(
sl@0
   219
          static_cast<const void *>(&x))+OffsetOfMember));
sl@0
   220
  }
sl@0
   221
sl@0
   222
  Type& operator()(Class& x)const
sl@0
   223
  { 
sl@0
   224
    return *static_cast<Type*>(
sl@0
   225
      static_cast<void*>(
sl@0
   226
        static_cast<char*>(static_cast<void *>(&x))+OffsetOfMember));
sl@0
   227
  }
sl@0
   228
sl@0
   229
  const Type& operator()(const reference_wrapper<const Class>& x,int=0)const
sl@0
   230
  {
sl@0
   231
    return operator()(x.get());
sl@0
   232
  }
sl@0
   233
sl@0
   234
  Type& operator()(const reference_wrapper<Class>& x)const
sl@0
   235
  {
sl@0
   236
    return operator()(x.get());
sl@0
   237
  }
sl@0
   238
};
sl@0
   239
sl@0
   240
} /* namespace multi_index::detail */
sl@0
   241
sl@0
   242
template<class Class,typename Type,std::size_t OffsetOfMember>
sl@0
   243
struct member_offset:
sl@0
   244
  mpl::if_c<
sl@0
   245
    is_const<Type>::value,
sl@0
   246
    detail::const_member_offset_base<Class,Type,OffsetOfMember>,
sl@0
   247
    detail::non_const_member_offset_base<Class,Type,OffsetOfMember>
sl@0
   248
  >::type
sl@0
   249
{
sl@0
   250
};
sl@0
   251
sl@0
   252
/* BOOST_MULTI_INDEX_MEMBER resolves to member in the normal cases,
sl@0
   253
 * and to member_offset as a workaround in those defective compilers for
sl@0
   254
 * which BOOST_NO_POINTER_TO_MEMBER_TEMPLATE_PARAMETERS is defined.
sl@0
   255
 */
sl@0
   256
sl@0
   257
#if defined(BOOST_NO_POINTER_TO_MEMBER_TEMPLATE_PARAMETERS)
sl@0
   258
#define BOOST_MULTI_INDEX_MEMBER(Class,Type,MemberName) \
sl@0
   259
::boost::multi_index::member_offset< Class,Type,offsetof(Class,MemberName) >
sl@0
   260
#else
sl@0
   261
#define BOOST_MULTI_INDEX_MEMBER(Class,Type,MemberName) \
sl@0
   262
::boost::multi_index::member< Class,Type,&Class::MemberName >
sl@0
   263
#endif
sl@0
   264
sl@0
   265
} /* namespace multi_index */
sl@0
   266
sl@0
   267
} /* namespace boost */
sl@0
   268
sl@0
   269
#endif