// Helper classes and functions for the circular buffer. // Copyright (c) 2003-2008 Jan Gaspar // Copyright 2014,2018 Glen Joseph Fernandes // (glenjofe@gmail.com) // Use, modification, and distribution is subject to the Boost Software // License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at // http://www.boost.org/LICENSE_1_0.txt) #if !defined(BOOST_CIRCULAR_BUFFER_DETAILS_HPP) #define BOOST_CIRCULAR_BUFFER_DETAILS_HPP #if defined(_MSC_VER) #pragma once #endif #include #include #include #include #include #include #include // Silence MS /W4 warnings like C4913: // "user defined binary operator ',' exists but no overload could convert all operands, default built-in binary operator ',' used" // This might happen when previously including some boost headers that overload the coma operator. #if defined(_MSC_VER) # pragma warning(push) # pragma warning(disable:4913) #endif namespace boost { namespace cb_details { template struct nonconst_traits; template void uninitialized_fill_n_with_alloc( ForwardIterator first, Diff n, const T& item, Alloc& alloc); template ForwardIterator uninitialized_copy(InputIterator first, InputIterator last, ForwardIterator dest, Alloc& a); template ForwardIterator uninitialized_move_if_noexcept(InputIterator first, InputIterator last, ForwardIterator dest, Alloc& a); /*! \struct const_traits \brief Defines the data types for a const iterator. */ template struct const_traits { // Basic types typedef typename Alloc::value_type value_type; typedef typename boost::allocator_const_pointer::type pointer; typedef const value_type& reference; typedef typename boost::allocator_size_type::type size_type; typedef typename boost::allocator_difference_type::type difference_type; // Non-const traits typedef nonconst_traits nonconst_self; }; /*! \struct nonconst_traits \brief Defines the data types for a non-const iterator. */ template struct nonconst_traits { // Basic types typedef typename Alloc::value_type value_type; typedef typename boost::allocator_pointer::type pointer; typedef value_type& reference; typedef typename boost::allocator_size_type::type size_type; typedef typename boost::allocator_difference_type::type difference_type; // Non-const traits typedef nonconst_traits nonconst_self; }; /*! \struct iterator_wrapper \brief Helper iterator dereference wrapper. */ template struct iterator_wrapper { mutable Iterator m_it; explicit iterator_wrapper(Iterator it) : m_it(it) {} Iterator operator () () const { return m_it++; } private: iterator_wrapper& operator = (const iterator_wrapper&); // do not generate }; /*! \struct item_wrapper \brief Helper item dereference wrapper. */ template struct item_wrapper { Value m_item; explicit item_wrapper(Value item) : m_item(item) {} Pointer operator () () const { return &m_item; } private: item_wrapper& operator = (const item_wrapper&); // do not generate }; /*! \struct assign_n \brief Helper functor for assigning n items. */ template struct assign_n { typedef typename boost::allocator_size_type::type size_type; size_type m_n; Value m_item; Alloc& m_alloc; assign_n(size_type n, Value item, Alloc& alloc) : m_n(n), m_item(item), m_alloc(alloc) {} template void operator () (Pointer p) const { uninitialized_fill_n_with_alloc(p, m_n, m_item, m_alloc); } private: assign_n& operator = (const assign_n&); // do not generate }; /*! \struct assign_range \brief Helper functor for assigning range of items. */ template struct assign_range { Iterator m_first; Iterator m_last; Alloc& m_alloc; assign_range(const Iterator& first, const Iterator& last, Alloc& alloc) : m_first(first), m_last(last), m_alloc(alloc) {} template void operator () (Pointer p) const { boost::cb_details::uninitialized_copy(m_first, m_last, p, m_alloc); } }; template inline assign_range make_assign_range(const Iterator& first, const Iterator& last, Alloc& a) { return assign_range(first, last, a); } /*! \class capacity_control \brief Capacity controller of the space optimized circular buffer. */ template class capacity_control { //! The capacity of the space-optimized circular buffer. Size m_capacity; //! The lowest guaranteed or minimum capacity of the adapted space-optimized circular buffer. Size m_min_capacity; public: //! Constructor. capacity_control(Size buffer_capacity, Size min_buffer_capacity = 0) : m_capacity(buffer_capacity), m_min_capacity(min_buffer_capacity) { // Check for capacity lower than min_capacity. BOOST_CB_ASSERT(buffer_capacity >= min_buffer_capacity); } // Default copy constructor. // Default assign operator. //! Get the capacity of the space optimized circular buffer. Size capacity() const { return m_capacity; } //! Get the minimal capacity of the space optimized circular buffer. Size min_capacity() const { return m_min_capacity; } //! Size operator - returns the capacity of the space optimized circular buffer. operator Size() const { return m_capacity; } }; /*! \struct iterator \brief Random access iterator for the circular buffer. \param Buff The type of the underlying circular buffer. \param Traits Basic iterator types. \note This iterator is not circular. It was designed for iterating from begin() to end() of the circular buffer. */ template struct iterator #if BOOST_CB_ENABLE_DEBUG : public debug_iterator_base #endif // #if BOOST_CB_ENABLE_DEBUG { // Helper types //! Non-const iterator. typedef iterator nonconst_self; // Basic types typedef std::random_access_iterator_tag iterator_category; //! The type of the elements stored in the circular buffer. typedef typename Traits::value_type value_type; //! Pointer to the element. typedef typename Traits::pointer pointer; //! Reference to the element. typedef typename Traits::reference reference; //! Size type. typedef typename Traits::size_type size_type; //! Difference type. typedef typename Traits::difference_type difference_type; // Member variables //! The circular buffer where the iterator points to. const Buff* m_buff; //! An internal iterator. pointer m_it; // Construction & assignment // Default copy constructor. //! Default constructor. iterator() : m_buff(0), m_it(0) {} #if BOOST_CB_ENABLE_DEBUG //! Copy constructor (used for converting from a non-const to a const iterator). iterator(const nonconst_self& it) : debug_iterator_base(it), m_buff(it.m_buff), m_it(it.m_it) {} //! Internal constructor. /*! \note This constructor is not intended to be used directly by the user. */ iterator(const Buff* cb, const pointer p) : debug_iterator_base(cb), m_buff(cb), m_it(p) {} #else iterator(const nonconst_self& it) : m_buff(it.m_buff), m_it(it.m_it) {} iterator(const Buff* cb, const pointer p) : m_buff(cb), m_it(p) {} #endif // #if BOOST_CB_ENABLE_DEBUG //! Assign operator. #if !defined(BOOST_NO_CXX11_DEFAULTED_FUNCTIONS) iterator& operator=(const iterator&) = default; #else iterator& operator=(const iterator& it) { if (this == &it) return *this; #if BOOST_CB_ENABLE_DEBUG debug_iterator_base::operator =(it); #endif // #if BOOST_CB_ENABLE_DEBUG m_buff = it.m_buff; m_it = it.m_it; return *this; } #endif // Random access iterator methods //! Dereferencing operator. reference operator * () const { BOOST_CB_ASSERT(is_valid(m_buff)); // check for uninitialized or invalidated iterator BOOST_CB_ASSERT(m_it != 0); // check for iterator pointing to end() return *m_it; } //! Dereferencing operator. pointer operator -> () const { return &(operator*()); } //! Difference operator. template difference_type operator - (const iterator& it) const { BOOST_CB_ASSERT(is_valid(m_buff)); // check for uninitialized or invalidated iterator BOOST_CB_ASSERT(it.is_valid(m_buff)); // check for uninitialized or invalidated iterator return linearize_pointer(*this) - linearize_pointer(it); } //! Increment operator (prefix). iterator& operator ++ () { BOOST_CB_ASSERT(is_valid(m_buff)); // check for uninitialized or invalidated iterator BOOST_CB_ASSERT(m_it != 0); // check for iterator pointing to end() m_buff->increment(m_it); if (m_it == m_buff->m_last) m_it = 0; return *this; } //! Increment operator (postfix). iterator operator ++ (int) { iterator tmp = *this; ++*this; return tmp; } //! Decrement operator (prefix). iterator& operator -- () { BOOST_CB_ASSERT(is_valid(m_buff)); // check for uninitialized or invalidated iterator BOOST_CB_ASSERT(m_it != m_buff->m_first); // check for iterator pointing to begin() if (m_it == 0) m_it = m_buff->m_last; m_buff->decrement(m_it); return *this; } //! Decrement operator (postfix). iterator operator -- (int) { iterator tmp = *this; --*this; return tmp; } //! Iterator addition. iterator& operator += (difference_type n) { BOOST_CB_ASSERT(is_valid(m_buff)); // check for uninitialized or invalidated iterator if (n > 0) { BOOST_CB_ASSERT(m_buff->end() - *this >= n); // check for too large n m_it = m_buff->add(m_it, n); if (m_it == m_buff->m_last) m_it = 0; } else if (n < 0) { *this -= -n; } return *this; } //! Iterator addition. iterator operator + (difference_type n) const { return iterator(*this) += n; } //! Iterator subtraction. iterator& operator -= (difference_type n) { BOOST_CB_ASSERT(is_valid(m_buff)); // check for uninitialized or invalidated iterator if (n > 0) { BOOST_CB_ASSERT(*this - m_buff->begin() >= n); // check for too large n m_it = m_buff->sub(m_it == 0 ? m_buff->m_last : m_it, n); } else if (n < 0) { *this += -n; } return *this; } //! Iterator subtraction. iterator operator - (difference_type n) const { return iterator(*this) -= n; } //! Element access operator. reference operator [] (difference_type n) const { return *(*this + n); } // Equality & comparison //! Equality. template bool operator == (const iterator& it) const { BOOST_CB_ASSERT(is_valid(m_buff)); // check for uninitialized or invalidated iterator BOOST_CB_ASSERT(it.is_valid(m_buff)); // check for uninitialized or invalidated iterator return m_it == it.m_it; } //! Inequality. template bool operator != (const iterator& it) const { BOOST_CB_ASSERT(is_valid(m_buff)); // check for uninitialized or invalidated iterator BOOST_CB_ASSERT(it.is_valid(m_buff)); // check for uninitialized or invalidated iterator return m_it != it.m_it; } //! Less. template bool operator < (const iterator& it) const { BOOST_CB_ASSERT(is_valid(m_buff)); // check for uninitialized or invalidated iterator BOOST_CB_ASSERT(it.is_valid(m_buff)); // check for uninitialized or invalidated iterator return linearize_pointer(*this) < linearize_pointer(it); } //! Greater. template bool operator > (const iterator& it) const { return it < *this; } //! Less or equal. template bool operator <= (const iterator& it) const { return !(it < *this); } //! Greater or equal. template bool operator >= (const iterator& it) const { return !(*this < it); } // Helpers //! Get a pointer which would point to the same element as the iterator in case the circular buffer is linearized. template typename Traits0::pointer linearize_pointer(const iterator& it) const { return it.m_it == 0 ? m_buff->m_buff + m_buff->size() : (it.m_it < m_buff->m_first ? it.m_it + (m_buff->m_end - m_buff->m_first) : m_buff->m_buff + (it.m_it - m_buff->m_first)); } }; //! Iterator addition. template inline iterator operator + (typename Traits::difference_type n, const iterator& it) { return it + n; } /*! \fn ForwardIterator uninitialized_copy(InputIterator first, InputIterator last, ForwardIterator dest) \brief Equivalent of std::uninitialized_copy but with explicit specification of value type. */ template inline ForwardIterator uninitialized_copy(InputIterator first, InputIterator last, ForwardIterator dest, Alloc& a) { ForwardIterator next = dest; BOOST_TRY { for (; first != last; ++first, ++dest) boost::allocator_construct(a, boost::to_address(dest), *first); } BOOST_CATCH(...) { for (; next != dest; ++next) boost::allocator_destroy(a, boost::to_address(next)); BOOST_RETHROW } BOOST_CATCH_END return dest; } template ForwardIterator uninitialized_move_if_noexcept_impl(InputIterator first, InputIterator last, ForwardIterator dest, Alloc& a, true_type) { for (; first != last; ++first, ++dest) boost::allocator_construct(a, boost::to_address(dest), boost::move(*first)); return dest; } template ForwardIterator uninitialized_move_if_noexcept_impl(InputIterator first, InputIterator last, ForwardIterator dest, Alloc& a, false_type) { return uninitialized_copy(first, last, dest, a); } /*! \fn ForwardIterator uninitialized_move_if_noexcept(InputIterator first, InputIterator last, ForwardIterator dest) \brief Equivalent of std::uninitialized_copy but with explicit specification of value type and moves elements if they have noexcept move constructors. */ template ForwardIterator uninitialized_move_if_noexcept(InputIterator first, InputIterator last, ForwardIterator dest, Alloc& a) { typedef typename boost::is_nothrow_move_constructible::type tag_t; return uninitialized_move_if_noexcept_impl(first, last, dest, a, tag_t()); } /*! \fn void uninitialized_fill_n_with_alloc(ForwardIterator first, Diff n, const T& item, Alloc& alloc) \brief Equivalent of std::uninitialized_fill_n with allocator. */ template inline void uninitialized_fill_n_with_alloc(ForwardIterator first, Diff n, const T& item, Alloc& alloc) { ForwardIterator next = first; BOOST_TRY { for (; n > 0; ++first, --n) boost::allocator_construct(alloc, boost::to_address(first), item); } BOOST_CATCH(...) { for (; next != first; ++next) boost::allocator_destroy(alloc, boost::to_address(next)); BOOST_RETHROW } BOOST_CATCH_END } } // namespace cb_details } // namespace boost #if defined(_MSC_VER) # pragma warning(pop) #endif #endif // #if !defined(BOOST_CIRCULAR_BUFFER_DETAILS_HPP)