#ifndef BOOST_ENABLE_SHARED_FROM_THIS_HPP_INCLUDED #define BOOST_ENABLE_SHARED_FROM_THIS_HPP_INCLUDED // // enable_shared_from_this.hpp // // Copyright (c) 2002 Peter Dimov // // Distributed under the Boost Software License, Version 1.0. (See // accompanying file LICENSE_1_0.txt or copy at // http://www.boost.org/LICENSE_1_0.txt) // // http://www.boost.org/libs/smart_ptr/enable_shared_from_this.html // #include #include #include #include namespace boost { template class enable_shared_from_this { void init_internal_shared_once() const { if( !owned() && _internal_shared_count.empty() ) { detail::shared_count( (void*)0, detail::sp_deleter_wrapper() ).swap(_internal_shared_count); _internal_weak_count = _internal_shared_count; } } bool owned() const { return _owned; } mutable detail::shared_count _internal_shared_count; mutable detail::weak_count _internal_weak_count; mutable bool _owned; protected: enable_shared_from_this(): _owned(false) { } enable_shared_from_this(enable_shared_from_this const &): _owned(false) { } enable_shared_from_this & operator=(enable_shared_from_this const &) { return *this; } // virtual destructor because we need a vtable for dynamic_cast from base to derived to work virtual ~enable_shared_from_this() { // make sure no dangling shared_ptr objects were created by the // user calling shared_from_this() but never passing ownership of the object // to a shared_ptr. BOOST_ASSERT(owned() || _internal_shared_count.use_count() <= 1); } public: shared_ptr shared_from_this() { init_internal_shared_once(); T * p = dynamic_cast(this); return shared_ptr( detail::shared_count( _internal_weak_count ), p ); } shared_ptr shared_from_this() const { init_internal_shared_once(); T const * p = dynamic_cast(this); return shared_ptr( detail::shared_count( _internal_weak_count ), p ); } template void _internal_accept_owner(shared_ptr &owner) const { if( !_owned ) { if( _internal_shared_count.empty() ) { _internal_weak_count = owner.get_shared_count(); }else { BOOST_ASSERT(owner.unique()); // no weak_ptrs to owner should exist either, but there's no way to check that typedef detail::sp_deleter_wrapper D; D * pd = static_cast(_internal_shared_count.get_deleter(BOOST_SP_TYPEID(D))); BOOST_ASSERT( pd != 0 ); pd->set_deleter(owner); shared_ptr( _internal_shared_count, owner.get() ).swap( owner ); detail::shared_count().swap(_internal_shared_count); } _owned = true; } } }; template< class T, class Y > inline void sp_accept_owner( boost::shared_ptr * ptr, boost::enable_shared_from_this const * pe ) { if( pe != 0 ) { pe->_internal_accept_owner( *ptr ); } } template< class T, class Y > inline void sp_accept_owner( boost::shared_ptr * ptr, boost::enable_shared_from_this const * pe, void * /*pd*/ ) { if( pe != 0 ) { pe->_internal_accept_owner( *ptr ); } } template< class T, class Y > inline void sp_accept_owner( boost::shared_ptr * /*ptr*/, boost::enable_shared_from_this const * /*pe*/, boost::detail::sp_deleter_wrapper * /*pd*/ ) { } } // namespace boost #endif // #ifndef BOOST_ENABLE_SHARED_FROM_THIS_HPP_INCLUDED