wasmer_wasix/syscalls/wasix/
sock_connect.rs1use super::*;
2use crate::syscalls::*;
3
4#[instrument(level = "trace", skip_all, fields(%sock, addr = field::Empty), ret)]
17pub fn sock_connect<M: MemorySize>(
18 mut ctx: FunctionEnvMut<'_, WasiEnv>,
19 sock: WasiFd,
20 addr: WasmPtr<__wasi_addr_port_t, M>,
21) -> Result<Errno, WasiError> {
22 WasiEnv::do_pending_operations(&mut ctx)?;
23
24 let env = ctx.data();
25 let memory = unsafe { env.memory_view(&ctx) };
26 let addr = wasi_try_ok!(crate::net::read_ip_port(&memory, addr));
27 let peer_addr = SocketAddr::new(addr.0, addr.1);
28 Span::current().record("addr", format!("{peer_addr:?}"));
29
30 wasi_try_ok!(sock_connect_internal(&mut ctx, sock, peer_addr)?);
31
32 #[cfg(feature = "journal")]
33 if ctx.data().enable_journal {
34 let local_addr = wasi_try_ok!(__sock_actor(
35 &mut ctx,
36 sock,
37 Rights::empty(),
38 |socket, _| socket.addr_local()
39 ));
40 JournalEffector::save_sock_connect(&mut ctx, sock, local_addr, peer_addr).map_err(
41 |err| {
42 tracing::error!("failed to save sock_connected event - {}", err);
43 WasiError::Exit(ExitCode::from(Errno::Fault))
44 },
45 )?;
46 }
47
48 Ok(Errno::Success)
49}
50
51fn nonblocking_connect_result(status: crate::net::socket::WasiSocketStatus) -> Result<(), Errno> {
52 match status {
53 crate::net::socket::WasiSocketStatus::Opening => Err(Errno::Inprogress),
54 crate::net::socket::WasiSocketStatus::Opened => Ok(()),
55 crate::net::socket::WasiSocketStatus::Closed
56 | crate::net::socket::WasiSocketStatus::Failed => Err(Errno::Notconn),
57 }
58}
59
60pub(crate) fn sock_connect_internal(
61 ctx: &mut FunctionEnvMut<'_, WasiEnv>,
62 sock: WasiFd,
63 addr: SocketAddr,
64) -> Result<Result<(), Errno>, WasiError> {
65 let env = ctx.data();
66 let net = env.net().clone();
67 let tasks = ctx.data().tasks().clone();
68 let nonblocking = match env.state.fs.get_fd(sock) {
69 Ok(fd_entry) => fd_entry.inner.flags.contains(Fdflags::NONBLOCK),
70 Err(err) => return Ok(Err(err)),
71 };
72 wasi_try_ok_ok!(__sock_upgrade(
73 ctx,
74 sock,
75 Rights::SOCK_CONNECT,
76 move |mut socket, flags| async move {
77 socket = socket
79 .auto_bind_udp(tasks.deref(), net.deref())
80 .await?
81 .unwrap_or(socket);
82 socket
83 .connect(
84 tasks.deref(),
85 net.deref(),
86 addr,
87 None,
88 flags.contains(Fdflags::NONBLOCK),
89 )
90 .await
91 }
92 ));
93
94 if nonblocking {
95 let status = match __sock_actor(ctx, sock, Rights::empty(), |socket, _| socket.status()) {
96 Ok(status) => status,
97 Err(err) => return Ok(Err(err)),
98 };
99 return Ok(nonblocking_connect_result(status));
100 }
101
102 Ok(Ok(()))
103}
104
105#[cfg(test)]
106mod tests {
107 use super::nonblocking_connect_result;
108 use crate::net::socket::WasiSocketStatus;
109 use wasmer_wasix_types::wasi::Errno;
110
111 #[test]
112 fn nonblocking_connect_result_maps_socket_states() {
113 assert_eq!(
114 nonblocking_connect_result(WasiSocketStatus::Opening),
115 Err(Errno::Inprogress)
116 );
117 assert_eq!(nonblocking_connect_result(WasiSocketStatus::Opened), Ok(()));
118 assert_eq!(
119 nonblocking_connect_result(WasiSocketStatus::Failed),
120 Err(Errno::Notconn)
121 );
122 assert_eq!(
123 nonblocking_connect_result(WasiSocketStatus::Closed),
124 Err(Errno::Notconn)
125 );
126 }
127}