broker_tokio/io/util/
buf_reader.rs1use crate::io::util::DEFAULT_BUF_SIZE;
2use crate::io::{AsyncBufRead, AsyncRead, AsyncWrite};
3
4use pin_project_lite::pin_project;
5use std::io::{self, Read};
6use std::mem::MaybeUninit;
7use std::pin::Pin;
8use std::task::{Context, Poll};
9use std::{cmp, fmt};
10
11pin_project! {
12 #[cfg_attr(docsrs, doc(cfg(feature = "io-util")))]
28 pub struct BufReader<R> {
29 #[pin]
30 pub(super) inner: R,
31 pub(super) buf: Box<[u8]>,
32 pub(super) pos: usize,
33 pub(super) cap: usize,
34 }
35}
36
37impl<R: AsyncRead> BufReader<R> {
38 pub fn new(inner: R) -> Self {
41 Self::with_capacity(DEFAULT_BUF_SIZE, inner)
42 }
43
44 pub fn with_capacity(capacity: usize, inner: R) -> Self {
46 unsafe {
47 let mut buffer = Vec::with_capacity(capacity);
48 buffer.set_len(capacity);
49
50 {
51 let b = &mut *(&mut buffer[..] as *mut [u8] as *mut [MaybeUninit<u8>]);
53 inner.prepare_uninitialized_buffer(b);
54 }
55 Self {
56 inner,
57 buf: buffer.into_boxed_slice(),
58 pos: 0,
59 cap: 0,
60 }
61 }
62 }
63
64 pub fn get_ref(&self) -> &R {
68 &self.inner
69 }
70
71 pub fn get_mut(&mut self) -> &mut R {
75 &mut self.inner
76 }
77
78 pub fn get_pin_mut(self: Pin<&mut Self>) -> Pin<&mut R> {
82 self.project().inner
83 }
84
85 pub fn into_inner(self) -> R {
89 self.inner
90 }
91
92 pub fn buffer(&self) -> &[u8] {
96 &self.buf[self.pos..self.cap]
97 }
98
99 #[inline]
101 fn discard_buffer(self: Pin<&mut Self>) {
102 let me = self.project();
103 *me.pos = 0;
104 *me.cap = 0;
105 }
106}
107
108impl<R: AsyncRead> AsyncRead for BufReader<R> {
109 fn poll_read(
110 mut self: Pin<&mut Self>,
111 cx: &mut Context<'_>,
112 buf: &mut [u8],
113 ) -> Poll<io::Result<usize>> {
114 if self.pos == self.cap && buf.len() >= self.buf.len() {
118 let res = ready!(self.as_mut().get_pin_mut().poll_read(cx, buf));
119 self.discard_buffer();
120 return Poll::Ready(res);
121 }
122 let mut rem = ready!(self.as_mut().poll_fill_buf(cx))?;
123 let nread = rem.read(buf)?;
124 self.consume(nread);
125 Poll::Ready(Ok(nread))
126 }
127
128 unsafe fn prepare_uninitialized_buffer(&self, buf: &mut [MaybeUninit<u8>]) -> bool {
130 self.inner.prepare_uninitialized_buffer(buf)
131 }
132}
133
134impl<R: AsyncRead> AsyncBufRead for BufReader<R> {
135 fn poll_fill_buf(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<io::Result<&[u8]>> {
136 let me = self.project();
137
138 if *me.pos >= *me.cap {
143 debug_assert!(*me.pos == *me.cap);
144 *me.cap = ready!(me.inner.poll_read(cx, me.buf))?;
145 *me.pos = 0;
146 }
147 Poll::Ready(Ok(&me.buf[*me.pos..*me.cap]))
148 }
149
150 fn consume(self: Pin<&mut Self>, amt: usize) {
151 let me = self.project();
152 *me.pos = cmp::min(*me.pos + amt, *me.cap);
153 }
154}
155
156impl<R: AsyncRead + AsyncWrite> AsyncWrite for BufReader<R> {
157 fn poll_write(
158 self: Pin<&mut Self>,
159 cx: &mut Context<'_>,
160 buf: &[u8],
161 ) -> Poll<io::Result<usize>> {
162 self.get_pin_mut().poll_write(cx, buf)
163 }
164
165 fn poll_flush(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<io::Result<()>> {
166 self.get_pin_mut().poll_flush(cx)
167 }
168
169 fn poll_shutdown(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<io::Result<()>> {
170 self.get_pin_mut().poll_shutdown(cx)
171 }
172}
173
174impl<R: fmt::Debug> fmt::Debug for BufReader<R> {
175 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
176 f.debug_struct("BufReader")
177 .field("reader", &self.inner)
178 .field(
179 "buffer",
180 &format_args!("{}/{}", self.cap - self.pos, self.buf.len()),
181 )
182 .finish()
183 }
184}
185
186#[cfg(test)]
187mod tests {
188 use super::*;
189
190 #[test]
191 fn assert_unpin() {
192 crate::is_unpin::<BufReader<()>>();
193 }
194}