sp_bml_unique_ptr_test.cpp 7.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294
  1. //
  2. // sp_bml_unique_ptr_test.cpp
  3. //
  4. // Copyright (c) 2012, 2015 Peter Dimov
  5. //
  6. // Distributed under the Boost Software License, Version 1.0.
  7. // See accompanying file LICENSE_1_0.txt or copy at
  8. // http://www.boost.org/LICENSE_1_0.txt
  9. //
  10. #include <boost/shared_ptr.hpp>
  11. #include <boost/enable_shared_from_this.hpp>
  12. #include <boost/detail/lightweight_test.hpp>
  13. #include <boost/move/unique_ptr.hpp>
  14. #include <boost/type_traits/remove_reference.hpp>
  15. #include <memory>
  16. #include <utility>
  17. struct X: public boost::enable_shared_from_this< X >
  18. {
  19. static int instances;
  20. X()
  21. {
  22. ++instances;
  23. }
  24. ~X()
  25. {
  26. --instances;
  27. }
  28. private:
  29. X( X const & );
  30. X & operator=( X const & );
  31. };
  32. int X::instances = 0;
  33. struct Y
  34. {
  35. static int instances;
  36. bool deleted_;
  37. Y(): deleted_( false )
  38. {
  39. ++instances;
  40. }
  41. ~Y()
  42. {
  43. BOOST_TEST( deleted_ );
  44. --instances;
  45. }
  46. private:
  47. Y( Y const & );
  48. Y & operator=( Y const & );
  49. };
  50. int Y::instances = 0;
  51. struct YD
  52. {
  53. void operator()( Y* p ) const
  54. {
  55. if( p )
  56. {
  57. p->deleted_ = true;
  58. delete p;
  59. }
  60. else
  61. {
  62. BOOST_ERROR( "YD::operator()(0) called" );
  63. }
  64. }
  65. };
  66. 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 )
  67. {
  68. BOOST_TEST( T::instances == 0 );
  69. boost::shared_ptr<U> sp( boost::move( p1 ) );
  70. BOOST_TEST( sp.get() == 0 );
  71. BOOST_TEST( sp.use_count() == 0 );
  72. sp.reset( new T, typename boost::remove_reference<D>::type() );
  73. BOOST_TEST( sp.get() != 0 );
  74. BOOST_TEST( sp.use_count() == 1 );
  75. BOOST_TEST( T::instances == 1 );
  76. sp = boost::move( p2 );
  77. BOOST_TEST( sp.get() == 0 );
  78. BOOST_TEST( sp.use_count() == 0 );
  79. BOOST_TEST( T::instances == 0 );
  80. }
  81. int main()
  82. {
  83. {
  84. BOOST_TEST( X::instances == 0 );
  85. boost::movelib::unique_ptr<X> p( new X );
  86. BOOST_TEST( X::instances == 1 );
  87. boost::shared_ptr<X> p2( boost::move( p ) );
  88. BOOST_TEST( X::instances == 1 );
  89. BOOST_TEST( p.get() == 0 );
  90. boost::shared_ptr<X> p3 = p2->shared_from_this();
  91. BOOST_TEST( p2 == p3 );
  92. BOOST_TEST( !(p2 < p3) && !(p3 < p2) );
  93. p2.reset();
  94. p3.reset();
  95. BOOST_TEST( X::instances == 0 );
  96. p2 = boost::movelib::unique_ptr<X>( new X );
  97. BOOST_TEST( X::instances == 1 );
  98. p2 = boost::movelib::unique_ptr<X>( new X );
  99. BOOST_TEST( X::instances == 1 );
  100. p3 = p2->shared_from_this();
  101. BOOST_TEST( p2 == p3 );
  102. BOOST_TEST( !(p2 < p3) && !(p3 < p2) );
  103. p2.reset();
  104. p3.reset();
  105. BOOST_TEST( X::instances == 0 );
  106. }
  107. {
  108. BOOST_TEST( X::instances == 0 );
  109. boost::movelib::unique_ptr<X> p( new X );
  110. BOOST_TEST( X::instances == 1 );
  111. boost::shared_ptr<X const> p2( boost::move( p ) );
  112. BOOST_TEST( X::instances == 1 );
  113. BOOST_TEST( p.get() == 0 );
  114. boost::shared_ptr<X const> p3 = p2->shared_from_this();
  115. BOOST_TEST( p2 == p3 );
  116. BOOST_TEST( !(p2 < p3) && !(p3 < p2) );
  117. p2.reset();
  118. p3.reset();
  119. BOOST_TEST( X::instances == 0 );
  120. p2 = boost::movelib::unique_ptr<X>( new X );
  121. BOOST_TEST( X::instances == 1 );
  122. p2 = boost::movelib::unique_ptr<X>( new X );
  123. BOOST_TEST( X::instances == 1 );
  124. p3 = p2->shared_from_this();
  125. BOOST_TEST( p2 == p3 );
  126. BOOST_TEST( !(p2 < p3) && !(p3 < p2) );
  127. p2.reset();
  128. p3.reset();
  129. BOOST_TEST( X::instances == 0 );
  130. }
  131. {
  132. BOOST_TEST( X::instances == 0 );
  133. boost::movelib::unique_ptr<X> p( new X );
  134. BOOST_TEST( X::instances == 1 );
  135. boost::shared_ptr<void> p2( boost::move( p ) );
  136. BOOST_TEST( X::instances == 1 );
  137. BOOST_TEST( p.get() == 0 );
  138. p2.reset();
  139. BOOST_TEST( X::instances == 0 );
  140. p2 = boost::movelib::unique_ptr<X>( new X );
  141. BOOST_TEST( X::instances == 1 );
  142. p2 = boost::movelib::unique_ptr<X>( new X );
  143. BOOST_TEST( X::instances == 1 );
  144. p2.reset();
  145. BOOST_TEST( X::instances == 0 );
  146. }
  147. {
  148. BOOST_TEST( Y::instances == 0 );
  149. boost::movelib::unique_ptr<Y, YD> p( new Y, YD() );
  150. BOOST_TEST( Y::instances == 1 );
  151. boost::shared_ptr<Y> p2( boost::move( p ) );
  152. BOOST_TEST( Y::instances == 1 );
  153. BOOST_TEST( p.get() == 0 );
  154. p2.reset();
  155. BOOST_TEST( Y::instances == 0 );
  156. p2 = boost::movelib::unique_ptr<Y, YD>( new Y, YD() );
  157. BOOST_TEST( Y::instances == 1 );
  158. p2 = boost::movelib::unique_ptr<Y, YD>( new Y, YD() );
  159. BOOST_TEST( Y::instances == 1 );
  160. p2.reset();
  161. BOOST_TEST( Y::instances == 0 );
  162. }
  163. {
  164. BOOST_TEST( Y::instances == 0 );
  165. YD yd;
  166. boost::movelib::unique_ptr<Y, YD&> p( new Y, yd );
  167. BOOST_TEST( Y::instances == 1 );
  168. boost::shared_ptr<Y> p2( boost::move( p ) );
  169. BOOST_TEST( Y::instances == 1 );
  170. BOOST_TEST( p.get() == 0 );
  171. p2.reset();
  172. BOOST_TEST( Y::instances == 0 );
  173. p2 = boost::movelib::unique_ptr<Y, YD&>( new Y, yd );
  174. BOOST_TEST( Y::instances == 1 );
  175. p2 = boost::movelib::unique_ptr<Y, YD&>( new Y, yd );
  176. BOOST_TEST( Y::instances == 1 );
  177. p2.reset();
  178. BOOST_TEST( Y::instances == 0 );
  179. }
  180. {
  181. BOOST_TEST( Y::instances == 0 );
  182. YD yd;
  183. boost::movelib::unique_ptr<Y, YD const&> p( new Y, yd );
  184. BOOST_TEST( Y::instances == 1 );
  185. boost::shared_ptr<Y> p2( boost::move( p ) );
  186. BOOST_TEST( Y::instances == 1 );
  187. BOOST_TEST( p.get() == 0 );
  188. p2.reset();
  189. BOOST_TEST( Y::instances == 0 );
  190. p2 = boost::movelib::unique_ptr<Y, YD const&>( new Y, yd );
  191. BOOST_TEST( Y::instances == 1 );
  192. p2 = boost::movelib::unique_ptr<Y, YD const&>( new Y, yd );
  193. BOOST_TEST( Y::instances == 1 );
  194. p2.reset();
  195. BOOST_TEST( Y::instances == 0 );
  196. }
  197. {
  198. test_null_unique_ptr<X>( boost::movelib::unique_ptr<X>(), boost::movelib::unique_ptr<X>() );
  199. test_null_unique_ptr<X const>( boost::movelib::unique_ptr<X>(), boost::movelib::unique_ptr<X>() );
  200. test_null_unique_ptr<void>( boost::movelib::unique_ptr<X>(), boost::movelib::unique_ptr<X>() );
  201. test_null_unique_ptr<void const>( boost::movelib::unique_ptr<X>(), boost::movelib::unique_ptr<X>() );
  202. }
  203. {
  204. test_null_unique_ptr<Y>( boost::movelib::unique_ptr<Y, YD>( 0, YD() ), boost::movelib::unique_ptr<Y, YD>( 0, YD() ) );
  205. test_null_unique_ptr<Y const>( boost::movelib::unique_ptr<Y, YD>( 0, YD() ), boost::movelib::unique_ptr<Y, YD>( 0, YD() ) );
  206. test_null_unique_ptr<void>( boost::movelib::unique_ptr<Y, YD>( 0, YD() ), boost::movelib::unique_ptr<Y, YD>( 0, YD() ) );
  207. test_null_unique_ptr<void const>( boost::movelib::unique_ptr<Y, YD>( 0, YD() ), boost::movelib::unique_ptr<Y, YD>( 0, YD() ) );
  208. }
  209. {
  210. YD yd;
  211. test_null_unique_ptr<Y>( boost::movelib::unique_ptr<Y, YD&>( 0, yd ), boost::movelib::unique_ptr<Y, YD&>( 0, yd ) );
  212. test_null_unique_ptr<Y const>( boost::movelib::unique_ptr<Y, YD&>( 0, yd ), boost::movelib::unique_ptr<Y, YD&>( 0, yd ) );
  213. test_null_unique_ptr<void>( boost::movelib::unique_ptr<Y, YD&>( 0, yd ), boost::movelib::unique_ptr<Y, YD&>( 0, yd ) );
  214. test_null_unique_ptr<void const>( boost::movelib::unique_ptr<Y, YD&>( 0, yd ), boost::movelib::unique_ptr<Y, YD&>( 0, yd ) );
  215. }
  216. return boost::report_errors();
  217. }