2002-11-18 14:37:02 +00:00
|
|
|
#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
|
|
|
|
//
|
2004-07-26 00:32:12 +00:00
|
|
|
// 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)
|
2002-11-18 14:37:02 +00:00
|
|
|
//
|
|
|
|
// http://www.boost.org/libs/smart_ptr/enable_shared_from_this.html
|
|
|
|
//
|
|
|
|
|
2008-04-22 19:48:39 +00:00
|
|
|
#include <boost/detail/shared_count.hpp>
|
2002-11-19 16:18:18 +00:00
|
|
|
#include <boost/shared_ptr.hpp>
|
|
|
|
#include <boost/assert.hpp>
|
|
|
|
#include <boost/config.hpp>
|
2002-11-18 14:37:02 +00:00
|
|
|
|
|
|
|
namespace boost
|
|
|
|
{
|
|
|
|
|
|
|
|
template<class T> class enable_shared_from_this
|
|
|
|
{
|
2008-03-20 19:32:43 +00:00
|
|
|
// dynamic cast to template type doesn't work in constructor, so we have
|
|
|
|
// to use lazy initialization
|
|
|
|
void init_internal_shared_once() const
|
|
|
|
{
|
2008-04-12 18:22:18 +00:00
|
|
|
if( !owned() && _internal_shared_this.get() == 0 )
|
2008-03-20 19:32:43 +00:00
|
|
|
{
|
2008-04-12 18:22:18 +00:00
|
|
|
T * p = dynamic_cast<T *>(const_cast<enable_shared_from_this*>(this));
|
|
|
|
_internal_shared_this = shared_ptr<T>( p, detail::sp_deleter_wrapper() );
|
2008-03-20 19:32:43 +00:00
|
|
|
BOOST_ASSERT(_internal_shared_this.get() == this);
|
2008-04-22 19:48:39 +00:00
|
|
|
_internal_weak_count = _internal_shared_this.get_shared_count();
|
2008-03-20 19:32:43 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool owned() const
|
|
|
|
{
|
|
|
|
return _owned;
|
|
|
|
}
|
|
|
|
|
|
|
|
typedef T _internal_element_type; // for bcc 5.5.1
|
|
|
|
mutable shared_ptr<_internal_element_type> _internal_shared_this;
|
2008-04-22 19:48:39 +00:00
|
|
|
mutable detail::weak_count _internal_weak_count;
|
2008-03-20 19:32:43 +00:00
|
|
|
mutable bool _owned;
|
2008-03-20 22:10:52 +00:00
|
|
|
|
2003-03-28 12:27:55 +00:00
|
|
|
protected:
|
|
|
|
|
2008-03-20 19:32:43 +00:00
|
|
|
enable_shared_from_this():
|
|
|
|
_owned(false)
|
2003-03-28 12:27:55 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2008-03-20 22:10:52 +00:00
|
|
|
enable_shared_from_this(enable_shared_from_this const &):
|
|
|
|
_owned(false)
|
2003-03-28 12:27:55 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
enable_shared_from_this & operator=(enable_shared_from_this const &)
|
|
|
|
{
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
2008-03-20 19:32:43 +00:00
|
|
|
// virtual destructor because we need a vtable for dynamic_cast from base to derived to work
|
|
|
|
virtual ~enable_shared_from_this()
|
2003-03-28 12:27:55 +00:00
|
|
|
{
|
2008-03-20 19:32:43 +00:00
|
|
|
// 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_this.use_count() <= 1);
|
2003-03-28 12:27:55 +00:00
|
|
|
}
|
|
|
|
|
2002-11-18 14:37:02 +00:00
|
|
|
public:
|
|
|
|
|
|
|
|
shared_ptr<T> shared_from_this()
|
|
|
|
{
|
2008-03-20 19:32:43 +00:00
|
|
|
init_internal_shared_once();
|
2008-04-22 19:48:39 +00:00
|
|
|
T * p = dynamic_cast<T *>(this);
|
|
|
|
return shared_ptr<T>( detail::shared_count( _internal_weak_count ), p );
|
2002-11-18 14:37:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
shared_ptr<T const> shared_from_this() const
|
|
|
|
{
|
2008-03-20 19:32:43 +00:00
|
|
|
init_internal_shared_once();
|
2008-04-22 19:48:39 +00:00
|
|
|
T const * p = dynamic_cast<T const *>(this);
|
|
|
|
return shared_ptr<T const>( detail::shared_count( _internal_weak_count ), p );
|
2002-11-18 14:37:02 +00:00
|
|
|
}
|
|
|
|
|
2008-03-20 19:32:43 +00:00
|
|
|
template<typename U>
|
|
|
|
void _internal_accept_owner(shared_ptr<U> &owner) const
|
|
|
|
{
|
2008-03-25 15:46:40 +00:00
|
|
|
if( !_owned )
|
|
|
|
{
|
2008-04-16 00:06:29 +00:00
|
|
|
if( !_internal_shared_this )
|
|
|
|
{
|
2008-04-22 19:48:39 +00:00
|
|
|
_internal_weak_count = owner.get_shared_count();
|
2008-04-16 00:06:29 +00:00
|
|
|
}else
|
|
|
|
{
|
2008-04-17 13:40:44 +00:00
|
|
|
BOOST_ASSERT(owner.unique()); // no weak_ptrs to owner should exist either, but there's no way to check that
|
2008-04-16 00:06:29 +00:00
|
|
|
detail::sp_deleter_wrapper * pd = get_deleter<detail::sp_deleter_wrapper>(_internal_shared_this);
|
|
|
|
BOOST_ASSERT( pd != 0 );
|
|
|
|
pd->set_deleter(owner);
|
|
|
|
|
|
|
|
owner.reset( _internal_shared_this, owner.get() );
|
|
|
|
_internal_shared_this.reset();
|
|
|
|
}
|
2008-03-25 15:46:40 +00:00
|
|
|
_owned = true;
|
|
|
|
}
|
2008-03-20 19:32:43 +00:00
|
|
|
}
|
2002-11-18 14:37:02 +00:00
|
|
|
};
|
|
|
|
|
2008-04-12 18:22:18 +00:00
|
|
|
template< class T, class Y > inline void sp_accept_owner( boost::shared_ptr<Y> * ptr, boost::enable_shared_from_this<T> const * pe )
|
|
|
|
{
|
|
|
|
if( pe != 0 )
|
|
|
|
{
|
|
|
|
pe->_internal_accept_owner( *ptr );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
template< class T, class Y > inline void sp_accept_owner( boost::shared_ptr<Y> * ptr, boost::enable_shared_from_this<T> 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<Y> * /*ptr*/, boost::enable_shared_from_this<T> const * /*pe*/, boost::detail::sp_deleter_wrapper * /*pd*/ )
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2002-11-18 14:37:02 +00:00
|
|
|
} // namespace boost
|
|
|
|
|
|
|
|
#endif // #ifndef BOOST_ENABLE_SHARED_FROM_THIS_HPP_INCLUDED
|