asterinas/kernel/src/syscall/recvfrom.rs
2024-12-16 21:12:08 +08:00

47 lines
1.3 KiB
Rust

// SPDX-License-Identifier: MPL-2.0
use super::SyscallReturn;
use crate::{
fs::file_table::FileDesc,
net::socket::SendRecvFlags,
prelude::*,
util::net::{get_socket_from_fd, write_socket_addr_to_user},
};
pub fn sys_recvfrom(
sockfd: FileDesc,
buf: Vaddr,
len: usize,
flags: i32,
src_addr: Vaddr,
addrlen_ptr: Vaddr,
ctx: &Context,
) -> Result<SyscallReturn> {
let flags = SendRecvFlags::from_bits_truncate(flags);
debug!("sockfd = {sockfd}, buf = 0x{buf:x}, len = {len}, flags = {flags:?}, src_addr = 0x{src_addr:x}, addrlen_ptr = 0x{addrlen_ptr:x}");
let socket = get_socket_from_fd(sockfd)?;
let mut writers = {
let vm_space = ctx.process.root_vmar().vm_space();
vm_space.writer(buf, len)?
};
let (recv_size, message_header) =
socket
.recvmsg(&mut writers, flags)
.map_err(|err| match err.error() {
// FIXME: `recvfrom` should not be restarted if a timeout has been set on the socket using `setsockopt`.
Errno::EINTR => Error::new(Errno::ERESTARTSYS),
_ => err,
})?;
if let Some(socket_addr) = message_header.addr()
&& src_addr != 0
{
write_socket_addr_to_user(socket_addr, src_addr, addrlen_ptr)?;
}
Ok(SyscallReturn::Return(recv_size as _))
}