|
@@ -10,6 +10,10 @@ use time::{Duration, Instant};
|
|
|
use socket::{Socket, SocketMeta, SocketHandle, PollAt};
|
|
|
use storage::{Assembler, RingBuffer};
|
|
|
use wire::{IpProtocol, IpRepr, IpAddress, IpEndpoint, TcpSeqNumber, TcpRepr, TcpControl};
|
|
|
+#[cfg(feature = "async")]
|
|
|
+use socket::WakerRegistration;
|
|
|
+#[cfg(feature = "async")]
|
|
|
+use core::task::Waker;
|
|
|
|
|
|
/// A TCP socket ring buffer.
|
|
|
pub type SocketBuffer<'a> = RingBuffer<'a, u8>;
|
|
@@ -248,6 +252,12 @@ pub struct TcpSocket<'a> {
|
|
|
/// The number of packets recived directly after
|
|
|
/// each other which have the same ACK number.
|
|
|
local_rx_dup_acks: u8,
|
|
|
+
|
|
|
+ #[cfg(feature = "async")]
|
|
|
+ rx_waker: WakerRegistration,
|
|
|
+ #[cfg(feature = "async")]
|
|
|
+ tx_waker: WakerRegistration,
|
|
|
+
|
|
|
}
|
|
|
|
|
|
const DEFAULT_MSS: usize = 536;
|
|
@@ -298,9 +308,49 @@ impl<'a> TcpSocket<'a> {
|
|
|
local_rx_last_ack: None,
|
|
|
local_rx_last_seq: None,
|
|
|
local_rx_dup_acks: 0,
|
|
|
+
|
|
|
+ #[cfg(feature = "async")]
|
|
|
+ rx_waker: WakerRegistration::new(),
|
|
|
+ #[cfg(feature = "async")]
|
|
|
+ tx_waker: WakerRegistration::new(),
|
|
|
}
|
|
|
}
|
|
|
|
|
|
+ /// Register a waker for receive operations.
|
|
|
+ ///
|
|
|
+ /// The waker is woken on state changes that might affect the return value
|
|
|
+ /// of `recv` method calls, such as receiving data, or the socket closing.
|
|
|
+ ///
|
|
|
+ /// Notes:
|
|
|
+ ///
|
|
|
+ /// - Only one waker can be registered at a time. If another waker was previously registered,
|
|
|
+ /// it is overwritten and will no longer be woken.
|
|
|
+ /// - The Waker is woken only once. Once woken, you must register it again to receive more wakes.
|
|
|
+ /// - "Spurious wakes" are allowed: a wake doesn't guarantee the result of `recv` has
|
|
|
+ /// necessarily changed.
|
|
|
+ #[cfg(feature = "async")]
|
|
|
+ pub fn register_recv_waker(&mut self, waker: &Waker) {
|
|
|
+ self.rx_waker.register(waker)
|
|
|
+ }
|
|
|
+
|
|
|
+ /// Register a waker for send operations.
|
|
|
+ ///
|
|
|
+ /// The waker is woken on state changes that might affect the return value
|
|
|
+ /// of `send` method calls, such as space becoming available in the transmit
|
|
|
+ /// buffer, or the socket closing.
|
|
|
+ ///
|
|
|
+ /// Notes:
|
|
|
+ ///
|
|
|
+ /// - Only one waker can be registered at a time. If another waker was previously registered,
|
|
|
+ /// it is overwritten and will no longer be woken.
|
|
|
+ /// - The Waker is woken only once. Once woken, you must register it again to receive more wakes.
|
|
|
+ /// - "Spurious wakes" are allowed: a wake doesn't guarantee the result of `send` has
|
|
|
+ /// necessarily changed.
|
|
|
+ #[cfg(feature = "async")]
|
|
|
+ pub fn register_send_waker(&mut self, waker: &Waker) {
|
|
|
+ self.tx_waker.register(waker)
|
|
|
+ }
|
|
|
+
|
|
|
/// Return the socket handle.
|
|
|
#[inline]
|
|
|
pub fn handle(&self) -> SocketHandle {
|
|
@@ -438,6 +488,12 @@ impl<'a> TcpSocket<'a> {
|
|
|
self.remote_win_shift = rx_cap_log2.saturating_sub(16) as u8;
|
|
|
self.remote_mss = DEFAULT_MSS;
|
|
|
self.remote_last_ts = None;
|
|
|
+
|
|
|
+ #[cfg(feature = "async")]
|
|
|
+ {
|
|
|
+ self.rx_waker.wake();
|
|
|
+ self.tx_waker.wake();
|
|
|
+ }
|
|
|
}
|
|
|
|
|
|
/// Start listening on the given endpoint.
|
|
@@ -825,7 +881,17 @@ impl<'a> TcpSocket<'a> {
|
|
|
self.state, state);
|
|
|
}
|
|
|
}
|
|
|
- self.state = state
|
|
|
+
|
|
|
+ self.state = state;
|
|
|
+
|
|
|
+ #[cfg(feature = "async")]
|
|
|
+ {
|
|
|
+ // Wake all tasks waiting. Even if we haven't received/sent data, this
|
|
|
+ // is needed because return values of functions may change depending on the state.
|
|
|
+ // For example, a pending read has to fail with an error if the socket is closed.
|
|
|
+ self.rx_waker.wake();
|
|
|
+ self.tx_waker.wake();
|
|
|
+ }
|
|
|
}
|
|
|
|
|
|
pub(crate) fn reply(ip_repr: &IpRepr, repr: &TcpRepr) -> (IpRepr, TcpRepr<'static>) {
|
|
@@ -1283,6 +1349,10 @@ impl<'a> TcpSocket<'a> {
|
|
|
self.meta.handle, self.local_endpoint, self.remote_endpoint,
|
|
|
ack_len, self.tx_buffer.len() - ack_len);
|
|
|
self.tx_buffer.dequeue_allocated(ack_len);
|
|
|
+
|
|
|
+ // There's new room available in tx_buffer, wake the waiting task if any.
|
|
|
+ #[cfg(feature = "async")]
|
|
|
+ self.tx_waker.wake();
|
|
|
}
|
|
|
|
|
|
if let Some(ack_number) = repr.ack_number {
|
|
@@ -1367,6 +1437,10 @@ impl<'a> TcpSocket<'a> {
|
|
|
self.meta.handle, self.local_endpoint, self.remote_endpoint,
|
|
|
contig_len, self.rx_buffer.len() + contig_len);
|
|
|
self.rx_buffer.enqueue_unallocated(contig_len);
|
|
|
+
|
|
|
+ // There's new data in rx_buffer, notify waiting task if any.
|
|
|
+ #[cfg(feature = "async")]
|
|
|
+ self.rx_waker.wake();
|
|
|
}
|
|
|
|
|
|
if !self.assembler.is_empty() {
|