#![allow(unused_qualifications)]
use super::inotify;
pub use crate::backend::fs::inotify::{CreateFlags, ReadFlags, WatchFlags};
use crate::backend::fs::syscalls;
use crate::fd::{AsFd, OwnedFd};
use crate::ffi::CStr;
use crate::io;
use crate::io::{read, Errno};
use core::mem::{align_of, size_of, MaybeUninit};
use linux_raw_sys::general::inotify_event;
#[doc(alias = "inotify_init1")]
#[inline]
pub fn init(flags: inotify::CreateFlags) -> io::Result<OwnedFd> {
syscalls::inotify_init1(flags)
}
#[doc(alias = "inotify_add_watch")]
#[inline]
pub fn add_watch<P: crate::path::Arg, Fd: AsFd>(
inot: Fd,
path: P,
flags: inotify::WatchFlags,
) -> io::Result<i32> {
path.into_with_c_str(|path| syscalls::inotify_add_watch(inot.as_fd(), path, flags))
}
#[doc(alias = "inotify_rm_watch")]
#[inline]
pub fn remove_watch<Fd: AsFd>(inot: Fd, wd: i32) -> io::Result<()> {
syscalls::inotify_rm_watch(inot.as_fd(), wd)
}
pub struct Reader<'buf, Fd: AsFd> {
fd: Fd,
buf: &'buf mut [MaybeUninit<u8>],
initialized: usize,
offset: usize,
}
impl<'buf, Fd: AsFd> Reader<'buf, Fd> {
pub fn new(fd: Fd, buf: &'buf mut [MaybeUninit<u8>]) -> Self {
Self {
fd,
buf: {
let offset = buf.as_ptr().align_offset(align_of::<inotify_event>());
if offset < buf.len() {
&mut buf[offset..]
} else {
&mut []
}
},
initialized: 0,
offset: 0,
}
}
}
#[doc(alias = "inotify_event")]
#[derive(Debug)]
pub struct Event<'a> {
wd: i32,
events: ReadFlags,
cookie: u32,
file_name: Option<&'a CStr>,
}
impl<'a> Event<'a> {
#[inline]
pub fn wd(&self) -> i32 {
self.wd
}
#[inline]
#[doc(alias = "mask")]
pub fn events(&self) -> ReadFlags {
self.events
}
#[inline]
pub fn cookie(&self) -> u32 {
self.cookie
}
#[inline]
pub fn file_name(&self) -> Option<&CStr> {
self.file_name
}
}
impl<'buf, Fd: AsFd> Reader<'buf, Fd> {
#[allow(unsafe_code)]
#[allow(clippy::should_implement_trait)]
pub fn next(&mut self) -> io::Result<Event<'_>> {
if self.is_buffer_empty() {
match read(self.fd.as_fd(), &mut *self.buf).map(|(init, _)| init.len()) {
Ok(0) => return Err(Errno::INVAL),
Ok(bytes_read) => {
self.initialized = bytes_read;
self.offset = 0;
}
Err(e) => return Err(e),
}
}
let ptr = self.buf[self.offset..].as_ptr();
let event = unsafe { &*ptr.cast::<inotify_event>() };
self.offset += size_of::<inotify_event>() + usize::try_from(event.len).unwrap();
Ok(Event {
wd: event.wd,
events: ReadFlags::from_bits_retain(event.mask),
cookie: event.cookie,
file_name: if event.len > 0 {
Some(unsafe { CStr::from_ptr(event.name.as_ptr().cast()) })
} else {
None
},
})
}
pub fn is_buffer_empty(&self) -> bool {
self.offset >= self.initialized
}
}