optional_test_ref_portable_minimum.cpp 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499
  1. // Copyright (C) 2014 Andrzej Krzemienski.
  2. //
  3. // Use, modification, and distribution is subject to the Boost Software
  4. // License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
  5. // http://www.boost.org/LICENSE_1_0.txt)
  6. //
  7. // See http://www.boost.org/lib/optional for documentation.
  8. //
  9. // You are welcome to contact the author at: akrzemi1@gmail.com
  10. #include "boost/optional/optional.hpp"
  11. #ifdef __BORLANDC__
  12. #pragma hdrstop
  13. #endif
  14. #include "boost/core/addressof.hpp"
  15. #include "boost/core/enable_if.hpp"
  16. #include "boost/core/lightweight_test.hpp"
  17. #include "testable_classes.hpp"
  18. using boost::optional;
  19. using boost::none;
  20. struct CountingClass
  21. {
  22. static int count;
  23. static int assign_count;
  24. CountingClass() { ++count; }
  25. CountingClass(const CountingClass&) { ++count; }
  26. CountingClass& operator=(const CountingClass&) { ++assign_count; return *this; }
  27. ~CountingClass() { ++count; }
  28. };
  29. int CountingClass::count = 0;
  30. int CountingClass::assign_count = 0;
  31. void test_no_object_creation()
  32. {
  33. BOOST_TEST_EQ(0, CountingClass::count);
  34. BOOST_TEST_EQ(0, CountingClass::assign_count);
  35. {
  36. CountingClass v1, v2;
  37. optional<CountingClass&> oA(v1);
  38. optional<CountingClass&> oB;
  39. optional<CountingClass&> oC = oA;
  40. oB = oA;
  41. *oB = v2;
  42. oC = none;
  43. oC = optional<CountingClass&>(v2);
  44. oB = none;
  45. oA = oB;
  46. }
  47. BOOST_TEST_EQ(4, CountingClass::count);
  48. BOOST_TEST_EQ(1, CountingClass::assign_count);
  49. }
  50. template <typename T>
  51. typename boost::enable_if< has_arrow<T> >::type
  52. test_arrow_const()
  53. {
  54. const typename concrete_type_of<T>::type v(2);
  55. optional<const T&> o(v);
  56. BOOST_TEST(o);
  57. BOOST_TEST_EQ(o->val(), 2);
  58. BOOST_TEST(boost::addressof(o->val()) == boost::addressof(v.val()));
  59. }
  60. template <typename T>
  61. typename boost::disable_if< has_arrow<T> >::type
  62. test_arrow_const()
  63. {
  64. }
  65. template <typename T>
  66. typename boost::enable_if< has_arrow<T> >::type
  67. test_arrow_noconst_const()
  68. {
  69. typename concrete_type_of<T>::type v(2);
  70. optional<const T&> o(v);
  71. BOOST_TEST(o);
  72. BOOST_TEST_EQ(o->val(), 2);
  73. BOOST_TEST(boost::addressof(o->val()) == boost::addressof(v.val()));
  74. v.val() = 1;
  75. BOOST_TEST(o);
  76. BOOST_TEST_EQ(o->val(), 1);
  77. BOOST_TEST_EQ(v.val(), 1);
  78. BOOST_TEST(boost::addressof(o->val()) == boost::addressof(v.val()));
  79. }
  80. template <typename T>
  81. typename boost::disable_if< has_arrow<T> >::type
  82. test_arrow_noconst_const()
  83. {
  84. }
  85. template <typename T>
  86. typename boost::enable_if< has_arrow<T> >::type
  87. test_arrow()
  88. {
  89. typename concrete_type_of<T>::type v(2);
  90. optional<T&> o(v);
  91. BOOST_TEST(o);
  92. BOOST_TEST_EQ(o->val(), 2);
  93. BOOST_TEST(boost::addressof(o->val()) == boost::addressof(v.val()));
  94. v.val() = 1;
  95. BOOST_TEST(o);
  96. BOOST_TEST_EQ(o->val(), 1);
  97. BOOST_TEST_EQ(v.val(), 1);
  98. BOOST_TEST(boost::addressof(o->val()) == boost::addressof(v.val()));
  99. o->val() = 3;
  100. BOOST_TEST(o);
  101. BOOST_TEST_EQ(o->val(), 3);
  102. BOOST_TEST_EQ(v.val(), 3);
  103. BOOST_TEST(boost::addressof(o->val()) == boost::addressof(v.val()));
  104. }
  105. template <typename T>
  106. typename boost::disable_if< has_arrow<T> >::type
  107. test_arrow()
  108. {
  109. }
  110. template <typename T>
  111. void test_not_containing_value_for()
  112. {
  113. optional<T&> o1;
  114. optional<T&> o2 = none;
  115. optional<T&> o3 = o1;
  116. BOOST_TEST(!o1);
  117. BOOST_TEST(!o2);
  118. BOOST_TEST(!o3);
  119. BOOST_TEST(o1 == none);
  120. BOOST_TEST(o2 == none);
  121. BOOST_TEST(o3 == none);
  122. }
  123. template <typename T>
  124. void test_direct_init_for_const()
  125. {
  126. const typename concrete_type_of<T>::type v(2);
  127. optional<const T&> o(v);
  128. BOOST_TEST(o);
  129. BOOST_TEST(o != none);
  130. BOOST_TEST(boost::addressof(*o) == boost::addressof(v));
  131. BOOST_TEST_EQ(val(*o), val(v));
  132. BOOST_TEST_EQ(val(*o), 2);
  133. }
  134. template <typename T>
  135. void test_direct_init_for_noconst_const()
  136. {
  137. typename concrete_type_of<T>::type v(2);
  138. optional<const T&> o(v);
  139. BOOST_TEST(o);
  140. BOOST_TEST(o != none);
  141. BOOST_TEST(boost::addressof(*o) == boost::addressof(v));
  142. BOOST_TEST_EQ(val(*o), val(v));
  143. BOOST_TEST_EQ(val(*o), 2);
  144. val(v) = 9;
  145. BOOST_TEST(boost::addressof(*o) == boost::addressof(v));
  146. BOOST_TEST_EQ(val(*o), val(v));
  147. BOOST_TEST_EQ(val(*o), 9);
  148. BOOST_TEST_EQ(val(v), 9);
  149. }
  150. template <typename T>
  151. void test_direct_init_for()
  152. {
  153. typename concrete_type_of<T>::type v(2);
  154. optional<T&> o(v);
  155. BOOST_TEST(o);
  156. BOOST_TEST(o != none);
  157. BOOST_TEST(boost::addressof(*o) == boost::addressof(v));
  158. BOOST_TEST_EQ(val(*o), val(v));
  159. BOOST_TEST_EQ(val(*o), 2);
  160. val(v) = 9;
  161. BOOST_TEST(boost::addressof(*o) == boost::addressof(v));
  162. BOOST_TEST_EQ(val(*o), val(v));
  163. BOOST_TEST_EQ(val(*o), 9);
  164. BOOST_TEST_EQ(val(v), 9);
  165. val(*o) = 7;
  166. BOOST_TEST(boost::addressof(*o) == boost::addressof(v));
  167. BOOST_TEST_EQ(val(*o), val(v));
  168. BOOST_TEST_EQ(val(*o), 7);
  169. BOOST_TEST_EQ(val(v), 7);
  170. }
  171. template <typename T, typename U>
  172. void test_clearing_the_value()
  173. {
  174. typename concrete_type_of<T>::type v(2);
  175. optional<U&> o1(v), o2(v);
  176. BOOST_TEST(o1);
  177. BOOST_TEST(o1 != none);
  178. BOOST_TEST(o2);
  179. BOOST_TEST(o2 != none);
  180. o1 = none;
  181. BOOST_TEST(!o1);
  182. BOOST_TEST(o1 == none);
  183. BOOST_TEST(o2);
  184. BOOST_TEST(o2 != none);
  185. BOOST_TEST_EQ(val(*o2), 2);
  186. BOOST_TEST(boost::addressof(*o2) == boost::addressof(v));
  187. BOOST_TEST_EQ(val(v), 2);
  188. }
  189. template <typename T, typename U>
  190. void test_equality()
  191. {
  192. typename concrete_type_of<T>::type v1(1), v2(2), v2_(2), v3(3);
  193. optional<U&> o1(v1), o2(v2), o2_(v2_), o3(v3), o3_(v3), oN, oN_;
  194. // o2 and o2_ point to different objects; o3 and o3_ point to the same object
  195. BOOST_TEST(oN == oN);
  196. BOOST_TEST(oN == oN_);
  197. BOOST_TEST(oN_ == oN);
  198. BOOST_TEST(o1 == o1);
  199. BOOST_TEST(o2 == o2);
  200. BOOST_TEST(o2 == o2_);
  201. BOOST_TEST(o2_ == o2);
  202. BOOST_TEST(o3 == o3);
  203. BOOST_TEST(o3 == o3_);
  204. BOOST_TEST(!(oN == o1));
  205. BOOST_TEST(!(o1 == oN));
  206. BOOST_TEST(!(o2 == o1));
  207. BOOST_TEST(!(o2 == oN));
  208. BOOST_TEST(!(oN != oN));
  209. BOOST_TEST(!(oN != oN_));
  210. BOOST_TEST(!(oN_ != oN));
  211. BOOST_TEST(!(o1 != o1));
  212. BOOST_TEST(!(o2 != o2));
  213. BOOST_TEST(!(o2 != o2_));
  214. BOOST_TEST(!(o2_ != o2));
  215. BOOST_TEST(!(o3 != o3));
  216. BOOST_TEST(!(o3 != o3_));
  217. BOOST_TEST( (oN != o1));
  218. BOOST_TEST( (o1 != oN));
  219. BOOST_TEST( (o2 != o1));
  220. BOOST_TEST( (o2 != oN));
  221. }
  222. template <typename T, typename U>
  223. void test_order()
  224. {
  225. typename concrete_type_of<T>::type v1(1), v2(2), v2_(2), v3(3);
  226. optional<U&> o1(v1), o2(v2), o2_(v2_), o3(v3), o3_(v3), oN, oN_;
  227. // o2 and o2_ point to different objects; o3 and o3_ point to the same object
  228. BOOST_TEST(!(oN < oN));
  229. BOOST_TEST(!(oN < oN_));
  230. BOOST_TEST(!(oN_ < oN));
  231. BOOST_TEST(!(o1 < o1));
  232. BOOST_TEST(!(o2 < o2));
  233. BOOST_TEST(!(o2 < o2_));
  234. BOOST_TEST(!(o2_ < o2));
  235. BOOST_TEST(!(o3 < o3));
  236. BOOST_TEST(!(o3 < o3_));
  237. BOOST_TEST( (oN <= oN));
  238. BOOST_TEST( (oN <= oN_));
  239. BOOST_TEST( (oN_ <= oN));
  240. BOOST_TEST( (o1 <= o1));
  241. BOOST_TEST( (o2 <= o2));
  242. BOOST_TEST( (o2 <= o2_));
  243. BOOST_TEST( (o2_ <= o2));
  244. BOOST_TEST( (o3 <= o3));
  245. BOOST_TEST( (o3 <= o3_));
  246. BOOST_TEST(!(oN > oN));
  247. BOOST_TEST(!(oN > oN_));
  248. BOOST_TEST(!(oN_ > oN));
  249. BOOST_TEST(!(o1 > o1));
  250. BOOST_TEST(!(o2 > o2));
  251. BOOST_TEST(!(o2 > o2_));
  252. BOOST_TEST(!(o2_ > o2));
  253. BOOST_TEST(!(o3 > o3));
  254. BOOST_TEST(!(o3 > o3_));
  255. BOOST_TEST( (oN >= oN));
  256. BOOST_TEST( (oN >= oN_));
  257. BOOST_TEST( (oN_ >= oN));
  258. BOOST_TEST( (o1 >= o1));
  259. BOOST_TEST( (o2 >= o2));
  260. BOOST_TEST( (o2 >= o2_));
  261. BOOST_TEST( (o2_ >= o2));
  262. BOOST_TEST( (o3 >= o3));
  263. BOOST_TEST( (o3 >= o3_));
  264. BOOST_TEST( (oN < o1));
  265. BOOST_TEST( (oN_ < o1));
  266. BOOST_TEST( (oN < o2));
  267. BOOST_TEST( (oN_ < o2));
  268. BOOST_TEST( (oN < o2_));
  269. BOOST_TEST( (oN_ < o2_));
  270. BOOST_TEST( (oN < o3));
  271. BOOST_TEST( (oN_ < o3));
  272. BOOST_TEST( (oN < o3_));
  273. BOOST_TEST( (oN_ < o3_));
  274. BOOST_TEST( (o1 < o2));
  275. BOOST_TEST( (o1 < o2_));
  276. BOOST_TEST( (o1 < o3));
  277. BOOST_TEST( (o1 < o3_));
  278. BOOST_TEST( (o2 < o3));
  279. BOOST_TEST( (o2_ < o3));
  280. BOOST_TEST( (o2 < o3_));
  281. BOOST_TEST( (o2_ < o3_));
  282. BOOST_TEST( (oN <= o1));
  283. BOOST_TEST( (oN_ <= o1));
  284. BOOST_TEST( (oN <= o2));
  285. BOOST_TEST( (oN_ <= o2));
  286. BOOST_TEST( (oN <= o2_));
  287. BOOST_TEST( (oN_ <= o2_));
  288. BOOST_TEST( (oN <= o3));
  289. BOOST_TEST( (oN_ <= o3));
  290. BOOST_TEST( (oN <= o3_));
  291. BOOST_TEST( (oN_ <= o3_));
  292. BOOST_TEST( (o1 <= o2));
  293. BOOST_TEST( (o1 <= o2_));
  294. BOOST_TEST( (o1 <= o3));
  295. BOOST_TEST( (o1 <= o3_));
  296. BOOST_TEST( (o2 <= o3));
  297. BOOST_TEST( (o2_ <= o3));
  298. BOOST_TEST( (o2 <= o3_));
  299. BOOST_TEST( (o2_ <= o3_));
  300. BOOST_TEST(!(oN > o1));
  301. BOOST_TEST(!(oN_ > o1));
  302. BOOST_TEST(!(oN > o2));
  303. BOOST_TEST(!(oN_ > o2));
  304. BOOST_TEST(!(oN > o2_));
  305. BOOST_TEST(!(oN_ > o2_));
  306. BOOST_TEST(!(oN > o3));
  307. BOOST_TEST(!(oN_ > o3));
  308. BOOST_TEST(!(oN > o3_));
  309. BOOST_TEST(!(oN_ > o3_));
  310. BOOST_TEST(!(o1 > o2));
  311. BOOST_TEST(!(o1 > o2_));
  312. BOOST_TEST(!(o1 > o3));
  313. BOOST_TEST(!(o1 > o3_));
  314. BOOST_TEST(!(o2 > o3));
  315. BOOST_TEST(!(o2_ > o3));
  316. BOOST_TEST(!(o2 > o3_));
  317. BOOST_TEST(!(o2_ > o3_));
  318. BOOST_TEST(!(oN >= o1));
  319. BOOST_TEST(!(oN_ >= o1));
  320. BOOST_TEST(!(oN >= o2));
  321. BOOST_TEST(!(oN_ >= o2));
  322. BOOST_TEST(!(oN >= o2_));
  323. BOOST_TEST(!(oN_ >= o2_));
  324. BOOST_TEST(!(oN >= o3));
  325. BOOST_TEST(!(oN_ >= o3));
  326. BOOST_TEST(!(oN >= o3_));
  327. BOOST_TEST(!(oN_ >= o3_));
  328. BOOST_TEST(!(o1 >= o2));
  329. BOOST_TEST(!(o1 >= o2_));
  330. BOOST_TEST(!(o1 >= o3));
  331. BOOST_TEST(!(o1 >= o3_));
  332. BOOST_TEST(!(o2 >= o3));
  333. BOOST_TEST(!(o2_ >= o3));
  334. BOOST_TEST(!(o2 >= o3_));
  335. BOOST_TEST(!(o2_ >= o3_));
  336. BOOST_TEST(!(o1 < oN));
  337. BOOST_TEST(!(o1 < oN_));
  338. BOOST_TEST(!(o2 < oN));
  339. BOOST_TEST(!(o2 < oN_));
  340. BOOST_TEST(!(o2_ < oN));
  341. BOOST_TEST(!(o2_ < oN_));
  342. BOOST_TEST(!(o3 < oN));
  343. BOOST_TEST(!(o3 < oN_));
  344. BOOST_TEST(!(o3_ < oN));
  345. BOOST_TEST(!(o3_ < oN_));
  346. BOOST_TEST(!(o2 < oN));
  347. BOOST_TEST(!(o2_ < oN_));
  348. BOOST_TEST(!(o3 < oN));
  349. BOOST_TEST(!(o3_ < oN_));
  350. BOOST_TEST(!(o3 < oN));
  351. BOOST_TEST(!(o3 < oN_));
  352. BOOST_TEST(!(o3_ < oN));
  353. BOOST_TEST(!(o3_ < oN_));
  354. }
  355. template <typename T, typename U>
  356. void test_swap()
  357. {
  358. typename concrete_type_of<T>::type v1(1), v2(2);
  359. optional<U&> o1(v1), o1_(v1), o2(v2), o2_(v2), oN, oN_;
  360. swap(o1, o1);
  361. BOOST_TEST(o1);
  362. BOOST_TEST(boost::addressof(*o1) == boost::addressof(v1));
  363. swap(oN, oN_);
  364. BOOST_TEST(!oN);
  365. BOOST_TEST(!oN_);
  366. swap(o1, oN);
  367. BOOST_TEST(!o1);
  368. BOOST_TEST(oN);
  369. BOOST_TEST(boost::addressof(*oN) == boost::addressof(v1));
  370. swap(oN, o1);
  371. BOOST_TEST(!oN);
  372. BOOST_TEST(o1);
  373. BOOST_TEST(boost::addressof(*o1) == boost::addressof(v1));
  374. swap(o1_, o2_);
  375. BOOST_TEST(o1_);
  376. BOOST_TEST(o2_);
  377. BOOST_TEST(boost::addressof(*o1_) == boost::addressof(v2));
  378. BOOST_TEST(boost::addressof(*o2_) == boost::addressof(v1));
  379. }
  380. template <typename T, typename U>
  381. void test_convertability_of_compatible_reference_types()
  382. {
  383. typename concrete_type_of<T>::type v1(1);
  384. optional<T&> oN, o1(v1);
  385. optional<U&> uN(oN), u1(o1);
  386. BOOST_TEST(!uN);
  387. BOOST_TEST(u1);
  388. BOOST_TEST(boost::addressof(*u1) == boost::addressof(*o1));
  389. uN = o1;
  390. u1 = oN;
  391. BOOST_TEST(!u1);
  392. BOOST_TEST(uN);
  393. BOOST_TEST(boost::addressof(*uN) == boost::addressof(*o1));
  394. }
  395. template <typename T>
  396. void test_optional_ref()
  397. {
  398. test_not_containing_value_for<T>();
  399. test_direct_init_for<T>();
  400. test_clearing_the_value<T, T>();
  401. test_arrow<T>();
  402. test_equality<T, T>();
  403. test_order<T, T>();
  404. test_swap<T, T>();
  405. }
  406. template <typename T>
  407. void test_optional_const_ref()
  408. {
  409. test_not_containing_value_for<const T>();
  410. test_direct_init_for_const<T>();
  411. test_direct_init_for_noconst_const<T>();
  412. test_clearing_the_value<const T, const T>();
  413. test_clearing_the_value<T, const T>();
  414. test_arrow_const<T>();
  415. test_arrow_noconst_const<T>();
  416. test_equality<const T, const T>();
  417. test_equality<T, const T>();
  418. test_order<const T, const T>();
  419. test_order<T, const T>();
  420. test_swap<const T, const T>();
  421. test_swap<T, const T>();
  422. }
  423. int main()
  424. {
  425. test_optional_ref<int>();
  426. test_optional_ref<ScopeGuard>();
  427. test_optional_ref<Abstract>();
  428. test_optional_ref< optional<int> >();
  429. test_optional_const_ref<int>();
  430. test_optional_const_ref<ScopeGuard>();
  431. test_optional_const_ref<Abstract>();
  432. test_optional_const_ref< optional<int> >();
  433. test_convertability_of_compatible_reference_types<int, const int>();
  434. test_convertability_of_compatible_reference_types<Impl, Abstract>();
  435. test_convertability_of_compatible_reference_types<Impl, const Abstract>();
  436. test_convertability_of_compatible_reference_types<const Impl, const Abstract>();
  437. test_convertability_of_compatible_reference_types<optional<int>, const optional<int> >();
  438. return boost::report_errors();
  439. }