Add constructor/assignment taking boost::movelib::unique_ptr

This commit is contained in:
Peter Dimov
2015-05-04 01:06:42 +03:00
parent 290fe82a43
commit d875a68ceb
4 changed files with 311 additions and 7 deletions

View File

@ -40,13 +40,18 @@
# include <new> // std::bad_alloc
#endif
#if !defined( BOOST_NO_CXX11_SMART_PTR )
# include <boost/utility/addressof.hpp>
#endif
#include <boost/core/addressof.hpp>
namespace boost
{
namespace movelib
{
template< class T, class D > class unique_ptr;
} // namespace movelib
namespace detail
{
@ -63,8 +68,6 @@ template< class D > struct sp_inplace_tag
{
};
#if !defined( BOOST_NO_CXX11_SMART_PTR )
template< class T > class sp_reference_wrapper
{
public:
@ -93,8 +96,6 @@ template< class D > struct sp_convert_reference< D& >
typedef sp_reference_wrapper< D > type;
};
#endif
class weak_count;
class shared_count
@ -438,6 +439,29 @@ public:
#endif
template<class Y, class D>
explicit shared_count( boost::movelib::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< typename boost::movelib::unique_ptr<Y, D>::pointer, D2 >( r.get(), d2 );
#ifdef BOOST_NO_EXCEPTIONS
if( pi_ == 0 )
{
boost::throw_exception( std::bad_alloc() );
}
#endif
r.release();
}
~shared_count() // nothrow
{
if( pi_ != 0 ) pi_->release();

View File

@ -55,6 +55,13 @@ template<class T> class weak_ptr;
template<class T> class enable_shared_from_this;
class enable_shared_from_raw;
namespace movelib
{
template< class T, class D > class unique_ptr;
} // namespace movelib
namespace detail
{
@ -495,6 +502,17 @@ public:
#endif
template< class Y, class D >
shared_ptr( boost::movelib::unique_ptr< Y, D > r ): px( r.get() ), pn()
{
boost::detail::sp_assert_convertible< Y, T >();
typename boost::movelib::unique_ptr< Y, D >::pointer tmp = r.get();
pn = boost::detail::shared_count( r );
boost::detail::sp_deleter_construct( this, tmp );
}
// assignment
shared_ptr & operator=( shared_ptr const & r ) BOOST_NOEXCEPT
@ -556,6 +574,27 @@ public:
#endif
template<class Y, class D>
shared_ptr & operator=( boost::movelib::unique_ptr<Y, D> r )
{
// this_type( static_cast< unique_ptr<Y, D> && >( r ) ).swap( *this );
boost::detail::sp_assert_convertible< Y, T >();
typename boost::movelib::unique_ptr< Y, D >::pointer p = r.get();
shared_ptr tmp;
tmp.px = p;
tmp.pn = boost::detail::shared_count( r );
boost::detail::sp_deleter_construct( &tmp, p );
tmp.swap( *this );
return *this;
}
// Move support
#if !defined( BOOST_NO_CXX11_RVALUE_REFERENCES )

View File

@ -179,5 +179,7 @@ import testing ;
[ run weak_from_this_test.cpp ]
[ run weak_from_this_test2.cpp ]
[ run sp_bml_unique_ptr_test.cpp ]
;
}

View File

@ -0,0 +1,239 @@
//
// sp_bml_unique_ptr_test.cpp
//
// Copyright (c) 2012, 2015 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 <boost/move/unique_ptr.hpp>
#include <memory>
#include <utility>
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 );
boost::movelib::unique_ptr<X> p( new X );
BOOST_TEST( X::instances == 1 );
boost::shared_ptr<X> p2( boost::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 );
p2 = boost::movelib::unique_ptr<X>( new X );
BOOST_TEST( X::instances == 1 );
p2 = boost::movelib::unique_ptr<X>( new X );
BOOST_TEST( X::instances == 1 );
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( X::instances == 0 );
boost::movelib::unique_ptr<X> p( new X );
BOOST_TEST( X::instances == 1 );
boost::shared_ptr<X const> p2( boost::move( p ) );
BOOST_TEST( X::instances == 1 );
BOOST_TEST( p.get() == 0 );
boost::shared_ptr<X const> p3 = p2->shared_from_this();
BOOST_TEST( p2 == p3 );
BOOST_TEST( !(p2 < p3) && !(p3 < p2) );
p2.reset();
p3.reset();
BOOST_TEST( X::instances == 0 );
p2 = boost::movelib::unique_ptr<X>( new X );
BOOST_TEST( X::instances == 1 );
p2 = boost::movelib::unique_ptr<X>( new X );
BOOST_TEST( X::instances == 1 );
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( X::instances == 0 );
boost::movelib::unique_ptr<X> p( new X );
BOOST_TEST( X::instances == 1 );
boost::shared_ptr<void> p2( boost::move( p ) );
BOOST_TEST( X::instances == 1 );
BOOST_TEST( p.get() == 0 );
p2.reset();
BOOST_TEST( X::instances == 0 );
p2 = boost::movelib::unique_ptr<X>( new X );
BOOST_TEST( X::instances == 1 );
p2 = boost::movelib::unique_ptr<X>( new X );
BOOST_TEST( X::instances == 1 );
p2.reset();
BOOST_TEST( X::instances == 0 );
}
{
BOOST_TEST( Y::instances == 0 );
boost::movelib::unique_ptr<Y, YD> p( new Y, YD() );
BOOST_TEST( Y::instances == 1 );
boost::shared_ptr<Y> p2( boost::move( p ) );
BOOST_TEST( Y::instances == 1 );
BOOST_TEST( p.get() == 0 );
p2.reset();
BOOST_TEST( Y::instances == 0 );
p2 = boost::movelib::unique_ptr<Y, YD>( new Y, YD() );
BOOST_TEST( Y::instances == 1 );
p2 = boost::movelib::unique_ptr<Y, YD>( new Y, YD() );
BOOST_TEST( Y::instances == 1 );
p2.reset();
BOOST_TEST( Y::instances == 0 );
}
{
BOOST_TEST( Y::instances == 0 );
YD yd;
boost::movelib::unique_ptr<Y, YD&> p( new Y, yd );
BOOST_TEST( Y::instances == 1 );
boost::shared_ptr<Y> p2( boost::move( p ) );
BOOST_TEST( Y::instances == 1 );
BOOST_TEST( p.get() == 0 );
p2.reset();
BOOST_TEST( Y::instances == 0 );
p2 = boost::movelib::unique_ptr<Y, YD&>( new Y, yd );
BOOST_TEST( Y::instances == 1 );
p2 = boost::movelib::unique_ptr<Y, YD&>( new Y, yd );
BOOST_TEST( Y::instances == 1 );
p2.reset();
BOOST_TEST( Y::instances == 0 );
}
{
BOOST_TEST( Y::instances == 0 );
YD yd;
boost::movelib::unique_ptr<Y, YD const&> p( new Y, yd );
BOOST_TEST( Y::instances == 1 );
boost::shared_ptr<Y> p2( boost::move( p ) );
BOOST_TEST( Y::instances == 1 );
BOOST_TEST( p.get() == 0 );
p2.reset();
BOOST_TEST( Y::instances == 0 );
p2 = boost::movelib::unique_ptr<Y, YD const&>( new Y, yd );
BOOST_TEST( Y::instances == 1 );
p2 = boost::movelib::unique_ptr<Y, YD const&>( new Y, yd );
BOOST_TEST( Y::instances == 1 );
p2.reset();
BOOST_TEST( Y::instances == 0 );
}
return boost::report_errors();
}