#[cfg(unix)]
use std::os::unix::prelude::OpenOptionsExt;
use std::{io, path::Path};
#[cfg(windows)]
use windows_sys::Win32::{
Foundation::{ERROR_INVALID_PARAMETER, GENERIC_READ, GENERIC_WRITE},
Security::SECURITY_ATTRIBUTES,
Storage::FileSystem::{
CREATE_ALWAYS, CREATE_NEW, FILE_FLAG_OPEN_REPARSE_POINT, FILE_GENERIC_WRITE,
FILE_SHARE_DELETE, FILE_SHARE_READ, FILE_SHARE_WRITE, FILE_WRITE_DATA, OPEN_ALWAYS,
OPEN_EXISTING, TRUNCATE_EXISTING,
},
};
use crate::{
driver::{op::Op, shared_fd::SharedFd},
fs::File,
};
#[derive(Debug, Clone)]
pub struct OpenOptions {
read: bool,
write: bool,
append: bool,
truncate: bool,
create: bool,
create_new: bool,
#[cfg(unix)]
pub(crate) mode: libc::mode_t,
#[cfg(unix)]
pub(crate) custom_flags: libc::c_int,
#[cfg(windows)]
pub(crate) custom_flags: u32,
#[cfg(windows)]
pub(crate) access_mode: Option<u32>,
#[cfg(windows)]
pub(crate) attributes: u32,
#[cfg(windows)]
pub(crate) share_mode: u32,
#[cfg(windows)]
pub(crate) security_qos_flags: u32,
#[cfg(windows)]
pub(crate) security_attributes: *mut SECURITY_ATTRIBUTES,
}
impl OpenOptions {
#[allow(clippy::new_without_default)]
pub fn new() -> OpenOptions {
OpenOptions {
read: false,
write: false,
append: false,
truncate: false,
create: false,
create_new: false,
#[cfg(unix)]
mode: 0o666,
#[cfg(unix)]
custom_flags: 0,
#[cfg(windows)]
custom_flags: 0,
#[cfg(windows)]
access_mode: None,
#[cfg(windows)]
share_mode: FILE_SHARE_READ | FILE_SHARE_WRITE | FILE_SHARE_DELETE,
#[cfg(windows)]
attributes: 0,
#[cfg(windows)]
security_qos_flags: 0,
#[cfg(windows)]
security_attributes: std::ptr::null_mut(),
}
}
pub fn read(&mut self, read: bool) -> &mut OpenOptions {
self.read = read;
self
}
pub fn write(&mut self, write: bool) -> &mut OpenOptions {
self.write = write;
self
}
pub fn append(&mut self, append: bool) -> &mut OpenOptions {
self.append = append;
self
}
pub fn truncate(&mut self, truncate: bool) -> &mut OpenOptions {
self.truncate = truncate;
self
}
pub fn create(&mut self, create: bool) -> &mut OpenOptions {
self.create = create;
self
}
pub fn create_new(&mut self, create_new: bool) -> &mut OpenOptions {
self.create_new = create_new;
self
}
pub async fn open(&self, path: impl AsRef<Path>) -> io::Result<File> {
let op = Op::open(path.as_ref(), self)?;
let completion = op.await;
Ok(File::from_shared_fd(SharedFd::new_without_register(
completion.meta.result? as _,
)))
}
#[cfg(unix)]
pub(crate) fn access_mode(&self) -> io::Result<libc::c_int> {
match (self.read, self.write, self.append) {
(true, false, false) => Ok(libc::O_RDONLY),
(false, true, false) => Ok(libc::O_WRONLY),
(true, true, false) => Ok(libc::O_RDWR),
(false, _, true) => Ok(libc::O_WRONLY | libc::O_APPEND),
(true, _, true) => Ok(libc::O_RDWR | libc::O_APPEND),
(false, false, false) => Err(io::Error::from_raw_os_error(libc::EINVAL)),
}
}
#[cfg(windows)]
pub(crate) fn access_mode(&self) -> io::Result<u32> {
match (self.read, self.write, self.append, self.access_mode) {
(.., Some(mode)) => Ok(mode),
(true, false, false, None) => Ok(GENERIC_READ),
(false, true, false, None) => Ok(GENERIC_WRITE),
(true, true, false, None) => Ok(GENERIC_READ | GENERIC_WRITE),
(false, _, true, None) => Ok(FILE_GENERIC_WRITE & !FILE_WRITE_DATA),
(true, _, true, None) => Ok(GENERIC_READ | (FILE_GENERIC_WRITE & !FILE_WRITE_DATA)),
(false, false, false, None) => {
Err(io::Error::from_raw_os_error(ERROR_INVALID_PARAMETER as _))
}
}
}
#[cfg(unix)]
pub(crate) fn creation_mode(&self) -> io::Result<libc::c_int> {
match (self.write, self.append) {
(true, false) => {}
(false, false) => {
if self.truncate || self.create || self.create_new {
return Err(io::Error::from_raw_os_error(libc::EINVAL));
}
}
(_, true) => {
if self.truncate && !self.create_new {
return Err(io::Error::from_raw_os_error(libc::EINVAL));
}
}
}
Ok(match (self.create, self.truncate, self.create_new) {
(false, false, false) => 0,
(true, false, false) => libc::O_CREAT,
(false, true, false) => libc::O_TRUNC,
(true, true, false) => libc::O_CREAT | libc::O_TRUNC,
(_, _, true) => libc::O_CREAT | libc::O_EXCL,
})
}
#[cfg(windows)]
pub(crate) fn creation_mode(&self) -> io::Result<u32> {
match (self.write, self.append) {
(true, false) => {}
(false, false) => {
if self.truncate || self.create || self.create_new {
return Err(io::Error::from_raw_os_error(ERROR_INVALID_PARAMETER as _));
}
}
(_, true) => {
if self.truncate && !self.create_new {
return Err(io::Error::from_raw_os_error(ERROR_INVALID_PARAMETER as _));
}
}
}
Ok(match (self.create, self.truncate, self.create_new) {
(false, false, false) => OPEN_EXISTING,
(true, false, false) => OPEN_ALWAYS,
(false, true, false) => TRUNCATE_EXISTING,
(true, true, false) => CREATE_ALWAYS,
(_, _, true) => CREATE_NEW,
})
}
#[cfg(windows)]
pub(crate) fn get_flags_and_attributes(&self) -> u32 {
self.custom_flags
| self.attributes
| self.security_qos_flags
| if self.create_new {
FILE_FLAG_OPEN_REPARSE_POINT as _
} else {
0
}
}
}
#[cfg(unix)]
impl OpenOptionsExt for OpenOptions {
fn mode(&mut self, mode: u32) -> &mut Self {
self.mode = mode as libc::mode_t;
self
}
fn custom_flags(&mut self, flags: i32) -> &mut Self {
self.custom_flags = flags as libc::c_int;
self
}
}