mirror of
https://github.com/boostorg/smart_ptr.git
synced 2025-07-31 21:24:40 +02:00
Add shared_ptr constructor taking std::unique_ptr. Refs #6625.
[SVN r81131]
This commit is contained in:
@@ -37,6 +37,10 @@
|
|||||||
#include <functional> // std::less
|
#include <functional> // std::less
|
||||||
#include <new> // std::bad_alloc
|
#include <new> // std::bad_alloc
|
||||||
|
|
||||||
|
#if !defined( BOOST_NO_CXX11_SMART_PTR )
|
||||||
|
# include <boost/utility/addressof.hpp>
|
||||||
|
#endif
|
||||||
|
|
||||||
namespace boost
|
namespace boost
|
||||||
{
|
{
|
||||||
|
|
||||||
@@ -56,6 +60,38 @@ template< class D > struct sp_inplace_tag
|
|||||||
{
|
{
|
||||||
};
|
};
|
||||||
|
|
||||||
|
#if !defined( BOOST_NO_CXX11_SMART_PTR )
|
||||||
|
|
||||||
|
template< class T > class sp_reference_wrapper
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
|
||||||
|
explicit sp_reference_wrapper( T & t): t_( boost::addressof( t ) )
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
template< class Y > void operator()( Y * p ) const
|
||||||
|
{
|
||||||
|
(*t_)( p );
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
|
||||||
|
T * t_;
|
||||||
|
};
|
||||||
|
|
||||||
|
template< class D > struct sp_convert_reference
|
||||||
|
{
|
||||||
|
typedef D type;
|
||||||
|
};
|
||||||
|
|
||||||
|
template< class D > struct sp_convert_reference< D& >
|
||||||
|
{
|
||||||
|
typedef sp_reference_wrapper< D > type;
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
class weak_count;
|
class weak_count;
|
||||||
|
|
||||||
class shared_count
|
class shared_count
|
||||||
@@ -300,6 +336,33 @@ public:
|
|||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#if !defined( BOOST_NO_CXX11_SMART_PTR )
|
||||||
|
|
||||||
|
template<class Y, class D>
|
||||||
|
explicit shared_count( std::unique_ptr<Y, D> & r ): pi_( 0 )
|
||||||
|
#if defined(BOOST_SP_ENABLE_DEBUG_HOOKS)
|
||||||
|
, id_(shared_count_id)
|
||||||
|
#endif
|
||||||
|
{
|
||||||
|
typedef typename sp_convert_reference<D>::type D2;
|
||||||
|
|
||||||
|
D2 d2( r.get_deleter() );
|
||||||
|
pi_ = new sp_counted_impl_pd< Y*, D2 >( r.get(), d2 );
|
||||||
|
|
||||||
|
#ifdef BOOST_NO_EXCEPTIONS
|
||||||
|
|
||||||
|
if( pi_ == 0 )
|
||||||
|
{
|
||||||
|
boost::throw_exception( std::bad_alloc() );
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
|
r.release();
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
~shared_count() // nothrow
|
~shared_count() // nothrow
|
||||||
{
|
{
|
||||||
if( pi_ != 0 ) pi_->release();
|
if( pi_ != 0 ) pi_->release();
|
||||||
|
@@ -298,6 +298,18 @@ public:
|
|||||||
|
|
||||||
#endif // BOOST_NO_AUTO_PTR
|
#endif // BOOST_NO_AUTO_PTR
|
||||||
|
|
||||||
|
#if !defined( BOOST_NO_CXX11_SMART_PTR )
|
||||||
|
|
||||||
|
template< class Y, class D >
|
||||||
|
shared_ptr( std::unique_ptr< Y, D > && r): px( r.get() ), pn()
|
||||||
|
{
|
||||||
|
Y * tmp = r.get();
|
||||||
|
pn = boost::detail::shared_count( r );
|
||||||
|
boost::detail::sp_enable_shared_from_this( this, tmp, tmp );
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
// assignment
|
// assignment
|
||||||
|
|
||||||
shared_ptr & operator=( shared_ptr const & r ) // never throws
|
shared_ptr & operator=( shared_ptr const & r ) // never throws
|
||||||
|
@@ -69,5 +69,6 @@ import testing ;
|
|||||||
[ run get_deleter_array_test.cpp ]
|
[ run get_deleter_array_test.cpp ]
|
||||||
[ run ip_hash_test.cpp ]
|
[ run ip_hash_test.cpp ]
|
||||||
[ run owner_less_test.cpp ]
|
[ run owner_less_test.cpp ]
|
||||||
|
[ run sp_unique_ptr_test.cpp ]
|
||||||
;
|
;
|
||||||
}
|
}
|
||||||
|
152
test/sp_unique_ptr_test.cpp
Normal file
152
test/sp_unique_ptr_test.cpp
Normal file
@@ -0,0 +1,152 @@
|
|||||||
|
//
|
||||||
|
// sp_unique_ptr_test.cpp
|
||||||
|
//
|
||||||
|
// Copyright (c) 2012 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
|
||||||
|
//
|
||||||
|
|
||||||
|
#include <boost/shared_ptr.hpp>
|
||||||
|
#include <boost/enable_shared_from_this.hpp>
|
||||||
|
#include <boost/detail/lightweight_test.hpp>
|
||||||
|
#include <memory>
|
||||||
|
#include <utility>
|
||||||
|
|
||||||
|
#if !defined( BOOST_NO_CXX11_SMART_PTR )
|
||||||
|
|
||||||
|
struct X: public boost::enable_shared_from_this< X >
|
||||||
|
{
|
||||||
|
static int instances;
|
||||||
|
|
||||||
|
X()
|
||||||
|
{
|
||||||
|
++instances;
|
||||||
|
}
|
||||||
|
|
||||||
|
~X()
|
||||||
|
{
|
||||||
|
--instances;
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
|
||||||
|
X( X const & );
|
||||||
|
X & operator=( X const & );
|
||||||
|
};
|
||||||
|
|
||||||
|
int X::instances = 0;
|
||||||
|
|
||||||
|
struct Y
|
||||||
|
{
|
||||||
|
static int instances;
|
||||||
|
|
||||||
|
bool deleted_;
|
||||||
|
|
||||||
|
Y(): deleted_( false )
|
||||||
|
{
|
||||||
|
++instances;
|
||||||
|
}
|
||||||
|
|
||||||
|
~Y()
|
||||||
|
{
|
||||||
|
BOOST_TEST( deleted_ );
|
||||||
|
--instances;
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
|
||||||
|
Y( Y const & );
|
||||||
|
Y & operator=( Y const & );
|
||||||
|
};
|
||||||
|
|
||||||
|
int Y::instances = 0;
|
||||||
|
|
||||||
|
struct YD
|
||||||
|
{
|
||||||
|
void operator()( Y* p ) const
|
||||||
|
{
|
||||||
|
p->deleted_ = true;
|
||||||
|
delete p;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
int main()
|
||||||
|
{
|
||||||
|
{
|
||||||
|
BOOST_TEST( X::instances == 0 );
|
||||||
|
|
||||||
|
std::unique_ptr<X> p( new X );
|
||||||
|
BOOST_TEST( X::instances == 1 );
|
||||||
|
|
||||||
|
boost::shared_ptr<X> p2( std::move( p ) );
|
||||||
|
BOOST_TEST( X::instances == 1 );
|
||||||
|
BOOST_TEST( p.get() == 0 );
|
||||||
|
|
||||||
|
boost::shared_ptr<X> p3 = p2->shared_from_this();
|
||||||
|
BOOST_TEST( p2 == p3 );
|
||||||
|
BOOST_TEST( !(p2 < p3) && !(p3 < p2) );
|
||||||
|
|
||||||
|
p2.reset();
|
||||||
|
p3.reset();
|
||||||
|
BOOST_TEST( X::instances == 0 );
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
BOOST_TEST( Y::instances == 0 );
|
||||||
|
|
||||||
|
std::unique_ptr<Y, YD> p( new Y, YD() );
|
||||||
|
BOOST_TEST( Y::instances == 1 );
|
||||||
|
|
||||||
|
boost::shared_ptr<Y> p2( std::move( p ) );
|
||||||
|
BOOST_TEST( Y::instances == 1 );
|
||||||
|
BOOST_TEST( p.get() == 0 );
|
||||||
|
|
||||||
|
p2.reset();
|
||||||
|
BOOST_TEST( Y::instances == 0 );
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
BOOST_TEST( Y::instances == 0 );
|
||||||
|
|
||||||
|
YD yd;
|
||||||
|
|
||||||
|
std::unique_ptr<Y, YD&> p( new Y, yd );
|
||||||
|
BOOST_TEST( Y::instances == 1 );
|
||||||
|
|
||||||
|
boost::shared_ptr<Y> p2( std::move( p ) );
|
||||||
|
BOOST_TEST( Y::instances == 1 );
|
||||||
|
BOOST_TEST( p.get() == 0 );
|
||||||
|
|
||||||
|
p2.reset();
|
||||||
|
BOOST_TEST( Y::instances == 0 );
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
BOOST_TEST( Y::instances == 0 );
|
||||||
|
|
||||||
|
YD yd;
|
||||||
|
|
||||||
|
std::unique_ptr<Y, YD const&> p( new Y, yd );
|
||||||
|
BOOST_TEST( Y::instances == 1 );
|
||||||
|
|
||||||
|
boost::shared_ptr<Y> p2( std::move( p ) );
|
||||||
|
BOOST_TEST( Y::instances == 1 );
|
||||||
|
BOOST_TEST( p.get() == 0 );
|
||||||
|
|
||||||
|
p2.reset();
|
||||||
|
BOOST_TEST( Y::instances == 0 );
|
||||||
|
}
|
||||||
|
|
||||||
|
return boost::report_errors();
|
||||||
|
}
|
||||||
|
|
||||||
|
#else // !defined( BOOST_NO_CXX11_SMART_PTR )
|
||||||
|
|
||||||
|
int main()
|
||||||
|
{
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif
|
Reference in New Issue
Block a user