Use #[pin_project] in stream implementation to get rid of remaining unsafe code

Now the whole crate has no unsafe code left.
pull/3/head
Sebastian Dröge 5 years ago
parent 93d4b35d4d
commit 9b7f7a02c0
  1. 76
      src/stream.rs

@ -3,85 +3,65 @@
//! There is no dependency on actual TLS implementations. Everything like //! There is no dependency on actual TLS implementations. Everything like
//! `native_tls` or `openssl` will work as long as there is a TLS stream supporting standard //! `native_tls` or `openssl` will work as long as there is a TLS stream supporting standard
//! `AsyncRead + AsyncWrite` traits. //! `AsyncRead + AsyncWrite` traits.
use pin_project::{pin_project, project};
use std::pin::Pin; use std::pin::Pin;
use std::task::{Context, Poll}; use std::task::{Context, Poll};
use futures::io::{AsyncRead, AsyncWrite}; use futures::io::{AsyncRead, AsyncWrite};
/// Stream, either plain TCP or TLS. /// Stream, either plain TCP or TLS.
#[pin_project]
pub enum Stream<S, T> { pub enum Stream<S, T> {
/// Unencrypted socket stream. /// Unencrypted socket stream.
Plain(S), Plain(#[pin] S),
/// Encrypted socket stream. /// Encrypted socket stream.
Tls(T), Tls(#[pin] T),
} }
impl<S: AsyncRead + Unpin, T: AsyncRead + Unpin> AsyncRead for Stream<S, T> { impl<S: AsyncRead + Unpin, T: AsyncRead + Unpin> AsyncRead for Stream<S, T> {
#[project]
fn poll_read( fn poll_read(
mut self: Pin<&mut Self>, self: Pin<&mut Self>,
cx: &mut Context<'_>, cx: &mut Context<'_>,
buf: &mut [u8], buf: &mut [u8],
) -> Poll<std::io::Result<usize>> { ) -> Poll<std::io::Result<usize>> {
match *self { #[project]
Stream::Plain(ref mut s) => { match self.project() {
let pinned = unsafe { Pin::new_unchecked(s) }; Stream::Plain(ref mut s) => Pin::new(s).poll_read(cx, buf),
pinned.poll_read(cx, buf) Stream::Tls(ref mut s) => Pin::new(s).poll_read(cx, buf),
}
Stream::Tls(ref mut s) => {
let pinned = unsafe { Pin::new_unchecked(s) };
pinned.poll_read(cx, buf)
}
} }
} }
} }
impl<S: AsyncWrite + Unpin, T: AsyncWrite + Unpin> AsyncWrite for Stream<S, T> { impl<S: AsyncWrite + Unpin, T: AsyncWrite + Unpin> AsyncWrite for Stream<S, T> {
#[project]
fn poll_write( fn poll_write(
mut self: Pin<&mut Self>, self: Pin<&mut Self>,
cx: &mut Context<'_>, cx: &mut Context<'_>,
buf: &[u8], buf: &[u8],
) -> Poll<Result<usize, std::io::Error>> { ) -> Poll<Result<usize, std::io::Error>> {
match *self { #[project]
Stream::Plain(ref mut s) => { match self.project() {
let pinned = unsafe { Pin::new_unchecked(s) }; Stream::Plain(ref mut s) => Pin::new(s).poll_write(cx, buf),
pinned.poll_write(cx, buf) Stream::Tls(ref mut s) => Pin::new(s).poll_write(cx, buf),
}
Stream::Tls(ref mut s) => {
let pinned = unsafe { Pin::new_unchecked(s) };
pinned.poll_write(cx, buf)
}
} }
} }
fn poll_flush( #[project]
mut self: Pin<&mut Self>, fn poll_flush(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Result<(), std::io::Error>> {
cx: &mut Context<'_>, #[project]
) -> Poll<Result<(), std::io::Error>> { match self.project() {
match *self { Stream::Plain(ref mut s) => Pin::new(s).poll_flush(cx),
Stream::Plain(ref mut s) => { Stream::Tls(ref mut s) => Pin::new(s).poll_flush(cx),
let pinned = unsafe { Pin::new_unchecked(s) };
pinned.poll_flush(cx)
}
Stream::Tls(ref mut s) => {
let pinned = unsafe { Pin::new_unchecked(s) };
pinned.poll_flush(cx)
}
} }
} }
fn poll_close( #[project]
mut self: Pin<&mut Self>, fn poll_close(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Result<(), std::io::Error>> {
cx: &mut Context<'_>, #[project]
) -> Poll<Result<(), std::io::Error>> { match self.project() {
match *self { Stream::Plain(ref mut s) => Pin::new(s).poll_close(cx),
Stream::Plain(ref mut s) => { Stream::Tls(ref mut s) => Pin::new(s).poll_close(cx),
let pinned = unsafe { Pin::new_unchecked(s) };
pinned.poll_close(cx)
}
Stream::Tls(ref mut s) => {
let pinned = unsafe { Pin::new_unchecked(s) };
pinned.poll_close(cx)
}
} }
} }
} }

Loading…
Cancel
Save