1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
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);
}
}
}