consuming_buffers.hpp 7.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294
  1. //
  2. // detail/consuming_buffers.hpp
  3. // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~
  4. //
  5. // Copyright (c) 2003-2015 Christopher M. Kohlhoff (chris at kohlhoff dot com)
  6. //
  7. // Distributed under the Boost Software License, Version 1.0. (See accompanying
  8. // file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
  9. //
  10. #ifndef BOOST_ASIO_DETAIL_CONSUMING_BUFFERS_HPP
  11. #define BOOST_ASIO_DETAIL_CONSUMING_BUFFERS_HPP
  12. #if defined(_MSC_VER) && (_MSC_VER >= 1200)
  13. # pragma once
  14. #endif // defined(_MSC_VER) && (_MSC_VER >= 1200)
  15. #include <boost/asio/detail/config.hpp>
  16. #include <cstddef>
  17. #include <iterator>
  18. #include <boost/asio/buffer.hpp>
  19. #include <boost/asio/detail/limits.hpp>
  20. #include <boost/asio/detail/push_options.hpp>
  21. namespace boost {
  22. namespace asio {
  23. namespace detail {
  24. // A proxy iterator for a sub-range in a list of buffers.
  25. template <typename Buffer, typename Buffer_Iterator>
  26. class consuming_buffers_iterator
  27. {
  28. public:
  29. /// The type used for the distance between two iterators.
  30. typedef std::ptrdiff_t difference_type;
  31. /// The type of the value pointed to by the iterator.
  32. typedef Buffer value_type;
  33. /// The type of the result of applying operator->() to the iterator.
  34. typedef const Buffer* pointer;
  35. /// The type of the result of applying operator*() to the iterator.
  36. typedef const Buffer& reference;
  37. /// The iterator category.
  38. typedef std::forward_iterator_tag iterator_category;
  39. // Default constructor creates an end iterator.
  40. consuming_buffers_iterator()
  41. : at_end_(true)
  42. {
  43. }
  44. // Construct with a buffer for the first entry and an iterator
  45. // range for the remaining entries.
  46. consuming_buffers_iterator(bool at_end, const Buffer& first,
  47. Buffer_Iterator begin_remainder, Buffer_Iterator end_remainder,
  48. std::size_t max_size)
  49. : at_end_(max_size > 0 ? at_end : true),
  50. first_(buffer(first, max_size)),
  51. begin_remainder_(begin_remainder),
  52. end_remainder_(end_remainder),
  53. offset_(0),
  54. max_size_(max_size)
  55. {
  56. }
  57. // Dereference an iterator.
  58. const Buffer& operator*() const
  59. {
  60. return dereference();
  61. }
  62. // Dereference an iterator.
  63. const Buffer* operator->() const
  64. {
  65. return &dereference();
  66. }
  67. // Increment operator (prefix).
  68. consuming_buffers_iterator& operator++()
  69. {
  70. increment();
  71. return *this;
  72. }
  73. // Increment operator (postfix).
  74. consuming_buffers_iterator operator++(int)
  75. {
  76. consuming_buffers_iterator tmp(*this);
  77. ++*this;
  78. return tmp;
  79. }
  80. // Test two iterators for equality.
  81. friend bool operator==(const consuming_buffers_iterator& a,
  82. const consuming_buffers_iterator& b)
  83. {
  84. return a.equal(b);
  85. }
  86. // Test two iterators for inequality.
  87. friend bool operator!=(const consuming_buffers_iterator& a,
  88. const consuming_buffers_iterator& b)
  89. {
  90. return !a.equal(b);
  91. }
  92. private:
  93. void increment()
  94. {
  95. if (!at_end_)
  96. {
  97. if (begin_remainder_ == end_remainder_
  98. || offset_ + buffer_size(first_) >= max_size_)
  99. {
  100. at_end_ = true;
  101. }
  102. else
  103. {
  104. offset_ += buffer_size(first_);
  105. first_ = buffer(*begin_remainder_++, max_size_ - offset_);
  106. }
  107. }
  108. }
  109. bool equal(const consuming_buffers_iterator& other) const
  110. {
  111. if (at_end_ && other.at_end_)
  112. return true;
  113. return !at_end_ && !other.at_end_
  114. && buffer_cast<const void*>(first_)
  115. == buffer_cast<const void*>(other.first_)
  116. && buffer_size(first_) == buffer_size(other.first_)
  117. && begin_remainder_ == other.begin_remainder_
  118. && end_remainder_ == other.end_remainder_;
  119. }
  120. const Buffer& dereference() const
  121. {
  122. return first_;
  123. }
  124. bool at_end_;
  125. Buffer first_;
  126. Buffer_Iterator begin_remainder_;
  127. Buffer_Iterator end_remainder_;
  128. std::size_t offset_;
  129. std::size_t max_size_;
  130. };
  131. // A proxy for a sub-range in a list of buffers.
  132. template <typename Buffer, typename Buffers>
  133. class consuming_buffers
  134. {
  135. public:
  136. // The type for each element in the list of buffers.
  137. typedef Buffer value_type;
  138. // A forward-only iterator type that may be used to read elements.
  139. typedef consuming_buffers_iterator<Buffer, typename Buffers::const_iterator>
  140. const_iterator;
  141. // Construct to represent the entire list of buffers.
  142. consuming_buffers(const Buffers& buffers)
  143. : buffers_(buffers),
  144. at_end_(buffers_.begin() == buffers_.end()),
  145. begin_remainder_(buffers_.begin()),
  146. max_size_((std::numeric_limits<std::size_t>::max)())
  147. {
  148. if (!at_end_)
  149. {
  150. first_ = *buffers_.begin();
  151. ++begin_remainder_;
  152. }
  153. }
  154. // Copy constructor.
  155. consuming_buffers(const consuming_buffers& other)
  156. : buffers_(other.buffers_),
  157. at_end_(other.at_end_),
  158. first_(other.first_),
  159. begin_remainder_(buffers_.begin()),
  160. max_size_(other.max_size_)
  161. {
  162. typename Buffers::const_iterator first = other.buffers_.begin();
  163. typename Buffers::const_iterator second = other.begin_remainder_;
  164. std::advance(begin_remainder_, std::distance(first, second));
  165. }
  166. // Assignment operator.
  167. consuming_buffers& operator=(const consuming_buffers& other)
  168. {
  169. buffers_ = other.buffers_;
  170. at_end_ = other.at_end_;
  171. first_ = other.first_;
  172. begin_remainder_ = buffers_.begin();
  173. typename Buffers::const_iterator first = other.buffers_.begin();
  174. typename Buffers::const_iterator second = other.begin_remainder_;
  175. std::advance(begin_remainder_, std::distance(first, second));
  176. max_size_ = other.max_size_;
  177. return *this;
  178. }
  179. // Get a forward-only iterator to the first element.
  180. const_iterator begin() const
  181. {
  182. return const_iterator(at_end_, first_,
  183. begin_remainder_, buffers_.end(), max_size_);
  184. }
  185. // Get a forward-only iterator for one past the last element.
  186. const_iterator end() const
  187. {
  188. return const_iterator();
  189. }
  190. // Set the maximum size for a single transfer.
  191. void prepare(std::size_t max_size)
  192. {
  193. max_size_ = max_size;
  194. }
  195. // Consume the specified number of bytes from the buffers.
  196. void consume(std::size_t size)
  197. {
  198. // Remove buffers from the start until the specified size is reached.
  199. while (size > 0 && !at_end_)
  200. {
  201. if (buffer_size(first_) <= size)
  202. {
  203. size -= buffer_size(first_);
  204. if (begin_remainder_ == buffers_.end())
  205. at_end_ = true;
  206. else
  207. first_ = *begin_remainder_++;
  208. }
  209. else
  210. {
  211. first_ = first_ + size;
  212. size = 0;
  213. }
  214. }
  215. // Remove any more empty buffers at the start.
  216. while (!at_end_ && buffer_size(first_) == 0)
  217. {
  218. if (begin_remainder_ == buffers_.end())
  219. at_end_ = true;
  220. else
  221. first_ = *begin_remainder_++;
  222. }
  223. }
  224. private:
  225. Buffers buffers_;
  226. bool at_end_;
  227. Buffer first_;
  228. typename Buffers::const_iterator begin_remainder_;
  229. std::size_t max_size_;
  230. };
  231. // Specialisation for null_buffers to ensure that the null_buffers type is
  232. // always passed through to the underlying read or write operation.
  233. template <typename Buffer>
  234. class consuming_buffers<Buffer, boost::asio::null_buffers>
  235. : public boost::asio::null_buffers
  236. {
  237. public:
  238. consuming_buffers(const boost::asio::null_buffers&)
  239. {
  240. // No-op.
  241. }
  242. void prepare(std::size_t)
  243. {
  244. // No-op.
  245. }
  246. void consume(std::size_t)
  247. {
  248. // No-op.
  249. }
  250. };
  251. } // namespace detail
  252. } // namespace asio
  253. } // namespace boost
  254. #include <boost/asio/detail/pop_options.hpp>
  255. #endif // BOOST_ASIO_DETAIL_CONSUMING_BUFFERS_HPP