2008-04-28 21:44:54 +00:00
|
|
|
/*
|
|
|
|
* memory.c: safer memory allocation
|
|
|
|
*
|
|
|
|
* Copyright (C) 2008 Daniel P. Berrange
|
|
|
|
*
|
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
|
|
* License as published by the Free Software Foundation; either
|
|
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This library is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
* Lesser General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
|
|
* License along with this library; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2008-04-29 13:37:20 +00:00
|
|
|
#include <config.h>
|
2008-04-28 21:44:54 +00:00
|
|
|
#include <stdlib.h>
|
2008-05-06 21:44:12 +00:00
|
|
|
#include <stddef.h>
|
2008-04-28 21:44:54 +00:00
|
|
|
|
|
|
|
#include "memory.h"
|
|
|
|
|
|
|
|
|
2008-05-29 15:13:07 +00:00
|
|
|
#if TEST_OOM
|
|
|
|
static int testMallocNext = 0;
|
|
|
|
static int testMallocFailFirst = 0;
|
|
|
|
static int testMallocFailLast = 0;
|
2008-05-29 18:47:00 +00:00
|
|
|
static void (*testMallocHook)(int, void*) = NULL;
|
2008-05-29 15:13:07 +00:00
|
|
|
static void *testMallocHookData = NULL;
|
|
|
|
|
|
|
|
void virAllocTestInit(void)
|
|
|
|
{
|
|
|
|
testMallocNext = 1;
|
|
|
|
testMallocFailFirst = 0;
|
|
|
|
testMallocFailLast = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int virAllocTestCount(void)
|
|
|
|
{
|
|
|
|
return testMallocNext - 1;
|
|
|
|
}
|
|
|
|
|
2008-05-29 18:47:00 +00:00
|
|
|
void virAllocTestHook(void (*func)(int, void*), void *data)
|
2008-05-29 15:13:07 +00:00
|
|
|
{
|
|
|
|
testMallocHook = func;
|
|
|
|
testMallocHookData = data;
|
|
|
|
}
|
|
|
|
|
|
|
|
void virAllocTestOOM(int n, int m)
|
|
|
|
{
|
|
|
|
testMallocNext = 1;
|
|
|
|
testMallocFailFirst = n;
|
|
|
|
testMallocFailLast = n + m - 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int virAllocTestFail(void)
|
|
|
|
{
|
|
|
|
int fail = 0;
|
|
|
|
if (testMallocNext == 0)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
fail =
|
|
|
|
testMallocNext >= testMallocFailFirst &&
|
|
|
|
testMallocNext <= testMallocFailLast;
|
|
|
|
|
|
|
|
if (fail && testMallocHook)
|
2008-05-29 18:47:00 +00:00
|
|
|
(testMallocHook)(testMallocNext, testMallocHookData);
|
2008-05-29 15:13:07 +00:00
|
|
|
|
|
|
|
testMallocNext++;
|
|
|
|
return fail;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2008-04-28 21:44:54 +00:00
|
|
|
/* Return 1 if an array of N objects, each of size S, cannot exist due
|
|
|
|
to size arithmetic overflow. S must be positive and N must be
|
|
|
|
nonnegative. This is a macro, not an inline function, so that it
|
|
|
|
works correctly even when SIZE_MAX < N.
|
|
|
|
|
|
|
|
By gnulib convention, SIZE_MAX represents overflow in size
|
|
|
|
calculations, so the conservative dividend to use here is
|
|
|
|
SIZE_MAX - 1, since SIZE_MAX might represent an overflowed value.
|
|
|
|
However, malloc (SIZE_MAX) fails on all known hosts where
|
|
|
|
sizeof (ptrdiff_t) <= sizeof (size_t), so do not bother to test for
|
|
|
|
exactly-SIZE_MAX allocations on such hosts; this avoids a test and
|
|
|
|
branch when S is known to be 1. */
|
|
|
|
# define xalloc_oversized(n, s) \
|
|
|
|
((size_t) (sizeof (ptrdiff_t) <= sizeof (size_t) ? -1 : -2) / (s) < (n))
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* virAlloc:
|
|
|
|
* @ptrptr: pointer to pointer for address of allocated memory
|
|
|
|
* @size: number of bytes to allocate
|
|
|
|
*
|
|
|
|
* Allocate 'size' bytes of memory. Return the address of the
|
|
|
|
* allocated memory in 'ptrptr'. The newly allocated memory is
|
|
|
|
* filled with zeros.
|
|
|
|
*
|
|
|
|
* Returns -1 on failure to allocate, zero on success
|
|
|
|
*/
|
|
|
|
int virAlloc(void *ptrptr, size_t size)
|
|
|
|
{
|
2008-05-29 15:13:07 +00:00
|
|
|
#if TEST_OOM
|
|
|
|
if (virAllocTestFail()) {
|
|
|
|
*(void **)ptrptr = NULL;
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2008-04-28 21:44:54 +00:00
|
|
|
if (size == 0) {
|
|
|
|
*(void **)ptrptr = NULL;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
*(void **)ptrptr = calloc(1, size);
|
|
|
|
if (*(void **)ptrptr == NULL)
|
|
|
|
return -1;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* virAllocN:
|
|
|
|
* @ptrptr: pointer to pointer for address of allocated memory
|
|
|
|
* @size: number of bytes to allocate
|
|
|
|
* @count: number of elements to allocate
|
|
|
|
*
|
2008-04-29 13:37:20 +00:00
|
|
|
* Allocate an array of memory 'count' elements long,
|
2008-04-28 21:44:54 +00:00
|
|
|
* each with 'size' bytes. Return the address of the
|
|
|
|
* allocated memory in 'ptrptr'. The newly allocated
|
|
|
|
* memory is filled with zeros.
|
|
|
|
*
|
|
|
|
* Returns -1 on failure to allocate, zero on success
|
|
|
|
*/
|
|
|
|
int virAllocN(void *ptrptr, size_t size, size_t count)
|
|
|
|
{
|
2008-05-29 15:13:07 +00:00
|
|
|
#if TEST_OOM
|
|
|
|
if (virAllocTestFail()) {
|
|
|
|
*(void **)ptrptr = NULL;
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2008-04-28 21:44:54 +00:00
|
|
|
if (size == 0 || count == 0) {
|
|
|
|
*(void **)ptrptr = NULL;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
*(void**)ptrptr = calloc(count, size);
|
|
|
|
if (*(void**)ptrptr == NULL)
|
|
|
|
return -1;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* virReallocN:
|
|
|
|
* @ptrptr: pointer to pointer for address of allocated memory
|
|
|
|
* @size: number of bytes to allocate
|
|
|
|
* @count: number of elements in array
|
|
|
|
*
|
|
|
|
* Resize the block of memory in 'ptrptr' to be an array of
|
|
|
|
* 'count' elements, each 'size' bytes in length. Update 'ptrptr'
|
|
|
|
* with the address of the newly allocated memory. On failure,
|
2008-04-29 13:37:20 +00:00
|
|
|
* 'ptrptr' is not changed and still points to the original memory
|
2008-04-28 21:44:54 +00:00
|
|
|
* block. The newly allocated memory is filled with zeros.
|
|
|
|
*
|
|
|
|
* Returns -1 on failure to allocate, zero on success
|
|
|
|
*/
|
|
|
|
int virReallocN(void *ptrptr, size_t size, size_t count)
|
|
|
|
{
|
|
|
|
void *tmp;
|
2008-05-29 15:13:07 +00:00
|
|
|
#if TEST_OOM
|
|
|
|
if (virAllocTestFail())
|
|
|
|
return -1;
|
|
|
|
#endif
|
|
|
|
|
2008-04-28 21:44:54 +00:00
|
|
|
if (size == 0 || count == 0) {
|
|
|
|
free(*(void **)ptrptr);
|
|
|
|
*(void **)ptrptr = NULL;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
if (xalloc_oversized(count, size)) {
|
|
|
|
errno = ENOMEM;
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
tmp = realloc(*(void**)ptrptr, size * count);
|
|
|
|
if (!tmp)
|
|
|
|
return -1;
|
|
|
|
*(void**)ptrptr = tmp;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* virFree:
|
|
|
|
* @ptrptr: pointer to pointer for address of memory to be freed
|
|
|
|
*
|
|
|
|
* Release the chunk of memory in the pointer pointed to by
|
|
|
|
* the 'ptrptr' variable. After release, 'ptrptr' will be
|
|
|
|
* updated to point to NULL.
|
|
|
|
*/
|
|
|
|
void virFree(void *ptrptr)
|
|
|
|
{
|
|
|
|
free(*(void**)ptrptr);
|
|
|
|
*(void**)ptrptr = NULL;
|
|
|
|
}
|