any_hook.hpp 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336
  1. /////////////////////////////////////////////////////////////////////////////
  2. //
  3. // (C) Copyright Ion Gaztanaga 2006-2013
  4. //
  5. // Distributed under the Boost Software License, Version 1.0.
  6. // (See accompanying file LICENSE_1_0.txt or copy at
  7. // http://www.boost.org/LICENSE_1_0.txt)
  8. //
  9. // See http://www.boost.org/libs/intrusive for documentation.
  10. //
  11. /////////////////////////////////////////////////////////////////////////////
  12. #ifndef BOOST_INTRUSIVE_ANY_HOOK_HPP
  13. #define BOOST_INTRUSIVE_ANY_HOOK_HPP
  14. #include <boost/intrusive/detail/config_begin.hpp>
  15. #include <boost/intrusive/intrusive_fwd.hpp>
  16. #include <boost/intrusive/detail/any_node_and_algorithms.hpp>
  17. #include <boost/intrusive/options.hpp>
  18. #include <boost/intrusive/detail/generic_hook.hpp>
  19. #include <boost/intrusive/detail/mpl.hpp>
  20. #include <boost/intrusive/pointer_rebind.hpp>
  21. #if defined(BOOST_HAS_PRAGMA_ONCE)
  22. # pragma once
  23. #endif
  24. namespace boost {
  25. namespace intrusive {
  26. //! Helper metafunction to define a \c \c any_base_hook that yields to the same
  27. //! type when the same options (either explicitly or implicitly) are used.
  28. #if defined(BOOST_INTRUSIVE_DOXYGEN_INVOKED) || defined(BOOST_INTRUSIVE_VARIADIC_TEMPLATES)
  29. template<class ...Options>
  30. #else
  31. template<class O1 = void, class O2 = void, class O3 = void>
  32. #endif
  33. struct make_any_base_hook
  34. {
  35. /// @cond
  36. typedef typename pack_options
  37. < hook_defaults,
  38. #if !defined(BOOST_INTRUSIVE_VARIADIC_TEMPLATES)
  39. O1, O2, O3
  40. #else
  41. Options...
  42. #endif
  43. >::type packed_options;
  44. typedef generic_hook
  45. < any_algorithms<typename packed_options::void_pointer>
  46. , typename packed_options::tag
  47. , packed_options::link_mode
  48. , AnyBaseHookId
  49. > implementation_defined;
  50. /// @endcond
  51. typedef implementation_defined type;
  52. };
  53. //! Derive a class from this hook in order to store objects of that class
  54. //! in an intrusive container.
  55. //!
  56. //! The hook admits the following options: \c tag<>, \c void_pointer<> and
  57. //! \c link_mode<>.
  58. //!
  59. //! \c tag<> defines a tag to identify the node.
  60. //! The same tag value can be used in different classes, but if a class is
  61. //! derived from more than one \c any_base_hook, then each \c any_base_hook needs its
  62. //! unique tag.
  63. //!
  64. //! \c link_mode<> will specify the linking mode of the hook (\c normal_link, \c safe_link).
  65. //!
  66. //! \c void_pointer<> is the pointer type that will be used internally in the hook
  67. //! and the container configured to use this hook.
  68. #if defined(BOOST_INTRUSIVE_DOXYGEN_INVOKED) || defined(BOOST_INTRUSIVE_VARIADIC_TEMPLATES)
  69. template<class ...Options>
  70. #else
  71. template<class O1, class O2, class O3>
  72. #endif
  73. class any_base_hook
  74. : public make_any_base_hook
  75. #if !defined(BOOST_INTRUSIVE_VARIADIC_TEMPLATES)
  76. <O1, O2, O3>
  77. #else
  78. <Options...>
  79. #endif
  80. ::type
  81. {
  82. #if defined(BOOST_INTRUSIVE_DOXYGEN_INVOKED)
  83. public:
  84. //! <b>Effects</b>: If link_mode is or \c safe_link
  85. //! initializes the node to an unlinked state.
  86. //!
  87. //! <b>Throws</b>: Nothing.
  88. any_base_hook();
  89. //! <b>Effects</b>: If link_mode is or \c safe_link
  90. //! initializes the node to an unlinked state. The argument is ignored.
  91. //!
  92. //! <b>Throws</b>: Nothing.
  93. //!
  94. //! <b>Rationale</b>: Providing a copy-constructor
  95. //! makes classes using the hook STL-compliant without forcing the
  96. //! user to do some additional work. \c swap can be used to emulate
  97. //! move-semantics.
  98. any_base_hook(const any_base_hook& );
  99. //! <b>Effects</b>: Empty function. The argument is ignored.
  100. //!
  101. //! <b>Throws</b>: Nothing.
  102. //!
  103. //! <b>Rationale</b>: Providing an assignment operator
  104. //! makes classes using the hook STL-compliant without forcing the
  105. //! user to do some additional work. \c swap can be used to emulate
  106. //! move-semantics.
  107. any_base_hook& operator=(const any_base_hook& );
  108. //! <b>Effects</b>: If link_mode is \c normal_link, the destructor does
  109. //! nothing (ie. no code is generated). If link_mode is \c safe_link and the
  110. //! object is stored in a container an assertion is raised.
  111. //!
  112. //! <b>Throws</b>: Nothing.
  113. ~any_base_hook();
  114. //! <b>Precondition</b>: link_mode must be \c safe_link.
  115. //!
  116. //! <b>Returns</b>: true, if the node belongs to a container, false
  117. //! otherwise. This function can be used to test whether \c container::iterator_to
  118. //! will return a valid iterator.
  119. //!
  120. //! <b>Complexity</b>: Constant
  121. bool is_linked() const;
  122. #endif
  123. };
  124. //! Helper metafunction to define a \c \c any_member_hook that yields to the same
  125. //! type when the same options (either explicitly or implicitly) are used.
  126. #if defined(BOOST_INTRUSIVE_DOXYGEN_INVOKED) || defined(BOOST_INTRUSIVE_VARIADIC_TEMPLATES)
  127. template<class ...Options>
  128. #else
  129. template<class O1 = void, class O2 = void, class O3 = void>
  130. #endif
  131. struct make_any_member_hook
  132. {
  133. /// @cond
  134. typedef typename pack_options
  135. < hook_defaults,
  136. #if !defined(BOOST_INTRUSIVE_VARIADIC_TEMPLATES)
  137. O1, O2, O3
  138. #else
  139. Options...
  140. #endif
  141. >::type packed_options;
  142. typedef generic_hook
  143. < any_algorithms<typename packed_options::void_pointer>
  144. , member_tag
  145. , packed_options::link_mode
  146. , NoBaseHookId
  147. > implementation_defined;
  148. /// @endcond
  149. typedef implementation_defined type;
  150. };
  151. //! Store this hook in a class to be inserted
  152. //! in an intrusive container.
  153. //!
  154. //! The hook admits the following options: \c void_pointer<> and
  155. //! \c link_mode<>.
  156. //!
  157. //! \c link_mode<> will specify the linking mode of the hook (\c normal_link or \c safe_link).
  158. //!
  159. //! \c void_pointer<> is the pointer type that will be used internally in the hook
  160. //! and the container configured to use this hook.
  161. #if defined(BOOST_INTRUSIVE_DOXYGEN_INVOKED) || defined(BOOST_INTRUSIVE_VARIADIC_TEMPLATES)
  162. template<class ...Options>
  163. #else
  164. template<class O1, class O2, class O3>
  165. #endif
  166. class any_member_hook
  167. : public make_any_member_hook
  168. #if !defined(BOOST_INTRUSIVE_VARIADIC_TEMPLATES)
  169. <O1, O2, O3>
  170. #else
  171. <Options...>
  172. #endif
  173. ::type
  174. {
  175. #if defined(BOOST_INTRUSIVE_DOXYGEN_INVOKED)
  176. public:
  177. //! <b>Effects</b>: If link_mode is or \c safe_link
  178. //! initializes the node to an unlinked state.
  179. //!
  180. //! <b>Throws</b>: Nothing.
  181. any_member_hook();
  182. //! <b>Effects</b>: If link_mode is or \c safe_link
  183. //! initializes the node to an unlinked state. The argument is ignored.
  184. //!
  185. //! <b>Throws</b>: Nothing.
  186. //!
  187. //! <b>Rationale</b>: Providing a copy-constructor
  188. //! makes classes using the hook STL-compliant without forcing the
  189. //! user to do some additional work. \c swap can be used to emulate
  190. //! move-semantics.
  191. any_member_hook(const any_member_hook& );
  192. //! <b>Effects</b>: Empty function. The argument is ignored.
  193. //!
  194. //! <b>Throws</b>: Nothing.
  195. //!
  196. //! <b>Rationale</b>: Providing an assignment operator
  197. //! makes classes using the hook STL-compliant without forcing the
  198. //! user to do some additional work. \c swap can be used to emulate
  199. //! move-semantics.
  200. any_member_hook& operator=(const any_member_hook& );
  201. //! <b>Effects</b>: If link_mode is \c normal_link, the destructor does
  202. //! nothing (ie. no code is generated). If link_mode is \c safe_link and the
  203. //! object is stored in a container an assertion is raised.
  204. //!
  205. //! <b>Throws</b>: Nothing.
  206. ~any_member_hook();
  207. //! <b>Precondition</b>: link_mode must be \c safe_link.
  208. //!
  209. //! <b>Returns</b>: true, if the node belongs to a container, false
  210. //! otherwise. This function can be used to test whether \c container::iterator_to
  211. //! will return a valid iterator.
  212. //!
  213. //! <b>Complexity</b>: Constant
  214. bool is_linked() const;
  215. #endif
  216. };
  217. /// @cond
  218. namespace detail{
  219. BOOST_INTRUSIVE_INTERNAL_STATIC_BOOL_IS_TRUE(old_proto_value_traits_base_hook, hooktags::is_base_hook)
  220. //!This option setter specifies that the container
  221. //!must use the specified base hook
  222. template<class BasicHook, template <class> class NodeTraits>
  223. struct any_to_some_hook
  224. {
  225. typedef typename BasicHook::template pack<empty>::proto_value_traits old_proto_value_traits;
  226. template<class Base>
  227. struct pack : public Base
  228. {
  229. struct proto_value_traits
  230. {
  231. //proto_value_traits::hooktags::is_base_hook is used by get_value_traits
  232. //to detect base hooks, so mark it in case BasicHook has it.
  233. struct hooktags
  234. {
  235. static const bool is_base_hook = old_proto_value_traits_base_hook_bool_is_true
  236. <old_proto_value_traits>::value;
  237. };
  238. typedef old_proto_value_traits basic_hook_t;
  239. static const bool is_any_hook = true;
  240. template<class VoidPtr>
  241. struct node_traits_from_voidptr
  242. { typedef NodeTraits<VoidPtr> type; };
  243. };
  244. };
  245. };
  246. } //namespace detail{
  247. /// @endcond
  248. //!This option setter specifies that
  249. //!any hook should behave as an slist hook
  250. template<class BasicHook>
  251. struct any_to_slist_hook
  252. /// @cond
  253. : public detail::any_to_some_hook<BasicHook, any_slist_node_traits>
  254. /// @endcond
  255. {};
  256. //!This option setter specifies that
  257. //!any hook should behave as an list hook
  258. template<class BasicHook>
  259. struct any_to_list_hook
  260. /// @cond
  261. : public detail::any_to_some_hook<BasicHook, any_list_node_traits>
  262. /// @endcond
  263. {};
  264. //!This option setter specifies that
  265. //!any hook should behave as a set hook
  266. template<class BasicHook>
  267. struct any_to_set_hook
  268. /// @cond
  269. : public detail::any_to_some_hook<BasicHook, any_rbtree_node_traits>
  270. /// @endcond
  271. {};
  272. //!This option setter specifies that
  273. //!any hook should behave as an avl_set hook
  274. template<class BasicHook>
  275. struct any_to_avl_set_hook
  276. /// @cond
  277. : public detail::any_to_some_hook<BasicHook, any_avltree_node_traits>
  278. /// @endcond
  279. {};
  280. //!This option setter specifies that any
  281. //!hook should behave as a bs_set hook
  282. template<class BasicHook>
  283. struct any_to_bs_set_hook
  284. /// @cond
  285. : public detail::any_to_some_hook<BasicHook, any_tree_node_traits>
  286. /// @endcond
  287. {};
  288. //!This option setter specifies that any hook
  289. //!should behave as an unordered set hook
  290. template<class BasicHook>
  291. struct any_to_unordered_set_hook
  292. /// @cond
  293. : public detail::any_to_some_hook<BasicHook, any_unordered_node_traits>
  294. /// @endcond
  295. {};
  296. } //namespace intrusive
  297. } //namespace boost
  298. #include <boost/intrusive/detail/config_end.hpp>
  299. #endif //BOOST_INTRUSIVE_ANY_HOOK_HPP