2 * util.c - utility functions
6 * Copyright (C) 2012, 2013, 2014 Eric Biggers
8 * This file is free software; you can redistribute it and/or modify it under
9 * the terms of the GNU Lesser General Public License as published by the Free
10 * Software Foundation; either version 3 of the License, or (at your option) any
13 * This file is distributed in the hope that it will be useful, but WITHOUT
14 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
15 * FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more
18 * You should have received a copy of the GNU Lesser General Public License
19 * along with this file; if not, see http://www.gnu.org/licenses/.
30 #ifdef HAVE_SYS_SYSCTL_H
31 # include <sys/types.h>
32 # include <sys/sysctl.h>
37 #include "wimlib/assert.h"
38 #include "wimlib/error.h"
39 #include "wimlib/timestamp.h"
40 #include "wimlib/util.h"
41 #include "wimlib/xml.h"
47 static void *(*wimlib_malloc_func) (size_t) = malloc;
48 static void (*wimlib_free_func) (void *) = free;
49 static void *(*wimlib_realloc_func)(void *, size_t) = realloc;
52 wimlib_malloc(size_t size)
57 ptr = (*wimlib_malloc_func)(size);
68 wimlib_free_memory(void *ptr)
70 (*wimlib_free_func)(ptr);
74 wimlib_realloc(void *ptr, size_t size)
78 return (*wimlib_realloc_func)(ptr, size);
82 wimlib_calloc(size_t nmemb, size_t size)
84 size_t total_size = nmemb * size;
85 void *p = MALLOC(total_size);
87 p = memset(p, 0, total_size);
92 wimlib_strdup(const char *str)
94 return memdup(str, strlen(str) + 1);
99 wimlib_wcsdup(const wchar_t *str)
101 return memdup(str, (wcslen(str) + 1) * sizeof(wchar_t));
106 wimlib_aligned_malloc(size_t size, size_t alignment)
108 wimlib_assert(alignment != 0 && is_power_of_2(alignment) &&
111 const uintptr_t mask = alignment - 1;
115 raw_ptr = MALLOC(mask + sizeof(size_t) + size);
117 ptr = (char *)raw_ptr + sizeof(size_t);
118 ptr = (void *)(((uintptr_t)ptr + mask) & ~mask);
119 *((size_t *)ptr - 1) = ptr - raw_ptr;
125 wimlib_aligned_free(void *ptr)
128 FREE((char *)ptr - *((size_t *)ptr - 1));
132 memdup(const void *mem, size_t size)
134 void *ptr = MALLOC(size);
136 ptr = memcpy(ptr, mem, size);
140 /* API function documented in wimlib.h */
142 wimlib_set_memory_allocator(void *(*malloc_func)(size_t),
143 void (*free_func)(void *),
144 void *(*realloc_func)(void *, size_t))
146 wimlib_malloc_func = malloc_func ? malloc_func : malloc;
147 wimlib_free_func = free_func ? free_func : free;
148 wimlib_realloc_func = realloc_func ? realloc_func : realloc;
150 xml_set_memory_allocator(wimlib_malloc_func, wimlib_free_func,
151 wimlib_realloc_func);
160 void *mempcpy(void *dst, const void *src, size_t n)
162 return memcpy(dst, src, n) + n;
166 static bool seeded = false;
171 srand(now_as_wim_timestamp());
175 /* Fills @n characters pointed to by @p with random alphanumeric characters. */
177 randomize_char_array_with_alnum(tchar *p, size_t n)
192 /* Fills @n bytes pointer to by @p with random numbers. */
194 randomize_byte_array(u8 *p, size_t n)
204 get_available_cpus(void)
206 long n = sysconf(_SC_NPROCESSORS_ONLN);
207 if (n < 1 || n >= UINT_MAX) {
208 WARNING("Failed to determine number of processors; assuming 1.");
213 #endif /* !__WIN32__ */
217 get_available_memory(void)
219 #if defined(_SC_PAGESIZE) && defined(_SC_PHYS_PAGES)
220 long page_size = sysconf(_SC_PAGESIZE);
221 long num_pages = sysconf(_SC_PHYS_PAGES);
222 if (page_size <= 0 || num_pages <= 0)
224 return ((u64)page_size * (u64)num_pages);
226 int mib[2] = {CTL_HW, HW_MEMSIZE};
228 size_t len = sizeof(memsize);
229 if (sysctl(mib, ARRAY_LEN(mib), &memsize, &len, NULL, 0) < 0 || len != 8)
235 WARNING("Failed to determine available memory; assuming 1 GiB");
238 #endif /* !__WIN32__ */