maybe.hpp 2.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120
  1. // Copyright Daniel Wallin 2006. Use, modification and distribution is
  2. // subject to the Boost Software License, Version 1.0. (See accompanying
  3. // file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
  4. //
  5. // 2009.10.21 TDS remove depenency on boost::python::detail::referent_storage
  6. //
  7. #ifndef BOOST_PARAMETER_MAYBE_091021_HPP
  8. # define BOOST_PARAMETER_MAYBE_091021_HPP
  9. # include <boost/mpl/if.hpp>
  10. # include <boost/mpl/identity.hpp>
  11. # include <boost/type_traits/is_reference.hpp>
  12. # include <boost/type_traits/add_reference.hpp>
  13. # include <boost/optional.hpp>
  14. # include <boost/aligned_storage.hpp>
  15. # include <boost/type_traits/remove_cv.hpp>
  16. # include <boost/type_traits/add_const.hpp>
  17. # include <boost/parameter/aux_/is_maybe.hpp>
  18. namespace boost { namespace parameter { namespace aux {
  19. template <class T> struct referent_size;
  20. template <class T>
  21. struct referent_size<T&>
  22. {
  23. BOOST_STATIC_CONSTANT(std::size_t, value = sizeof(T));
  24. };
  25. // A metafunction returning a POD type which can store U, where T ==
  26. // U&. If T is not a reference type, returns a POD which can store T.
  27. template <class T>
  28. struct referent_storage
  29. {
  30. typedef typename boost::aligned_storage<
  31. referent_size<T>::value
  32. >::type type;
  33. };
  34. template <class T>
  35. struct maybe : maybe_base
  36. {
  37. typedef typename add_reference<
  38. # if BOOST_WORKAROUND(__BORLANDC__, BOOST_TESTED_AT(0x564))
  39. T const
  40. # else
  41. typename add_const<T>::type
  42. # endif
  43. >::type reference;
  44. typedef typename remove_cv<
  45. BOOST_DEDUCED_TYPENAME remove_reference<reference>::type
  46. >::type non_cv_value;
  47. explicit maybe(T value_)
  48. : value(value_)
  49. , constructed(false)
  50. {}
  51. maybe()
  52. : constructed(false)
  53. {}
  54. ~maybe()
  55. {
  56. if (constructed)
  57. this->destroy();
  58. }
  59. reference construct(reference value_) const
  60. {
  61. return value_;
  62. }
  63. template <class U>
  64. reference construct2(U const& value_) const
  65. {
  66. new (m_storage.address()) non_cv_value(value_);
  67. constructed = true;
  68. return *(non_cv_value*)m_storage.address();
  69. }
  70. template <class U>
  71. reference construct(U const& value_) const
  72. {
  73. return this->construct2(value_);
  74. }
  75. void destroy()
  76. {
  77. ((non_cv_value*)m_storage.address())->~non_cv_value();
  78. }
  79. typedef reference(maybe<T>::*safe_bool)() const;
  80. operator safe_bool() const
  81. {
  82. return value ? &maybe<T>::get : 0 ;
  83. }
  84. reference get() const
  85. {
  86. return value.get();
  87. }
  88. private:
  89. boost::optional<T> value;
  90. mutable bool constructed;
  91. mutable typename referent_storage<
  92. reference
  93. >::type m_storage;
  94. };
  95. }}} // namespace boost::parameter::aux
  96. #endif // BOOST_PARAMETER_MAYBE_060211_HPP