use std::ffi::{c_char, CStr};
use std::slice;
use crate::platform_types::size_t;
macro_rules! extern_container_arg {
(nofail mut $name: ident) => {
let mut $name = $name.opencv_into_extern_container_nofail();
};
(nofail $name: ident) => {
let $name = $name.opencv_into_extern_container_nofail();
};
(mut $name: ident) => {
let mut $name = $name.opencv_into_extern_container()?;
};
($name: ident) => {
let $name = $name.opencv_into_extern_container()?;
};
}
macro_rules! string_arg_output_send {
(via $name_via: ident) => {
let mut $name_via = ::std::ptr::null_mut();
};
}
macro_rules! string_arg_output_receive {
($name_via: ident => $name: ident) => {
*$name = unsafe { $crate::templ::receive_string($name_via.cast()) };
};
}
macro_rules! callback_arg {
($tr_name: ident($($tr_arg_name: ident: $tr_arg_type: ty),*) -> $tr_ret: ty => $tr_userdata_name: ident in $callback_name: ident($($fw_arg_name: ident: $fw_arg_type: ty),*) -> $fw_ret: ty) => {
unsafe extern "C" fn trampoline($($tr_arg_name: $tr_arg_type),*) -> $tr_ret {
let callback: *mut Box<dyn FnMut($($fw_arg_type),*) -> $fw_ret> = $tr_userdata_name.cast();
let out = unsafe { (*callback)($($fw_arg_name),*) };
out
}
let $tr_name = if $callback_name.is_some() {
Some(trampoline as unsafe extern "C" fn($($tr_arg_type),*) -> $tr_ret)
} else {
None
};
}
}
macro_rules! userdata_arg {
($userdata_name: ident: $userdata_type: ty => $callback_name: ident) => {
let $userdata_name: $userdata_type = if let Some(callback) = $callback_name {
Box::into_raw(Box::new(callback)).cast()
} else {
::std::ptr::null_mut() };
};
}
macro_rules! input_array_arg {
($name: ident) => {
let $name = $name.input_array()?;
};
}
macro_rules! output_array_arg {
($name: ident) => {
let $name = $name.output_array()?;
};
}
macro_rules! input_output_array_arg {
($name: ident) => {
let $name = $name.input_output_array()?;
};
}
macro_rules! string_array_arg {
($name: ident) => {
let $name = $name
.iter()
.map(|x| ::std::ffi::CString::new(*x).expect("invalid C string"))
.collect::<::std::vec::Vec<_>>();
let $name = $name.iter().map(|x| x.as_ptr()).collect::<::std::vec::Vec<_>>();
};
}
#[allow(unused_macros)]
macro_rules! string_array_arg_mut {
($name: ident) => {
let $name = $name
.iter()
.map(|x| ::std::ffi::CString::new(*x).expect("invalid C string"))
.collect::<::std::vec::Vec<_>>();
let mut $name = $name.iter().map(|x| x.as_ptr().cast_mut()).collect::<::std::vec::Vec<_>>();
};
}
macro_rules! smart_ptr_option_arg {
(ref $name: ident) => {
let null = if $name.is_none() {
Some(unsafe { $crate::core::Ptr::new_null() })
} else {
None
};
let $name = $name.or(null.as_ref()).expect("Nullability should have been checked");
};
($name: ident) => {
let $name = $name.unwrap_or_else(|| unsafe { $crate::core::Ptr::new_null() });
};
}
macro_rules! return_send {
(via $name: ident) => {
let mut $name = ::std::mem::MaybeUninit::uninit();
};
}
macro_rules! return_receive {
($name_via: ident => $name: ident) => {
let $name = unsafe { $name_via.assume_init() };
};
}
#[inline(always)]
pub unsafe fn ocvrs_create_string(s: *const c_char) -> *mut String {
let s = unsafe { CStr::from_ptr(s) }.to_string_lossy().into_owned();
Box::into_raw(Box::new(s))
}
#[inline(always)]
pub unsafe fn ocvrs_create_byte_string(v: *const u8, len: size_t) -> *mut Vec<u8> {
let byte_slice = if v.is_null() {
&[]
} else {
unsafe { slice::from_raw_parts(v, len) }
};
let v = byte_slice.to_vec();
Box::into_raw(Box::new(v))
}
#[inline]
pub unsafe fn receive_string<T>(s: *mut T) -> T {
if s.is_null() {
panic!("Got null pointer for receive_string()");
}
unsafe { *Box::from_raw(s) }
}