sl@0: #ifndef BOOST_SHARED_PTR_HPP_INCLUDED sl@0: #define BOOST_SHARED_PTR_HPP_INCLUDED sl@0: sl@0: // sl@0: // shared_ptr.hpp sl@0: // sl@0: // (C) Copyright Greg Colvin and Beman Dawes 1998, 1999. sl@0: // Copyright (c) 2001-2006 Peter Dimov sl@0: // sl@0: // Distributed under the Boost Software License, Version 1.0. (See sl@0: // accompanying file LICENSE_1_0.txt or copy at sl@0: // http://www.boost.org/LICENSE_1_0.txt) sl@0: // sl@0: // See http://www.boost.org/libs/smart_ptr/shared_ptr.htm for documentation. sl@0: // sl@0: sl@0: #include // for broken compiler workarounds sl@0: sl@0: #if defined(BOOST_NO_MEMBER_TEMPLATES) && !defined(BOOST_MSVC6_MEMBER_TEMPLATES) sl@0: #include sl@0: #else sl@0: sl@0: #include // for std::auto_ptr sl@0: sl@0: #include sl@0: #include sl@0: #include sl@0: #include sl@0: #include sl@0: sl@0: #include // for std::swap sl@0: #include // for std::less sl@0: #include // for std::bad_cast sl@0: #include // for std::basic_ostream sl@0: sl@0: #ifdef BOOST_MSVC // moved here to work around VC++ compiler crash sl@0: # pragma warning(push) sl@0: # pragma warning(disable:4284) // odd return type for operator-> sl@0: #endif sl@0: sl@0: namespace boost sl@0: { sl@0: sl@0: template class weak_ptr; sl@0: template class enable_shared_from_this; sl@0: sl@0: namespace detail sl@0: { sl@0: sl@0: struct static_cast_tag {}; sl@0: struct const_cast_tag {}; sl@0: struct dynamic_cast_tag {}; sl@0: struct polymorphic_cast_tag {}; sl@0: sl@0: template struct shared_ptr_traits sl@0: { sl@0: typedef T & reference; sl@0: }; sl@0: sl@0: template<> struct shared_ptr_traits sl@0: { sl@0: typedef void reference; sl@0: }; sl@0: sl@0: #if !defined(BOOST_NO_CV_VOID_SPECIALIZATIONS) sl@0: sl@0: template<> struct shared_ptr_traits sl@0: { sl@0: typedef void reference; sl@0: }; sl@0: sl@0: template<> struct shared_ptr_traits sl@0: { sl@0: typedef void reference; sl@0: }; sl@0: sl@0: template<> struct shared_ptr_traits sl@0: { sl@0: typedef void reference; sl@0: }; sl@0: sl@0: #endif sl@0: sl@0: // enable_shared_from_this support sl@0: sl@0: template void sp_enable_shared_from_this( shared_count const & pn, boost::enable_shared_from_this const * pe, Y const * px ) sl@0: { sl@0: if(pe != 0) pe->_internal_weak_this._internal_assign(const_cast(px), pn); sl@0: } sl@0: sl@0: #ifdef sgi sl@0: // Turn off: the last argument of the varargs function "sp_enable_shared_from_this" is unnamed sl@0: # pragma set woff 3506 sl@0: #endif sl@0: sl@0: inline void sp_enable_shared_from_this( shared_count const & /*pn*/, ... ) sl@0: { sl@0: } sl@0: sl@0: #ifdef sgi sl@0: # pragma reset woff 3506 sl@0: #endif sl@0: sl@0: #if !defined( BOOST_NO_SFINAE ) && !defined( BOOST_NO_TEMPLATE_PARTIAL_SPECIALIZATION ) && !defined( BOOST_NO_AUTO_PTR ) sl@0: sl@0: // rvalue auto_ptr support based on a technique by Dave Abrahams sl@0: sl@0: template< class T, class R > struct sp_enable_if_auto_ptr sl@0: { sl@0: }; sl@0: sl@0: template< class T, class R > struct sp_enable_if_auto_ptr< std::auto_ptr< T >, R > sl@0: { sl@0: typedef R type; sl@0: }; sl@0: sl@0: #endif sl@0: sl@0: } // namespace detail sl@0: sl@0: sl@0: // sl@0: // shared_ptr sl@0: // sl@0: // An enhanced relative of scoped_ptr with reference counted copy semantics. sl@0: // The object pointed to is deleted when the last shared_ptr pointing to it sl@0: // is destroyed or reset. sl@0: // sl@0: sl@0: template class shared_ptr sl@0: { sl@0: private: sl@0: sl@0: // Borland 5.5.1 specific workaround sl@0: typedef shared_ptr this_type; sl@0: sl@0: public: sl@0: sl@0: typedef T element_type; sl@0: typedef T value_type; sl@0: typedef T * pointer; sl@0: typedef typename boost::detail::shared_ptr_traits::reference reference; sl@0: sl@0: shared_ptr(): px(0), pn() // never throws in 1.30+ sl@0: { sl@0: } sl@0: sl@0: template sl@0: explicit shared_ptr( Y * p ): px( p ), pn( p ) // Y must be complete sl@0: { sl@0: boost::detail::sp_enable_shared_from_this( pn, p, p ); sl@0: } sl@0: sl@0: // sl@0: // Requirements: D's copy constructor must not throw sl@0: // sl@0: // shared_ptr will release p by calling d(p) sl@0: // sl@0: sl@0: template shared_ptr(Y * p, D d): px(p), pn(p, d) sl@0: { sl@0: boost::detail::sp_enable_shared_from_this( pn, p, p ); sl@0: } sl@0: sl@0: // As above, but with allocator. A's copy constructor shall not throw. sl@0: sl@0: template shared_ptr( Y * p, D d, A a ): px( p ), pn( p, d, a ) sl@0: { sl@0: boost::detail::sp_enable_shared_from_this( pn, p, p ); sl@0: } sl@0: sl@0: // generated copy constructor, assignment, destructor are fine... sl@0: sl@0: // except that Borland C++ has a bug, and g++ with -Wsynth warns sl@0: #if defined(__BORLANDC__) || defined(__GNUC__) sl@0: sl@0: shared_ptr & operator=(shared_ptr const & r) // never throws sl@0: { sl@0: px = r.px; sl@0: pn = r.pn; // shared_count::op= doesn't throw sl@0: return *this; sl@0: } sl@0: sl@0: #endif sl@0: sl@0: template sl@0: explicit shared_ptr(weak_ptr const & r): pn(r.pn) // may throw sl@0: { sl@0: // it is now safe to copy r.px, as pn(r.pn) did not throw sl@0: px = r.px; sl@0: } sl@0: sl@0: template sl@0: shared_ptr(shared_ptr const & r): px(r.px), pn(r.pn) // never throws sl@0: { sl@0: } sl@0: sl@0: template sl@0: shared_ptr(shared_ptr const & r, boost::detail::static_cast_tag): px(static_cast(r.px)), pn(r.pn) sl@0: { sl@0: } sl@0: sl@0: template sl@0: shared_ptr(shared_ptr const & r, boost::detail::const_cast_tag): px(const_cast(r.px)), pn(r.pn) sl@0: { sl@0: } sl@0: sl@0: template sl@0: shared_ptr(shared_ptr const & r, boost::detail::dynamic_cast_tag): px(dynamic_cast(r.px)), pn(r.pn) sl@0: { sl@0: if(px == 0) // need to allocate new counter -- the cast failed sl@0: { sl@0: pn = boost::detail::shared_count(); sl@0: } sl@0: } sl@0: sl@0: template sl@0: shared_ptr(shared_ptr const & r, boost::detail::polymorphic_cast_tag): px(dynamic_cast(r.px)), pn(r.pn) sl@0: { sl@0: if(px == 0) sl@0: { sl@0: boost::throw_exception(std::bad_cast()); sl@0: } sl@0: } sl@0: sl@0: #ifndef BOOST_NO_AUTO_PTR sl@0: sl@0: template sl@0: explicit shared_ptr(std::auto_ptr & r): px(r.get()), pn() sl@0: { sl@0: Y * tmp = r.get(); sl@0: pn = boost::detail::shared_count(r); sl@0: boost::detail::sp_enable_shared_from_this( pn, tmp, tmp ); sl@0: } sl@0: sl@0: #if !defined( BOOST_NO_SFINAE ) && !defined( BOOST_NO_TEMPLATE_PARTIAL_SPECIALIZATION ) sl@0: sl@0: template sl@0: explicit shared_ptr( Ap r, typename boost::detail::sp_enable_if_auto_ptr::type = 0 ): px( r.get() ), pn() sl@0: { sl@0: typename Ap::element_type * tmp = r.get(); sl@0: pn = boost::detail::shared_count( r ); sl@0: boost::detail::sp_enable_shared_from_this( pn, tmp, tmp ); sl@0: } sl@0: sl@0: sl@0: #endif // BOOST_NO_SFINAE, BOOST_NO_TEMPLATE_PARTIAL_SPECIALIZATION sl@0: sl@0: #endif // BOOST_NO_AUTO_PTR sl@0: sl@0: #if !defined(BOOST_MSVC) || (BOOST_MSVC >= 1300) sl@0: sl@0: template sl@0: shared_ptr & operator=(shared_ptr const & r) // never throws sl@0: { sl@0: px = r.px; sl@0: pn = r.pn; // shared_count::op= doesn't throw sl@0: return *this; sl@0: } sl@0: sl@0: #endif sl@0: sl@0: #ifndef BOOST_NO_AUTO_PTR sl@0: sl@0: template sl@0: shared_ptr & operator=( std::auto_ptr & r ) sl@0: { sl@0: this_type(r).swap(*this); sl@0: return *this; sl@0: } sl@0: sl@0: #if !defined( BOOST_NO_SFINAE ) && !defined( BOOST_NO_TEMPLATE_PARTIAL_SPECIALIZATION ) sl@0: sl@0: template sl@0: typename boost::detail::sp_enable_if_auto_ptr< Ap, shared_ptr & >::type operator=( Ap r ) sl@0: { sl@0: this_type( r ).swap( *this ); sl@0: return *this; sl@0: } sl@0: sl@0: sl@0: #endif // BOOST_NO_SFINAE, BOOST_NO_TEMPLATE_PARTIAL_SPECIALIZATION sl@0: sl@0: #endif // BOOST_NO_AUTO_PTR sl@0: sl@0: void reset() // never throws in 1.30+ sl@0: { sl@0: this_type().swap(*this); sl@0: } sl@0: sl@0: template void reset(Y * p) // Y must be complete sl@0: { sl@0: BOOST_ASSERT(p == 0 || p != px); // catch self-reset errors sl@0: this_type(p).swap(*this); sl@0: } sl@0: sl@0: template void reset( Y * p, D d ) sl@0: { sl@0: this_type( p, d ).swap( *this ); sl@0: } sl@0: sl@0: template void reset( Y * p, D d, A a ) sl@0: { sl@0: this_type( p, d, a ).swap( *this ); sl@0: } sl@0: sl@0: reference operator* () const // never throws sl@0: { sl@0: BOOST_ASSERT(px != 0); sl@0: return *px; sl@0: } sl@0: sl@0: T * operator-> () const // never throws sl@0: { sl@0: BOOST_ASSERT(px != 0); sl@0: return px; sl@0: } sl@0: sl@0: T * get() const // never throws sl@0: { sl@0: return px; sl@0: } sl@0: sl@0: // implicit conversion to "bool" sl@0: sl@0: #if defined(__SUNPRO_CC) && BOOST_WORKAROUND(__SUNPRO_CC, <= 0x580) sl@0: sl@0: operator bool () const sl@0: { sl@0: return px != 0; sl@0: } sl@0: sl@0: #elif defined( _MANAGED ) sl@0: sl@0: static void unspecified_bool( this_type*** ) sl@0: { sl@0: } sl@0: sl@0: typedef void (*unspecified_bool_type)( this_type*** ); sl@0: sl@0: operator unspecified_bool_type() const // never throws sl@0: { sl@0: return px == 0? 0: unspecified_bool; sl@0: } sl@0: sl@0: #elif \ sl@0: ( defined(__MWERKS__) && BOOST_WORKAROUND(__MWERKS__, < 0x3200) ) || \ sl@0: ( defined(__GNUC__) && (__GNUC__ * 100 + __GNUC_MINOR__ < 304) ) sl@0: sl@0: typedef T * (this_type::*unspecified_bool_type)() const; sl@0: sl@0: operator unspecified_bool_type() const // never throws sl@0: { sl@0: return px == 0? 0: &this_type::get; sl@0: } sl@0: sl@0: #else sl@0: sl@0: typedef T * this_type::*unspecified_bool_type; sl@0: sl@0: operator unspecified_bool_type() const // never throws sl@0: { sl@0: return px == 0? 0: &this_type::px; sl@0: } sl@0: sl@0: #endif sl@0: sl@0: // operator! is redundant, but some compilers need it sl@0: sl@0: bool operator! () const // never throws sl@0: { sl@0: return px == 0; sl@0: } sl@0: sl@0: bool unique() const // never throws sl@0: { sl@0: return pn.unique(); sl@0: } sl@0: sl@0: long use_count() const // never throws sl@0: { sl@0: return pn.use_count(); sl@0: } sl@0: sl@0: void swap(shared_ptr & other) // never throws sl@0: { sl@0: std::swap(px, other.px); sl@0: pn.swap(other.pn); sl@0: } sl@0: sl@0: template bool _internal_less(shared_ptr const & rhs) const sl@0: { sl@0: return pn < rhs.pn; sl@0: } sl@0: sl@0: void * _internal_get_deleter(std::type_info const & ti) const sl@0: { sl@0: return pn.get_deleter(ti); sl@0: } sl@0: sl@0: // Tasteless as this may seem, making all members public allows member templates sl@0: // to work in the absence of member template friends. (Matthew Langston) sl@0: sl@0: #ifndef BOOST_NO_MEMBER_TEMPLATE_FRIENDS sl@0: sl@0: private: sl@0: sl@0: template friend class shared_ptr; sl@0: template friend class weak_ptr; sl@0: sl@0: sl@0: #endif sl@0: sl@0: T * px; // contained pointer sl@0: boost::detail::shared_count pn; // reference counter sl@0: sl@0: }; // shared_ptr sl@0: sl@0: template inline bool operator==(shared_ptr const & a, shared_ptr const & b) sl@0: { sl@0: return a.get() == b.get(); sl@0: } sl@0: sl@0: template inline bool operator!=(shared_ptr const & a, shared_ptr const & b) sl@0: { sl@0: return a.get() != b.get(); sl@0: } sl@0: sl@0: #if __GNUC__ == 2 && __GNUC_MINOR__ <= 96 sl@0: sl@0: // Resolve the ambiguity between our op!= and the one in rel_ops sl@0: sl@0: template inline bool operator!=(shared_ptr const & a, shared_ptr const & b) sl@0: { sl@0: return a.get() != b.get(); sl@0: } sl@0: sl@0: #endif sl@0: sl@0: template inline bool operator<(shared_ptr const & a, shared_ptr const & b) sl@0: { sl@0: return a._internal_less(b); sl@0: } sl@0: sl@0: template inline void swap(shared_ptr & a, shared_ptr & b) sl@0: { sl@0: a.swap(b); sl@0: } sl@0: sl@0: template shared_ptr static_pointer_cast(shared_ptr const & r) sl@0: { sl@0: return shared_ptr(r, boost::detail::static_cast_tag()); sl@0: } sl@0: sl@0: template shared_ptr const_pointer_cast(shared_ptr const & r) sl@0: { sl@0: return shared_ptr(r, boost::detail::const_cast_tag()); sl@0: } sl@0: sl@0: template shared_ptr dynamic_pointer_cast(shared_ptr const & r) sl@0: { sl@0: return shared_ptr(r, boost::detail::dynamic_cast_tag()); sl@0: } sl@0: sl@0: // shared_*_cast names are deprecated. Use *_pointer_cast instead. sl@0: sl@0: template shared_ptr shared_static_cast(shared_ptr const & r) sl@0: { sl@0: return shared_ptr(r, boost::detail::static_cast_tag()); sl@0: } sl@0: sl@0: template shared_ptr shared_dynamic_cast(shared_ptr const & r) sl@0: { sl@0: return shared_ptr(r, boost::detail::dynamic_cast_tag()); sl@0: } sl@0: sl@0: template shared_ptr shared_polymorphic_cast(shared_ptr const & r) sl@0: { sl@0: return shared_ptr(r, boost::detail::polymorphic_cast_tag()); sl@0: } sl@0: sl@0: template shared_ptr shared_polymorphic_downcast(shared_ptr const & r) sl@0: { sl@0: BOOST_ASSERT(dynamic_cast(r.get()) == r.get()); sl@0: return shared_static_cast(r); sl@0: } sl@0: sl@0: // get_pointer() enables boost::mem_fn to recognize shared_ptr sl@0: sl@0: template inline T * get_pointer(shared_ptr const & p) sl@0: { sl@0: return p.get(); sl@0: } sl@0: sl@0: // operator<< sl@0: sl@0: #if defined(__GNUC__) && (__GNUC__ < 3) sl@0: sl@0: template std::ostream & operator<< (std::ostream & os, shared_ptr const & p) sl@0: { sl@0: os << p.get(); sl@0: return os; sl@0: } sl@0: sl@0: #else sl@0: sl@0: // in STLport's no-iostreams mode no iostream symbols can be used sl@0: #ifndef _STLP_NO_IOSTREAMS sl@0: sl@0: # if defined(BOOST_MSVC) && BOOST_WORKAROUND(BOOST_MSVC, < 1300 && __SGI_STL_PORT) sl@0: // MSVC6 has problems finding std::basic_ostream through the using declaration in namespace _STL sl@0: using std::basic_ostream; sl@0: template basic_ostream & operator<< (basic_ostream & os, shared_ptr const & p) sl@0: # else sl@0: template std::basic_ostream & operator<< (std::basic_ostream & os, shared_ptr const & p) sl@0: # endif sl@0: { sl@0: os << p.get(); sl@0: return os; sl@0: } sl@0: sl@0: #endif // _STLP_NO_IOSTREAMS sl@0: sl@0: #endif // __GNUC__ < 3 sl@0: sl@0: // get_deleter (experimental) sl@0: sl@0: #if ( defined(__GNUC__) && BOOST_WORKAROUND(__GNUC__, < 3) ) || \ sl@0: ( defined(__EDG_VERSION__) && BOOST_WORKAROUND(__EDG_VERSION__, <= 238) ) || \ sl@0: ( defined(__HP_aCC) && BOOST_WORKAROUND(__HP_aCC, <= 33500) ) sl@0: sl@0: // g++ 2.9x doesn't allow static_cast(void *) sl@0: // apparently EDG 2.38 and HP aCC A.03.35 also don't accept it sl@0: sl@0: template D * get_deleter(shared_ptr const & p) sl@0: { sl@0: void const * q = p._internal_get_deleter(typeid(D)); sl@0: return const_cast(static_cast(q)); sl@0: } sl@0: sl@0: #else sl@0: sl@0: template D * get_deleter(shared_ptr const & p) sl@0: { sl@0: return static_cast(p._internal_get_deleter(typeid(D))); sl@0: } sl@0: sl@0: #endif sl@0: sl@0: } // namespace boost sl@0: sl@0: #ifdef BOOST_MSVC sl@0: # pragma warning(pop) sl@0: #endif sl@0: sl@0: #endif // #if defined(BOOST_NO_MEMBER_TEMPLATES) && !defined(BOOST_MSVC6_MEMBER_TEMPLATES) sl@0: sl@0: #endif // #ifndef BOOST_SHARED_PTR_HPP_INCLUDED