• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 use crate::io::util::read_line::finish_string_read;
2 use crate::io::util::read_to_end::read_to_end_internal;
3 use crate::io::util::vec_with_initialized::VecWithInitialized;
4 use crate::io::AsyncRead;
5 
6 use pin_project_lite::pin_project;
7 use std::future::Future;
8 use std::marker::PhantomPinned;
9 use std::pin::Pin;
10 use std::task::{Context, Poll};
11 use std::{io, mem};
12 
13 pin_project! {
14     /// Future for the [`read_to_string`](super::AsyncReadExt::read_to_string) method.
15     #[derive(Debug)]
16     #[must_use = "futures do nothing unless you `.await` or poll them"]
17     pub struct ReadToString<'a, R: ?Sized> {
18         reader: &'a mut R,
19         // This is the buffer we were provided. It will be replaced with an empty string
20         // while reading to postpone utf-8 handling until after reading.
21         output: &'a mut String,
22         // The actual allocation of the string is moved into this vector instead.
23         buf: VecWithInitialized<Vec<u8>>,
24         // The number of bytes appended to buf. This can be less than buf.len() if
25         // the buffer was not empty when the operation was started.
26         read: usize,
27         // Make this future `!Unpin` for compatibility with async trait methods.
28         #[pin]
29         _pin: PhantomPinned,
30     }
31 }
32 
read_to_string<'a, R>( reader: &'a mut R, string: &'a mut String, ) -> ReadToString<'a, R> where R: AsyncRead + ?Sized + Unpin,33 pub(crate) fn read_to_string<'a, R>(
34     reader: &'a mut R,
35     string: &'a mut String,
36 ) -> ReadToString<'a, R>
37 where
38     R: AsyncRead + ?Sized + Unpin,
39 {
40     let buf = mem::replace(string, String::new()).into_bytes();
41     ReadToString {
42         reader,
43         buf: VecWithInitialized::new(buf),
44         output: string,
45         read: 0,
46         _pin: PhantomPinned,
47     }
48 }
49 
read_to_string_internal<R: AsyncRead + ?Sized>( reader: Pin<&mut R>, output: &mut String, buf: &mut VecWithInitialized<Vec<u8>>, read: &mut usize, cx: &mut Context<'_>, ) -> Poll<io::Result<usize>>50 fn read_to_string_internal<R: AsyncRead + ?Sized>(
51     reader: Pin<&mut R>,
52     output: &mut String,
53     buf: &mut VecWithInitialized<Vec<u8>>,
54     read: &mut usize,
55     cx: &mut Context<'_>,
56 ) -> Poll<io::Result<usize>> {
57     let io_res = ready!(read_to_end_internal(buf, reader, read, cx));
58     let utf8_res = String::from_utf8(buf.take());
59 
60     // At this point both buf and output are empty. The allocation is in utf8_res.
61 
62     debug_assert!(buf.is_empty());
63     debug_assert!(output.is_empty());
64     finish_string_read(io_res, utf8_res, *read, output, true)
65 }
66 
67 impl<A> Future for ReadToString<'_, A>
68 where
69     A: AsyncRead + ?Sized + Unpin,
70 {
71     type Output = io::Result<usize>;
72 
poll(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Self::Output>73     fn poll(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Self::Output> {
74         let me = self.project();
75 
76         read_to_string_internal(Pin::new(*me.reader), me.output, me.buf, me.read, cx)
77     }
78 }
79