diff --git a/src/buffers.tex b/src/buffers.tex index 837920f..b8bfaf2 100644 --- a/src/buffers.tex +++ b/src/buffers.tex @@ -174,14 +174,14 @@ error_code& ec); template - size_t read(SyncReadStream& stream, DynamicBuffer&& b); + size_t read(SyncReadStream& stream, DynamicBuffer b); template - size_t read(SyncReadStream& stream, DynamicBuffer&& b, error_code& ec); + size_t read(SyncReadStream& stream, DynamicBuffer b, error_code& ec); template - size_t read(SyncReadStream& stream, DynamicBuffer&& b, + size_t read(SyncReadStream& stream, DynamicBuffer b, CompletionCondition completion_condition); template - size_t read(SyncReadStream& stream, DynamicBuffer&& b, + size_t read(SyncReadStream& stream, DynamicBuffer b, CompletionCondition completion_condition, error_code& ec); // \ref{buffer.async.read}, asynchronous read operations @@ -200,11 +200,11 @@ template @\DEDUCED@ async_read(AsyncReadStream& stream, - DynamicBuffer&& b, CompletionToken&& token); + DynamicBuffer b, CompletionToken&& token); template @\DEDUCED@ async_read(AsyncReadStream& stream, - DynamicBuffer&& b, + DynamicBuffer b, CompletionCondition completion_condition, CompletionToken&& token); @@ -229,14 +229,14 @@ error_code& ec); template - size_t write(SyncWriteStream& stream, DynamicBuffer&& b); + size_t write(SyncWriteStream& stream, DynamicBuffer b); template - size_t write(SyncWriteStream& stream, DynamicBuffer&& b, error_code& ec); + size_t write(SyncWriteStream& stream, DynamicBuffer b, error_code& ec); template - size_t write(SyncWriteStream& stream, DynamicBuffer&& b, + size_t write(SyncWriteStream& stream, DynamicBuffer b, CompletionCondition completion_condition); template - size_t write(SyncWriteStream& stream, DynamicBuffer&& b, + size_t write(SyncWriteStream& stream, DynamicBuffer b, CompletionCondition completion_condition, error_code& ec); // \ref{buffer.async.write}, asynchronous write operations @@ -255,36 +255,36 @@ template @\DEDUCED@ async_write(AsyncWriteStream& stream, - DynamicBuffer&& b, CompletionToken&& token); + DynamicBuffer b, CompletionToken&& token); template @\DEDUCED@ async_write(AsyncWriteStream& stream, - DynamicBuffer&& b, + DynamicBuffer b, CompletionCondition completion_condition, CompletionToken&& token); // \ref{buffer.read.until}, synchronous delimited read operations template - size_t read_until(SyncReadStream& s, DynamicBuffer&& b, char delim); + size_t read_until(SyncReadStream& s, DynamicBuffer b, char delim); template - size_t read_until(SyncReadStream& s, DynamicBuffer&& b, + size_t read_until(SyncReadStream& s, DynamicBuffer b, char delim, error_code& ec); template - size_t read_until(SyncReadStream& s, DynamicBuffer&& b, string_view delim); + size_t read_until(SyncReadStream& s, DynamicBuffer b, string_view delim); template - size_t read_until(SyncReadStream& s, DynamicBuffer&& b, + size_t read_until(SyncReadStream& s, DynamicBuffer b, string_view delim, error_code& ec); // \ref{buffer.async.read.until}, asynchronous delimited read operations template @\DEDUCED@ async_read_until(AsyncReadStream& s, - DynamicBuffer&& b, char delim, + DynamicBuffer b, char delim, CompletionToken&& token); template @\DEDUCED@ async_read_until(AsyncReadStream& s, - DynamicBuffer&& b, string_view delim, + DynamicBuffer b, string_view delim, CompletionToken&& token); } // inline namespace experimental::net::\namespacever @@ -455,15 +455,16 @@ \rSec2[buffer.reqmts.dynamicbuffer]{Dynamic buffer requirements} \pnum -A \defn{dynamic buffer} encapsulates memory storage that may be automatically resized as required, where the memory is divided into two regions: readable bytes followed by writable bytes. These memory regions are internal to the dynamic buffer, but direct access to the elements is provided to permit them to be efficiently used with I/O operations. \begin{note} Such as the \tcode{send} or \tcode{receive} operations of a socket. The readable bytes would be used as the constant buffer sequence for \tcode{send}, and the writable bytes used as the mutable buffer sequence for \tcode{receive}. \end{note} Data written to the writable bytes of a dynamic buffer object is appended to the readable bytes of the same object. +A \defn{dynamic buffer} encapsulates memory storage that may be automatically resized as required. Direct access to the elements is provided to permit them to be efficiently used with I/O operations. \begin{note} Such as the \tcode{send} or \tcode{receive} operations of a socket. \end{note} \pnum -A type \tcode{X} meets the \defnnewoldconcept{DynamicBuffer} requirements if it meets the requirements of \oldconcept{Destructible}~(\CppXref{destructible}) and \oldconcept{MoveConstructible}~(\CppXref{moveconstructible}), as well as the additional requirements listed in Table~\ref{tab:buffer.reqmts.dynamicbuffer.requirements}. +A type \tcode{X} meets the \defnnewoldconcept{DynamicBuffer} requirements if it meets the requirements of \oldconcept{Destructible} (\CppXref{destructible}) and \oldconcept{CopyConstructible} (\CppXref{copyconstructible}), as well as the additional requirements listed in Table~\ref{tab:buffer.reqmts.dynamicbuffer.requirements}. \pnum In Table~\ref{tab:buffer.reqmts.dynamicbuffer.requirements}, \tcode{x} denotes a value of type \tcode{X}, \tcode{x1} denotes a (possibly const) value of type \tcode{X}, +\tcode{pos} denotes a (possibly const) value of type \tcode{size_t}, and \tcode{n} denotes a (possibly const) value of type \tcode{size_t}. \indextext{requirements!\idxnewoldconcept{DynamicBuffer}}% @@ -484,39 +485,43 @@ \tcode{X::const_buffers_type} & type meeting \newoldconcept{ConstBufferSequence}~(\ref{buffer.reqmts.constbuffersequence}) requirements. & - This type represents the memory associated with the readable bytes. \\ \rowsep + This type represents the underlying memory as non-modifiable bytes. \\ \rowsep \tcode{X::mutable_buffers_type} & type meeting \newoldconcept{MutableBufferSequence}~(\ref{buffer.reqmts.constbuffersequence}) requirements. & - This type represents the memory associated with the writable bytes. \\ \rowsep + This type represents the underlying memory as modifiable bytes. \\ \rowsep \tcode{x1.size()} & \tcode{size_t} & - Returns the number of readable bytes. \\ \rowsep + Returns the number of bytes. \\ \rowsep \tcode{x1.max_size()} & \tcode{size_t} & -Returns the maximum number of bytes, both readable and writable, that can be held by \tcode{x1}. \\ \rowsep +Returns the maximum number of bytes that can be held by \tcode{x1}. \\ \rowsep \tcode{x1.capacity()} & \tcode{size_t} & -Returns the maximum number of bytes, both readable and writable, that can be held by \tcode{x1} without requiring reallocation. \\ \rowsep +Returns the maximum number of bytes that can be held by \tcode{x1} without requiring reallocation. \\ \rowsep -\tcode{x1.data()} & +\tcode{x1.data(pos, n)} & \tcode{X::const_buffers_type} & -Returns a constant buffer sequence \tcode{u} that represents the readable bytes, and where \tcode{buffer_size(u) == size()}. \\ \rowsep +Returns a constant buffer sequence \tcode{u} that represents the region of underlying memory at offset \tcode{pos} and length \tcode{n}. \\ \rowsep -\tcode{x.prepare(n)} & +\tcode{x.data(pos, n)} & \tcode{X::mutable_buffers_type} & -Returns a mutable buffer sequence \tcode{u} representing the writable bytes, and where \tcode{buffer_size(u) == n}. The dynamic buffer reallocates memory as required. All constant or mutable buffer sequences previously obtained using \tcode{data()} or \tcode{prepare()} are invalidated.\br \throws\ \tcode{length_error} if \tcode{size() + n} exceeds \tcode{max_size()}. \\ \rowsep +Returns a mutable buffer sequence \tcode{u} that represents the region of underlying memory at offset \tcode{pos} and length \tcode{n}. \\ \rowsep -\tcode{x.commit(n)} & +\tcode{x.grow(n)} & & -Appends \tcode{n} bytes from the start of the writable bytes to the end of the readable bytes. The remainder of the writable bytes are discarded. If \tcode{n} is greater than the number of writable bytes, all writable bytes are appended to the readable bytes. All constant or mutable buffer sequences previously obtained using \tcode{data()} or \tcode{prepare()} are invalidated. \\ \rowsep +Adds \tcode{n} bytes of space at the end of the underlying memory. All constant or mutable buffer sequences previously obtained using \tcode{data()} are invalidated. \\ \rowsep + +\tcode{x.shrink(n)} & + & +Removes \tcode{n} bytes of space from the end of the underlying memory. If \tcode{n} is greater than the number of bytes, all bytes are discarded. All constant or mutable buffer sequences previously obtained using \tcode{data()} are invalidated. \\ \rowsep \tcode{x.consume(n)} & & -Removes \tcode{n} bytes from beginning of the readable bytes. If \tcode{n} is greater than the number of readable bytes, all readable bytes are removed. All constant or mutable buffer sequences previously obtained using \tcode{data()} or \tcode{prepare()} are invalidated. \\ +Removes \tcode{n} bytes from beginning of the bytes. If \tcode{n} is greater than the number of bytes, all bytes are removed. All constant or mutable buffer sequences previously obtained using \tcode{data()} are invalidated. \\ \end{libreqtab3} @@ -1120,20 +1125,19 @@ explicit dynamic_vector_buffer(vector& vec) noexcept; dynamic_vector_buffer(vector& vec, size_t maximum_size) noexcept; - dynamic_vector_buffer(dynamic_vector_buffer&&) = default; // members: size_t size() const noexcept; size_t max_size() const noexcept; size_t capacity() const noexcept; - const_buffers_type data() const noexcept; - mutable_buffers_type prepare(size_t n); - void commit(size_t n); + mutable_buffers_type data(size_t pos, size_t n) noexcept; + const_buffers_type data(size_t pos, size_t n) const noexcept; + void grow(size_t n); + void shrink(size_t n); void consume(size_t n); private: vector& vec_; // \expos - size_t size_; // \expos const size_t max_size_; // \expos }; @@ -1153,7 +1157,7 @@ \begin{itemdescr} \pnum -\effects Initializes \tcode{vec_} with \tcode{vec}, \tcode{size_} with \tcode{vec.size()}, and \tcode{max_size_} with \tcode{vec.max_size()}. +\effects Initializes \tcode{vec_} with \tcode{vec} and \tcode{max_size_} with \tcode{vec.max_size()}. \end{itemdescr} \indexlibrary{\idxcode{dynamic_vector_buffer}!constructor}% @@ -1167,7 +1171,7 @@ \expects \tcode{vec.size() <= maximum_size}. \pnum -\effects Initializes \tcode{vec_} with \tcode{vec}, \tcode{size_} with \tcode{vec.size()}, and \tcode{max_size_} with \tcode{maximum_size}. +\effects Initializes \tcode{vec_} with \tcode{vec} and \tcode{max_size_} with \tcode{maximum_size}. \end{itemdescr} \indexlibrarymember{size}{dynamic_vector_buffer}% @@ -1177,7 +1181,7 @@ \begin{itemdescr} \pnum -\returns \tcode{size_}. +\returns \tcode{min(vec_.size(), max_size_)}. \end{itemdescr} \indexlibrarymember{max_size}{dynamic_vector_buffer}% @@ -1197,48 +1201,41 @@ \begin{itemdescr} \pnum -\returns \tcode{vec_.capacity()}. +\returns \tcode{min(vec_.capacity(), max_size_)}. \end{itemdescr} \indexlibrarymember{data}{dynamic_vector_buffer}% \begin{itemdecl} -const_buffers_type data() const noexcept; +mutable_buffers_type data(size_t pos, size_t n) noexcept; +const_buffers_type data(size_t pos, size_t n) const noexcept; \end{itemdecl} \begin{itemdescr} \pnum -\returns \tcode{buffer(vec_, size_)}. +\returns \tcode{buffer(buffer(vec_, max_size_) + pos, n)}. \end{itemdescr} -\indexlibrarymember{prepare}{dynamic_vector_buffer}% +\indexlibrarymember{grow}{dynamic_vector_buffer}% \begin{itemdecl} -mutable_buffers_type prepare(size_t n); +void grow(size_t n); \end{itemdecl} \begin{itemdescr} \pnum -\effects Performs \tcode{vec_.resize(size_ + n)}. +\effects Performs \tcode{vec_.resize(size() + n);}. \pnum -\returns \tcode{buffer(buffer(vec_) + size_, n)}. - -\pnum -\throws \tcode{length_error} if \tcode{size() + n} exceeds \tcode{max_size()}. +\throws \tcode{length_error} if \tcode{size() > max_size() || max_size() - size() < n}. \end{itemdescr} -\indexlibrarymember{commit}{dynamic_vector_buffer}% +\indexlibrarymember{shrink}{dynamic_vector_buffer}% \begin{itemdecl} -void commit(size_t n); +void shrink(size_t n); \end{itemdecl} \begin{itemdescr} \pnum -\effects Performs: -\begin{codeblock} -size_ += min(n, vec_.size() - size_); -vec_.resize(size_); -\end{codeblock} - +\effects Performs \tcode{vec_.resize(n > size() ? 0 : size() - n);}. \end{itemdescr} \indexlibrarymember{consume}{dynamic_vector_buffer}% @@ -1250,9 +1247,7 @@ \pnum \effects Performs: \begin{codeblock} -size_t m = min(n, size_); -vec_.erase(vec_.begin(), vec_.begin() + m); -size_ -= m; +vec_.erase(vec_.begin(), vec_.begin() + min(n, size())); \end{codeblock} \end{itemdescr} @@ -1280,20 +1275,19 @@ explicit dynamic_string_buffer(basic_string& str) noexcept; dynamic_string_buffer(basic_string& str, size_t maximum_size) noexcept; - dynamic_string_buffer(dynamic_string_buffer&&) = default; // members: size_t size() const noexcept; size_t max_size() const noexcept; size_t capacity() const noexcept; - const_buffers_type data() const noexcept; - mutable_buffers_type prepare(size_t n); - void commit(size_t n) noexcept; + mutable_buffers_type data(size_t pos, size_t n) noexcept; + const_buffers_type data(size_t pos, size_t n) const noexcept; + void grow(size_t n); + void shrink(size_t n); void consume(size_t n); private: basic_string& str_; // \expos - size_t size_; // \expos const size_t max_size_; // \expos }; @@ -1313,7 +1307,7 @@ \begin{itemdescr} \pnum -\effects Initializes \tcode{str_} with \tcode{str}, \tcode{size_} with \tcode{str.size()}, and \tcode{max_size_} with \tcode{str.max_size()}. +\effects Initializes \tcode{str_} with \tcode{str} and \tcode{max_size_} with \tcode{str.max_size()}. \end{itemdescr} \indexlibrary{\idxcode{dynamic_string_buffer}!constructor}% @@ -1327,7 +1321,7 @@ \expects \tcode{str.size() <= maximum_size}. \pnum -\effects Initializes \tcode{str_} with \tcode{str}, \tcode{size_} with \tcode{str.size()}, and \tcode{max_size_} with \tcode{maximum_size}. +\effects Initializes \tcode{str_} with \tcode{str} and \tcode{max_size_} with \tcode{maximum_size}. \end{itemdescr} \indexlibrarymember{size}{dynamic_string_buffer}% @@ -1337,7 +1331,7 @@ \begin{itemdescr} \pnum -\returns \tcode{size_}. +\returns \tcode{min(str_.size(), max_size_)}. \end{itemdescr} \indexlibrarymember{max_size}{dynamic_string_buffer}% @@ -1357,48 +1351,41 @@ \begin{itemdescr} \pnum -\returns \tcode{str_.capacity()}. +\returns \tcode{min(str_.capacity(), max_size_)}. \end{itemdescr} \indexlibrarymember{data}{dynamic_string_buffer}% \begin{itemdecl} -const_buffers_type data() const noexcept; +mutable_buffers_type data(size_t pos, size_t n) noexcept; +const_buffers_type data(size_t pos, size_t n) const noexcept; \end{itemdecl} \begin{itemdescr} \pnum -\returns \tcode{buffer(str_, size_)}. +\returns \tcode{buffer(buffer(str_, max_size_) + pos, n)}. \end{itemdescr} -\indexlibrarymember{prepare}{dynamic_string_buffer}% +\indexlibrarymember{grow}{dynamic_string_buffer}% \begin{itemdecl} -mutable_buffers_type prepare(size_t n); +void grow(size_t n); \end{itemdecl} \begin{itemdescr} \pnum -\effects Performs \tcode{str_.resize(size_ + n)}. - -\pnum -\returns \tcode{buffer(buffer(str_) + size_, n)}. +\effects Performs \tcode{str_.resize(size() + n);}. \pnum -\throws \tcode{length_error} if \tcode{size() + n} exceeds \tcode{max_size()}. +\throws \tcode{length_error} if \tcode{size() > max_size() || max_size() - size() < n}. \end{itemdescr} -\indexlibrarymember{commit}{dynamic_string_buffer}% +\indexlibrarymember{shrink}{dynamic_string_buffer}% \begin{itemdecl} -void commit(size_t n) noexcept; +void shrink(size_t n); \end{itemdecl} \begin{itemdescr} \pnum -\effects Performs: -\begin{codeblock} -size_ += min(n, str_.size() - size_); -str_.resize(size_); -\end{codeblock} - +\effects Performs \tcode{str_.resize(n > size() ? 0 : size() - n);}. \end{itemdescr} \indexlibrarymember{consume}{dynamic_string_buffer}% @@ -1410,9 +1397,7 @@ \pnum \effects Performs: \begin{codeblock} -size_t m = min(n, size_); -str_.erase(0, m); -size_ -= m; +str_.erase(0, min(n, size())); \end{codeblock} \end{itemdescr} diff --git a/src/bufferstreams.tex b/src/bufferstreams.tex index b1df4e1..7d171fe 100644 --- a/src/bufferstreams.tex +++ b/src/bufferstreams.tex @@ -384,16 +384,16 @@ \indexlibrary{\idxcode{read}}% \begin{itemdecl} template - size_t read(SyncReadStream& stream, DynamicBuffer&& b); + size_t read(SyncReadStream& stream, DynamicBuffer b); template - size_t read(SyncReadStream& stream, DynamicBuffer&& b, error_code& ec); + size_t read(SyncReadStream& stream, DynamicBuffer b, error_code& ec); template - size_t read(SyncReadStream& stream, DynamicBuffer&& b, + size_t read(SyncReadStream& stream, DynamicBuffer b, CompletionCondition completion_condition); template - size_t read(SyncReadStream& stream, DynamicBuffer&& b, + size_t read(SyncReadStream& stream, DynamicBuffer b, CompletionCondition completion_condition, error_code& ec); \end{itemdecl} @@ -407,7 +407,15 @@ \effects Clears \tcode{ec}, then reads data from the synchronous read stream~(\ref{buffer.stream.reqmts.syncreadstream}) object \tcode{stream} by performing zero or more calls to the stream's \tcode{read_some} member function. \pnum -Data is placed into the dynamic buffer~(\ref{buffer.reqmts.dynamicbuffer}) object \tcode{b}. A mutable buffer sequence~(\ref{buffer.reqmts.mutablebuffersequence}) is obtained prior to each \tcode{read_some} call using \tcode{b.prepare(N)}, where \tcode{N} is an unspecified value less than or equal to \tcode{b.max_size() - b.size()}. \begin{note} Implementations can use \tcode{b.capacity()} when determining \tcode{N}, to minimize the number of \tcode{read_some} calls performed on the stream. \end{note} After each \tcode{read_some} call, the implementation performs \tcode{b.commit(n)}, where \tcode{n} is the return value from \tcode{read_some}. +Data is placed into the dynamic buffer~(\ref{buffer.reqmts.dynamicbuffer}) object \tcode{b}. A mutable buffer sequence~(\ref{buffer.reqmts.mutablebuffersequence}) \tcode{x} is obtained prior to each \tcode{read_some} call by performing: + +\begin{codeblock} +auto orig_size = b.size(); +b.grow(N); +auto x = b.data(orig_size, N); +\end{codeblock} + +where \tcode{N} is an unspecified value less than or equal to \tcode{b.max_size() - b.size()}. \begin{note} Implementations can use \tcode{b.capacity()} when determining \tcode{N}, to minimize the number of \tcode{read_some} calls performed on the stream. \end{note} After each \tcode{read_some} call, the implementation performs \tcode{b.shrink(N - n)}, where \tcode{n} is the return value from \tcode{read_some}. \pnum The \tcode{completion_condition} parameter specifies a completion condition to be called prior to each call to the stream's \tcode{read_some} member function. The completion condition is passed the \tcode{error_code} value from the most recent \tcode{read_some} call, and the total number of bytes transferred in the synchronous read operation so far. The completion condition return value specifies the maximum number of bytes to be read on the subsequent \tcode{read_some} call. Overloads where a completion condition is not specified behave as if called with an object of class \tcode{transfer_all}. @@ -483,11 +491,11 @@ \begin{itemdecl} template @\DEDUCED@ async_read(AsyncReadStream& stream, - DynamicBuffer&& b, CompletionToken&& token); + DynamicBuffer b, CompletionToken&& token); template @\DEDUCED@ async_read(AsyncReadStream& stream, - DynamicBuffer&& b, + DynamicBuffer b, CompletionCondition completion_condition, CompletionToken&& token); \end{itemdecl} @@ -504,7 +512,15 @@ \effects Initiates an asynchronous operation to read data from the buffer-oriented asynchronous read stream~(\ref{buffer.stream.reqmts.asyncreadstream}) object \tcode{stream} by performing one or more asynchronous read_some operations on the stream. \pnum -Data is placed into the dynamic buffer~(\ref{buffer.reqmts.dynamicbuffer}) object \tcode{b}. A mutable buffer sequence~(\ref{buffer.reqmts.mutablebuffersequence}) is obtained prior to each \tcode{async_read_some} call using \tcode{b.prepare(N)}, where \tcode{N} is an unspecified value such that \tcode{N} is less than or equal to \tcode{b.max_size() - b.size()}. \begin{note} Implementations can use \tcode{b.capacity()} when determining \tcode{N}, to minimize the number of asynchronous read_some operations performed on the stream. \end{note} After the completion of each asynchronous read_some operation, the implementation performs \tcode{b.commit(n)}, where \tcode{n} is the value passed to the asynchronous read_some operation's completion handler. +Data is placed into the dynamic buffer~(\ref{buffer.reqmts.dynamicbuffer}) object \tcode{b}. A mutable buffer sequence~(\ref{buffer.reqmts.mutablebuffersequence}) \tcode{x} is obtained prior to each \tcode{async_read_some} call by performing: + +\begin{codeblock} +auto orig_size = b.size(); +b.grow(N); +auto x = b.data(orig_size, N); +\end{codeblock} + +where \tcode{N} is an unspecified value such that \tcode{N} is less than or equal to \tcode{b.max_size() - b.size()}. \begin{note} Implementations can use \tcode{b.capacity()} when determining \tcode{N}, to minimize the number of asynchronous read_some operations performed on the stream. \end{note} After the completion of each asynchronous read_some operation, the implementation performs \tcode{b.shrink(N - n)}, where \tcode{n} is the value passed to the asynchronous read_some operation's completion handler. \pnum The \tcode{completion_condition} parameter specifies a completion condition to be called prior to each asynchronous read_some operation. The completion condition is passed the \tcode{error_code} value from the most recent asynchronous read_some operation, and the total number of bytes transferred in the asynchronous read operation so far. The completion condition return value specifies the maximum number of bytes to be read on the subsequent asynchronous read_some operation. Overloads where a completion condition is not specified behave as if called with an object of class \tcode{transfer_all}. @@ -585,14 +601,14 @@ \indexlibrary{\idxcode{write}}% \begin{itemdecl} template - size_t write(SyncWriteStream& stream, DynamicBuffer&& b); + size_t write(SyncWriteStream& stream, DynamicBuffer b); template - size_t write(SyncWriteStream& stream, DynamicBuffer&& b, error_code& ec); + size_t write(SyncWriteStream& stream, DynamicBuffer b, error_code& ec); template - size_t write(SyncWriteStream& stream, DynamicBuffer&& b, + size_t write(SyncWriteStream& stream, DynamicBuffer b, CompletionCondition completion_condition); template - size_t write(SyncWriteStream& stream, DynamicBuffer&& b, + size_t write(SyncWriteStream& stream, DynamicBuffer b, CompletionCondition completion_condition, error_code& ec); \end{itemdecl} @@ -683,11 +699,11 @@ \begin{itemdecl} template @\DEDUCED@ async_write(AsyncWriteStream& stream, - DynamicBuffer&& b, CompletionToken&& token); + DynamicBuffer b, CompletionToken&& token); template @\DEDUCED@ async_write(AsyncWriteStream& stream, - DynamicBuffer&& b, + DynamicBuffer b, CompletionCondition completion_condition, CompletionToken&& token); \end{itemdecl} @@ -731,30 +747,38 @@ \indexlibrary{\idxcode{read_until}}% \begin{itemdecl} template - size_t read_until(SyncReadStream& s, DynamicBuffer&& b, char delim); + size_t read_until(SyncReadStream& s, DynamicBuffer b, char delim); template - size_t read_until(SyncReadStream& s, DynamicBuffer&& b, + size_t read_until(SyncReadStream& s, DynamicBuffer b, char delim, error_code& ec); template - size_t read_until(SyncReadStream& s, DynamicBuffer&& b, string_view delim); + size_t read_until(SyncReadStream& s, DynamicBuffer b, string_view delim); template - size_t read_until(SyncReadStream& s, DynamicBuffer&& b, + size_t read_until(SyncReadStream& s, DynamicBuffer b, string_view delim, error_code& ec); \end{itemdecl} \begin{itemdescr} \pnum -\effects Reads data from the buffer-oriented synchronous read stream~(\ref{buffer.stream.reqmts.syncreadstream}) object \tcode{stream} by performing zero or more calls to the stream's \tcode{read_some} member function, until the readable bytes of the dynamic buffer~(\ref{buffer.reqmts.dynamicbuffer}) object \tcode{b} contains the specified delimiter \tcode{delim}. +\effects Reads data from the buffer-oriented synchronous read stream~(\ref{buffer.stream.reqmts.syncreadstream}) object \tcode{stream} by performing zero or more calls to the stream's \tcode{read_some} member function, until the dynamic buffer~(\ref{buffer.reqmts.dynamicbuffer}) object \tcode{b} contains the specified delimiter \tcode{delim}. \pnum -Data is placed into the dynamic buffer object \tcode{b}. A mutable buffer sequence~(\ref{buffer.reqmts.mutablebuffersequence}) is obtained prior to each \tcode{read_some} call using \tcode{b.prepare(N)}, where \tcode{N} is an unspecified value such that \tcode{N <= max_size() - size()}. \begin{note} Implementations can use \tcode{b.capacity()} when determining \tcode{N}, to minimize the number of \tcode{read_some} calls performed on the stream. \end{note} After each \tcode{read_some} call, the implementation performs \tcode{b.commit(n)}, where \tcode{n} is the return value from \tcode{read_some}. +Data is placed into the dynamic buffer object \tcode{b}. A mutable buffer sequence~(\ref{buffer.reqmts.mutablebuffersequence}) \tcode{x} is obtained prior to each \tcode{read_some} call by performing: + +\begin{codeblock} +auto orig_size = b.size(); +b.grow(N); +auto x = b.data(orig_size, N); +\end{codeblock} + +where \tcode{N} is an unspecified value such that \tcode{N <= max_size() - size()}. \begin{note} Implementations can use \tcode{b.capacity()} when determining \tcode{N}, to minimize the number of \tcode{read_some} calls performed on the stream. \end{note} After each \tcode{read_some} call, the implementation performs \tcode{b.shrink(N - n)}, where \tcode{n} is the return value from \tcode{read_some}. \pnum The synchronous read_until operation continues until: \begin{itemize} \item -the readable bytes of \tcode{b} contains the delimiter \tcode{delim}; or +\tcode{b} contains the delimiter \tcode{delim}; or \item \tcode{b.size() == b.max_size()}; or \item @@ -762,10 +786,10 @@ \end{itemize} \pnum -On exit, if the readable bytes of \tcode{b} contains the delimiter, \tcode{ec} is set such that \tcode{!ec} is \tcode{true}. Otherwise, if \tcode{b.size() == b.max_size()}, \tcode{ec} is set such that \tcode{ec == stream_errc::not_found}. If \tcode{b.size() < b.max_size()}, \tcode{ec} contains the \tcode{error_code} from the most recent \tcode{read_some} call. +On exit, if \tcode{b} contains the delimiter, \tcode{ec} is set such that \tcode{!ec} is \tcode{true}. Otherwise, if \tcode{b.size() == b.max_size()}, \tcode{ec} is set such that \tcode{ec == stream_errc::not_found}. If \tcode{b.size() < b.max_size()}, \tcode{ec} contains the \tcode{error_code} from the most recent \tcode{read_some} call. \pnum -\returns The number of bytes in the readable bytes of \tcode{b} up to and including the delimiter, if present. \begin{note} On completion, the buffer can contain additional bytes following the delimiter. \end{note} Otherwise returns \tcode{0}. +\returns The number of bytes in \tcode{b} up to and including the delimiter, if present. \begin{note} On completion, the buffer can contain additional bytes following the delimiter. \end{note} Otherwise returns \tcode{0}. \end{itemdescr} @@ -776,11 +800,11 @@ \begin{itemdecl} template @\DEDUCED@ async_read_until(AsyncReadStream& s, - DynamicBuffer&& b, char delim, + DynamicBuffer b, char delim, CompletionToken&& token); template @\DEDUCED@ async_read_until(AsyncReadStream& s, - DynamicBuffer&& b, string_view delim, + DynamicBuffer b, string_view delim, CompletionToken&& token); \end{itemdecl} @@ -792,17 +816,25 @@ \completionsig \tcode{void(error_code ec, size_t n)}. \pnum -\effects Initiates an asynchronous operation to read data from the buffer-oriented asynchronous read stream~(\ref{buffer.stream.reqmts.asyncreadstream}) object \tcode{stream} by performing zero or more asynchronous read_some operations on the stream, until the readable bytes of the dynamic buffer~(\ref{buffer.reqmts.dynamicbuffer}) object \tcode{b} contain the specified delimiter \tcode{delim}. +\effects Initiates an asynchronous operation to read data from the buffer-oriented asynchronous read stream~(\ref{buffer.stream.reqmts.asyncreadstream}) object \tcode{stream} by performing zero or more asynchronous read_some operations on the stream, until the dynamic buffer~(\ref{buffer.reqmts.dynamicbuffer}) object \tcode{b} contains the specified delimiter \tcode{delim}. \pnum -Data is placed into the dynamic buffer object \tcode{b}. A mutable buffer sequence~(\ref{buffer.reqmts.mutablebuffersequence}) is obtained prior to each \tcode{async_read_some} call using \tcode{b.prepare(N)}, where \tcode{N} is an unspecified value such that \tcode{N <= max_size() - size()}. \begin{note} Implementations can use \tcode{b.capacity()} when determining \tcode{N}, to minimize the number of asynchronous read_some operations performed on the stream. \end{note} After the completion of each asynchronous read_some operation, the implementation performs \tcode{b.commit(n)}, where \tcode{n} is the value passed to the asynchronous read_some operation's completion handler. +Data is placed into the dynamic buffer object \tcode{b}. A mutable buffer sequence~(\ref{buffer.reqmts.mutablebuffersequence}) \tcode{x} is obtained prior to each \tcode{async_read_some} call by performing: + +\begin{codeblock} +auto orig_size = b.size(); +b.grow(N); +auto x = b.data(orig_size, N); +\end{codeblock} + +where \tcode{N} is an unspecified value such that \tcode{N <= max_size() - size()}. \begin{note} Implementations can use \tcode{b.capacity()} when determining \tcode{N}, to minimize the number of asynchronous read_some operations performed on the stream. \end{note} After the completion of each asynchronous read_some operation, the implementation performs \tcode{b.shrink(N - n)}, where \tcode{n} is the value passed to the asynchronous read_some operation's completion handler. \pnum The asynchronous read_until operation continues until: \begin{itemize} \item -the readable bytes of \tcode{b} contain the delimiter \tcode{delim}; or +\tcode{b} contains the delimiter \tcode{delim}; or \item \tcode{b.size() == b.max_size()}; or \item @@ -816,7 +848,7 @@ If \tcode{delim} is of type \tcode{string_view}, the implementation copies the underlying sequence of characters prior to initiating an asynchronous read_some operation on the stream. \begin{note} This means that the caller is not required to guarantee the validity of the delimiter string after the call to \tcode{async_read_until} returns. \end{note} \pnum -On completion of the asynchronous operation, if the readable bytes of \tcode{b} contain the delimiter, \tcode{ec} is set such that \tcode{!ec} is \tcode{true}. Otherwise, if \tcode{b.size() == b.max_size()}, \tcode{ec} is set such that \tcode{ec == stream_errc::not_found}. If \tcode{b.size() < b.max_size()}, \tcode{ec} is the \tcode{error_code} from the most recent asynchronous read_some operation. \tcode{n} is the number of readable bytes in \tcode{b} up to and including the delimiter, if present, otherwise \tcode{0}. +On completion of the asynchronous operation, if \tcode{b} contains the delimiter, \tcode{ec} is set such that \tcode{!ec} is \tcode{true}. Otherwise, if \tcode{b.size() == b.max_size()}, \tcode{ec} is set such that \tcode{ec == stream_errc::not_found}. If \tcode{b.size() < b.max_size()}, \tcode{ec} is the \tcode{error_code} from the most recent asynchronous read_some operation. \tcode{n} is the number of readable bytes in \tcode{b} up to and including the delimiter, if present, otherwise \tcode{0}. \end{itemdescr}