os/ossrv/ossrv_pub/boost_apis/boost/intrusive_ptr.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
#ifndef BOOST_INTRUSIVE_PTR_HPP_INCLUDED
sl@0
     2
#define BOOST_INTRUSIVE_PTR_HPP_INCLUDED
sl@0
     3
sl@0
     4
//
sl@0
     5
//  intrusive_ptr.hpp
sl@0
     6
//
sl@0
     7
//  Copyright (c) 2001, 2002 Peter Dimov
sl@0
     8
//
sl@0
     9
// Distributed under the Boost Software License, Version 1.0. (See
sl@0
    10
// accompanying file LICENSE_1_0.txt or copy at
sl@0
    11
// http://www.boost.org/LICENSE_1_0.txt)
sl@0
    12
//
sl@0
    13
//  See http://www.boost.org/libs/smart_ptr/intrusive_ptr.html for documentation.
sl@0
    14
//
sl@0
    15
sl@0
    16
#include <boost/config.hpp>
sl@0
    17
sl@0
    18
#ifdef BOOST_MSVC  // moved here to work around VC++ compiler crash
sl@0
    19
# pragma warning(push)
sl@0
    20
# pragma warning(disable:4284) // odd return type for operator->
sl@0
    21
#endif
sl@0
    22
sl@0
    23
#include <boost/assert.hpp>
sl@0
    24
#include <boost/detail/workaround.hpp>
sl@0
    25
sl@0
    26
#include <functional>           // for std::less
sl@0
    27
#include <iosfwd>               // for std::basic_ostream
sl@0
    28
sl@0
    29
sl@0
    30
