use std::{ffi::c_void, fmt::Display, ptr::NonNull};
use crate::ffi::abort_on_panic;
#[repr(C)]
#[derive(Clone, Copy, Debug)]
pub struct CVec {
pub ptr: *mut c_void,
pub len: usize,
pub cap: usize,
}
impl CVec {
#[must_use]
pub fn empty() -> Self {
Self {
ptr: NonNull::<u8>::dangling().as_ptr().cast::<c_void>(),
len: 0,
cap: 0,
}
}
}
impl<T> From<Vec<T>> for CVec {
fn from(mut data: Vec<T>) -> Self {
if data.is_empty() {
Self::empty()
} else {
let len = data.len();
let cap = data.capacity();
let ptr = data.as_mut_ptr();
#[allow(
clippy::mem_forget,
reason = "intentional ownership transfer to C; matching CVec::drop reclaims via Vec::from_raw_parts"
)]
std::mem::forget(data);
Self {
ptr: ptr.cast::<std::ffi::c_void>(),
len,
cap,
}
}
}
}
impl Display for CVec {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
write!(
f,
"CVec {{ ptr: {:?}, len: {}, cap: {} }}",
self.ptr, self.len, self.cap,
)
}
}
#[cfg(feature = "ffi")]
#[unsafe(no_mangle)]
pub extern "C" fn cvec_new() -> CVec {
abort_on_panic(CVec::empty)
}
#[cfg(test)]
mod tests {
use rstest::*;
use super::CVec;
#[rstest]
#[allow(unused_assignments)]
fn access_values_test() {
let test_data = vec![1_u64, 2, 3];
let mut vec_len = 0;
let mut vec_cap = 0;
let cvec: CVec = {
let data = test_data.clone();
vec_len = data.len();
vec_cap = data.capacity();
data.into()
};
let CVec { ptr, len, cap } = cvec;
assert_eq!(len, vec_len);
assert_eq!(cap, vec_cap);
let data = ptr.cast::<u64>();
#[allow(
clippy::multiple_unsafe_ops_per_block,
reason = "test asserts on three pointer reads in sequence"
)]
unsafe {
assert_eq!(*data, test_data[0]);
assert_eq!(*data.add(1), test_data[1]);
assert_eq!(*data.add(2), test_data[2]);
}
unsafe {
let _ = Vec::from_raw_parts(ptr.cast::<u64>(), len, cap);
}
}
#[rstest]
fn empty_vec_should_give_dangling_ptr() {
let data: Vec<u64> = vec![];
let cvec: CVec = data.into();
assert!(!cvec.ptr.is_null());
assert_eq!(cvec.len, 0);
assert_eq!(cvec.cap, 0);
}
}