factory.hpp 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179
  1. /*=============================================================================
  2. Copyright (c) 2007 Tobias Schwinger
  3. Use modification and distribution are 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. #ifndef BOOST_FUNCTIONAL_FACTORY_HPP_INCLUDED
  8. # ifndef BOOST_PP_IS_ITERATING
  9. # include <boost/preprocessor/iteration/iterate.hpp>
  10. # include <boost/preprocessor/repetition/enum_params.hpp>
  11. # include <boost/preprocessor/repetition/enum_binary_params.hpp>
  12. # include <new>
  13. # include <boost/pointee.hpp>
  14. # include <boost/get_pointer.hpp>
  15. # include <boost/non_type.hpp>
  16. # include <boost/type_traits/remove_cv.hpp>
  17. # if defined(BOOST_FUNCTIONAL_FACTORY_SUPPORT_NONE_T)
  18. # include <boost/none_t.hpp>
  19. # endif
  20. # ifndef BOOST_FUNCTIONAL_FACTORY_MAX_ARITY
  21. # define BOOST_FUNCTIONAL_FACTORY_MAX_ARITY 10
  22. # elif BOOST_FUNCTIONAL_FACTORY_MAX_ARITY < 3
  23. # undef BOOST_FUNCTIONAL_FACTORY_MAX_ARITY
  24. # define BOOST_FUNCTIONAL_FACTORY_MAX_ARITY 3
  25. # endif
  26. namespace boost
  27. {
  28. enum factory_alloc_propagation
  29. {
  30. factory_alloc_for_pointee_and_deleter,
  31. factory_passes_alloc_to_smart_pointer
  32. };
  33. #if defined(BOOST_FUNCTIONAL_FACTORY_SUPPORT_NONE_T)
  34. template< typename Pointer, class Allocator = boost::none_t,
  35. factory_alloc_propagation AP = factory_alloc_for_pointee_and_deleter >
  36. class factory;
  37. #else
  38. template< typename Pointer, class Allocator = void,
  39. factory_alloc_propagation AP = factory_alloc_for_pointee_and_deleter >
  40. class factory;
  41. #endif
  42. //----- ---- --- -- - - - -
  43. template< typename Pointer, factory_alloc_propagation AP >
  44. class factory<Pointer, void, AP>
  45. {
  46. public:
  47. typedef typename boost::remove_cv<Pointer>::type result_type;
  48. typedef typename boost::pointee<result_type>::type value_type;
  49. factory()
  50. { }
  51. # define BOOST_PP_FILENAME_1 <boost/functional/factory.hpp>
  52. # define BOOST_PP_ITERATION_LIMITS (0,BOOST_FUNCTIONAL_FACTORY_MAX_ARITY)
  53. # include BOOST_PP_ITERATE()
  54. };
  55. #if defined(BOOST_FUNCTIONAL_FACTORY_SUPPORT_NONE_T)
  56. template< typename Pointer, factory_alloc_propagation AP >
  57. class factory<Pointer, boost::none_t, AP>
  58. : public factory<Pointer, void, AP>
  59. {};
  60. #endif
  61. template< class Pointer, class Allocator, factory_alloc_propagation AP >
  62. class factory
  63. : private Allocator::template rebind< typename boost::pointee<
  64. typename boost::remove_cv<Pointer>::type >::type >::other
  65. {
  66. public:
  67. typedef typename boost::remove_cv<Pointer>::type result_type;
  68. typedef typename boost::pointee<result_type>::type value_type;
  69. typedef typename Allocator::template rebind<value_type>::other
  70. allocator_type;
  71. explicit factory(allocator_type const & a = allocator_type())
  72. : allocator_type(a)
  73. { }
  74. private:
  75. struct deleter
  76. : allocator_type
  77. {
  78. inline deleter(allocator_type const& that)
  79. : allocator_type(that)
  80. { }
  81. allocator_type& get_allocator() const
  82. {
  83. return *const_cast<allocator_type*>(
  84. static_cast<allocator_type const*>(this));
  85. }
  86. void operator()(value_type* ptr) const
  87. {
  88. if (!! ptr) ptr->~value_type();
  89. const_cast<allocator_type*>(static_cast<allocator_type const*>(
  90. this))->deallocate(ptr,1);
  91. }
  92. };
  93. inline allocator_type& get_allocator() const
  94. {
  95. return *const_cast<allocator_type*>(
  96. static_cast<allocator_type const*>(this));
  97. }
  98. inline result_type make_pointer(value_type* ptr, boost::non_type<
  99. factory_alloc_propagation,factory_passes_alloc_to_smart_pointer>)
  100. const
  101. {
  102. return result_type(ptr,deleter(this->get_allocator()));
  103. }
  104. inline result_type make_pointer(value_type* ptr, boost::non_type<
  105. factory_alloc_propagation,factory_alloc_for_pointee_and_deleter>)
  106. const
  107. {
  108. return result_type(ptr,deleter(this->get_allocator()),
  109. this->get_allocator());
  110. }
  111. public:
  112. # define BOOST_TMP_MACRO
  113. # define BOOST_PP_FILENAME_1 <boost/functional/factory.hpp>
  114. # define BOOST_PP_ITERATION_LIMITS (0,BOOST_FUNCTIONAL_FACTORY_MAX_ARITY)
  115. # include BOOST_PP_ITERATE()
  116. # undef BOOST_TMP_MACRO
  117. };
  118. template< typename Pointer, class Allocator, factory_alloc_propagation AP >
  119. class factory<Pointer&, Allocator, AP>;
  120. // forbidden, would create a dangling reference
  121. }
  122. # define BOOST_FUNCTIONAL_FACTORY_HPP_INCLUDED
  123. # else // defined(BOOST_PP_IS_ITERATING)
  124. # define N BOOST_PP_ITERATION()
  125. # if !defined(BOOST_TMP_MACRO)
  126. # if N > 0
  127. template< BOOST_PP_ENUM_PARAMS(N, typename T) >
  128. # endif
  129. inline result_type operator()(BOOST_PP_ENUM_BINARY_PARAMS(N,T,& a)) const
  130. {
  131. return result_type( new value_type(BOOST_PP_ENUM_PARAMS(N,a)) );
  132. }
  133. # else // defined(BOOST_TMP_MACRO)
  134. # if N > 0
  135. template< BOOST_PP_ENUM_PARAMS(N, typename T) >
  136. # endif
  137. inline result_type operator()(BOOST_PP_ENUM_BINARY_PARAMS(N,T,& a)) const
  138. {
  139. value_type* memory = this->get_allocator().allocate(1);
  140. try
  141. {
  142. return make_pointer(
  143. new(memory) value_type(BOOST_PP_ENUM_PARAMS(N,a)),
  144. boost::non_type<factory_alloc_propagation,AP>() );
  145. }
  146. catch (...) { this->get_allocator().deallocate(memory,1); throw; }
  147. }
  148. # endif
  149. # undef N
  150. # endif // defined(BOOST_PP_IS_ITERATING)
  151. #endif // include guard