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