Files
boost_smart_ptr/test/local_sp_test.cpp

1163 lines
32 KiB
C++
Raw Normal View History

2017-06-04 21:24:20 +03:00
//
// local_sp_test.cpp
//
// Copyright 2002, 2003, 2017 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/smart_ptr/local_shared_ptr.hpp>
#include <boost/core/lightweight_test.hpp>
struct X
{
static long instances;
X()
{
++instances;
}
~X()
{
--instances;
}
private:
X( X const & );
X & operator=( X const & );
};
long X::instances = 0;
class incomplete;
2017-06-05 16:38:41 +03:00
template<class T> static void test_empty( boost::local_shared_ptr<T> const & p )
{
BOOST_TEST( p? false: true );
BOOST_TEST( !p );
BOOST_TEST( p.get() == 0 );
BOOST_TEST( p.local_use_count() == 0 );
}
2017-06-11 20:35:51 +03:00
// default constructor
2017-06-04 21:24:20 +03:00
static void default_constructor()
{
{
boost::local_shared_ptr<int> p;
2017-06-05 16:38:41 +03:00
test_empty( p );
2017-06-04 21:24:20 +03:00
}
{
boost::local_shared_ptr<void> p;
2017-06-05 16:38:41 +03:00
test_empty( p );
2017-06-04 21:24:20 +03:00
}
{
boost::local_shared_ptr<incomplete> p;
2017-06-05 16:38:41 +03:00
test_empty( p );
2017-06-04 21:24:20 +03:00
}
BOOST_TEST( X::instances == 0 );
{
boost::local_shared_ptr<X> p;
BOOST_TEST( X::instances == 0 );
2017-06-05 16:38:41 +03:00
test_empty( p );
}
}
2017-06-11 20:35:51 +03:00
// nullptr_constructor
2017-06-05 16:38:41 +03:00
static void nullptr_constructor()
{
#if !defined( BOOST_NO_CXX11_NULLPTR )
{
boost::local_shared_ptr<int> p( nullptr );
test_empty( p );
}
{
boost::local_shared_ptr<void> p( nullptr );
test_empty( p );
}
{
boost::local_shared_ptr<incomplete> p( nullptr );
test_empty( p );
}
BOOST_TEST( X::instances == 0 );
{
boost::local_shared_ptr<X> p( nullptr );
BOOST_TEST( X::instances == 0 );
test_empty( p );
2017-06-04 21:24:20 +03:00
}
2017-06-05 16:38:41 +03:00
#endif
2017-06-04 21:24:20 +03:00
}
2017-06-11 20:35:51 +03:00
// pointer constructor
2017-06-04 21:24:20 +03:00
template<class T, class U> static void pc0_test_()
{
boost::local_shared_ptr<T> p( static_cast<U*>( 0 ) );
BOOST_TEST( p? false: true );
BOOST_TEST( !p );
BOOST_TEST( p.get() == 0 );
BOOST_TEST( p.local_use_count() == 1 );
}
template<class T> static void pc0_test()
{
pc0_test_<T, T>();
pc0_test_<T const, T const>();
pc0_test_<T volatile, T volatile>();
pc0_test_<T const volatile, T const volatile>();
pc0_test_<T const, T>();
pc0_test_<T volatile, T>();
pc0_test_<T const volatile, T>();
pc0_test_<void, T>();
pc0_test_<void const, T>();
pc0_test_<void volatile, T>();
pc0_test_<void const volatile, T>();
}
template<class T, class U> static void pc1_test_()
{
boost::local_shared_ptr<T> p( new U() );
BOOST_TEST( p? true: false );
BOOST_TEST( !!p );
BOOST_TEST( p.get() != 0 );
BOOST_TEST( p.local_use_count() == 1 );
}
template<class T> static void pc1_test()
{
pc1_test_<T, T>();
pc1_test_<T const, T const>();
pc1_test_<T volatile, T volatile>();
pc1_test_<T const volatile, T const volatile>();
pc1_test_<T const, T>();
pc1_test_<T volatile, T>();
pc1_test_<T const volatile, T>();
pc1_test_<void, T>();
pc1_test_<void const, T>();
pc1_test_<void volatile, T>();
pc1_test_<void const volatile, T>();
}
static void pointer_constructor()
{
pc0_test<int>();
pc0_test<X>();
pc1_test<int>();
BOOST_TEST( X::instances == 0 );
pc1_test<X>();
BOOST_TEST( X::instances == 0 );
}
2017-06-11 20:35:51 +03:00
// deleter constructor
2017-06-04 21:24:20 +03:00
int m = 0;
void deleter2( int * p )
{
BOOST_TEST( p == &m );
++*p;
}
template<class T> static void deleter2_test_()
{
{
m = 0;
boost::local_shared_ptr<T> p( &m, deleter2 );
BOOST_TEST( p? true: false );
BOOST_TEST( !!p );
BOOST_TEST( p.get() == &m );
BOOST_TEST( p.local_use_count() == 1 );
}
BOOST_TEST( m == 1 );
}
static void deleter_constructor()
{
deleter2_test_<int>();
deleter2_test_<int const>();
deleter2_test_<int volatile>();
deleter2_test_<int const volatile>();
deleter2_test_<void>();
deleter2_test_<void const>();
deleter2_test_<void volatile>();
deleter2_test_<void const volatile>();
}
2017-06-11 20:35:51 +03:00
// nullptr_deleter_constructor
#if !defined( BOOST_NO_CXX11_NULLPTR )
void deleter3( boost::detail::sp_nullptr_t )
{
++m;
}
template<class T> static void deleter3_test_()
{
{
m = 0;
boost::local_shared_ptr<T> p( nullptr, deleter3 );
BOOST_TEST( !p );
BOOST_TEST( p.get() == 0 );
BOOST_TEST( p.local_use_count() == 1 );
}
BOOST_TEST( m == 1 );
}
static void nullptr_deleter_constructor()
{
deleter3_test_<int>();
deleter3_test_<int const>();
deleter3_test_<int volatile>();
deleter3_test_<int const volatile>();
deleter3_test_<void>();
deleter3_test_<void const>();
deleter3_test_<void volatile>();
deleter3_test_<void const volatile>();
}
#else
static void nullptr_deleter_constructor()
{
}
#endif
// allocator constructor
template<class T> static void allocator_test_()
{
{
m = 0;
boost::local_shared_ptr<T> p( &m, deleter2, std::allocator<void>() );
BOOST_TEST( p? true: false );
BOOST_TEST( !!p );
BOOST_TEST( p.get() == &m );
BOOST_TEST( p.local_use_count() == 1 );
}
BOOST_TEST( m == 1 );
}
static void allocator_constructor()
{
allocator_test_<int>();
allocator_test_<int const>();
allocator_test_<int volatile>();
allocator_test_<int const volatile>();
allocator_test_<void>();
allocator_test_<void const>();
allocator_test_<void volatile>();
allocator_test_<void const volatile>();
}
// nullptr_allocator_constructor
#if !defined( BOOST_NO_CXX11_NULLPTR )
template<class T> static void allocator3_test_()
{
{
m = 0;
boost::local_shared_ptr<T> p( nullptr, deleter3, std::allocator<void>() );
BOOST_TEST( !p );
BOOST_TEST( p.get() == 0 );
BOOST_TEST( p.local_use_count() == 1 );
}
BOOST_TEST( m == 1 );
}
static void nullptr_allocator_constructor()
{
allocator3_test_<int>();
allocator3_test_<int const>();
allocator3_test_<int volatile>();
allocator3_test_<int const volatile>();
allocator3_test_<void>();
allocator3_test_<void const>();
allocator3_test_<void volatile>();
allocator3_test_<void const volatile>();
}
#else
static void nullptr_allocator_constructor()
{
}
#endif
// copy constructor
2017-06-04 21:24:20 +03:00
template<class T> static void empty_copy_test()
{
boost::local_shared_ptr<T> p1;
test_empty( p1 );
boost::local_shared_ptr<T> p2( p1 );
test_empty( p2 );
boost::local_shared_ptr<T const> p3( p1 );
test_empty( p3 );
boost::local_shared_ptr<void> p4( p1 );
test_empty( p4 );
boost::local_shared_ptr<void const> p5( p3 );
test_empty( p5 );
}
template<class T, class U> static void test_nonempty_copy( boost::local_shared_ptr<U> const & p1 )
{
long k = p1.local_use_count();
{
boost::local_shared_ptr<T> p2( p1 );
BOOST_TEST( p2.get() == p1.get() );
BOOST_TEST( p2.local_use_count() == p1.local_use_count() );
BOOST_TEST( p2.local_use_count() == k + 1 );
}
BOOST_TEST( p1.local_use_count() == k );
}
template<class T> static void null_copy_test()
{
boost::local_shared_ptr<T> p1( static_cast<T*>(0) );
test_nonempty_copy<T>( p1 );
test_nonempty_copy<T const>( p1 );
test_nonempty_copy<T volatile>( p1 );
test_nonempty_copy<T const volatile>( p1 );
test_nonempty_copy<void>( p1 );
test_nonempty_copy<void const>( p1 );
test_nonempty_copy<void volatile>( p1 );
test_nonempty_copy<void const volatile>( p1 );
}
template<class T> static void new_copy_test()
{
boost::local_shared_ptr<T> p1( new T() );
test_nonempty_copy<T>( p1 );
test_nonempty_copy<T const>( p1 );
test_nonempty_copy<T volatile>( p1 );
test_nonempty_copy<T const volatile>( p1 );
test_nonempty_copy<void>( p1 );
test_nonempty_copy<void const>( p1 );
test_nonempty_copy<void volatile>( p1 );
test_nonempty_copy<void const volatile>( p1 );
}
static void copy_constructor()
{
empty_copy_test<int>();
empty_copy_test<incomplete>();
empty_copy_test<X>();
2017-06-11 20:35:51 +03:00
2017-06-05 16:38:41 +03:00
BOOST_TEST( X::instances == 0 );
2017-06-04 21:24:20 +03:00
null_copy_test<int>();
null_copy_test<X>();
2017-06-11 20:35:51 +03:00
2017-06-05 16:38:41 +03:00
BOOST_TEST( X::instances == 0 );
2017-06-04 21:24:20 +03:00
new_copy_test<int>();
new_copy_test<X>();
2017-06-11 20:35:51 +03:00
2017-06-05 16:38:41 +03:00
BOOST_TEST( X::instances == 0 );
}
2017-06-11 20:35:51 +03:00
// move constructor
#if !defined( BOOST_NO_CXX11_RVALUE_REFERENCES )
template<class T> static void empty_move_test()
{
boost::local_shared_ptr<T> p2(( boost::local_shared_ptr<T>() ));
test_empty( p2 );
boost::local_shared_ptr<T const> p3(( boost::local_shared_ptr<T>() ));
test_empty( p3 );
boost::local_shared_ptr<void> p4(( boost::local_shared_ptr<T>() ));
test_empty( p4 );
boost::local_shared_ptr<void const> p5( std::move(p3) );
test_empty( p5 );
}
template<class T, class U> static void test_nonempty_move( boost::local_shared_ptr<U> && p1 )
{
U* q = p1.get();
long k = p1.local_use_count();
boost::local_shared_ptr<T> p2( std::move(p1) );
BOOST_TEST( p2.get() == q );
BOOST_TEST( p2.local_use_count() == k );
BOOST_TEST( p1.get() == 0 );
BOOST_TEST( p1.local_use_count() == 0 );
}
template<class T> static void null_move_test()
{
test_nonempty_move<T>( boost::local_shared_ptr<T>( static_cast<T*>(0) ) );
test_nonempty_move<T const>( boost::local_shared_ptr<T>( static_cast<T*>(0) ) );
test_nonempty_move<T volatile>( boost::local_shared_ptr<T>( static_cast<T*>(0) ) );
test_nonempty_move<T const volatile>( boost::local_shared_ptr<T>( static_cast<T*>(0) ) );
test_nonempty_move<void>( boost::local_shared_ptr<T>( static_cast<T*>(0) ) );
test_nonempty_move<void const>( boost::local_shared_ptr<T>( static_cast<T*>(0) ) );
test_nonempty_move<void volatile>( boost::local_shared_ptr<T>( static_cast<T*>(0) ) );
test_nonempty_move<void const volatile>( boost::local_shared_ptr<T>( static_cast<T*>(0) ) );
}
template<class T> static void new_move_test()
{
test_nonempty_move<T>( boost::local_shared_ptr<T>( new T() ) );
test_nonempty_move<T const>( boost::local_shared_ptr<T>( new T() ) );
test_nonempty_move<T volatile>( boost::local_shared_ptr<T>( new T() ) );
test_nonempty_move<T const volatile>( boost::local_shared_ptr<T>( new T() ) );
test_nonempty_move<void>( boost::local_shared_ptr<T>( new T() ) );
test_nonempty_move<void const>( boost::local_shared_ptr<T>( new T() ) );
test_nonempty_move<void volatile>( boost::local_shared_ptr<T>( new T() ) );
test_nonempty_move<void const volatile>( boost::local_shared_ptr<T>( new T() ) );
}
static void move_constructor()
{
empty_move_test<int>();
empty_move_test<incomplete>();
empty_move_test<X>();
BOOST_TEST( X::instances == 0 );
null_move_test<int>();
null_move_test<X>();
BOOST_TEST( X::instances == 0 );
new_move_test<int>();
new_move_test<X>();
BOOST_TEST( X::instances == 0 );
}
#else
static void move_constructor()
{
}
#endif
// aliasing constructor
2017-06-05 16:38:41 +03:00
template<class T, class U> void test_aliasing_( boost::local_shared_ptr<T> const & p1, U * p2 )
{
boost::local_shared_ptr<U> p3( p1, p2 );
BOOST_TEST( p3.get() == p2 );
BOOST_TEST( p3.local_use_count() == p1.local_use_count() );
BOOST_TEST( !p3.owner_before( p1 ) && !p1.owner_before( p3 ) );
}
template<class T, class U> void test_01_aliasing_()
{
U u;
boost::local_shared_ptr<T> p1;
test_aliasing_( p1, &u );
}
template<class T, class U> void test_01_aliasing()
{
test_01_aliasing_<T, U>();
test_01_aliasing_<T const, U>();
test_01_aliasing_<T volatile, U>();
test_01_aliasing_<T const volatile, U>();
test_01_aliasing_<T, U volatile>();
test_01_aliasing_<T const, U volatile>();
test_01_aliasing_<T volatile, U volatile>();
test_01_aliasing_<T const volatile, U volatile>();
}
template<class T, class U> void test_10_aliasing_()
{
boost::local_shared_ptr<T> p1( new T() );
test_aliasing_( p1, static_cast<U*>(0) );
}
template<class T, class U> void test_10_aliasing()
{
test_10_aliasing_<T, U>();
test_10_aliasing_<T const, U>();
test_10_aliasing_<T volatile, U>();
test_10_aliasing_<T const volatile, U>();
test_10_aliasing_<T, U const>();
test_10_aliasing_<T const, U const>();
test_10_aliasing_<T volatile, U const>();
test_10_aliasing_<T const volatile, U const>();
test_10_aliasing_<T, U volatile>();
test_10_aliasing_<T const, U volatile>();
test_10_aliasing_<T volatile, U volatile>();
test_10_aliasing_<T const volatile, U volatile>();
test_10_aliasing_<T, U const volatile>();
test_10_aliasing_<T const, U const volatile>();
test_10_aliasing_<T volatile, U const volatile>();
test_10_aliasing_<T const volatile, U const volatile>();
}
template<class T, class U> void test_11_aliasing_()
{
U u;
boost::local_shared_ptr<T> p1( new T() );
test_aliasing_( p1, &u );
}
template<class T, class U> void test_11_aliasing()
{
test_11_aliasing_<T, U>();
test_11_aliasing_<T const, U>();
test_11_aliasing_<T volatile, U>();
test_11_aliasing_<T const volatile, U>();
test_11_aliasing_<T, U volatile>();
test_11_aliasing_<T const, U volatile>();
test_11_aliasing_<T volatile, U volatile>();
test_11_aliasing_<T const volatile, U volatile>();
}
static void aliasing_constructor()
{
test_01_aliasing<int, int>();
test_10_aliasing<int, int>();
test_11_aliasing<int, int>();
test_01_aliasing<void, int>();
test_10_aliasing<int, void>();
test_10_aliasing<int, incomplete>();
test_01_aliasing<X, X>();
BOOST_TEST( X::instances == 0 );
test_10_aliasing<X, X>();
BOOST_TEST( X::instances == 0 );
test_11_aliasing<X, X>();
BOOST_TEST( X::instances == 0 );
test_01_aliasing<int, X>();
BOOST_TEST( X::instances == 0 );
test_10_aliasing<int, X>();
BOOST_TEST( X::instances == 0 );
test_11_aliasing<int, X>();
BOOST_TEST( X::instances == 0 );
test_01_aliasing<X, int>();
BOOST_TEST( X::instances == 0 );
test_10_aliasing<X, int>();
BOOST_TEST( X::instances == 0 );
test_11_aliasing<X, int>();
BOOST_TEST( X::instances == 0 );
test_01_aliasing<void, X>();
BOOST_TEST( X::instances == 0 );
test_10_aliasing<X, void>();
BOOST_TEST( X::instances == 0 );
test_10_aliasing<X, incomplete>();
BOOST_TEST( X::instances == 0 );
2017-06-04 21:24:20 +03:00
}
2017-06-11 20:35:51 +03:00
// shared_ptr copy constructor
template<class T> static void empty_shared_ptr_copy_test()
{
boost::shared_ptr<T> p1;
boost::local_shared_ptr<T> p2( p1 );
test_empty( p2 );
boost::local_shared_ptr<T const> p3( p1 );
test_empty( p3 );
boost::local_shared_ptr<void> p4( p1 );
test_empty( p4 );
boost::local_shared_ptr<void const> p5( p3 );
test_empty( p5 );
}
template<class T, class U> static void test_nonempty_shared_ptr_copy( boost::shared_ptr<U> const & p1 )
{
long k = p1.use_count();
{
boost::local_shared_ptr<T> p2( p1 );
BOOST_TEST( p2.get() == p1.get() );
BOOST_TEST( p2.local_use_count() == 1 );
BOOST_TEST( p1.use_count() == k + 1 );
}
BOOST_TEST( p1.use_count() == k );
}
template<class T> static void null_shared_ptr_copy_test()
{
boost::shared_ptr<T> p1( static_cast<T*>(0) );
test_nonempty_shared_ptr_copy<T>( p1 );
test_nonempty_shared_ptr_copy<T const>( p1 );
test_nonempty_shared_ptr_copy<T volatile>( p1 );
test_nonempty_shared_ptr_copy<T const volatile>( p1 );
test_nonempty_shared_ptr_copy<void>( p1 );
test_nonempty_shared_ptr_copy<void const>( p1 );
test_nonempty_shared_ptr_copy<void volatile>( p1 );
test_nonempty_shared_ptr_copy<void const volatile>( p1 );
}
template<class T> static void new_shared_ptr_copy_test()
{
boost::shared_ptr<T> p1( new T() );
test_nonempty_shared_ptr_copy<T>( p1 );
test_nonempty_shared_ptr_copy<T const>( p1 );
test_nonempty_shared_ptr_copy<T volatile>( p1 );
test_nonempty_shared_ptr_copy<T const volatile>( p1 );
test_nonempty_shared_ptr_copy<void>( p1 );
test_nonempty_shared_ptr_copy<void const>( p1 );
test_nonempty_shared_ptr_copy<void volatile>( p1 );
test_nonempty_shared_ptr_copy<void const volatile>( p1 );
}
static void shared_ptr_copy_constructor()
{
empty_shared_ptr_copy_test<int>();
empty_shared_ptr_copy_test<incomplete>();
empty_shared_ptr_copy_test<X>();
BOOST_TEST( X::instances == 0 );
null_shared_ptr_copy_test<int>();
null_shared_ptr_copy_test<X>();
BOOST_TEST( X::instances == 0 );
new_shared_ptr_copy_test<int>();
new_shared_ptr_copy_test<X>();
BOOST_TEST( X::instances == 0 );
}
// shared_ptr_move constructor
#if !defined( BOOST_NO_CXX11_RVALUE_REFERENCES )
template<class T> static void empty_shared_ptr_move_test()
{
boost::local_shared_ptr<T> p2(( boost::shared_ptr<T>() ));
test_empty( p2 );
boost::local_shared_ptr<T const> p3(( boost::shared_ptr<T>() ));
test_empty( p3 );
boost::local_shared_ptr<void> p4(( boost::shared_ptr<T>() ));
test_empty( p4 );
boost::local_shared_ptr<void const> p5( std::move(p3) );
test_empty( p5 );
}
template<class T, class U> static void test_nonempty_shared_ptr_move( boost::shared_ptr<U> && p1 )
{
U* q = p1.get();
boost::local_shared_ptr<T> p2( std::move(p1) );
BOOST_TEST( p2.get() == q );
BOOST_TEST( p2.local_use_count() == 1 );
BOOST_TEST( p1.get() == 0 );
BOOST_TEST( p1.use_count() == 0 );
}
template<class T> static void null_shared_ptr_move_test()
{
test_nonempty_shared_ptr_move<T>( boost::shared_ptr<T>( static_cast<T*>(0) ) );
test_nonempty_shared_ptr_move<T const>( boost::shared_ptr<T>( static_cast<T*>(0) ) );
test_nonempty_shared_ptr_move<T volatile>( boost::shared_ptr<T>( static_cast<T*>(0) ) );
test_nonempty_shared_ptr_move<T const volatile>( boost::shared_ptr<T>( static_cast<T*>(0) ) );
test_nonempty_shared_ptr_move<void>( boost::shared_ptr<T>( static_cast<T*>(0) ) );
test_nonempty_shared_ptr_move<void const>( boost::shared_ptr<T>( static_cast<T*>(0) ) );
test_nonempty_shared_ptr_move<void volatile>( boost::shared_ptr<T>( static_cast<T*>(0) ) );
test_nonempty_shared_ptr_move<void const volatile>( boost::shared_ptr<T>( static_cast<T*>(0) ) );
}
template<class T> static void new_shared_ptr_move_test()
{
test_nonempty_shared_ptr_move<T>( boost::shared_ptr<T>( new T() ) );
test_nonempty_shared_ptr_move<T const>( boost::shared_ptr<T>( new T() ) );
test_nonempty_shared_ptr_move<T volatile>( boost::shared_ptr<T>( new T() ) );
test_nonempty_shared_ptr_move<T const volatile>( boost::shared_ptr<T>( new T() ) );
test_nonempty_shared_ptr_move<void>( boost::shared_ptr<T>( new T() ) );
test_nonempty_shared_ptr_move<void const>( boost::shared_ptr<T>( new T() ) );
test_nonempty_shared_ptr_move<void volatile>( boost::shared_ptr<T>( new T() ) );
test_nonempty_shared_ptr_move<void const volatile>( boost::shared_ptr<T>( new T() ) );
}
static void shared_ptr_move_constructor()
{
empty_shared_ptr_move_test<int>();
empty_shared_ptr_move_test<incomplete>();
empty_shared_ptr_move_test<X>();
BOOST_TEST( X::instances == 0 );
null_shared_ptr_move_test<int>();
null_shared_ptr_move_test<X>();
BOOST_TEST( X::instances == 0 );
new_shared_ptr_move_test<int>();
new_shared_ptr_move_test<X>();
BOOST_TEST( X::instances == 0 );
}
#else
static void shared_ptr_move_constructor()
{
}
#endif
// copy assignment
template<class T> static void empty_copy_assign_test()
{
boost::local_shared_ptr<T> p1;
test_empty( p1 );
boost::local_shared_ptr<T> p2;
p2 = p1;
test_empty( p2 );
boost::local_shared_ptr<T const> p3;
p3 = p1;
test_empty( p3 );
boost::local_shared_ptr<void> p4;
p4 = p1;
test_empty( p4 );
boost::local_shared_ptr<void const> p5;
p5 = p3;
test_empty( p5 );
}
template<class T> static void empty_copy_assign_test_()
{
boost::local_shared_ptr<T> p1;
test_empty( p1 );
boost::local_shared_ptr<T> p2( static_cast<T*>(0) );
p2 = p1;
test_empty( p2 );
boost::local_shared_ptr<T const> p3( static_cast<T const*>(0) );
p3 = p1;
test_empty( p3 );
boost::local_shared_ptr<void> p4( static_cast<T*>(0) );
p4 = p1;
test_empty( p4 );
boost::local_shared_ptr<void const> p5( static_cast<T const*>(0) );
p5 = p3;
test_empty( p5 );
}
template<class T, class U> static void test_nonempty_copy_assign( boost::local_shared_ptr<T> p2, boost::local_shared_ptr<U> const & p1 )
{
long k = p1.local_use_count();
p2 = p1;
BOOST_TEST( p2.get() == p1.get() );
BOOST_TEST( p2.local_use_count() == p1.local_use_count() );
BOOST_TEST( p2.local_use_count() == k + 1 );
p2.reset();
BOOST_TEST( p1.local_use_count() == k );
}
template<class T> static void null_copy_assign_test()
{
boost::local_shared_ptr<T> p1( static_cast<T*>(0) );
test_nonempty_copy_assign( boost::local_shared_ptr<T>(), p1 );
test_nonempty_copy_assign( boost::local_shared_ptr<T const>(), p1 );
test_nonempty_copy_assign( boost::local_shared_ptr<T volatile>(), p1 );
test_nonempty_copy_assign( boost::local_shared_ptr<T const volatile>(), p1 );
test_nonempty_copy_assign( boost::local_shared_ptr<T>( static_cast<T*>(0) ), p1 );
test_nonempty_copy_assign( boost::local_shared_ptr<T const>( static_cast<T const*>(0) ), p1 );
test_nonempty_copy_assign( boost::local_shared_ptr<T volatile>( static_cast<T volatile*>(0) ), p1 );
test_nonempty_copy_assign( boost::local_shared_ptr<T const volatile>( static_cast<T const volatile*>(0) ), p1 );
test_nonempty_copy_assign( boost::local_shared_ptr<void>(), p1 );
test_nonempty_copy_assign( boost::local_shared_ptr<void const>(), p1 );
test_nonempty_copy_assign( boost::local_shared_ptr<void volatile>(), p1 );
test_nonempty_copy_assign( boost::local_shared_ptr<void const volatile>(), p1 );
}
template<class T> static void new_copy_assign_test()
{
boost::local_shared_ptr<T> p1( new T() );
test_nonempty_copy_assign( boost::local_shared_ptr<T>(), p1 );
test_nonempty_copy_assign( boost::local_shared_ptr<T const>(), p1 );
test_nonempty_copy_assign( boost::local_shared_ptr<T volatile>(), p1 );
test_nonempty_copy_assign( boost::local_shared_ptr<T const volatile>(), p1 );
test_nonempty_copy_assign( boost::local_shared_ptr<T>( static_cast<T*>(0) ), p1 );
test_nonempty_copy_assign( boost::local_shared_ptr<T const>( static_cast<T const*>(0) ), p1 );
test_nonempty_copy_assign( boost::local_shared_ptr<T volatile>( static_cast<T volatile*>(0) ), p1 );
test_nonempty_copy_assign( boost::local_shared_ptr<T const volatile>( static_cast<T const volatile*>(0) ), p1 );
test_nonempty_copy_assign( boost::local_shared_ptr<T>( new T() ), p1 );
test_nonempty_copy_assign( boost::local_shared_ptr<T const>( new T const() ), p1 );
test_nonempty_copy_assign( boost::local_shared_ptr<T volatile>( new T volatile() ), p1 );
test_nonempty_copy_assign( boost::local_shared_ptr<T const volatile>( new T const volatile() ), p1 );
test_nonempty_copy_assign( boost::local_shared_ptr<void>(), p1 );
test_nonempty_copy_assign( boost::local_shared_ptr<void const>(), p1 );
test_nonempty_copy_assign( boost::local_shared_ptr<void volatile>(), p1 );
test_nonempty_copy_assign( boost::local_shared_ptr<void const volatile>(), p1 );
}
static void copy_assignment()
{
empty_copy_assign_test<incomplete>();
empty_copy_assign_test<int>();
empty_copy_assign_test_<int>();
empty_copy_assign_test<X>();
empty_copy_assign_test_<X>();
BOOST_TEST( X::instances == 0 );
null_copy_assign_test<int>();
null_copy_assign_test<X>();
BOOST_TEST( X::instances == 0 );
new_copy_assign_test<int>();
new_copy_assign_test<X>();
BOOST_TEST( X::instances == 0 );
}
// move assignment
#if !defined( BOOST_NO_CXX11_RVALUE_REFERENCES )
template<class T> static void empty_move_assign_test()
{
boost::local_shared_ptr<T> p2;
p2 = boost::local_shared_ptr<T>();
test_empty( p2 );
boost::local_shared_ptr<T const> p3;
p3 = boost::local_shared_ptr<T>();
test_empty( p3 );
boost::local_shared_ptr<void> p4;
p4 = boost::local_shared_ptr<T>();
test_empty( p4 );
boost::local_shared_ptr<void const> p5;
p5 = std::move( p3 );
test_empty( p5 );
}
template<class T> static void empty_move_assign_test_()
{
boost::local_shared_ptr<T> p2( static_cast<T*>(0) );
p2 = boost::local_shared_ptr<T>();
test_empty( p2 );
boost::local_shared_ptr<T const> p3( static_cast<T const*>(0) );
p3 = boost::local_shared_ptr<T>();
test_empty( p3 );
boost::local_shared_ptr<void> p4( static_cast<T*>(0) );
p4 = boost::local_shared_ptr<T>();
test_empty( p4 );
boost::local_shared_ptr<void const> p5( static_cast<T const*>(0) );
p5 = std::move( p3 );
test_empty( p5 );
}
template<class T, class U> static void test_nonempty_move_assign( boost::local_shared_ptr<T> p2, boost::local_shared_ptr<U> && p1 )
{
U* q = p1.get();
long k = p1.local_use_count();
p2 = std::move( p1 );
BOOST_TEST( p2.get() == q );
BOOST_TEST( p2.local_use_count() == k );
BOOST_TEST( p1.get() == 0 );
BOOST_TEST( p1.local_use_count() == 0 );
}
template<class T> static void null_move_assign_test()
{
test_nonempty_move_assign( boost::local_shared_ptr<T>(), boost::local_shared_ptr<T>( static_cast<T*>(0) ) );
test_nonempty_move_assign( boost::local_shared_ptr<T const>(), boost::local_shared_ptr<T>( static_cast<T*>(0) ) );
test_nonempty_move_assign( boost::local_shared_ptr<T volatile>(), boost::local_shared_ptr<T>( static_cast<T*>(0) ) );
test_nonempty_move_assign( boost::local_shared_ptr<T const volatile>(), boost::local_shared_ptr<T>( static_cast<T*>(0) ) );
test_nonempty_move_assign( boost::local_shared_ptr<T>( static_cast<T*>(0) ), boost::local_shared_ptr<T>( static_cast<T*>(0) ) );
test_nonempty_move_assign( boost::local_shared_ptr<T const>( static_cast<T const*>(0) ), boost::local_shared_ptr<T>( static_cast<T*>(0) ) );
test_nonempty_move_assign( boost::local_shared_ptr<T volatile>( static_cast<T volatile*>(0) ), boost::local_shared_ptr<T>( static_cast<T*>(0) ) );
test_nonempty_move_assign( boost::local_shared_ptr<T const volatile>( static_cast<T const volatile*>(0) ), boost::local_shared_ptr<T>( static_cast<T*>(0) ) );
test_nonempty_move_assign( boost::local_shared_ptr<void>(), boost::local_shared_ptr<T>( static_cast<T*>(0) ) );
test_nonempty_move_assign( boost::local_shared_ptr<void const>(), boost::local_shared_ptr<T>( static_cast<T*>(0) ) );
test_nonempty_move_assign( boost::local_shared_ptr<void volatile>(), boost::local_shared_ptr<T>( static_cast<T*>(0) ) );
test_nonempty_move_assign( boost::local_shared_ptr<void const volatile>(), boost::local_shared_ptr<T>( static_cast<T*>(0) ) );
}
template<class T> static void new_move_assign_test()
{
test_nonempty_move_assign( boost::local_shared_ptr<T>(), boost::local_shared_ptr<T>( new T() ) );
test_nonempty_move_assign( boost::local_shared_ptr<T const>(), boost::local_shared_ptr<T>( new T() ) );
test_nonempty_move_assign( boost::local_shared_ptr<T volatile>(), boost::local_shared_ptr<T>( new T() ) );
test_nonempty_move_assign( boost::local_shared_ptr<T const volatile>(), boost::local_shared_ptr<T>( new T() ) );
test_nonempty_move_assign( boost::local_shared_ptr<T>( static_cast<T*>(0) ), boost::local_shared_ptr<T>( new T() ) );
test_nonempty_move_assign( boost::local_shared_ptr<T const>( static_cast<T const*>(0) ), boost::local_shared_ptr<T>( new T() ) );
test_nonempty_move_assign( boost::local_shared_ptr<T volatile>( static_cast<T volatile*>(0) ), boost::local_shared_ptr<T>( new T() ) );
test_nonempty_move_assign( boost::local_shared_ptr<T const volatile>( static_cast<T const volatile*>(0) ), boost::local_shared_ptr<T>( new T() ) );
test_nonempty_move_assign( boost::local_shared_ptr<T>( new T() ), boost::local_shared_ptr<T>( new T() ) );
test_nonempty_move_assign( boost::local_shared_ptr<T const>( new T const() ), boost::local_shared_ptr<T>( new T() ) );
test_nonempty_move_assign( boost::local_shared_ptr<T volatile>( new T volatile() ), boost::local_shared_ptr<T>( new T() ) );
test_nonempty_move_assign( boost::local_shared_ptr<T const volatile>( new T const volatile() ), boost::local_shared_ptr<T>( new T() ) );
test_nonempty_move_assign( boost::local_shared_ptr<void>(), boost::local_shared_ptr<T>( new T() ) );
test_nonempty_move_assign( boost::local_shared_ptr<void const>(), boost::local_shared_ptr<T>( new T() ) );
test_nonempty_move_assign( boost::local_shared_ptr<void volatile>(), boost::local_shared_ptr<T>( new T() ) );
test_nonempty_move_assign( boost::local_shared_ptr<void const volatile>(), boost::local_shared_ptr<T>( new T() ) );
}
static void move_assignment()
{
empty_move_assign_test<incomplete>();
empty_move_assign_test<int>();
empty_move_assign_test_<int>();
empty_move_assign_test<X>();
empty_move_assign_test_<X>();
BOOST_TEST( X::instances == 0 );
null_move_assign_test<int>();
null_move_assign_test<X>();
BOOST_TEST( X::instances == 0 );
new_move_assign_test<int>();
new_move_assign_test<X>();
BOOST_TEST( X::instances == 0 );
}
#else
static void move_assignment()
{
}
#endif
// nullptr assignment
#if !defined( BOOST_NO_CXX11_NULLPTR )
template<class T> static void test_nullptr_assign( boost::local_shared_ptr<T> p1 )
{
p1 = nullptr;
BOOST_TEST( p1.get() == 0 );
BOOST_TEST( p1.local_use_count() == 0 );
}
template<class T> static void empty_nullptr_assign_test()
{
test_nullptr_assign( boost::local_shared_ptr<T>() );
test_nullptr_assign( boost::local_shared_ptr<T const>() );
test_nullptr_assign( boost::local_shared_ptr<T volatile>() );
test_nullptr_assign( boost::local_shared_ptr<T const volatile>() );
}
template<class T> static void null_nullptr_assign_test()
{
test_nullptr_assign( boost::local_shared_ptr<T>( static_cast<T*>(0) ) );
test_nullptr_assign( boost::local_shared_ptr<T const>( static_cast<T const*>(0) ) );
test_nullptr_assign( boost::local_shared_ptr<T volatile>( static_cast<T volatile*>(0) ) );
test_nullptr_assign( boost::local_shared_ptr<T const volatile>( static_cast<T const volatile*>(0) ) );
}
template<class T> static void new_nullptr_assign_test()
{
test_nullptr_assign( boost::local_shared_ptr<T>( new T() ) );
test_nullptr_assign( boost::local_shared_ptr<T const>( new T const() ) );
test_nullptr_assign( boost::local_shared_ptr<T volatile>( new T volatile() ) );
test_nullptr_assign( boost::local_shared_ptr<T const volatile>( new T const volatile() ) );
}
static void nullptr_assignment()
{
empty_nullptr_assign_test<incomplete>();
empty_nullptr_assign_test<void>();
empty_nullptr_assign_test<int>();
empty_nullptr_assign_test<X>();
BOOST_TEST( X::instances == 0 );
null_nullptr_assign_test<int>();
null_nullptr_assign_test<X>();
BOOST_TEST( X::instances == 0 );
new_nullptr_assign_test<int>();
new_nullptr_assign_test<X>();
BOOST_TEST( X::instances == 0 );
}
#else
static void nullptr_assignment()
{
}
#endif
// main
2017-06-04 21:24:20 +03:00
int main()
{
default_constructor();
2017-06-05 16:38:41 +03:00
nullptr_constructor();
2017-06-04 21:24:20 +03:00
pointer_constructor();
deleter_constructor();
copy_constructor();
2017-06-11 20:35:51 +03:00
move_constructor();
2017-06-05 16:38:41 +03:00
aliasing_constructor();
2017-06-11 20:35:51 +03:00
nullptr_deleter_constructor();
allocator_constructor();
nullptr_allocator_constructor();
shared_ptr_copy_constructor();
shared_ptr_move_constructor();
// unique_ptr_constructor();
copy_assignment();
move_assignment();
nullptr_assignment();
// shared_ptr_copy_assignment();
// shared_ptr_move_assignment();
// unique_ptr_assignment();
// pointer_reset();
// deleter_reset();
// allocator_reset();
// aliasing_reset();
// element_access();
// swap_test();
// owner_before_test();
// equal_test();
2017-06-04 21:24:20 +03:00
return boost::report_errors();
}