#include "config.h" #include "almalloc.h" #include #include #include #include #include #ifdef HAVE_MALLOC_H #include #endif void *al_malloc(size_t alignment, size_t size) { assert((alignment & (alignment-1)) == 0); alignment = std::max(alignment, alignof(std::max_align_t)); #if defined(HAVE_STD_ALIGNED_ALLOC) size = (size+(alignment-1))&~(alignment-1); return std::aligned_alloc(alignment, size); #elif defined(HAVE_POSIX_MEMALIGN) void *ret{}; if(posix_memalign(&ret, alignment, size) == 0) return ret; return nullptr; #elif defined(HAVE__ALIGNED_MALLOC) return _aligned_malloc(size, alignment); #else size_t total_size{size + alignment-1 + sizeof(void*)}; void *base{std::malloc(total_size)}; if(base != nullptr) { void *aligned_ptr{static_cast(base) + sizeof(void*)}; total_size -= sizeof(void*); std::align(alignment, size, aligned_ptr, total_size); *(static_cast(aligned_ptr)-1) = base; base = aligned_ptr; } return base; #endif } void *al_calloc(size_t alignment, size_t size) { void *ret{al_malloc(alignment, size)}; if(ret) std::memset(ret, 0, size); return ret; } void al_free(void *ptr) noexcept { #if defined(HAVE_STD_ALIGNED_ALLOC) || defined(HAVE_POSIX_MEMALIGN) std::free(ptr); #elif defined(HAVE__ALIGNED_MALLOC) _aligned_free(ptr); #else if(ptr != nullptr) std::free(*(static_cast(ptr) - 1)); #endif }