2020-04-01 04:34:32 +08:00
|
|
|
//! async TCP interface
|
|
|
|
//!
|
|
|
|
//! TODO: implement futures AsyncRead/AsyncWrite/Stream/Sink interfaces
|
|
|
|
|
2020-03-31 07:16:58 +08:00
|
|
|
use core::{
|
|
|
|
future::Future,
|
|
|
|
pin::Pin,
|
|
|
|
task::{Context, Poll},
|
|
|
|
};
|
|
|
|
use alloc::vec;
|
|
|
|
use smoltcp::{
|
|
|
|
socket::{
|
2020-04-01 04:34:32 +08:00
|
|
|
SocketHandle, SocketRef,
|
2020-03-31 07:16:58 +08:00
|
|
|
TcpSocketBuffer, TcpSocket,
|
|
|
|
},
|
|
|
|
};
|
|
|
|
use super::Sockets;
|
|
|
|
|
2020-04-01 04:34:32 +08:00
|
|
|
/// References a smoltcp TcpSocket
|
2020-03-31 07:16:58 +08:00
|
|
|
pub struct TcpStream {
|
|
|
|
handle: SocketHandle,
|
|
|
|
}
|
|
|
|
|
2020-04-01 04:34:32 +08:00
|
|
|
/// Wait while polling a stream
|
|
|
|
macro_rules! poll_stream {
|
|
|
|
($stream: expr, $output: ty, $f: expr) => (async {
|
|
|
|
struct Adhoc<'a> {
|
|
|
|
stream: &'a TcpStream,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a> Future for Adhoc<'a> {
|
|
|
|
type Output = $output;
|
|
|
|
|
|
|
|
fn poll(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Self::Output> {
|
|
|
|
let result = self.stream.with_socket($f);
|
|
|
|
if !result.is_ready() {
|
|
|
|
Sockets::register_waker(cx.waker().clone());
|
|
|
|
}
|
|
|
|
result
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Adhoc { stream: $stream }.await
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2020-03-31 07:16:58 +08:00
|
|
|
impl TcpStream {
|
|
|
|
fn new(rx_bufsize: usize, tx_bufsize: usize) -> Self {
|
2020-04-01 04:34:32 +08:00
|
|
|
// TODO: Uninitialized is faster than zeroed
|
2020-03-31 07:16:58 +08:00
|
|
|
let rx_buffer = TcpSocketBuffer::new(vec![0u8; rx_bufsize]);
|
|
|
|
let tx_buffer = TcpSocketBuffer::new(vec![0u8; tx_bufsize]);
|
|
|
|
let socket = TcpSocket::new(rx_buffer, tx_buffer);
|
|
|
|
let handle = Sockets::instance().sockets.borrow_mut()
|
|
|
|
.add(socket);
|
|
|
|
TcpStream { handle }
|
|
|
|
}
|
|
|
|
|
2020-04-01 04:34:32 +08:00
|
|
|
/// Operate on the referenced TCP socket
|
2020-03-31 07:16:58 +08:00
|
|
|
fn with_socket<F, R>(&self, f: F) -> R
|
|
|
|
where
|
|
|
|
F: FnOnce(SocketRef<TcpSocket>) -> R,
|
|
|
|
{
|
|
|
|
let mut sockets = Sockets::instance().sockets.borrow_mut();
|
|
|
|
let socket_ref = sockets.get::<TcpSocket>(self.handle);
|
|
|
|
f(socket_ref)
|
|
|
|
}
|
|
|
|
|
2020-04-01 04:34:32 +08:00
|
|
|
/// Listen for the next incoming connection on a TCP
|
|
|
|
/// port. Succeeds on connection attempt.
|
2020-03-31 07:16:58 +08:00
|
|
|
pub async fn listen(port: u16, rx_bufsize: usize, tx_bufsize: usize) -> Self {
|
|
|
|
struct Accept {
|
|
|
|
stream: Option<TcpStream>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Future for Accept {
|
|
|
|
type Output = TcpStream;
|
|
|
|
|
|
|
|
fn poll(mut self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Self::Output> {
|
|
|
|
let is_active = self.stream.as_ref()
|
|
|
|
.map(|s| s.with_socket(|s| s.is_active()))
|
|
|
|
.unwrap_or(true);
|
|
|
|
if is_active {
|
|
|
|
Poll::Ready(self.stream.take().unwrap())
|
|
|
|
} else {
|
|
|
|
Sockets::register_waker(cx.waker().clone());
|
|
|
|
|
|
|
|
//asm::sev();
|
|
|
|
Poll::Pending
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
let stream = Self::new(rx_bufsize, tx_bufsize);
|
|
|
|
stream.with_socket(|mut s| s.listen(port)).expect("listen");
|
|
|
|
Accept {
|
|
|
|
stream: Some(stream),
|
|
|
|
}.await
|
|
|
|
}
|
|
|
|
|
2020-04-01 04:34:32 +08:00
|
|
|
/// Probe the receive buffer
|
|
|
|
///
|
|
|
|
/// Instead of handing you the data on the heap all at once,
|
|
|
|
/// smoltcp's read interface is wrapped so that your callback can
|
|
|
|
/// just return `Poll::Pending` if there is not enough data
|
|
|
|
/// yet. Likewise, return the amount of bytes consumed from the
|
|
|
|
/// buffer in the `Poll::Ready` result.
|
|
|
|
pub async fn recv<F, R>(&self, f: F) -> smoltcp::Result<R>
|
|
|
|
where
|
|
|
|
F: Fn(&[u8]) -> Poll<(usize, R)>,
|
|
|
|
{
|
|
|
|
struct Recv<'a, F: FnOnce(&[u8]) -> Poll<(usize, R)>, R> {
|
|
|
|
stream: &'a TcpStream,
|
|
|
|
f: F,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a, F: Fn(&[u8]) -> Poll<(usize, R)>, R> Future for Recv<'a, F, R> {
|
|
|
|
type Output = smoltcp::Result<R>;
|
|
|
|
|
|
|
|
fn poll(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Self::Output> {
|
|
|
|
let result = self.stream.with_socket(|mut socket| {
|
|
|
|
socket.recv(|buf| match (self.f)(buf) {
|
|
|
|
Poll::Ready((amount, result)) => (amount, Poll::Ready(Ok(result))),
|
|
|
|
Poll::Pending => (0, Poll::Pending),
|
|
|
|
})
|
|
|
|
});
|
|
|
|
match result {
|
|
|
|
Ok(result) => {
|
|
|
|
if !result.is_ready() {
|
|
|
|
Sockets::register_waker(cx.waker().clone());
|
|
|
|
}
|
|
|
|
result
|
|
|
|
}
|
|
|
|
Err(e) =>
|
|
|
|
Poll::Ready(Err(e)),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Recv {
|
|
|
|
stream: self,
|
|
|
|
f,
|
|
|
|
}.await
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Wait until there is any space in the socket's send queue
|
|
|
|
async fn wait_can_send(&self) -> smoltcp::Result<()> {
|
|
|
|
poll_stream!(self, smoltcp::Result<()>, |socket| {
|
|
|
|
if !socket.is_active() {
|
|
|
|
Poll::Ready(Err(smoltcp::Error::Illegal))
|
|
|
|
} else if socket.can_send() {
|
|
|
|
Poll::Ready(Ok(()))
|
|
|
|
} else {
|
|
|
|
Poll::Pending
|
|
|
|
}
|
|
|
|
}).await
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Yields to wait for more buffer space
|
|
|
|
pub async fn send<I: IntoIterator<Item = u8>>(&self, data: I) -> Result<(), smoltcp::Error> {
|
|
|
|
let mut data = data.into_iter();
|
|
|
|
let mut done = false;
|
|
|
|
while !done {
|
|
|
|
self.wait_can_send().await?;
|
|
|
|
|
|
|
|
self.with_socket(|mut socket| {
|
|
|
|
socket.send(|buf| {
|
|
|
|
for i in 0..buf.len() {
|
|
|
|
if let Some(byte) = data.next() {
|
|
|
|
buf[i] = byte;
|
|
|
|
} else {
|
|
|
|
done = true;
|
|
|
|
return (i, ())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
(buf.len(), ())
|
|
|
|
})
|
|
|
|
})?;
|
|
|
|
}
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Wait for all queued data to be sent and ACKed
|
|
|
|
///
|
|
|
|
/// **Warning:** this may not work as immediately as expected! The
|
|
|
|
/// other side may wait until it sends packets to you for
|
|
|
|
/// piggybacking the ACKs.
|
|
|
|
pub async fn flush(&self) {
|
|
|
|
poll_stream!(self, (), |socket| {
|
|
|
|
if socket.may_send() && socket.send_queue() > 0 {
|
|
|
|
Poll::Pending
|
|
|
|
} else {
|
|
|
|
Poll::Ready(())
|
|
|
|
}
|
|
|
|
}).await
|
2020-03-31 07:16:58 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Drop for TcpStream {
|
|
|
|
fn drop(&mut self) {
|
|
|
|
Sockets::instance().sockets.borrow_mut()
|
|
|
|
.remove(self.handle);
|
|
|
|
}
|
|
|
|
}
|