forked from boostorg/smart_ptr
Make null unique_ptr convert to empty shared_ptr
This commit is contained in:
@ -502,9 +502,12 @@ public:
|
||||
boost::detail::sp_assert_convertible< Y, T >();
|
||||
|
||||
typename std::unique_ptr< Y, D >::pointer tmp = r.get();
|
||||
pn = boost::detail::shared_count( r );
|
||||
|
||||
boost::detail::sp_deleter_construct( this, tmp );
|
||||
if( tmp != 0 )
|
||||
{
|
||||
pn = boost::detail::shared_count( r );
|
||||
boost::detail::sp_deleter_construct( this, tmp );
|
||||
}
|
||||
}
|
||||
|
||||
#endif
|
||||
@ -515,9 +518,12 @@ public:
|
||||
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 );
|
||||
if( tmp != 0 )
|
||||
{
|
||||
pn = boost::detail::shared_count( r );
|
||||
boost::detail::sp_deleter_construct( this, tmp );
|
||||
}
|
||||
}
|
||||
|
||||
// assignment
|
||||
@ -592,10 +598,13 @@ public:
|
||||
|
||||
shared_ptr tmp;
|
||||
|
||||
tmp.px = p;
|
||||
tmp.pn = boost::detail::shared_count( r );
|
||||
if( p != 0 )
|
||||
{
|
||||
tmp.px = p;
|
||||
tmp.pn = boost::detail::shared_count( r );
|
||||
|
||||
boost::detail::sp_deleter_construct( &tmp, p );
|
||||
boost::detail::sp_deleter_construct( &tmp, p );
|
||||
}
|
||||
|
||||
tmp.swap( *this );
|
||||
|
||||
|
@ -12,6 +12,7 @@
|
||||
#include <boost/enable_shared_from_this.hpp>
|
||||
#include <boost/detail/lightweight_test.hpp>
|
||||
#include <boost/move/unique_ptr.hpp>
|
||||
#include <boost/type_traits/remove_reference.hpp>
|
||||
#include <memory>
|
||||
#include <utility>
|
||||
|
||||
@ -66,11 +67,42 @@ struct YD
|
||||
{
|
||||
void operator()( Y* p ) const
|
||||
{
|
||||
p->deleted_ = true;
|
||||
delete p;
|
||||
if( p )
|
||||
{
|
||||
p->deleted_ = true;
|
||||
delete p;
|
||||
}
|
||||
else
|
||||
{
|
||||
BOOST_ERROR( "YD::operator()(0) called" );
|
||||
}
|
||||
}
|
||||
};
|
||||
|
||||
template<class U, class T, class D> static void test_null_unique_ptr( boost::movelib::unique_ptr<T, D> p1, boost::movelib::unique_ptr<T, D> p2 )
|
||||
{
|
||||
BOOST_TEST( T::instances == 0 );
|
||||
|
||||
boost::shared_ptr<U> sp( boost::move( p1 ) );
|
||||
|
||||
BOOST_TEST( sp.get() == 0 );
|
||||
BOOST_TEST( sp.use_count() == 0 );
|
||||
|
||||
sp.reset( new T, typename boost::remove_reference<D>::type() );
|
||||
|
||||
BOOST_TEST( sp.get() != 0 );
|
||||
BOOST_TEST( sp.use_count() == 1 );
|
||||
|
||||
BOOST_TEST( T::instances == 1 );
|
||||
|
||||
sp = boost::move( p2 );
|
||||
|
||||
BOOST_TEST( sp.get() == 0 );
|
||||
BOOST_TEST( sp.use_count() == 0 );
|
||||
|
||||
BOOST_TEST( T::instances == 0 );
|
||||
}
|
||||
|
||||
int main()
|
||||
{
|
||||
{
|
||||
@ -235,5 +267,28 @@ int main()
|
||||
BOOST_TEST( Y::instances == 0 );
|
||||
}
|
||||
|
||||
{
|
||||
test_null_unique_ptr<X>( boost::movelib::unique_ptr<X>(), boost::movelib::unique_ptr<X>() );
|
||||
test_null_unique_ptr<X const>( boost::movelib::unique_ptr<X>(), boost::movelib::unique_ptr<X>() );
|
||||
test_null_unique_ptr<void>( boost::movelib::unique_ptr<X>(), boost::movelib::unique_ptr<X>() );
|
||||
test_null_unique_ptr<void const>( boost::movelib::unique_ptr<X>(), boost::movelib::unique_ptr<X>() );
|
||||
}
|
||||
|
||||
{
|
||||
test_null_unique_ptr<Y>( boost::movelib::unique_ptr<Y, YD>( 0, YD() ), boost::movelib::unique_ptr<Y, YD>( 0, YD() ) );
|
||||
test_null_unique_ptr<Y const>( boost::movelib::unique_ptr<Y, YD>( 0, YD() ), boost::movelib::unique_ptr<Y, YD>( 0, YD() ) );
|
||||
test_null_unique_ptr<void>( boost::movelib::unique_ptr<Y, YD>( 0, YD() ), boost::movelib::unique_ptr<Y, YD>( 0, YD() ) );
|
||||
test_null_unique_ptr<void const>( boost::movelib::unique_ptr<Y, YD>( 0, YD() ), boost::movelib::unique_ptr<Y, YD>( 0, YD() ) );
|
||||
}
|
||||
|
||||
{
|
||||
YD yd;
|
||||
|
||||
test_null_unique_ptr<Y>( boost::movelib::unique_ptr<Y, YD&>( 0, yd ), boost::movelib::unique_ptr<Y, YD&>( 0, yd ) );
|
||||
test_null_unique_ptr<Y const>( boost::movelib::unique_ptr<Y, YD&>( 0, yd ), boost::movelib::unique_ptr<Y, YD&>( 0, yd ) );
|
||||
test_null_unique_ptr<void>( boost::movelib::unique_ptr<Y, YD&>( 0, yd ), boost::movelib::unique_ptr<Y, YD&>( 0, yd ) );
|
||||
test_null_unique_ptr<void const>( boost::movelib::unique_ptr<Y, YD&>( 0, yd ), boost::movelib::unique_ptr<Y, YD&>( 0, yd ) );
|
||||
}
|
||||
|
||||
return boost::report_errors();
|
||||
}
|
||||
|
@ -11,6 +11,7 @@
|
||||
#include <boost/shared_ptr.hpp>
|
||||
#include <boost/enable_shared_from_this.hpp>
|
||||
#include <boost/detail/lightweight_test.hpp>
|
||||
#include <boost/type_traits/remove_reference.hpp>
|
||||
#include <memory>
|
||||
#include <utility>
|
||||
|
||||
@ -67,11 +68,42 @@ struct YD
|
||||
{
|
||||
void operator()( Y* p ) const
|
||||
{
|
||||
p->deleted_ = true;
|
||||
delete p;
|
||||
if( p )
|
||||
{
|
||||
p->deleted_ = true;
|
||||
delete p;
|
||||
}
|
||||
else
|
||||
{
|
||||
BOOST_ERROR( "YD::operator()(0) called" );
|
||||
}
|
||||
}
|
||||
};
|
||||
|
||||
template<class U, class T, class D> static void test_null_unique_ptr( std::unique_ptr<T, D> p1, std::unique_ptr<T, D> p2 )
|
||||
{
|
||||
BOOST_TEST( T::instances == 0 );
|
||||
|
||||
boost::shared_ptr<U> sp( std::move( p1 ) );
|
||||
|
||||
BOOST_TEST( sp.get() == 0 );
|
||||
BOOST_TEST( sp.use_count() == 0 );
|
||||
|
||||
sp.reset( new T, typename boost::remove_reference<D>::type() );
|
||||
|
||||
BOOST_TEST( sp.get() != 0 );
|
||||
BOOST_TEST( sp.use_count() == 1 );
|
||||
|
||||
BOOST_TEST( T::instances == 1 );
|
||||
|
||||
sp = std::move( p2 );
|
||||
|
||||
BOOST_TEST( sp.get() == 0 );
|
||||
BOOST_TEST( sp.use_count() == 0 );
|
||||
|
||||
BOOST_TEST( T::instances == 0 );
|
||||
}
|
||||
|
||||
int main()
|
||||
{
|
||||
{
|
||||
@ -226,6 +258,29 @@ int main()
|
||||
BOOST_TEST( Y::instances == 0 );
|
||||
}
|
||||
|
||||
{
|
||||
test_null_unique_ptr<X>( std::unique_ptr<X>(), std::unique_ptr<X>() );
|
||||
test_null_unique_ptr<X const>( std::unique_ptr<X>(), std::unique_ptr<X>() );
|
||||
test_null_unique_ptr<void>( std::unique_ptr<X>(), std::unique_ptr<X>() );
|
||||
test_null_unique_ptr<void const>( std::unique_ptr<X>(), std::unique_ptr<X>() );
|
||||
}
|
||||
|
||||
{
|
||||
test_null_unique_ptr<Y>( std::unique_ptr<Y, YD>( 0, YD() ), std::unique_ptr<Y, YD>( 0, YD() ) );
|
||||
test_null_unique_ptr<Y const>( std::unique_ptr<Y, YD>( 0, YD() ), std::unique_ptr<Y, YD>( 0, YD() ) );
|
||||
test_null_unique_ptr<void>( std::unique_ptr<Y, YD>( 0, YD() ), std::unique_ptr<Y, YD>( 0, YD() ) );
|
||||
test_null_unique_ptr<void const>( std::unique_ptr<Y, YD>( 0, YD() ), std::unique_ptr<Y, YD>( 0, YD() ) );
|
||||
}
|
||||
|
||||
{
|
||||
YD yd;
|
||||
|
||||
test_null_unique_ptr<Y>( std::unique_ptr<Y, YD&>( 0, yd ), std::unique_ptr<Y, YD&>( 0, yd ) );
|
||||
test_null_unique_ptr<Y const>( std::unique_ptr<Y, YD&>( 0, yd ), std::unique_ptr<Y, YD&>( 0, yd ) );
|
||||
test_null_unique_ptr<void>( std::unique_ptr<Y, YD&>( 0, yd ), std::unique_ptr<Y, YD&>( 0, yd ) );
|
||||
test_null_unique_ptr<void const>( std::unique_ptr<Y, YD&>( 0, yd ), std::unique_ptr<Y, YD&>( 0, yd ) );
|
||||
}
|
||||
|
||||
return boost::report_errors();
|
||||
}
|
||||
|
||||
|
Reference in New Issue
Block a user