use core::ffi::c_int;
use ax_errno::AxResult;
use bitflags::bitflags;
use linux_raw_sys::general::{O_CLOEXEC, O_NONBLOCK};
use starry_vm::VmMutPtr;
use crate::file::{FileLike, Pipe, close_file_like};
bitflags! {
#[derive(Debug, Clone, Copy, Default)]
pub struct PipeFlags: u32 {
const CLOEXEC = O_CLOEXEC;
const NONBLOCK = O_NONBLOCK;
}
}
pub fn sys_pipe2(fds: *mut [c_int; 2], flags: u32) -> AxResult<isize> {
let flags = {
let new_flags = PipeFlags::from_bits_truncate(flags);
if new_flags.bits() != flags {
warn!("sys_pipe2 <= unrecognized flags: {flags}");
}
new_flags
};
let cloexec = flags.contains(PipeFlags::CLOEXEC);
let (read_end, write_end) = Pipe::new();
if flags.contains(PipeFlags::NONBLOCK) {
read_end.set_nonblocking(true)?;
write_end.set_nonblocking(true)?;
}
let read_fd = read_end.add_to_fd_table(cloexec)?;
let write_fd = write_end
.add_to_fd_table(cloexec)
.inspect_err(|_| close_file_like(read_fd).unwrap())?;
fds.vm_write([read_fd, write_fd])?;
debug!(
"sys_pipe2 <= fds: {:?}, flags: {:?}",
[read_fd, write_fd],
flags
);
Ok(0)
}