use crate::{FromInner, Inner, IntoInner};
use uv::uv_write_t;
callbacks! {
pub WriteCB(req: WriteReq, status: crate::Result<u32>);
}
pub(crate) struct WriteDataFields<'a> {
bufs_ptr: *mut uv::uv_buf_t,
bufs_len: usize,
bufs_capacity: usize,
write_cb: WriteCB<'a>,
}
pub(crate) extern "C" fn uv_write_cb(req: *mut uv_write_t, status: std::os::raw::c_int) {
let dataptr = crate::Req::get_data(uv_handle!(req));
if !dataptr.is_null() {
unsafe {
if let super::WriteData(d) = &mut *dataptr {
let status = if status < 0 {
Err(crate::Error::from_inner(status as uv::uv_errno_t))
} else {
Ok(status as _)
};
d.write_cb.call(req.into_inner(), status);
}
}
}
let mut req = WriteReq::from_inner(req);
req.destroy();
}
#[derive(Clone, Copy)]
pub struct WriteReq {
req: *mut uv_write_t,
pub(crate) bufs_ptr: *const uv::uv_buf_t,
}
impl WriteReq {
pub fn new<CB: Into<WriteCB<'static>>>(
bufs: &[impl crate::BufTrait],
cb: CB,
) -> crate::Result<WriteReq> {
let layout = std::alloc::Layout::new::<uv_write_t>();
let req = unsafe { std::alloc::alloc(layout) as *mut uv_write_t };
if req.is_null() {
return Err(crate::Error::ENOMEM);
}
let (bufs_ptr, bufs_len, bufs_capacity) = bufs.into_inner();
let write_cb = cb.into();
crate::Req::initialize_data(
uv_handle!(req),
super::WriteData(WriteDataFields {
bufs_ptr,
bufs_len,
bufs_capacity,
write_cb,
}),
);
Ok(WriteReq { req, bufs_ptr })
}
pub fn handle(&self) -> crate::StreamHandle {
unsafe { (*self.req).handle }.into_inner()
}
pub fn send_handle(&self) -> crate::StreamHandle {
unsafe { (*self.req).send_handle }.into_inner()
}
pub fn destroy(&mut self) {
let dataptr = crate::Req::get_data(uv_handle!(self.req));
if !dataptr.is_null() {
if let super::WriteData(d) = unsafe { &mut *dataptr } {
if !d.bufs_ptr.is_null() {
unsafe {
std::mem::drop(Vec::from_raw_parts(d.bufs_ptr, d.bufs_len, d.bufs_capacity))
};
}
}
}
crate::Req::free_data(uv_handle!(self.req));
let layout = std::alloc::Layout::new::<uv_write_t>();
unsafe { std::alloc::dealloc(self.req as _, layout) };
}
}
impl FromInner<*mut uv_write_t> for WriteReq {
fn from_inner(req: *mut uv_write_t) -> WriteReq {
WriteReq {
req,
bufs_ptr: std::ptr::null(),
}
}
}
impl Inner<*mut uv_write_t> for WriteReq {
fn inner(&self) -> *mut uv_write_t {
self.req
}
}
impl Inner<*mut uv::uv_req_t> for WriteReq {
fn inner(&self) -> *mut uv::uv_req_t {
uv_handle!(self.req)
}
}
impl From<WriteReq> for crate::Req {
fn from(write: WriteReq) -> crate::Req {
crate::Req::from_inner(Inner::<*mut uv::uv_req_t>::inner(&write))
}
}
impl crate::ToReq for WriteReq {
fn to_req(&self) -> crate::Req {
crate::Req::from_inner(Inner::<*mut uv::uv_req_t>::inner(self))
}
}
impl crate::ReqTrait for WriteReq {}