1 /**************************************************************************
3 * Copyright 2008-2010 VMware, Inc.
6 * Permission is hereby granted, free of charge, to any person obtaining a
7 * copy of this software and associated documentation files (the
8 * "Software"), to deal in the Software without restriction, including
9 * without limitation the rights to use, copy, modify, merge, publish,
10 * distribute, sub license, and/or sell copies of the Software, and to
11 * permit persons to whom the Software is furnished to do so, subject to
12 * the following conditions:
14 * The above copyright notice and this permission notice (including the
15 * next paragraph) shall be included in all copies or substantial portions
18 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
19 * OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
20 * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT.
21 * IN NO EVENT SHALL VMWARE AND/OR ITS SUPPLIERS BE LIABLE FOR
22 * ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
23 * TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
24 * SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
26 **************************************************************************/
30 * Memory alignment wrappers.
35 #error "Must not be included directly. Include os_memory.h instead"
40 * Add two size_t values with integer overflow check.
41 * TODO: leverage __builtin_add_overflow where available
44 add_overflow_size_t(size_t a
, size_t b
, size_t *res
)
47 return *res
< a
|| *res
< b
;
51 #if defined(HAVE_POSIX_MEMALIGN)
54 os_malloc_aligned(size_t size
, size_t alignment
)
57 alignment
= (alignment
+ sizeof(void*) - 1) & ~(sizeof(void*) - 1);
58 if(posix_memalign(&ptr
, alignment
, size
) != 0)
63 #define os_free_aligned(_ptr) free(_ptr)
68 * Return memory on given byte alignment
71 os_malloc_aligned(size_t size
, size_t alignment
)
79 * alloc_size = size + alignment + sizeof(void *)
81 * while checking for overflow.
83 if (add_overflow_size_t(size
, alignment
, &alloc_size
) ||
84 add_overflow_size_t(alloc_size
, sizeof(void *), &alloc_size
)) {
88 ptr
= (char *) os_malloc(alloc_size
);
92 buf
= (char *)(((uintptr_t)ptr
+ sizeof(void *) + alignment
- 1) & ~((uintptr_t)(alignment
- 1)));
93 *(char **)(buf
- sizeof(void *)) = ptr
;
100 * Free memory returned by os_malloc_aligned().
103 os_free_aligned(void *ptr
)
106 void **cubbyHole
= (void **) ((char *) ptr
- sizeof(void *));
107 void *realAddr
= *cubbyHole
;
115 * Reallocate memeory, with alignment
118 os_realloc_aligned(void *ptr
, size_t oldsize
, size_t newsize
, size_t alignment
)
120 const size_t copySize
= MIN2(oldsize
, newsize
);
121 void *newBuf
= os_malloc_aligned(newsize
, alignment
);
122 if (newBuf
&& ptr
&& copySize
> 0) {
123 memcpy(newBuf
, ptr
, copySize
);
126 os_free_aligned(ptr
);