optional.hpp 35 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146
  1. // Copyright (C) 2003, 2008 Fernando Luis Cacciola Carballal.
  2. // Copyright (C) 2014, 2015 Andrzej Krzemienski.
  3. //
  4. // Use, modification, and distribution is subject to the Boost Software
  5. // License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
  6. // http://www.boost.org/LICENSE_1_0.txt)
  7. //
  8. // See http://www.boost.org/libs/optional for documentation.
  9. //
  10. // You are welcome to contact the author at:
  11. // fernando_cacciola@hotmail.com
  12. //
  13. // Revisions:
  14. // 27 Apr 2008 (improved swap) Fernando Cacciola, Niels Dekker, Thorsten Ottosen
  15. // 05 May 2014 (Added move semantics) Andrzej Krzemienski
  16. //
  17. #ifndef BOOST_OPTIONAL_OPTIONAL_FLC_19NOV2002_HPP
  18. #define BOOST_OPTIONAL_OPTIONAL_FLC_19NOV2002_HPP
  19. #include <new>
  20. #include <iosfwd>
  21. #include <boost/assert.hpp>
  22. #include <boost/core/addressof.hpp>
  23. #include <boost/core/enable_if.hpp>
  24. #include <boost/core/explicit_operator_bool.hpp>
  25. #include <boost/core/swap.hpp>
  26. #include <boost/optional/bad_optional_access.hpp>
  27. #include <boost/static_assert.hpp>
  28. #include <boost/throw_exception.hpp>
  29. #include <boost/type.hpp>
  30. #include <boost/type_traits/alignment_of.hpp>
  31. #include <boost/type_traits/conditional.hpp>
  32. #include <boost/type_traits/has_nothrow_constructor.hpp>
  33. #include <boost/type_traits/type_with_alignment.hpp>
  34. #include <boost/type_traits/remove_const.hpp>
  35. #include <boost/type_traits/remove_reference.hpp>
  36. #include <boost/type_traits/decay.hpp>
  37. #include <boost/type_traits/is_base_of.hpp>
  38. #include <boost/type_traits/is_constructible.hpp>
  39. #include <boost/type_traits/is_lvalue_reference.hpp>
  40. #include <boost/type_traits/is_nothrow_move_assignable.hpp>
  41. #include <boost/type_traits/is_nothrow_move_constructible.hpp>
  42. #include <boost/type_traits/is_rvalue_reference.hpp>
  43. #include <boost/type_traits/is_same.hpp>
  44. #include <boost/move/utility.hpp>
  45. #include <boost/none.hpp>
  46. #include <boost/utility/compare_pointees.hpp>
  47. #include <boost/optional/optional_fwd.hpp>
  48. #include <boost/optional/detail/optional_config.hpp>
  49. #include <boost/optional/detail/optional_factory_support.hpp>
  50. #include <boost/optional/detail/optional_aligned_storage.hpp>
  51. #ifdef BOOST_OPTIONAL_CONFIG_USE_OLD_IMPLEMENTATION_OF_OPTIONAL
  52. #include <boost/optional/detail/old_optional_implementation.hpp>
  53. #else
  54. namespace boost {
  55. namespace optional_detail {
  56. struct optional_tag {} ;
  57. template<class T>
  58. class optional_base : public optional_tag
  59. {
  60. private :
  61. typedef aligned_storage<T> storage_type ;
  62. typedef optional_base<T> this_type ;
  63. protected :
  64. typedef T value_type ;
  65. protected:
  66. typedef T & reference_type ;
  67. typedef T const& reference_const_type ;
  68. #ifndef BOOST_OPTIONAL_DETAIL_NO_RVALUE_REFERENCES
  69. typedef T && rval_reference_type ;
  70. typedef T && reference_type_of_temporary_wrapper ;
  71. #endif
  72. typedef T * pointer_type ;
  73. typedef T const* pointer_const_type ;
  74. typedef T const& argument_type ;
  75. // Creates an optional<T> uninitialized.
  76. // No-throw
  77. optional_base()
  78. :
  79. m_initialized(false) {}
  80. // Creates an optional<T> uninitialized.
  81. // No-throw
  82. optional_base ( none_t )
  83. :
  84. m_initialized(false) {}
  85. // Creates an optional<T> initialized with 'val'.
  86. // Can throw if T::T(T const&) does
  87. optional_base ( argument_type val )
  88. :
  89. m_initialized(false)
  90. {
  91. construct(val);
  92. }
  93. #ifndef BOOST_OPTIONAL_DETAIL_NO_RVALUE_REFERENCES
  94. // move-construct an optional<T> initialized from an rvalue-ref to 'val'.
  95. // Can throw if T::T(T&&) does
  96. optional_base ( rval_reference_type val )
  97. :
  98. m_initialized(false)
  99. {
  100. construct( boost::move(val) );
  101. }
  102. #endif
  103. // Creates an optional<T> initialized with 'val' IFF cond is true, otherwise creates an uninitialzed optional<T>.
  104. // Can throw if T::T(T const&) does
  105. optional_base ( bool cond, argument_type val )
  106. :
  107. m_initialized(false)
  108. {
  109. if ( cond )
  110. construct(val);
  111. }
  112. // Creates a deep copy of another optional<T>
  113. // Can throw if T::T(T const&) does
  114. optional_base ( optional_base const& rhs )
  115. :
  116. m_initialized(false)
  117. {
  118. if ( rhs.is_initialized() )
  119. construct(rhs.get_impl());
  120. }
  121. #ifndef BOOST_OPTIONAL_DETAIL_NO_RVALUE_REFERENCES
  122. // Creates a deep move of another optional<T>
  123. // Can throw if T::T(T&&) does
  124. optional_base ( optional_base&& rhs )
  125. :
  126. m_initialized(false)
  127. {
  128. if ( rhs.is_initialized() )
  129. construct( boost::move(rhs.get_impl()) );
  130. }
  131. #endif
  132. #ifndef BOOST_OPTIONAL_DETAIL_NO_RVALUE_REFERENCES
  133. template<class Expr, class PtrExpr>
  134. explicit optional_base ( Expr&& expr, PtrExpr const* tag )
  135. :
  136. m_initialized(false)
  137. {
  138. construct(boost::forward<Expr>(expr),tag);
  139. }
  140. #else
  141. // This is used for both converting and in-place constructions.
  142. // Derived classes use the 'tag' to select the appropriate
  143. // implementation (the correct 'construct()' overload)
  144. template<class Expr>
  145. explicit optional_base ( Expr const& expr, Expr const* tag )
  146. :
  147. m_initialized(false)
  148. {
  149. construct(expr,tag);
  150. }
  151. #endif
  152. // No-throw (assuming T::~T() doesn't)
  153. ~optional_base() { destroy() ; }
  154. // Assigns from another optional<T> (deep-copies the rhs value)
  155. void assign ( optional_base const& rhs )
  156. {
  157. if (is_initialized())
  158. {
  159. if ( rhs.is_initialized() )
  160. assign_value(rhs.get_impl());
  161. else destroy();
  162. }
  163. else
  164. {
  165. if ( rhs.is_initialized() )
  166. construct(rhs.get_impl());
  167. }
  168. }
  169. #ifndef BOOST_OPTIONAL_DETAIL_NO_RVALUE_REFERENCES
  170. // Assigns from another optional<T> (deep-moves the rhs value)
  171. void assign ( optional_base&& rhs )
  172. {
  173. if (is_initialized())
  174. {
  175. if ( rhs.is_initialized() )
  176. assign_value( boost::move(rhs.get_impl()) );
  177. else destroy();
  178. }
  179. else
  180. {
  181. if ( rhs.is_initialized() )
  182. construct(boost::move(rhs.get_impl()));
  183. }
  184. }
  185. #endif
  186. // Assigns from another _convertible_ optional<U> (deep-copies the rhs value)
  187. template<class U>
  188. void assign ( optional<U> const& rhs )
  189. {
  190. if (is_initialized())
  191. {
  192. if ( rhs.is_initialized() )
  193. #ifndef BOOST_OPTIONAL_CONFIG_RESTORE_ASSIGNMENT_OF_NONCONVERTIBLE_TYPES
  194. assign_value( rhs.get() );
  195. #else
  196. assign_value( static_cast<value_type>(rhs.get()) );
  197. #endif
  198. else destroy();
  199. }
  200. else
  201. {
  202. if ( rhs.is_initialized() )
  203. #ifndef BOOST_OPTIONAL_CONFIG_RESTORE_ASSIGNMENT_OF_NONCONVERTIBLE_TYPES
  204. construct(rhs.get());
  205. #else
  206. construct(static_cast<value_type>(rhs.get()));
  207. #endif
  208. }
  209. }
  210. #ifndef BOOST_OPTIONAL_DETAIL_NO_RVALUE_REFERENCES
  211. // move-assigns from another _convertible_ optional<U> (deep-moves from the rhs value)
  212. template<class U>
  213. void assign ( optional<U>&& rhs )
  214. {
  215. typedef BOOST_DEDUCED_TYPENAME optional<U>::rval_reference_type ref_type;
  216. if (is_initialized())
  217. {
  218. if ( rhs.is_initialized() )
  219. assign_value( static_cast<ref_type>(rhs.get()) );
  220. else destroy();
  221. }
  222. else
  223. {
  224. if ( rhs.is_initialized() )
  225. construct(static_cast<ref_type>(rhs.get()));
  226. }
  227. }
  228. #endif
  229. // Assigns from a T (deep-copies the rhs value)
  230. void assign ( argument_type val )
  231. {
  232. if (is_initialized())
  233. assign_value(val);
  234. else construct(val);
  235. }
  236. #ifndef BOOST_OPTIONAL_DETAIL_NO_RVALUE_REFERENCES
  237. // Assigns from a T (deep-moves the rhs value)
  238. void assign ( rval_reference_type val )
  239. {
  240. if (is_initialized())
  241. assign_value( boost::move(val) );
  242. else construct( boost::move(val) );
  243. }
  244. #endif
  245. // Assigns from "none", destroying the current value, if any, leaving this UNINITIALIZED
  246. // No-throw (assuming T::~T() doesn't)
  247. void assign ( none_t ) BOOST_NOEXCEPT { destroy(); }
  248. #ifndef BOOST_OPTIONAL_NO_INPLACE_FACTORY_SUPPORT
  249. #ifndef BOOST_OPTIONAL_DETAIL_NO_RVALUE_REFERENCES
  250. template<class Expr, class ExprPtr>
  251. void assign_expr ( Expr&& expr, ExprPtr const* tag )
  252. {
  253. if (is_initialized())
  254. assign_expr_to_initialized(boost::forward<Expr>(expr),tag);
  255. else construct(boost::forward<Expr>(expr),tag);
  256. }
  257. #else
  258. template<class Expr>
  259. void assign_expr ( Expr const& expr, Expr const* tag )
  260. {
  261. if (is_initialized())
  262. assign_expr_to_initialized(expr,tag);
  263. else construct(expr,tag);
  264. }
  265. #endif
  266. #endif
  267. public :
  268. // **DEPPRECATED** Destroys the current value, if any, leaving this UNINITIALIZED
  269. // No-throw (assuming T::~T() doesn't)
  270. void reset() BOOST_NOEXCEPT { destroy(); }
  271. // **DEPPRECATED** Replaces the current value -if any- with 'val'
  272. void reset ( argument_type val ) { assign(val); }
  273. // Returns a pointer to the value if this is initialized, otherwise,
  274. // returns NULL.
  275. // No-throw
  276. pointer_const_type get_ptr() const { return m_initialized ? get_ptr_impl() : 0 ; }
  277. pointer_type get_ptr() { return m_initialized ? get_ptr_impl() : 0 ; }
  278. bool is_initialized() const { return m_initialized ; }
  279. protected :
  280. void construct ( argument_type val )
  281. {
  282. ::new (m_storage.address()) value_type(val) ;
  283. m_initialized = true ;
  284. }
  285. #ifndef BOOST_OPTIONAL_DETAIL_NO_RVALUE_REFERENCES
  286. void construct ( rval_reference_type val )
  287. {
  288. ::new (m_storage.address()) value_type( boost::move(val) ) ;
  289. m_initialized = true ;
  290. }
  291. #endif
  292. #if (!defined BOOST_OPTIONAL_DETAIL_NO_RVALUE_REFERENCES) && (!defined BOOST_NO_CXX11_VARIADIC_TEMPLATES)
  293. // Constructs in-place
  294. // upon exception *this is always uninitialized
  295. template<class... Args>
  296. void emplace_assign ( Args&&... args )
  297. {
  298. destroy();
  299. ::new (m_storage.address()) value_type( boost::forward<Args>(args)... );
  300. m_initialized = true ;
  301. }
  302. #elif (!defined BOOST_OPTIONAL_DETAIL_NO_RVALUE_REFERENCES)
  303. template<class Arg>
  304. void emplace_assign ( Arg&& arg )
  305. {
  306. destroy();
  307. ::new (m_storage.address()) value_type( boost::forward<Arg>(arg) );
  308. m_initialized = true ;
  309. }
  310. void emplace_assign ()
  311. {
  312. destroy();
  313. ::new (m_storage.address()) value_type();
  314. m_initialized = true ;
  315. }
  316. #else
  317. template<class Arg>
  318. void emplace_assign ( const Arg& arg )
  319. {
  320. destroy();
  321. ::new (m_storage.address()) value_type( arg );
  322. m_initialized = true ;
  323. }
  324. template<class Arg>
  325. void emplace_assign ( Arg& arg )
  326. {
  327. destroy();
  328. ::new (m_storage.address()) value_type( arg );
  329. m_initialized = true ;
  330. }
  331. void emplace_assign ()
  332. {
  333. destroy();
  334. ::new (m_storage.address()) value_type();
  335. m_initialized = true ;
  336. }
  337. #endif
  338. #ifndef BOOST_OPTIONAL_NO_INPLACE_FACTORY_SUPPORT
  339. #ifndef BOOST_OPTIONAL_DETAIL_NO_RVALUE_REFERENCES
  340. // Constructs in-place using the given factory
  341. template<class Expr>
  342. void construct ( Expr&& factory, in_place_factory_base const* )
  343. {
  344. boost_optional_detail::construct<value_type>(factory, m_storage.address());
  345. m_initialized = true ;
  346. }
  347. // Constructs in-place using the given typed factory
  348. template<class Expr>
  349. void construct ( Expr&& factory, typed_in_place_factory_base const* )
  350. {
  351. factory.apply(m_storage.address()) ;
  352. m_initialized = true ;
  353. }
  354. template<class Expr>
  355. void assign_expr_to_initialized ( Expr&& factory, in_place_factory_base const* tag )
  356. {
  357. destroy();
  358. construct(factory,tag);
  359. }
  360. // Constructs in-place using the given typed factory
  361. template<class Expr>
  362. void assign_expr_to_initialized ( Expr&& factory, typed_in_place_factory_base const* tag )
  363. {
  364. destroy();
  365. construct(factory,tag);
  366. }
  367. #else
  368. // Constructs in-place using the given factory
  369. template<class Expr>
  370. void construct ( Expr const& factory, in_place_factory_base const* )
  371. {
  372. boost_optional_detail::construct<value_type>(factory, m_storage.address());
  373. m_initialized = true ;
  374. }
  375. // Constructs in-place using the given typed factory
  376. template<class Expr>
  377. void construct ( Expr const& factory, typed_in_place_factory_base const* )
  378. {
  379. factory.apply(m_storage.address()) ;
  380. m_initialized = true ;
  381. }
  382. template<class Expr>
  383. void assign_expr_to_initialized ( Expr const& factory, in_place_factory_base const* tag )
  384. {
  385. destroy();
  386. construct(factory,tag);
  387. }
  388. // Constructs in-place using the given typed factory
  389. template<class Expr>
  390. void assign_expr_to_initialized ( Expr const& factory, typed_in_place_factory_base const* tag )
  391. {
  392. destroy();
  393. construct(factory,tag);
  394. }
  395. #endif
  396. #endif
  397. #ifndef BOOST_OPTIONAL_DETAIL_NO_RVALUE_REFERENCES
  398. // Constructs using any expression implicitly convertible to the single argument
  399. // of a one-argument T constructor.
  400. // Converting constructions of optional<T> from optional<U> uses this function with
  401. // 'Expr' being of type 'U' and relying on a converting constructor of T from U.
  402. template<class Expr>
  403. void construct ( Expr&& expr, void const* )
  404. {
  405. new (m_storage.address()) value_type(boost::forward<Expr>(expr)) ;
  406. m_initialized = true ;
  407. }
  408. // Assigns using a form any expression implicitly convertible to the single argument
  409. // of a T's assignment operator.
  410. // Converting assignments of optional<T> from optional<U> uses this function with
  411. // 'Expr' being of type 'U' and relying on a converting assignment of T from U.
  412. template<class Expr>
  413. void assign_expr_to_initialized ( Expr&& expr, void const* )
  414. {
  415. assign_value( boost::forward<Expr>(expr) );
  416. }
  417. #else
  418. // Constructs using any expression implicitly convertible to the single argument
  419. // of a one-argument T constructor.
  420. // Converting constructions of optional<T> from optional<U> uses this function with
  421. // 'Expr' being of type 'U' and relying on a converting constructor of T from U.
  422. template<class Expr>
  423. void construct ( Expr const& expr, void const* )
  424. {
  425. new (m_storage.address()) value_type(expr) ;
  426. m_initialized = true ;
  427. }
  428. // Assigns using a form any expression implicitly convertible to the single argument
  429. // of a T's assignment operator.
  430. // Converting assignments of optional<T> from optional<U> uses this function with
  431. // 'Expr' being of type 'U' and relying on a converting assignment of T from U.
  432. template<class Expr>
  433. void assign_expr_to_initialized ( Expr const& expr, void const* )
  434. {
  435. assign_value(expr);
  436. }
  437. #endif
  438. #ifdef BOOST_OPTIONAL_WEAK_OVERLOAD_RESOLUTION
  439. // BCB5.64 (and probably lower versions) workaround.
  440. // The in-place factories are supported by means of catch-all constructors
  441. // and assignment operators (the functions are parameterized in terms of
  442. // an arbitrary 'Expr' type)
  443. // This compiler incorrectly resolves the overload set and sinks optional<T> and optional<U>
  444. // to the 'Expr'-taking functions even though explicit overloads are present for them.
  445. // Thus, the following overload is needed to properly handle the case when the 'lhs'
  446. // is another optional.
  447. //
  448. // For VC<=70 compilers this workaround dosen't work becasue the comnpiler issues and error
  449. // instead of choosing the wrong overload
  450. //
  451. #ifndef BOOST_OPTIONAL_DETAIL_NO_RVALUE_REFERENCES
  452. // Notice that 'Expr' will be optional<T> or optional<U> (but not optional_base<..>)
  453. template<class Expr>
  454. void construct ( Expr&& expr, optional_tag const* )
  455. {
  456. if ( expr.is_initialized() )
  457. {
  458. // An exception can be thrown here.
  459. // It it happens, THIS will be left uninitialized.
  460. new (m_storage.address()) value_type(boost::move(expr.get())) ;
  461. m_initialized = true ;
  462. }
  463. }
  464. #else
  465. // Notice that 'Expr' will be optional<T> or optional<U> (but not optional_base<..>)
  466. template<class Expr>
  467. void construct ( Expr const& expr, optional_tag const* )
  468. {
  469. if ( expr.is_initialized() )
  470. {
  471. // An exception can be thrown here.
  472. // It it happens, THIS will be left uninitialized.
  473. new (m_storage.address()) value_type(expr.get()) ;
  474. m_initialized = true ;
  475. }
  476. }
  477. #endif
  478. #endif // defined BOOST_OPTIONAL_WEAK_OVERLOAD_RESOLUTION
  479. void assign_value ( argument_type val ) { get_impl() = val; }
  480. #ifndef BOOST_OPTIONAL_DETAIL_NO_RVALUE_REFERENCES
  481. void assign_value ( rval_reference_type val ) { get_impl() = static_cast<rval_reference_type>(val); }
  482. #endif
  483. void destroy()
  484. {
  485. if ( m_initialized )
  486. destroy_impl() ;
  487. }
  488. reference_const_type get_impl() const { return m_storage.ref() ; }
  489. reference_type get_impl() { return m_storage.ref() ; }
  490. pointer_const_type get_ptr_impl() const { return m_storage.ptr_ref(); }
  491. pointer_type get_ptr_impl() { return m_storage.ptr_ref(); }
  492. private :
  493. #if BOOST_WORKAROUND(BOOST_MSVC, <= 1600)
  494. void destroy_impl ( ) { m_storage.ptr_ref()->~T() ; m_initialized = false ; }
  495. #else
  496. void destroy_impl ( ) { m_storage.ref().T::~T() ; m_initialized = false ; }
  497. #endif
  498. bool m_initialized ;
  499. storage_type m_storage ;
  500. } ;
  501. // definition of metafunciton is_optional_val_init_candidate
  502. template <typename U>
  503. struct is_optional_related
  504. : boost::conditional< boost::is_base_of<optional_detail::optional_tag, BOOST_DEDUCED_TYPENAME boost::decay<U>::type>::value
  505. || boost::is_same<BOOST_DEDUCED_TYPENAME boost::decay<U>::type, none_t>::value,
  506. boost::true_type, boost::false_type>::type
  507. {};
  508. #if !defined(BOOST_NO_CXX11_VARIADIC_TEMPLATES) && !defined(BOOST_NO_CXX11_DECLTYPE) && !BOOST_WORKAROUND(BOOST_MSVC, < 1800) && !BOOST_WORKAROUND(BOOST_GCC_VERSION, < 40500) && !defined(__SUNPRO_CC)
  509. // this condition is a copy paste from is_constructible.hpp
  510. // I also disable SUNPRO, as it seems not to support type_traits correctly
  511. template <typename T, typename U>
  512. struct is_convertible_to_T_or_factory
  513. : boost::conditional< boost::is_base_of<boost::in_place_factory_base, BOOST_DEDUCED_TYPENAME boost::decay<U>::type>::value
  514. || boost::is_base_of<boost::typed_in_place_factory_base, BOOST_DEDUCED_TYPENAME boost::decay<U>::type>::value
  515. || boost::is_constructible<T, U&&>::value
  516. , boost::true_type, boost::false_type>::type
  517. {};
  518. #else
  519. #define BOOST_OPTIONAL_DETAIL_NO_IS_CONSTRUCTIBLE_TRAIT
  520. template <typename, typename>
  521. struct is_convertible_to_T_or_factory : boost::true_type
  522. {};
  523. #endif // is_convertible condition
  524. template <typename T, typename U>
  525. struct is_optional_val_init_candidate
  526. : boost::conditional< !is_optional_related<U>::value && is_convertible_to_T_or_factory<T, U>::value
  527. , boost::true_type, boost::false_type>::type
  528. {};
  529. } // namespace optional_detail
  530. template<class T>
  531. class optional : public optional_detail::optional_base<T>
  532. {
  533. typedef optional_detail::optional_base<T> base ;
  534. public :
  535. typedef optional<T> this_type ;
  536. typedef BOOST_DEDUCED_TYPENAME base::value_type value_type ;
  537. typedef BOOST_DEDUCED_TYPENAME base::reference_type reference_type ;
  538. typedef BOOST_DEDUCED_TYPENAME base::reference_const_type reference_const_type ;
  539. #ifndef BOOST_OPTIONAL_DETAIL_NO_RVALUE_REFERENCES
  540. typedef BOOST_DEDUCED_TYPENAME base::rval_reference_type rval_reference_type ;
  541. typedef BOOST_DEDUCED_TYPENAME base::reference_type_of_temporary_wrapper reference_type_of_temporary_wrapper ;
  542. #endif
  543. typedef BOOST_DEDUCED_TYPENAME base::pointer_type pointer_type ;
  544. typedef BOOST_DEDUCED_TYPENAME base::pointer_const_type pointer_const_type ;
  545. typedef BOOST_DEDUCED_TYPENAME base::argument_type argument_type ;
  546. // Creates an optional<T> uninitialized.
  547. // No-throw
  548. optional() BOOST_NOEXCEPT : base() {}
  549. // Creates an optional<T> uninitialized.
  550. // No-throw
  551. optional( none_t none_ ) BOOST_NOEXCEPT : base(none_) {}
  552. // Creates an optional<T> initialized with 'val'.
  553. // Can throw if T::T(T const&) does
  554. optional ( argument_type val ) : base(val) {}
  555. #ifndef BOOST_OPTIONAL_DETAIL_NO_RVALUE_REFERENCES
  556. // Creates an optional<T> initialized with 'move(val)'.
  557. // Can throw if T::T(T &&) does
  558. optional ( rval_reference_type val ) : base( boost::forward<T>(val) )
  559. {}
  560. #endif
  561. // Creates an optional<T> initialized with 'val' IFF cond is true, otherwise creates an uninitialized optional.
  562. // Can throw if T::T(T const&) does
  563. optional ( bool cond, argument_type val ) : base(cond,val) {}
  564. // NOTE: MSVC needs templated versions first
  565. // Creates a deep copy of another convertible optional<U>
  566. // Requires a valid conversion from U to T.
  567. // Can throw if T::T(U const&) does
  568. template<class U>
  569. explicit optional ( optional<U> const& rhs )
  570. :
  571. base()
  572. {
  573. if ( rhs.is_initialized() )
  574. this->construct(rhs.get());
  575. }
  576. #ifndef BOOST_OPTIONAL_DETAIL_NO_RVALUE_REFERENCES
  577. // Creates a deep move of another convertible optional<U>
  578. // Requires a valid conversion from U to T.
  579. // Can throw if T::T(U&&) does
  580. template<class U>
  581. explicit optional ( optional<U> && rhs )
  582. :
  583. base()
  584. {
  585. if ( rhs.is_initialized() )
  586. this->construct( boost::move(rhs.get()) );
  587. }
  588. #endif
  589. #ifndef BOOST_OPTIONAL_NO_INPLACE_FACTORY_SUPPORT
  590. // Creates an optional<T> with an expression which can be either
  591. // (a) An instance of InPlaceFactory (i.e. in_place(a,b,...,n);
  592. // (b) An instance of TypedInPlaceFactory ( i.e. in_place<T>(a,b,...,n);
  593. // (c) Any expression implicitly convertible to the single type
  594. // of a one-argument T's constructor.
  595. // (d*) Weak compilers (BCB) might also resolved Expr as optional<T> and optional<U>
  596. // even though explicit overloads are present for these.
  597. // Depending on the above some T ctor is called.
  598. // Can throw if the resolved T ctor throws.
  599. #ifndef BOOST_OPTIONAL_DETAIL_NO_RVALUE_REFERENCES
  600. template<class Expr>
  601. explicit optional ( Expr&& expr,
  602. BOOST_DEDUCED_TYPENAME boost::enable_if< optional_detail::is_optional_val_init_candidate<T, Expr> >::type* = 0
  603. )
  604. : base(boost::forward<Expr>(expr),boost::addressof(expr))
  605. {}
  606. #else
  607. template<class Expr>
  608. explicit optional ( Expr const& expr ) : base(expr,boost::addressof(expr)) {}
  609. #endif // !defined BOOST_OPTIONAL_DETAIL_NO_RVALUE_REFERENCES
  610. #endif // !defined BOOST_OPTIONAL_NO_INPLACE_FACTORY_SUPPORT
  611. // Creates a deep copy of another optional<T>
  612. // Can throw if T::T(T const&) does
  613. optional ( optional const& rhs ) : base( static_cast<base const&>(rhs) ) {}
  614. #ifndef BOOST_OPTIONAL_DETAIL_NO_RVALUE_REFERENCES
  615. // Creates a deep move of another optional<T>
  616. // Can throw if T::T(T&&) does
  617. optional ( optional && rhs )
  618. BOOST_NOEXCEPT_IF(::boost::is_nothrow_move_constructible<T>::value)
  619. : base( boost::move(rhs) )
  620. {}
  621. #endif
  622. #if BOOST_WORKAROUND(_MSC_VER, <= 1600)
  623. // On old MSVC compilers the implicitly declared dtor is not called
  624. ~optional() {}
  625. #endif
  626. #if !defined(BOOST_OPTIONAL_NO_INPLACE_FACTORY_SUPPORT) && !defined(BOOST_OPTIONAL_WEAK_OVERLOAD_RESOLUTION)
  627. // Assigns from an expression. See corresponding constructor.
  628. // Basic Guarantee: If the resolved T ctor throws, this is left UNINITIALIZED
  629. #ifndef BOOST_OPTIONAL_DETAIL_NO_RVALUE_REFERENCES
  630. template<class Expr>
  631. BOOST_DEDUCED_TYPENAME boost::enable_if<optional_detail::is_optional_val_init_candidate<T, Expr>, optional&>::type
  632. operator= ( Expr&& expr )
  633. {
  634. this->assign_expr(boost::forward<Expr>(expr),boost::addressof(expr));
  635. return *this ;
  636. }
  637. #else
  638. template<class Expr>
  639. optional& operator= ( Expr const& expr )
  640. {
  641. this->assign_expr(expr,boost::addressof(expr));
  642. return *this ;
  643. }
  644. #endif // !defined BOOST_OPTIONAL_DETAIL_NO_RVALUE_REFERENCES
  645. #endif // !defined(BOOST_OPTIONAL_NO_INPLACE_FACTORY_SUPPORT) && !defined(BOOST_OPTIONAL_WEAK_OVERLOAD_RESOLUTION)
  646. // Copy-assigns from another convertible optional<U> (converts && deep-copies the rhs value)
  647. // Requires a valid conversion from U to T.
  648. // Basic Guarantee: If T::T( U const& ) throws, this is left UNINITIALIZED
  649. template<class U>
  650. optional& operator= ( optional<U> const& rhs )
  651. {
  652. this->assign(rhs);
  653. return *this ;
  654. }
  655. #ifndef BOOST_OPTIONAL_DETAIL_NO_RVALUE_REFERENCES
  656. // Move-assigns from another convertible optional<U> (converts && deep-moves the rhs value)
  657. // Requires a valid conversion from U to T.
  658. // Basic Guarantee: If T::T( U && ) throws, this is left UNINITIALIZED
  659. template<class U>
  660. optional& operator= ( optional<U> && rhs )
  661. {
  662. this->assign(boost::move(rhs));
  663. return *this ;
  664. }
  665. #endif
  666. // Assigns from another optional<T> (deep-copies the rhs value)
  667. // Basic Guarantee: If T::T( T const& ) throws, this is left UNINITIALIZED
  668. // (NOTE: On BCB, this operator is not actually called and left is left UNMODIFIED in case of a throw)
  669. optional& operator= ( optional const& rhs )
  670. {
  671. this->assign( static_cast<base const&>(rhs) ) ;
  672. return *this ;
  673. }
  674. #ifndef BOOST_OPTIONAL_DETAIL_NO_RVALUE_REFERENCES
  675. // Assigns from another optional<T> (deep-moves the rhs value)
  676. optional& operator= ( optional && rhs )
  677. BOOST_NOEXCEPT_IF(::boost::is_nothrow_move_constructible<T>::value && ::boost::is_nothrow_move_assignable<T>::value)
  678. {
  679. this->assign( static_cast<base &&>(rhs) ) ;
  680. return *this ;
  681. }
  682. #endif
  683. // Assigns from a T (deep-copies the rhs value)
  684. // Basic Guarantee: If T::( T const& ) throws, this is left UNINITIALIZED
  685. optional& operator= ( argument_type val )
  686. {
  687. this->assign( val ) ;
  688. return *this ;
  689. }
  690. #ifndef BOOST_OPTIONAL_DETAIL_NO_RVALUE_REFERENCES
  691. // Assigns from a T (deep-moves the rhs value)
  692. optional& operator= ( rval_reference_type val )
  693. {
  694. this->assign( boost::move(val) ) ;
  695. return *this ;
  696. }
  697. #endif
  698. // Assigns from a "none"
  699. // Which destroys the current value, if any, leaving this UNINITIALIZED
  700. // No-throw (assuming T::~T() doesn't)
  701. optional& operator= ( none_t none_ ) BOOST_NOEXCEPT
  702. {
  703. this->assign( none_ ) ;
  704. return *this ;
  705. }
  706. #if (!defined BOOST_OPTIONAL_DETAIL_NO_RVALUE_REFERENCES) && (!defined BOOST_NO_CXX11_VARIADIC_TEMPLATES)
  707. // Constructs in-place
  708. // upon exception *this is always uninitialized
  709. template<class... Args>
  710. void emplace ( Args&&... args )
  711. {
  712. this->emplace_assign( boost::forward<Args>(args)... );
  713. }
  714. #elif (!defined BOOST_OPTIONAL_DETAIL_NO_RVALUE_REFERENCES)
  715. template<class Arg>
  716. void emplace ( Arg&& arg )
  717. {
  718. this->emplace_assign( boost::forward<Arg>(arg) );
  719. }
  720. void emplace ()
  721. {
  722. this->emplace_assign();
  723. }
  724. #else
  725. template<class Arg>
  726. void emplace ( const Arg& arg )
  727. {
  728. this->emplace_assign( arg );
  729. }
  730. template<class Arg>
  731. void emplace ( Arg& arg )
  732. {
  733. this->emplace_assign( arg );
  734. }
  735. void emplace ()
  736. {
  737. this->emplace_assign();
  738. }
  739. #endif
  740. void swap( optional & arg )
  741. BOOST_NOEXCEPT_IF(::boost::is_nothrow_move_constructible<T>::value && ::boost::is_nothrow_move_assignable<T>::value)
  742. {
  743. // allow for Koenig lookup
  744. boost::swap(*this, arg);
  745. }
  746. // Returns a reference to the value if this is initialized, otherwise,
  747. // the behaviour is UNDEFINED
  748. // No-throw
  749. reference_const_type get() const { BOOST_ASSERT(this->is_initialized()) ; return this->get_impl(); }
  750. reference_type get() { BOOST_ASSERT(this->is_initialized()) ; return this->get_impl(); }
  751. // Returns a copy of the value if this is initialized, 'v' otherwise
  752. reference_const_type get_value_or ( reference_const_type v ) const { return this->is_initialized() ? get() : v ; }
  753. reference_type get_value_or ( reference_type v ) { return this->is_initialized() ? get() : v ; }
  754. // Returns a pointer to the value if this is initialized, otherwise,
  755. // the behaviour is UNDEFINED
  756. // No-throw
  757. pointer_const_type operator->() const { BOOST_ASSERT(this->is_initialized()) ; return this->get_ptr_impl() ; }
  758. pointer_type operator->() { BOOST_ASSERT(this->is_initialized()) ; return this->get_ptr_impl() ; }
  759. // Returns a reference to the value if this is initialized, otherwise,
  760. // the behaviour is UNDEFINED
  761. // No-throw
  762. #if (!defined BOOST_NO_CXX11_REF_QUALIFIERS) && (!defined BOOST_OPTIONAL_DETAIL_NO_RVALUE_REFERENCES)
  763. reference_const_type operator *() const& { return this->get() ; }
  764. reference_type operator *() & { return this->get() ; }
  765. reference_type_of_temporary_wrapper operator *() && { return boost::move(this->get()) ; }
  766. #else
  767. reference_const_type operator *() const { return this->get() ; }
  768. reference_type operator *() { return this->get() ; }
  769. #endif // !defined BOOST_NO_CXX11_REF_QUALIFIERS
  770. #if (!defined BOOST_NO_CXX11_REF_QUALIFIERS) && (!defined BOOST_OPTIONAL_DETAIL_NO_RVALUE_REFERENCES)
  771. reference_const_type value() const&
  772. {
  773. if (this->is_initialized())
  774. return this->get() ;
  775. else
  776. throw_exception(bad_optional_access());
  777. }
  778. reference_type value() &
  779. {
  780. if (this->is_initialized())
  781. return this->get() ;
  782. else
  783. throw_exception(bad_optional_access());
  784. }
  785. reference_type_of_temporary_wrapper value() &&
  786. {
  787. if (this->is_initialized())
  788. return boost::move(this->get()) ;
  789. else
  790. throw_exception(bad_optional_access());
  791. }
  792. #else
  793. reference_const_type value() const
  794. {
  795. if (this->is_initialized())
  796. return this->get() ;
  797. else
  798. throw_exception(bad_optional_access());
  799. }
  800. reference_type value()
  801. {
  802. if (this->is_initialized())
  803. return this->get() ;
  804. else
  805. throw_exception(bad_optional_access());
  806. }
  807. #endif
  808. #ifndef BOOST_NO_CXX11_REF_QUALIFIERS
  809. template <class U>
  810. value_type value_or ( U&& v ) const&
  811. {
  812. if (this->is_initialized())
  813. return get();
  814. else
  815. return boost::forward<U>(v);
  816. }
  817. template <class U>
  818. value_type value_or ( U&& v ) &&
  819. {
  820. if (this->is_initialized())
  821. return boost::move(get());
  822. else
  823. return boost::forward<U>(v);
  824. }
  825. #elif !defined BOOST_OPTIONAL_DETAIL_NO_RVALUE_REFERENCES
  826. template <class U>
  827. value_type value_or ( U&& v ) const
  828. {
  829. if (this->is_initialized())
  830. return get();
  831. else
  832. return boost::forward<U>(v);
  833. }
  834. #else
  835. template <class U>
  836. value_type value_or ( U const& v ) const
  837. {
  838. if (this->is_initialized())
  839. return get();
  840. else
  841. return v;
  842. }
  843. template <class U>
  844. value_type value_or ( U& v ) const
  845. {
  846. if (this->is_initialized())
  847. return get();
  848. else
  849. return v;
  850. }
  851. #endif
  852. #ifndef BOOST_NO_CXX11_REF_QUALIFIERS
  853. template <typename F>
  854. value_type value_or_eval ( F f ) const&
  855. {
  856. if (this->is_initialized())
  857. return get();
  858. else
  859. return f();
  860. }
  861. template <typename F>
  862. value_type value_or_eval ( F f ) &&
  863. {
  864. if (this->is_initialized())
  865. return boost::move(get());
  866. else
  867. return f();
  868. }
  869. #else
  870. template <typename F>
  871. value_type value_or_eval ( F f ) const
  872. {
  873. if (this->is_initialized())
  874. return get();
  875. else
  876. return f();
  877. }
  878. #endif
  879. bool operator!() const BOOST_NOEXCEPT { return !this->is_initialized() ; }
  880. BOOST_EXPLICIT_OPERATOR_BOOL_NOEXCEPT()
  881. } ;
  882. } // namespace boost
  883. #endif // BOOST_OPTIONAL_CONFIG_USE_OLD_IMPLEMENTATION_OF_OPTIONAL
  884. namespace boost {
  885. #ifndef BOOST_OPTIONAL_DETAIL_NO_RVALUE_REFERENCES
  886. template<class T>
  887. class optional<T&&>
  888. {
  889. BOOST_STATIC_ASSERT_MSG(sizeof(T) == 0, "Optional rvalue references are illegal.");
  890. } ;
  891. #endif
  892. } // namespace boost
  893. #ifndef BOOST_OPTIONAL_CONFIG_DONT_SPECIALIZE_OPTIONAL_REFS
  894. # include <boost/optional/detail/optional_reference_spec.hpp>
  895. #endif
  896. namespace boost {
  897. // Returns optional<T>(v)
  898. template<class T>
  899. inline
  900. optional<T> make_optional ( T const& v )
  901. {
  902. return optional<T>(v);
  903. }
  904. // Returns optional<T>(cond,v)
  905. template<class T>
  906. inline
  907. optional<T> make_optional ( bool cond, T const& v )
  908. {
  909. return optional<T>(cond,v);
  910. }
  911. // Returns a reference to the value if this is initialized, otherwise, the behaviour is UNDEFINED.
  912. // No-throw
  913. template<class T>
  914. inline
  915. BOOST_DEDUCED_TYPENAME optional<T>::reference_const_type
  916. get ( optional<T> const& opt )
  917. {
  918. return opt.get() ;
  919. }
  920. template<class T>
  921. inline
  922. BOOST_DEDUCED_TYPENAME optional<T>::reference_type
  923. get ( optional<T>& opt )
  924. {
  925. return opt.get() ;
  926. }
  927. // Returns a pointer to the value if this is initialized, otherwise, returns NULL.
  928. // No-throw
  929. template<class T>
  930. inline
  931. BOOST_DEDUCED_TYPENAME optional<T>::pointer_const_type
  932. get ( optional<T> const* opt )
  933. {
  934. return opt->get_ptr() ;
  935. }
  936. template<class T>
  937. inline
  938. BOOST_DEDUCED_TYPENAME optional<T>::pointer_type
  939. get ( optional<T>* opt )
  940. {
  941. return opt->get_ptr() ;
  942. }
  943. // Returns a reference to the value if this is initialized, otherwise, the behaviour is UNDEFINED.
  944. // No-throw
  945. template<class T>
  946. inline
  947. BOOST_DEDUCED_TYPENAME optional<T>::reference_const_type
  948. get_optional_value_or ( optional<T> const& opt, BOOST_DEDUCED_TYPENAME optional<T>::reference_const_type v )
  949. {
  950. return opt.get_value_or(v) ;
  951. }
  952. template<class T>
  953. inline
  954. BOOST_DEDUCED_TYPENAME optional<T>::reference_type
  955. get_optional_value_or ( optional<T>& opt, BOOST_DEDUCED_TYPENAME optional<T>::reference_type v )
  956. {
  957. return opt.get_value_or(v) ;
  958. }
  959. // Returns a pointer to the value if this is initialized, otherwise, returns NULL.
  960. // No-throw
  961. template<class T>
  962. inline
  963. BOOST_DEDUCED_TYPENAME optional<T>::pointer_const_type
  964. get_pointer ( optional<T> const& opt )
  965. {
  966. return opt.get_ptr() ;
  967. }
  968. template<class T>
  969. inline
  970. BOOST_DEDUCED_TYPENAME optional<T>::pointer_type
  971. get_pointer ( optional<T>& opt )
  972. {
  973. return opt.get_ptr() ;
  974. }
  975. } // namespace boost
  976. namespace boost {
  977. // The following declaration prevents a bug where operator safe-bool is used upon streaming optional object if you forget the IO header.
  978. template<class CharType, class CharTrait>
  979. std::basic_ostream<CharType, CharTrait>&
  980. operator<<(std::basic_ostream<CharType, CharTrait>& os, optional_detail::optional_tag const&)
  981. {
  982. BOOST_STATIC_ASSERT_MSG(sizeof(CharType) == 0, "If you want to output boost::optional, include header <boost/optional/optional_io.hpp>");
  983. return os;
  984. }
  985. } // namespace boost
  986. #include <boost/optional/detail/optional_relops.hpp>
  987. #include <boost/optional/detail/optional_swap.hpp>
  988. #endif // header guard