use std::{mem, ptr, slice};
use libc::{
mlock as libc_mlock, mmap as libc_mmap, munmap as libc_munmap, MAP_ANONYMOUS, MAP_FAILED,
MAP_PRIVATE, PROT_READ, PROT_WRITE,
};
pub const PAGE_SIZE: usize = 1 << 12;
pub struct ResultArray<T: Sized> {
ptr: *mut u8,
size: usize,
cap: usize, len: usize,
_phantom: core::marker::PhantomData<T>,
}
impl<T: Sized> ResultArray<T> {
pub fn new(cap: usize) -> Self {
let size = cap * mem::size_of::<T>();
assert!(size % PAGE_SIZE == 0);
assert!(size < (std::isize::MAX as usize));
let mapped = unsafe {
let addr = libc_mmap(
ptr::null_mut(),
size,
PROT_READ | PROT_WRITE,
MAP_PRIVATE | MAP_ANONYMOUS,
-1,
0,
);
if addr == MAP_FAILED {
panic!("Unable to mmap");
}
addr as *mut _
};
unsafe {
let ret = libc_mlock(mapped as *const _, size);
assert_eq!(ret, 0);
}
Self {
ptr: mapped,
size,
len: 0,
cap,
_phantom: core::marker::PhantomData,
}
}
pub fn iter(&self) -> slice::Iter<T> {
self.as_slice().iter()
}
pub fn push(&mut self, item: T) {
assert!(self.len < self.cap);
let ptr = unsafe { (self.ptr as *mut T).add(self.len) };
self.len += 1;
unsafe {
std::ptr::write(ptr, item);
}
}
fn as_slice(&self) -> &[T] {
unsafe { core::slice::from_raw_parts(self.ptr as *const T, self.len) }
}
fn pop(&mut self) {
assert!(self.len > 0);
self.len -= 1;
let ptr = unsafe { self.ptr.add(self.len) };
unsafe {
let _ = std::ptr::read(ptr as *const T);
}
}
}
impl<T: Sized> Drop for ResultArray<T> {
fn drop(&mut self) {
while self.len > 0 {
self.pop();
}
unsafe {
libc_munmap(self.ptr as *mut _, self.size);
}
}
}