132 lines
2.9 KiB
C
132 lines
2.9 KiB
C
|
/******************************************************************************
|
||
|
Copyright (c) 2013 by Hugh Bailey <obs.jim@gmail.com>
|
||
|
|
||
|
This software is provided 'as-is', without any express or implied
|
||
|
warranty. In no event will the authors be held liable for any damages
|
||
|
arising from the use of this software.
|
||
|
|
||
|
Permission is granted to anyone to use this software for any purpose,
|
||
|
including commercial applications, and to alter it and redistribute it
|
||
|
freely, subject to the following restrictions:
|
||
|
|
||
|
1. The origin of this software must not be misrepresented; you must not
|
||
|
claim that you wrote the original software. If you use this software
|
||
|
in a product, an acknowledgment in the product documentation would be
|
||
|
appreciated but is not required.
|
||
|
|
||
|
2. Altered source versions must be plainly marked as such, and must not be
|
||
|
misrepresented as being the original software.
|
||
|
|
||
|
3. This notice may not be removed or altered from any source
|
||
|
distribution.
|
||
|
******************************************************************************/
|
||
|
|
||
|
#include <malloc.h>
|
||
|
#include <stdlib.h>
|
||
|
#include <string.h>
|
||
|
#include "base.h"
|
||
|
#include "bmem.h"
|
||
|
|
||
|
static void *a_malloc(size_t size, size_t align)
|
||
|
{
|
||
|
#if defined(_WIN32)
|
||
|
return _aligned_malloc(size, align);
|
||
|
#elif defined (__posix__)
|
||
|
void *ptr;
|
||
|
if (posix_memalign(&ptr, align, size) != 0)
|
||
|
return NULL;
|
||
|
|
||
|
return ptr;
|
||
|
#else
|
||
|
#error unknown OS
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
static void a_free(void *ptr)
|
||
|
{
|
||
|
#if defined(_WIN32)
|
||
|
_aligned_free(ptr);
|
||
|
#elif defined (__posix__)
|
||
|
free(ptr);
|
||
|
#else
|
||
|
#error unknown OS
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
static struct base_allocator alloc = {malloc, realloc, free, a_malloc, a_free};
|
||
|
static size_t num_allocs = 0;
|
||
|
|
||
|
void base_set_allocator(struct base_allocator *defs)
|
||
|
{
|
||
|
memcpy(&alloc, defs, sizeof(struct base_allocator));
|
||
|
}
|
||
|
|
||
|
void *bmalloc(size_t size)
|
||
|
{
|
||
|
void *ptr = alloc.malloc(size);
|
||
|
if (!ptr && !size)
|
||
|
ptr = alloc.malloc(1);
|
||
|
if (!ptr)
|
||
|
bcrash("Out of memory while trying to allocate %lu bytes",
|
||
|
(unsigned long)size);
|
||
|
|
||
|
num_allocs++;
|
||
|
return ptr;
|
||
|
}
|
||
|
|
||
|
void *brealloc(void *ptr, size_t size)
|
||
|
{
|
||
|
if (!ptr)
|
||
|
num_allocs++;
|
||
|
|
||
|
ptr = alloc.realloc(ptr, size);
|
||
|
if (!ptr && !size)
|
||
|
ptr = alloc.realloc(ptr, 1);
|
||
|
if (!ptr)
|
||
|
bcrash("Out of memory while trying to allocate %lu bytes",
|
||
|
(unsigned long)size);
|
||
|
|
||
|
return ptr;
|
||
|
}
|
||
|
|
||
|
void bfree(void *ptr)
|
||
|
{
|
||
|
if (ptr)
|
||
|
num_allocs--;
|
||
|
alloc.free(ptr);
|
||
|
}
|
||
|
|
||
|
void *baligned_malloc(size_t size, size_t align)
|
||
|
{
|
||
|
void *ptr = alloc.aligned_malloc(size, align);
|
||
|
if (!ptr && !size)
|
||
|
ptr = alloc.aligned_malloc(1, align);
|
||
|
if (!ptr)
|
||
|
bcrash("Out of memory while trying to allocate %lu bytes",
|
||
|
(unsigned long)size);
|
||
|
|
||
|
num_allocs++;
|
||
|
return ptr;
|
||
|
}
|
||
|
|
||
|
void baligned_free(void *ptr)
|
||
|
{
|
||
|
if (ptr)
|
||
|
num_allocs--;
|
||
|
alloc.aligned_free(ptr);
|
||
|
}
|
||
|
|
||
|
size_t bnum_allocs(void)
|
||
|
{
|
||
|
return num_allocs;
|
||
|
}
|
||
|
|
||
|
void *bmemdup(const void *ptr, size_t size)
|
||
|
{
|
||
|
void *out = bmalloc(size);
|
||
|
if (size)
|
||
|
memcpy(out, ptr, size);
|
||
|
|
||
|
return out;
|
||
|
}
|