stack.hpp 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583
  1. // Copyright (C) 2008-2013 Tim Blechmann
  2. //
  3. // Distributed under the Boost Software License, Version 1.0. (See
  4. // accompanying file LICENSE_1_0.txt or copy at
  5. // http://www.boost.org/LICENSE_1_0.txt)
  6. #ifndef BOOST_LOCKFREE_STACK_HPP_INCLUDED
  7. #define BOOST_LOCKFREE_STACK_HPP_INCLUDED
  8. #include <boost/assert.hpp>
  9. #include <boost/checked_delete.hpp>
  10. #include <boost/integer_traits.hpp>
  11. #include <boost/static_assert.hpp>
  12. #include <boost/tuple/tuple.hpp>
  13. #include <boost/type_traits/has_trivial_assign.hpp>
  14. #include <boost/type_traits/has_trivial_destructor.hpp>
  15. #include <boost/lockfree/detail/atomic.hpp>
  16. #include <boost/lockfree/detail/copy_payload.hpp>
  17. #include <boost/lockfree/detail/freelist.hpp>
  18. #include <boost/lockfree/detail/parameter.hpp>
  19. #include <boost/lockfree/detail/tagged_ptr.hpp>
  20. #ifdef BOOST_HAS_PRAGMA_ONCE
  21. #pragma once
  22. #endif
  23. namespace boost {
  24. namespace lockfree {
  25. namespace detail {
  26. typedef parameter::parameters<boost::parameter::optional<tag::allocator>,
  27. boost::parameter::optional<tag::capacity>
  28. > stack_signature;
  29. }
  30. /** The stack class provides a multi-writer/multi-reader stack, pushing and popping is lock-free,
  31. * construction/destruction has to be synchronized. It uses a freelist for memory management,
  32. * freed nodes are pushed to the freelist and not returned to the OS before the stack is destroyed.
  33. *
  34. * \b Policies:
  35. *
  36. * - \c boost::lockfree::fixed_sized<>, defaults to \c boost::lockfree::fixed_sized<false> <br>
  37. * Can be used to completely disable dynamic memory allocations during push in order to ensure lockfree behavior.<br>
  38. * If the data structure is configured as fixed-sized, the internal nodes are stored inside an array and they are addressed
  39. * by array indexing. This limits the possible size of the stack to the number of elements that can be addressed by the index
  40. * type (usually 2**16-2), but on platforms that lack double-width compare-and-exchange instructions, this is the best way
  41. * to achieve lock-freedom.
  42. *
  43. * - \c boost::lockfree::capacity<>, optional <br>
  44. * If this template argument is passed to the options, the size of the stack is set at compile-time. <br>
  45. * It this option implies \c fixed_sized<true>
  46. *
  47. * - \c boost::lockfree::allocator<>, defaults to \c boost::lockfree::allocator<std::allocator<void>> <br>
  48. * Specifies the allocator that is used for the internal freelist
  49. *
  50. * \b Requirements:
  51. * - T must have a copy constructor
  52. * */
  53. #ifndef BOOST_DOXYGEN_INVOKED
  54. template <typename T,
  55. class A0 = boost::parameter::void_,
  56. class A1 = boost::parameter::void_,
  57. class A2 = boost::parameter::void_>
  58. #else
  59. template <typename T, ...Options>
  60. #endif
  61. class stack
  62. {
  63. private:
  64. #ifndef BOOST_DOXYGEN_INVOKED
  65. BOOST_STATIC_ASSERT(boost::has_trivial_assign<T>::value);
  66. BOOST_STATIC_ASSERT(boost::has_trivial_destructor<T>::value);
  67. typedef typename detail::stack_signature::bind<A0, A1, A2>::type bound_args;
  68. static const bool has_capacity = detail::extract_capacity<bound_args>::has_capacity;
  69. static const size_t capacity = detail::extract_capacity<bound_args>::capacity;
  70. static const bool fixed_sized = detail::extract_fixed_sized<bound_args>::value;
  71. static const bool node_based = !(has_capacity || fixed_sized);
  72. static const bool compile_time_sized = has_capacity;
  73. struct node
  74. {
  75. node(T const & val):
  76. v(val)
  77. {}
  78. typedef typename detail::select_tagged_handle<node, node_based>::handle_type handle_t;
  79. handle_t next;
  80. const T v;
  81. };
  82. typedef typename detail::extract_allocator<bound_args, node>::type node_allocator;
  83. typedef typename detail::select_freelist<node, node_allocator, compile_time_sized, fixed_sized, capacity>::type pool_t;
  84. typedef typename pool_t::tagged_node_handle tagged_node_handle;
  85. // check compile-time capacity
  86. BOOST_STATIC_ASSERT((mpl::if_c<has_capacity,
  87. mpl::bool_<capacity - 1 < boost::integer_traits<boost::uint16_t>::const_max>,
  88. mpl::true_
  89. >::type::value));
  90. struct implementation_defined
  91. {
  92. typedef node_allocator allocator;
  93. typedef std::size_t size_type;
  94. };
  95. #endif
  96. BOOST_DELETED_FUNCTION(stack(stack const&))
  97. BOOST_DELETED_FUNCTION(stack& operator= (stack const&))
  98. public:
  99. typedef T value_type;
  100. typedef typename implementation_defined::allocator allocator;
  101. typedef typename implementation_defined::size_type size_type;
  102. /**
  103. * \return true, if implementation is lock-free.
  104. *
  105. * \warning It only checks, if the top stack node and the freelist can be modified in a lock-free manner.
  106. * On most platforms, the whole implementation is lock-free, if this is true. Using c++0x-style atomics,
  107. * there is no possibility to provide a completely accurate implementation, because one would need to test
  108. * every internal node, which is impossible if further nodes will be allocated from the operating system.
  109. *
  110. * */
  111. bool is_lock_free (void) const
  112. {
  113. return tos.is_lock_free() && pool.is_lock_free();
  114. }
  115. //! Construct stack
  116. // @{
  117. stack(void):
  118. pool(node_allocator(), capacity)
  119. {
  120. BOOST_ASSERT(has_capacity);
  121. initialize();
  122. }
  123. template <typename U>
  124. explicit stack(typename node_allocator::template rebind<U>::other const & alloc):
  125. pool(alloc, capacity)
  126. {
  127. BOOST_STATIC_ASSERT(has_capacity);
  128. initialize();
  129. }
  130. explicit stack(allocator const & alloc):
  131. pool(alloc, capacity)
  132. {
  133. BOOST_ASSERT(has_capacity);
  134. initialize();
  135. }
  136. // @}
  137. //! Construct stack, allocate n nodes for the freelist.
  138. // @{
  139. explicit stack(size_type n):
  140. pool(node_allocator(), n)
  141. {
  142. BOOST_ASSERT(!has_capacity);
  143. initialize();
  144. }
  145. template <typename U>
  146. stack(size_type n, typename node_allocator::template rebind<U>::other const & alloc):
  147. pool(alloc, n)
  148. {
  149. BOOST_STATIC_ASSERT(!has_capacity);
  150. initialize();
  151. }
  152. // @}
  153. /** Allocate n nodes for freelist
  154. *
  155. * \pre only valid if no capacity<> argument given
  156. * \note thread-safe, may block if memory allocator blocks
  157. *
  158. * */
  159. void reserve(size_type n)
  160. {
  161. BOOST_STATIC_ASSERT(!has_capacity);
  162. pool.template reserve<true>(n);
  163. }
  164. /** Allocate n nodes for freelist
  165. *
  166. * \pre only valid if no capacity<> argument given
  167. * \note not thread-safe, may block if memory allocator blocks
  168. *
  169. * */
  170. void reserve_unsafe(size_type n)
  171. {
  172. BOOST_STATIC_ASSERT(!has_capacity);
  173. pool.template reserve<false>(n);
  174. }
  175. /** Destroys stack, free all nodes from freelist.
  176. *
  177. * \note not thread-safe
  178. *
  179. * */
  180. ~stack(void)
  181. {
  182. T dummy;
  183. while(unsynchronized_pop(dummy))
  184. {}
  185. }
  186. private:
  187. #ifndef BOOST_DOXYGEN_INVOKED
  188. void initialize(void)
  189. {
  190. tos.store(tagged_node_handle(pool.null_handle(), 0));
  191. }
  192. void link_nodes_atomic(node * new_top_node, node * end_node)
  193. {
  194. tagged_node_handle old_tos = tos.load(detail::memory_order_relaxed);
  195. for (;;) {
  196. tagged_node_handle new_tos (pool.get_handle(new_top_node), old_tos.get_tag());
  197. end_node->next = pool.get_handle(old_tos);
  198. if (tos.compare_exchange_weak(old_tos, new_tos))
  199. break;
  200. }
  201. }
  202. void link_nodes_unsafe(node * new_top_node, node * end_node)
  203. {
  204. tagged_node_handle old_tos = tos.load(detail::memory_order_relaxed);
  205. tagged_node_handle new_tos (pool.get_handle(new_top_node), old_tos.get_tag());
  206. end_node->next = pool.get_pointer(old_tos);
  207. tos.store(new_tos, memory_order_relaxed);
  208. }
  209. template <bool Threadsafe, bool Bounded, typename ConstIterator>
  210. tuple<node*, node*> prepare_node_list(ConstIterator begin, ConstIterator end, ConstIterator & ret)
  211. {
  212. ConstIterator it = begin;
  213. node * end_node = pool.template construct<Threadsafe, Bounded>(*it++);
  214. if (end_node == NULL) {
  215. ret = begin;
  216. return make_tuple<node*, node*>(NULL, NULL);
  217. }
  218. node * new_top_node = end_node;
  219. end_node->next = NULL;
  220. try {
  221. /* link nodes */
  222. for (; it != end; ++it) {
  223. node * newnode = pool.template construct<Threadsafe, Bounded>(*it);
  224. if (newnode == NULL)
  225. break;
  226. newnode->next = new_top_node;
  227. new_top_node = newnode;
  228. }
  229. } catch (...) {
  230. for (node * current_node = new_top_node; current_node != NULL;) {
  231. node * next = current_node->next;
  232. pool.template destruct<Threadsafe>(current_node);
  233. current_node = next;
  234. }
  235. throw;
  236. }
  237. ret = it;
  238. return make_tuple(new_top_node, end_node);
  239. }
  240. #endif
  241. public:
  242. /** Pushes object t to the stack.
  243. *
  244. * \post object will be pushed to the stack, if internal node can be allocated
  245. * \returns true, if the push operation is successful.
  246. *
  247. * \note Thread-safe. If internal memory pool is exhausted and the memory pool is not fixed-sized, a new node will be allocated
  248. * from the OS. This may not be lock-free.
  249. * \throws if memory allocator throws
  250. * */
  251. bool push(T const & v)
  252. {
  253. return do_push<false>(v);
  254. }
  255. /** Pushes object t to the stack.
  256. *
  257. * \post object will be pushed to the stack, if internal node can be allocated
  258. * \returns true, if the push operation is successful.
  259. *
  260. * \note Thread-safe and non-blocking. If internal memory pool is exhausted, the push operation will fail
  261. * */
  262. bool bounded_push(T const & v)
  263. {
  264. return do_push<true>(v);
  265. }
  266. #ifndef BOOST_DOXYGEN_INVOKED
  267. private:
  268. template <bool Bounded>
  269. bool do_push(T const & v)
  270. {
  271. node * newnode = pool.template construct<true, Bounded>(v);
  272. if (newnode == 0)
  273. return false;
  274. link_nodes_atomic(newnode, newnode);
  275. return true;
  276. }
  277. template <bool Bounded, typename ConstIterator>
  278. ConstIterator do_push(ConstIterator begin, ConstIterator end)
  279. {
  280. node * new_top_node;
  281. node * end_node;
  282. ConstIterator ret;
  283. tie(new_top_node, end_node) = prepare_node_list<true, Bounded>(begin, end, ret);
  284. if (new_top_node)
  285. link_nodes_atomic(new_top_node, end_node);
  286. return ret;
  287. }
  288. public:
  289. #endif
  290. /** Pushes as many objects from the range [begin, end) as freelist node can be allocated.
  291. *
  292. * \return iterator to the first element, which has not been pushed
  293. *
  294. * \note Operation is applied atomically
  295. * \note Thread-safe. If internal memory pool is exhausted and the memory pool is not fixed-sized, a new node will be allocated
  296. * from the OS. This may not be lock-free.
  297. * \throws if memory allocator throws
  298. */
  299. template <typename ConstIterator>
  300. ConstIterator push(ConstIterator begin, ConstIterator end)
  301. {
  302. return do_push<false, ConstIterator>(begin, end);
  303. }
  304. /** Pushes as many objects from the range [begin, end) as freelist node can be allocated.
  305. *
  306. * \return iterator to the first element, which has not been pushed
  307. *
  308. * \note Operation is applied atomically
  309. * \note Thread-safe and non-blocking. If internal memory pool is exhausted, the push operation will fail
  310. * \throws if memory allocator throws
  311. */
  312. template <typename ConstIterator>
  313. ConstIterator bounded_push(ConstIterator begin, ConstIterator end)
  314. {
  315. return do_push<true, ConstIterator>(begin, end);
  316. }
  317. /** Pushes object t to the stack.
  318. *
  319. * \post object will be pushed to the stack, if internal node can be allocated
  320. * \returns true, if the push operation is successful.
  321. *
  322. * \note Not thread-safe. If internal memory pool is exhausted and the memory pool is not fixed-sized, a new node will be allocated
  323. * from the OS. This may not be lock-free.
  324. * \throws if memory allocator throws
  325. * */
  326. bool unsynchronized_push(T const & v)
  327. {
  328. node * newnode = pool.template construct<false, false>(v);
  329. if (newnode == 0)
  330. return false;
  331. link_nodes_unsafe(newnode, newnode);
  332. return true;
  333. }
  334. /** Pushes as many objects from the range [begin, end) as freelist node can be allocated.
  335. *
  336. * \return iterator to the first element, which has not been pushed
  337. *
  338. * \note Not thread-safe. If internal memory pool is exhausted and the memory pool is not fixed-sized, a new node will be allocated
  339. * from the OS. This may not be lock-free.
  340. * \throws if memory allocator throws
  341. */
  342. template <typename ConstIterator>
  343. ConstIterator unsynchronized_push(ConstIterator begin, ConstIterator end)
  344. {
  345. node * new_top_node;
  346. node * end_node;
  347. ConstIterator ret;
  348. tie(new_top_node, end_node) = prepare_node_list<false, false>(begin, end, ret);
  349. if (new_top_node)
  350. link_nodes_unsafe(new_top_node, end_node);
  351. return ret;
  352. }
  353. /** Pops object from stack.
  354. *
  355. * \post if pop operation is successful, object will be copied to ret.
  356. * \returns true, if the pop operation is successful, false if stack was empty.
  357. *
  358. * \note Thread-safe and non-blocking
  359. *
  360. * */
  361. bool pop(T & ret)
  362. {
  363. return pop<T>(ret);
  364. }
  365. /** Pops object from stack.
  366. *
  367. * \pre type T must be convertible to U
  368. * \post if pop operation is successful, object will be copied to ret.
  369. * \returns true, if the pop operation is successful, false if stack was empty.
  370. *
  371. * \note Thread-safe and non-blocking
  372. *
  373. * */
  374. template <typename U>
  375. bool pop(U & ret)
  376. {
  377. BOOST_STATIC_ASSERT((boost::is_convertible<T, U>::value));
  378. detail::consume_via_copy<U> consumer(ret);
  379. return consume_one(consumer);
  380. }
  381. /** Pops object from stack.
  382. *
  383. * \post if pop operation is successful, object will be copied to ret.
  384. * \returns true, if the pop operation is successful, false if stack was empty.
  385. *
  386. * \note Not thread-safe, but non-blocking
  387. *
  388. * */
  389. bool unsynchronized_pop(T & ret)
  390. {
  391. return unsynchronized_pop<T>(ret);
  392. }
  393. /** Pops object from stack.
  394. *
  395. * \pre type T must be convertible to U
  396. * \post if pop operation is successful, object will be copied to ret.
  397. * \returns true, if the pop operation is successful, false if stack was empty.
  398. *
  399. * \note Not thread-safe, but non-blocking
  400. *
  401. * */
  402. template <typename U>
  403. bool unsynchronized_pop(U & ret)
  404. {
  405. BOOST_STATIC_ASSERT((boost::is_convertible<T, U>::value));
  406. tagged_node_handle old_tos = tos.load(detail::memory_order_relaxed);
  407. node * old_tos_pointer = pool.get_pointer(old_tos);
  408. if (!pool.get_pointer(old_tos))
  409. return false;
  410. node * new_tos_ptr = pool.get_pointer(old_tos_pointer->next);
  411. tagged_node_handle new_tos(pool.get_handle(new_tos_ptr), old_tos.get_next_tag());
  412. tos.store(new_tos, memory_order_relaxed);
  413. detail::copy_payload(old_tos_pointer->v, ret);
  414. pool.template destruct<false>(old_tos);
  415. return true;
  416. }
  417. /** consumes one element via a functor
  418. *
  419. * pops one element from the stack and applies the functor on this object
  420. *
  421. * \returns true, if one element was consumed
  422. *
  423. * \note Thread-safe and non-blocking, if functor is thread-safe and non-blocking
  424. * */
  425. template <typename Functor>
  426. bool consume_one(Functor & f)
  427. {
  428. tagged_node_handle old_tos = tos.load(detail::memory_order_consume);
  429. for (;;) {
  430. node * old_tos_pointer = pool.get_pointer(old_tos);
  431. if (!old_tos_pointer)
  432. return false;
  433. tagged_node_handle new_tos(old_tos_pointer->next, old_tos.get_next_tag());
  434. if (tos.compare_exchange_weak(old_tos, new_tos)) {
  435. f(old_tos_pointer->v);
  436. pool.template destruct<true>(old_tos);
  437. return true;
  438. }
  439. }
  440. }
  441. /// \copydoc boost::lockfree::stack::consume_one(Functor & rhs)
  442. template <typename Functor>
  443. bool consume_one(Functor const & f)
  444. {
  445. tagged_node_handle old_tos = tos.load(detail::memory_order_consume);
  446. for (;;) {
  447. node * old_tos_pointer = pool.get_pointer(old_tos);
  448. if (!old_tos_pointer)
  449. return false;
  450. tagged_node_handle new_tos(old_tos_pointer->next, old_tos.get_next_tag());
  451. if (tos.compare_exchange_weak(old_tos, new_tos)) {
  452. f(old_tos_pointer->v);
  453. pool.template destruct<true>(old_tos);
  454. return true;
  455. }
  456. }
  457. }
  458. /** consumes all elements via a functor
  459. *
  460. * sequentially pops all elements from the stack and applies the functor on each object
  461. *
  462. * \returns number of elements that are consumed
  463. *
  464. * \note Thread-safe and non-blocking, if functor is thread-safe and non-blocking
  465. * */
  466. template <typename Functor>
  467. size_t consume_all(Functor & f)
  468. {
  469. size_t element_count = 0;
  470. while (consume_one(f))
  471. element_count += 1;
  472. return element_count;
  473. }
  474. /// \copydoc boost::lockfree::stack::consume_all(Functor & rhs)
  475. template <typename Functor>
  476. size_t consume_all(Functor const & f)
  477. {
  478. size_t element_count = 0;
  479. while (consume_one(f))
  480. element_count += 1;
  481. return element_count;
  482. }
  483. /**
  484. * \return true, if stack is empty.
  485. *
  486. * \note It only guarantees that at some point during the execution of the function the stack has been empty.
  487. * It is rarely practical to use this value in program logic, because the stack can be modified by other threads.
  488. * */
  489. bool empty(void) const
  490. {
  491. return pool.get_pointer(tos.load()) == NULL;
  492. }
  493. private:
  494. #ifndef BOOST_DOXYGEN_INVOKED
  495. detail::atomic<tagged_node_handle> tos;
  496. static const int padding_size = BOOST_LOCKFREE_CACHELINE_BYTES - sizeof(tagged_node_handle);
  497. char padding[padding_size];
  498. pool_t pool;
  499. #endif
  500. };
  501. } /* namespace lockfree */
  502. } /* namespace boost */
  503. #endif /* BOOST_LOCKFREE_STACK_HPP_INCLUDED */