mirror of
https://github.com/asterinas/asterinas.git
synced 2025-06-22 08:53:29 +00:00
209 lines
7.1 KiB
Rust
209 lines
7.1 KiB
Rust
// SPDX-License-Identifier: MPL-2.0
|
|
|
|
use core::sync::atomic::{AtomicBool, Ordering};
|
|
|
|
use aster_bigtcp::{
|
|
errors::tcp::{RecvError, SendError},
|
|
socket::{NeedIfacePoll, TcpStateCheck},
|
|
wire::IpEndpoint,
|
|
};
|
|
|
|
use super::StreamObserver;
|
|
use crate::{
|
|
events::IoEvents,
|
|
net::{
|
|
iface::{BoundTcpSocket, Iface},
|
|
socket::util::{send_recv_flags::SendRecvFlags, shutdown_cmd::SockShutdownCmd},
|
|
},
|
|
prelude::*,
|
|
process::signal::Pollee,
|
|
util::{MultiRead, MultiWrite},
|
|
};
|
|
|
|
pub struct ConnectedStream {
|
|
bound_socket: BoundTcpSocket,
|
|
remote_endpoint: IpEndpoint,
|
|
/// Indicates whether this connection is "new" in a `connect()` system call.
|
|
///
|
|
/// If the connection is not new, `connect()` will fail with the error code `EISCONN`,
|
|
/// otherwise it will succeed. This means that `connect()` will succeed _exactly_ once,
|
|
/// regardless of whether the connection is established synchronously or asynchronously.
|
|
///
|
|
/// If the connection is established synchronously, the synchronous `connect()` will succeed
|
|
/// and any subsequent `connect()` will fail; otherwise, the first `connect()` after the
|
|
/// connection is established asynchronously will succeed and any subsequent `connect()` will
|
|
/// fail.
|
|
is_new_connection: bool,
|
|
/// Indicates if the receiving side of this socket is closed.
|
|
///
|
|
/// The receiving side may be closed if this side disables reading
|
|
/// or if the peer side closes its sending half.
|
|
is_receiving_closed: AtomicBool,
|
|
/// Indicates if the sending side of this socket is closed.
|
|
///
|
|
/// The sending side can only be closed if this side disables writing.
|
|
is_sending_closed: AtomicBool,
|
|
}
|
|
|
|
impl ConnectedStream {
|
|
pub fn new(
|
|
bound_socket: BoundTcpSocket,
|
|
remote_endpoint: IpEndpoint,
|
|
is_new_connection: bool,
|
|
) -> Self {
|
|
Self {
|
|
bound_socket,
|
|
remote_endpoint,
|
|
is_new_connection,
|
|
is_receiving_closed: AtomicBool::new(false),
|
|
is_sending_closed: AtomicBool::new(false),
|
|
}
|
|
}
|
|
|
|
pub fn shutdown(&self, cmd: SockShutdownCmd, pollee: &Pollee) -> Result<()> {
|
|
let mut events = IoEvents::empty();
|
|
|
|
if cmd.shut_read() {
|
|
self.is_receiving_closed.store(true, Ordering::Relaxed);
|
|
events |= IoEvents::IN | IoEvents::RDHUP;
|
|
}
|
|
|
|
if cmd.shut_write() {
|
|
self.is_sending_closed.store(true, Ordering::Relaxed);
|
|
self.bound_socket.close();
|
|
events |= IoEvents::OUT | IoEvents::HUP;
|
|
}
|
|
|
|
pollee.notify(events);
|
|
|
|
Ok(())
|
|
}
|
|
|
|
pub fn try_recv(
|
|
&self,
|
|
writer: &mut dyn MultiWrite,
|
|
_flags: SendRecvFlags,
|
|
) -> Result<(usize, NeedIfacePoll)> {
|
|
let result = self.bound_socket.recv(|socket_buffer| {
|
|
match writer.write(&mut VmReader::from(&*socket_buffer)) {
|
|
Ok(len) => (len, Ok(len)),
|
|
Err(e) => (0, Err(e)),
|
|
}
|
|
});
|
|
|
|
match result {
|
|
Ok((Ok(0), need_poll)) if self.is_receiving_closed.load(Ordering::Relaxed) => {
|
|
Ok((0, need_poll))
|
|
}
|
|
Ok((Ok(0), need_poll)) => {
|
|
debug_assert!(!*need_poll);
|
|
return_errno_with_message!(Errno::EAGAIN, "the receive buffer is empty")
|
|
}
|
|
Ok((Ok(recv_bytes), need_poll)) => Ok((recv_bytes, need_poll)),
|
|
Ok((Err(e), need_poll)) => {
|
|
debug_assert!(!*need_poll);
|
|
Err(e)
|
|
}
|
|
Err(RecvError::Finished) => Ok((0, NeedIfacePoll::FALSE)),
|
|
Err(RecvError::InvalidState) => {
|
|
return_errno_with_message!(Errno::ECONNRESET, "the connection is reset")
|
|
}
|
|
}
|
|
}
|
|
|
|
pub fn try_send(
|
|
&self,
|
|
reader: &mut dyn MultiRead,
|
|
_flags: SendRecvFlags,
|
|
) -> Result<(usize, NeedIfacePoll)> {
|
|
let result = self.bound_socket.send(|socket_buffer| {
|
|
match reader.read(&mut VmWriter::from(socket_buffer)) {
|
|
Ok(len) => (len, Ok(len)),
|
|
Err(e) => (0, Err(e)),
|
|
}
|
|
});
|
|
|
|
match result {
|
|
Ok((Ok(0), need_poll)) => {
|
|
debug_assert!(!*need_poll);
|
|
return_errno_with_message!(Errno::EAGAIN, "the send buffer is full")
|
|
}
|
|
Ok((Ok(sent_bytes), need_poll)) => Ok((sent_bytes, need_poll)),
|
|
Ok((Err(e), need_poll)) => {
|
|
debug_assert!(!*need_poll);
|
|
Err(e)
|
|
}
|
|
Err(SendError::InvalidState) => {
|
|
// FIXME: `EPIPE` is another possibility, which means that the socket is shut down
|
|
// for writing. In that case, we should also trigger a `SIGPIPE` if `MSG_NOSIGNAL`
|
|
// is not specified.
|
|
return_errno_with_message!(Errno::ECONNRESET, "the connection is reset");
|
|
}
|
|
}
|
|
}
|
|
|
|
pub fn local_endpoint(&self) -> IpEndpoint {
|
|
self.bound_socket.local_endpoint().unwrap()
|
|
}
|
|
|
|
pub fn remote_endpoint(&self) -> IpEndpoint {
|
|
self.remote_endpoint
|
|
}
|
|
|
|
pub fn iface(&self) -> &Arc<Iface> {
|
|
self.bound_socket.iface()
|
|
}
|
|
|
|
pub fn check_new(&mut self) -> Result<()> {
|
|
if !self.is_new_connection {
|
|
return_errno_with_message!(Errno::EISCONN, "the socket is already connected");
|
|
}
|
|
|
|
self.is_new_connection = false;
|
|
Ok(())
|
|
}
|
|
|
|
pub(super) fn check_io_events(&self) -> IoEvents {
|
|
self.bound_socket.raw_with(|socket| {
|
|
if socket.is_peer_closed() {
|
|
// Only the sending side of peer socket is closed
|
|
self.is_receiving_closed.store(true, Ordering::Relaxed);
|
|
} else if socket.is_closed() {
|
|
// The sending side of both peer socket and this socket are closed
|
|
self.is_receiving_closed.store(true, Ordering::Relaxed);
|
|
self.is_sending_closed.store(true, Ordering::Relaxed);
|
|
}
|
|
|
|
let is_receiving_closed = self.is_receiving_closed.load(Ordering::Relaxed);
|
|
let is_sending_closed = self.is_sending_closed.load(Ordering::Relaxed);
|
|
|
|
let mut events = IoEvents::empty();
|
|
|
|
// If the receiving side is closed, always add events IN and RDHUP;
|
|
// otherwise, check if the socket can receive.
|
|
if is_receiving_closed {
|
|
events |= IoEvents::IN | IoEvents::RDHUP;
|
|
} else if socket.can_recv() {
|
|
events |= IoEvents::IN;
|
|
}
|
|
|
|
// If the sending side is closed, always add an OUT event;
|
|
// otherwise, check if the socket can send.
|
|
if is_sending_closed || socket.can_send() {
|
|
events |= IoEvents::OUT;
|
|
}
|
|
|
|
// If both sending and receiving sides are closed, add a HUP event.
|
|
if is_receiving_closed && is_sending_closed {
|
|
events |= IoEvents::HUP;
|
|
}
|
|
|
|
events
|
|
})
|
|
}
|
|
|
|
pub(super) fn set_observer(&self, observer: StreamObserver) {
|
|
self.bound_socket.set_observer(observer)
|
|
}
|
|
}
|