namespace boost
sl@0
    31
{
sl@0
    32
sl@0
    33
//
sl@0
    34
//  intrusive_ptr
sl@0
    35
//
sl@0
    36
//  A smart pointer that uses intrusive reference counting.
sl@0
    37
//
sl@0
    38
//  Relies on unqualified calls to
sl@0
    39
//  
sl@0
    40
//      void intrusive_ptr_add_ref(T * p);
sl@0
    41
//      void intrusive_ptr_release(T * p);
sl@0
    42
//
sl@0
    43
//          (p != 0)
sl@0
    44
//
sl@0
    45
//  The object is responsible for destroying itself.
sl@0
    46
//
sl@0
    47
sl@0
    48
template<class T> class intrusive_ptr
sl@0
    49
{
sl@0
    50
private:
sl@0
    51
sl@0
    52
    typedef intrusive_ptr this_type;
sl@0
    53
sl@0
    54
public:
sl@0
    55
sl@0
    56
    typedef T element_type;
sl@0
    57
sl@0
    58
    intrusive_ptr(): p_(0)
sl@0
    59
    {
sl@0
    60
    }
sl@0
    61
sl@0
    62
    intrusive_ptr(T * p, bool add_ref = true): p_(p)
sl@0
    63
    {
sl@0
    64
        if(p_ != 0 && add_ref) intrusive_ptr_add_ref(p_);
sl@0
    65
    }
sl@0
    66
sl@0
    67
#if !defined(BOOST_NO_MEMBER_TEMPLATES) || defined(BOOST_MSVC6_MEMBER_TEMPLATES)
sl@0
    68
sl@0
    69
    template<class U> intrusive_ptr(intrusive_ptr<U> const & rhs): p_(rhs.get())
sl@0
    70
    {
sl@0
    71
        if(p_ != 0) intrusive_ptr_add_ref(p_);
sl@0
    72
    }
sl@0
    73
sl@0
    74
#endif
sl@0
    75
sl@0
    76
    intrusive_ptr(intrusive_ptr const & rhs): p_(rhs.p_)
sl@0
    77
    {
sl@0
    78
        if(p_ != 0) intrusive_ptr_add_ref(p_);
sl@0
    79
    }
sl@0
    80
sl@0
    81
    ~intrusive_ptr()
sl@0
    82
    {
sl@0
    83
        if(p_ != 0) intrusive_ptr_release(p_);
sl@0
    84
    }
sl@0
    85
sl@0
    86
#if !defined(BOOST_NO_MEMBER_TEMPLATES) || defined(BOOST_MSVC6_MEMBER_TEMPLATES)
sl@0
    87
sl@0
    88
    template<class U> intrusive_ptr & operator=(intrusive_ptr<U> const & rhs)
sl@0
    89
    {
sl@0
    90
        this_type(rhs).swap(*this);
sl@0
    91
        return *this;
sl@0
    92
    }
sl@0
    93
sl@0
    94
#endif
sl@0
    95
sl@0
    96
    intrusive_ptr & operator=(intrusive_ptr const & rhs)
sl@0
    97
    {
sl@0
    98
        this_type(rhs).swap(*this);
sl@0
    99
        return *this;
sl@0
   100
    }
sl@0
   101
sl@0
   102
    intrusive_ptr & operator=(T * rhs)
sl@0
   103
    {
sl@0
   104
        this_type(rhs).swap(*this);
sl@0
   105
        return *this;
sl@0
   106
    }
sl@0
   107
sl@0
   108
    T * get() const
sl@0
   109
    {
sl@0
   110
        return p_;
sl@0
   111
    }
sl@0
   112
sl@0
   113
    T & operator*() const
sl@0
   114
    {
sl@0
   115
        return *p_;
sl@0
   116
    }
sl@0
   117
sl@0
   118
    T * operator->() const
sl@0
   119
    {
sl@0
   120
        return p_;
sl@0
   121
    }
sl@0
   122
sl@0
   123
#if defined(__SUNPRO_CC) && BOOST_WORKAROUND(__SUNPRO_CC, <= 0x530)
sl@0
   124
sl@0
   125
    operator bool () const
sl@0
   126
    {
sl@0
   127
        return p_ != 0;
sl@0
   128
    }
sl@0
   129
sl@0
   130
#elif defined(__MWERKS__) && BOOST_WORKAROUND(__MWERKS__, BOOST_TESTED_AT(0x3003))
sl@0
   131
    typedef T * (this_type::*unspecified_bool_type)() const;
sl@0
   132
    
sl@0
   133
    operator unspecified_bool_type() const // never throws
sl@0
   134
    {
sl@0
   135
        return p_ == 0? 0: &this_type::get;
sl@0
   136
    }
sl@0
   137
sl@0
   138
#else 
sl@0
   139
sl@0
   140
    typedef T * this_type::*unspecified_bool_type;
sl@0
   141
sl@0
   142
    operator unspecified_bool_type () const
sl@0
   143
    {
sl@0
   144
        return p_ == 0? 0: &this_type::p_;
sl@0
   145
    }
sl@0
   146
sl@0
   147
#endif
sl@0
   148
sl@0
   149
    // operator! is a Borland-specific workaround
sl@0
   150
    bool operator! () const
sl@0
   151
    {
sl@0
   152
        return p_ == 0;
sl@0
   153
    }
sl@0
   154
sl@0
   155
    void swap(intrusive_ptr & rhs)
sl@0
   156
    {
sl@0
   157
        T * tmp = p_;
sl@0
   158
        p_ = rhs.p_;
sl@0
   159
        rhs.p_ = tmp;
sl@0
   160
    }
sl@0
   161
sl@0
   162
private:
sl@0
   163
sl@0
   164
    T * p_;
sl@0
   165
};
sl@0
   166
sl@0
   167
template<class T, class U> inline bool operator==(intrusive_ptr<T> const & a, intrusive_ptr<U> const & b)
sl@0
   168
{
sl@0
   169
    return a.get() == b.get();
sl@0
   170
}
sl@0
   171
sl@0
   172
template<class T, class U> inline bool operator!=(intrusive_ptr<T> const & a, intrusive_ptr<U> const & b)
sl@0
   173
{
sl@0
   174
    return a.get() != b.get();
sl@0
   175
}
sl@0
   176
sl@0
   177
template<class T, class U> inline bool operator==(intrusive_ptr<T> const & a, U * b)
sl@0
   178
{
sl@0
   179
    return a.get() == b;
sl@0
   180
}
sl@0
   181
sl@0
   182
template<class T, class U> inline bool operator!=(intrusive_ptr<T> const & a, U * b)
sl@0
   183
{
sl@0
   184
    return a.get() != b;
sl@0
   185
}
sl@0
   186
sl@0
   187
template<class T, class U> inline bool operator==(T * a, intrusive_ptr<U> const & b)
sl@0
   188
{
sl@0
   189
    return a == b.get();
sl@0
   190
}
sl@0
   191
sl@0
   192
template<class T, class U> inline bool operator!=(T * a, intrusive_ptr<U> const & b)
sl@0
   193
{
sl@0
   194
    return a != b.get();
sl@0
   195
}
sl@0
   196
sl@0
   197
#if __GNUC__ == 2 && __GNUC_MINOR__ <= 96
sl@0
   198
sl@0
   199
// Resolve the ambiguity between our op!= and the one in rel_ops
sl@0
   200
sl@0
   201
template<class T> inline bool operator!=(intrusive_ptr<T> const & a, intrusive_ptr<T> const & b)
sl@0
   202
{
sl@0
   203
    return a.get() != b.get();
sl@0
   204
}
sl@0
   205
sl@0
   206
#endif
sl@0
   207
sl@0
   208
template<class T> inline bool operator<(intrusive_ptr<T> const & a, intrusive_ptr<T> const & b)
sl@0
   209
{
sl@0
   210
    return std::less<T *>()(a.get(), b.get());
sl@0
   211
}
sl@0
   212
sl@0
   213
template<class T> void swap(intrusive_ptr<T> & lhs, intrusive_ptr<T> & rhs)
sl@0
   214
{
sl@0
   215
    lhs.swap(rhs);
sl@0
   216
}
sl@0
   217
sl@0
   218
// mem_fn support
sl@0
   219
sl@0
   220
template<class T> T * get_pointer(intrusive_ptr<T> const & p)
sl@0
   221
{
sl@0
   222
    return p.get();
sl@0
   223
}
sl@0
   224
sl@0
   225
template<class T, class U> intrusive_ptr<T> static_pointer_cast(intrusive_ptr<U> const & p)
sl@0
   226
{
sl@0
   227
    return static_cast<T *>(p.get());
sl@0
   228
}
sl@0
   229
sl@0
   230
template<class T, class U> intrusive_ptr<T> const_pointer_cast(intrusive_ptr<U> const & p)
sl@0
   231
{
sl@0
   232
    return const_cast<T *>(p.get());
sl@0
   233
}
sl@0
   234
sl@0
   235
template<class T, class U> intrusive_ptr<T> dynamic_pointer_cast(intrusive_ptr<U> const & p)
sl@0
   236
{
sl@0
   237
    return dynamic_cast<T *>(p.get());
sl@0
   238
}
sl@0
   239
sl@0
   240
// operator<<
sl@0
   241
sl@0
   242
#if defined(__GNUC__) &&  (__GNUC__ < 3)
sl@0
   243
sl@0
   244
template<class Y> std::ostream & operator<< (std::ostream & os, intrusive_ptr<Y> const & p)
sl@0
   245
{
sl@0
   246
    os << p.get();
sl@0
   247
    return os;
sl@0
   248
}
sl@0
   249
sl@0
   250
#else
sl@0
   251
sl@0
   252
// in STLport's no-iostreams mode no iostream symbols can be used
sl@0
   253
#ifndef _STLP_NO_IOSTREAMS
sl@0
   254
sl@0
   255
# if defined(BOOST_MSVC) && BOOST_WORKAROUND(BOOST_MSVC, < 1300 && __SGI_STL_PORT)
sl@0
   256
// MSVC6 has problems finding std::basic_ostream through the using declaration in namespace _STL
sl@0
   257
using std::basic_ostream;
sl@0
   258
template<class E, class T, class Y> basic_ostream<E, T> & operator<< (basic_ostream<E, T> & os, intrusive_ptr<Y> const & p)
sl@0
   259
# else
sl@0
   260
template<class E, class T, class Y> std::basic_ostream<E, T> & operator<< (std::basic_ostream<E, T> & os, intrusive_ptr<Y> const & p)
sl@0
   261
# endif 
sl@0
   262
{
sl@0
   263
    os << p.get();
sl@0
   264
    return os;
sl@0
   265
}
sl@0
   266
sl@0
   267
#endif // _STLP_NO_IOSTREAMS
sl@0
   268
sl@0
   269
#endif // __GNUC__ < 3
sl@0
   270
sl@0
   271
} // namespace boost
sl@0
   272
sl@0
   273
#ifdef BOOST_MSVC
sl@0
   274
# pragma warning(pop)
sl@0
   275
#endif    
sl@0
   276
sl@0
   277
#endif  // #ifndef BOOST_INTRUSIVE_PTR_HPP_INCLUDED