use std::os::unix::io::RawFd;
use std::{fmt, io};
use crate::sys;
pub(crate) fn execute(
    fd: RawFd,
    opcode: libc::c_uint,
    arg: *const libc::c_void,
    len: libc::c_uint,
) -> io::Result<i32> {
    unsafe { sys::io_uring_register(fd, opcode, arg, len) }
}
pub struct Probe(ProbeAndOps);
#[repr(C)]
struct ProbeAndOps(sys::io_uring_probe, [sys::io_uring_probe_op; Probe::COUNT]);
impl Probe {
    pub(crate) const COUNT: usize = 256;
    pub fn new() -> Probe {
        Probe(ProbeAndOps(
            sys::io_uring_probe::default(),
            [sys::io_uring_probe_op::default(); Probe::COUNT],
        ))
    }
    #[inline]
    pub(crate) fn as_mut_ptr(&mut self) -> *mut sys::io_uring_probe {
        &mut (self.0).0
    }
    pub fn is_supported(&self, opcode: u8) -> bool {
        unsafe {
            let probe = &(self.0).0;
            if opcode <= probe.last_op {
                let ops = probe.ops.as_slice(Self::COUNT);
                ops[opcode as usize].flags & (sys::IO_URING_OP_SUPPORTED as u16) != 0
            } else {
                false
            }
        }
    }
}
impl Default for Probe {
    #[inline]
    fn default() -> Probe {
        Probe::new()
    }
}
impl fmt::Debug for Probe {
    fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
        struct Op<'a>(&'a sys::io_uring_probe_op);
        impl fmt::Debug for Op<'_> {
            #[inline]
            fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
                f.debug_struct("Op").field("code", &self.0.op).finish()
            }
        }
        let probe = &(self.0).0;
        let list = unsafe { probe.ops.as_slice(probe.last_op as usize + 1) };
        let list = list
            .iter()
            .filter(|op| op.flags & (sys::IO_URING_OP_SUPPORTED as u16) != 0)
            .map(Op);
        f.debug_set().entries(list).finish()
    }
}
#[repr(transparent)]
pub struct Restriction(sys::io_uring_restriction);
#[inline(always)]
fn res_zeroed() -> sys::io_uring_restriction {
    unsafe { std::mem::zeroed() }
}
impl Restriction {
    pub fn register_op(op: u8) -> Restriction {
        let mut res = res_zeroed();
        res.opcode = sys::IORING_RESTRICTION_REGISTER_OP as _;
        res.__bindgen_anon_1.register_op = op;
        Restriction(res)
    }
    pub fn sqe_op(op: u8) -> Restriction {
        let mut res = res_zeroed();
        res.opcode = sys::IORING_RESTRICTION_SQE_OP as _;
        res.__bindgen_anon_1.sqe_op = op;
        Restriction(res)
    }
    pub fn sqe_flags_allowed(flags: u8) -> Restriction {
        let mut res = res_zeroed();
        res.opcode = sys::IORING_RESTRICTION_SQE_FLAGS_ALLOWED as _;
        res.__bindgen_anon_1.sqe_flags = flags;
        Restriction(res)
    }
    pub fn sqe_flags_required(flags: u8) -> Restriction {
        let mut res = res_zeroed();
        res.opcode = sys::IORING_RESTRICTION_SQE_FLAGS_REQUIRED as _;
        res.__bindgen_anon_1.sqe_flags = flags;
        Restriction(res)
    }
}
pub const SKIP_FILE: RawFd = sys::IORING_REGISTER_FILES_SKIP;
#[test]
fn test_probe_layout() {
    use std::alloc::Layout;
    use std::mem;
    let probe = Probe::new();
    assert_eq!(
        Layout::new::<sys::io_uring_probe>().size()
            + mem::size_of::<sys::io_uring_probe_op>() * 256,
        Layout::for_value(&probe.0).size()
    );
    assert_eq!(
        Layout::new::<sys::io_uring_probe>().align(),
        Layout::for_value(&probe.0).align()
    );
}