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
use core::{mem::align_of, ptr::write, slice};

/// A memory block on a heap.
pub struct Heap<T> {
    ptr: *mut T,
    len: usize,
}

impl<T> Heap<T> {
    /// Creates a heap.
    pub fn new(len: usize, default: impl Fn() -> T) -> Self {
        let mut this = Self {
            ptr: unsafe { libc::malloc(len * align_of::<T>()) } as _,
            len,
        };

        for x in this.as_slice_mut() {
            unsafe { write(x, default()) };
        }

        this
    }

    /// Returns a slice.
    pub fn as_slice(&mut self) -> &[T] {
        unsafe { slice::from_raw_parts(self.ptr as _, self.len) }
    }

    /// Returns a mutable slice.
    pub fn as_slice_mut(&mut self) -> &mut [T] {
        unsafe { slice::from_raw_parts_mut(self.ptr as _, self.len) }
    }
}

impl<T> Drop for Heap<T> {
    fn drop(&mut self) {
        unsafe { libc::free(self.ptr as _) }
    }
}

#[cfg(test)]
mod tests {
    use super::*;

    #[test]
    fn new() {
        Heap::<usize>::new(42, || 42);
    }

    #[test]
    fn new_zero_sized() {
        Heap::<usize>::new(0, || 42);
    }
}