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
use std::alloc::{GlobalAlloc, Layout};
use crate::raw;
/// Panics with a message without using an allocator.
/// Useful when using the allocator should be avoided or it is
/// inaccessible. The default [std::panic] performs allocations and so
/// will cause a double panic without a meaningful message if the
/// allocator can't be used. This function makes sure we can panic with
/// a reasonable message even without the allocator working.
fn allocation_free_panic(message: &'static str) -> ! {
use std::os::unix::io::AsRawFd;
let _ = nix::unistd::write(std::io::stderr().as_raw_fd(), message.as_bytes());
std::process::abort();
}
const REDIS_ALLOCATOR_NOT_AVAILABLE_MESSAGE: &str =
"Critical error: the Redis Allocator isn't available.\n";
/// Defines the Redis allocator. This allocator delegates the allocation
/// and deallocation tasks to the Redis server when available, otherwise
/// it panics.
#[derive(Copy, Clone)]
pub struct RedisAlloc;
unsafe impl GlobalAlloc for RedisAlloc {
unsafe fn alloc(&self, layout: Layout) -> *mut u8 {
/*
* To make sure the memory allocation by Redis is aligned to the according to the layout,
* we need to align the size of the allocation to the layout.
*
* "Memory is conceptually broken into equal-sized chunks,
* where the chunk size is a power of two that is greater than the page size.
* Chunks are always aligned to multiples of the chunk size.
* This alignment makes it possible to find metadata for user objects very quickly."
*
* From: https://linux.die.net/man/3/jemalloc
*/
let size = (layout.size() + layout.align() - 1) & (!(layout.align() - 1));
match raw::RedisModule_Alloc {
Some(alloc) => alloc(size).cast(),
None => allocation_free_panic(REDIS_ALLOCATOR_NOT_AVAILABLE_MESSAGE),
}
}
unsafe fn dealloc(&self, ptr: *mut u8, _layout: Layout) {
match raw::RedisModule_Free {
Some(f) => f(ptr.cast()),
None => allocation_free_panic(REDIS_ALLOCATOR_NOT_AVAILABLE_MESSAGE),
};
}
}