use std::{ops::Deref, ptr, slice};
use netcorehost::{
pdcstr,
hostfxr::AssemblyDelegateLoader,
};
#[repr(C)]
pub struct RawVec<T> {
pub(crate) data: *mut T,
pub(crate) len: usize,
pub(crate) capacity: usize,
}
unsafe impl<T> Send for RawVec<T> {}
unsafe impl<T> Sync for RawVec<T> {}
impl<T> RawVec<T> {
pub fn free(&mut self) {
if !self.data.is_null() {
let owned = unsafe { Box::from_raw(self.data) };
drop(owned);
self.capacity = 0;
self.len = 0;
self.data = ptr::null_mut();
}
}
pub fn from_vec(buf: Vec<T>) -> Self {
let capacity = buf.capacity();
let len = buf.len();
let data = buf.leak();
{
RawVec {
data: data.as_mut_ptr(),
len,
capacity,
}
}
}
}
impl<T> Deref for RawVec<T> {
type Target = [T];
fn deref(&self) -> &Self::Target {
unsafe { slice::from_raw_parts_mut(self.data, self.len) }
}
}
impl<T> Drop for RawVec<T> {
fn drop(&mut self) {
self.free()
}
}
pub(crate) extern "system" fn rust_allocate_memory(size: usize, vec: *mut RawVec<u8>) {
let buf = vec![0; size];
unsafe {
*vec = RawVec::from_vec(buf)
};
}
pub fn configure_allocator(delegate_loader: &AssemblyDelegateLoader) {
let set_rust_allocate_memory = delegate_loader
.get_function_with_unmanaged_callers_only::<fn(extern "system" fn(usize, *mut RawVec<u8>))>(
pdcstr!("librawfilereader.Exports, librawfilereader"),
pdcstr!("SetForeignAllocateMemory"),
)
.unwrap();
set_rust_allocate_memory(rust_allocate_memory);
}