Path: blob/main/system/lib/mimalloc/src/alloc-posix.c
6175 views
/* ----------------------------------------------------------------------------1Copyright (c) 2018-2021, Microsoft Research, Daan Leijen2This is free software; you can redistribute it and/or modify it under the3terms of the MIT license. A copy of the license can be found in the file4"LICENSE" at the root of this distribution.5-----------------------------------------------------------------------------*/67// ------------------------------------------------------------------------8// mi prefixed publi definitions of various Posix, Unix, and C++ functions9// for convenience and used when overriding these functions.10// ------------------------------------------------------------------------11#include "mimalloc.h"12#include "mimalloc/internal.h"1314// ------------------------------------------------------15// Posix & Unix functions definitions16// ------------------------------------------------------1718#include <errno.h>19#include <string.h> // memset20#include <stdlib.h> // getenv2122#ifdef _MSC_VER23#pragma warning(disable:4996) // getenv _wgetenv24#endif2526#ifndef EINVAL27#define EINVAL 2228#endif29#ifndef ENOMEM30#define ENOMEM 1231#endif323334mi_decl_nodiscard size_t mi_malloc_size(const void* p) mi_attr_noexcept {35// if (!mi_is_in_heap_region(p)) return 0;36return mi_usable_size(p);37}3839mi_decl_nodiscard size_t mi_malloc_usable_size(const void *p) mi_attr_noexcept {40// if (!mi_is_in_heap_region(p)) return 0;41return mi_usable_size(p);42}4344mi_decl_nodiscard size_t mi_malloc_good_size(size_t size) mi_attr_noexcept {45return mi_good_size(size);46}4748void mi_cfree(void* p) mi_attr_noexcept {49if (mi_is_in_heap_region(p)) {50mi_free(p);51}52}5354int mi_posix_memalign(void** p, size_t alignment, size_t size) mi_attr_noexcept {55// Note: The spec dictates we should not modify `*p` on an error. (issue#27)56// <http://man7.org/linux/man-pages/man3/posix_memalign.3.html>57if (p == NULL) return EINVAL;58if ((alignment % sizeof(void*)) != 0) return EINVAL; // natural alignment59// it is also required that alignment is a power of 2 and > 0; this is checked in `mi_malloc_aligned`60if (alignment==0 || !_mi_is_power_of_two(alignment)) return EINVAL; // not a power of 261void* q = mi_malloc_aligned(size, alignment);62if (q==NULL && size != 0) return ENOMEM;63mi_assert_internal(((uintptr_t)q % alignment) == 0);64*p = q;65return 0;66}6768mi_decl_nodiscard mi_decl_restrict void* mi_memalign(size_t alignment, size_t size) mi_attr_noexcept {69void* p = mi_malloc_aligned(size, alignment);70mi_assert_internal(((uintptr_t)p % alignment) == 0);71return p;72}7374mi_decl_nodiscard mi_decl_restrict void* mi_valloc(size_t size) mi_attr_noexcept {75return mi_memalign( _mi_os_page_size(), size );76}7778mi_decl_nodiscard mi_decl_restrict void* mi_pvalloc(size_t size) mi_attr_noexcept {79size_t psize = _mi_os_page_size();80if (size >= SIZE_MAX - psize) return NULL; // overflow81size_t asize = _mi_align_up(size, psize);82return mi_malloc_aligned(asize, psize);83}8485mi_decl_nodiscard mi_decl_restrict void* mi_aligned_alloc(size_t alignment, size_t size) mi_attr_noexcept {86// C11 requires the size to be an integral multiple of the alignment, see <https://en.cppreference.com/w/c/memory/aligned_alloc>.87// unfortunately, it turns out quite some programs pass a size that is not an integral multiple so skip this check..88/* if mi_unlikely((size & (alignment - 1)) != 0) { // C11 requires alignment>0 && integral multiple, see <https://en.cppreference.com/w/c/memory/aligned_alloc>89#if MI_DEBUG > 090_mi_error_message(EOVERFLOW, "(mi_)aligned_alloc requires the size to be an integral multiple of the alignment (size %zu, alignment %zu)\n", size, alignment);91#endif92return NULL;93}94*/95// C11 also requires alignment to be a power-of-two (and > 0) which is checked in mi_malloc_aligned96void* p = mi_malloc_aligned(size, alignment);97mi_assert_internal(((uintptr_t)p % alignment) == 0);98return p;99}100101mi_decl_nodiscard void* mi_reallocarray( void* p, size_t count, size_t size ) mi_attr_noexcept { // BSD102void* newp = mi_reallocn(p,count,size);103if (newp==NULL) { errno = ENOMEM; }104return newp;105}106107mi_decl_nodiscard int mi_reallocarr( void* p, size_t count, size_t size ) mi_attr_noexcept { // NetBSD108mi_assert(p != NULL);109if (p == NULL) {110errno = EINVAL;111return EINVAL;112}113void** op = (void**)p;114void* newp = mi_reallocarray(*op, count, size);115if mi_unlikely(newp == NULL) { return errno; }116*op = newp;117return 0;118}119120void* mi__expand(void* p, size_t newsize) mi_attr_noexcept { // Microsoft121void* res = mi_expand(p, newsize);122if (res == NULL) { errno = ENOMEM; }123return res;124}125126mi_decl_nodiscard mi_decl_restrict unsigned short* mi_wcsdup(const unsigned short* s) mi_attr_noexcept {127if (s==NULL) return NULL;128size_t len;129for(len = 0; s[len] != 0; len++) { }130size_t size = (len+1)*sizeof(unsigned short);131unsigned short* p = (unsigned short*)mi_malloc(size);132if (p != NULL) {133_mi_memcpy(p,s,size);134}135return p;136}137138mi_decl_nodiscard mi_decl_restrict unsigned char* mi_mbsdup(const unsigned char* s) mi_attr_noexcept {139return (unsigned char*)mi_strdup((const char*)s);140}141142int mi_dupenv_s(char** buf, size_t* size, const char* name) mi_attr_noexcept {143if (buf==NULL || name==NULL) return EINVAL;144if (size != NULL) *size = 0;145char* p = getenv(name); // mscver warning 4996146if (p==NULL) {147*buf = NULL;148}149else {150*buf = mi_strdup(p);151if (*buf==NULL) return ENOMEM;152if (size != NULL) *size = _mi_strlen(p);153}154return 0;155}156157int mi_wdupenv_s(unsigned short** buf, size_t* size, const unsigned short* name) mi_attr_noexcept {158if (buf==NULL || name==NULL) return EINVAL;159if (size != NULL) *size = 0;160#if !defined(_WIN32) || (defined(WINAPI_FAMILY) && (WINAPI_FAMILY != WINAPI_FAMILY_DESKTOP_APP))161// not supported162*buf = NULL;163return EINVAL;164#else165unsigned short* p = (unsigned short*)_wgetenv((const wchar_t*)name); // msvc warning 4996166if (p==NULL) {167*buf = NULL;168}169else {170*buf = mi_wcsdup(p);171if (*buf==NULL) return ENOMEM;172if (size != NULL) *size = wcslen((const wchar_t*)p);173}174return 0;175#endif176}177178mi_decl_nodiscard void* mi_aligned_offset_recalloc(void* p, size_t newcount, size_t size, size_t alignment, size_t offset) mi_attr_noexcept { // Microsoft179return mi_recalloc_aligned_at(p, newcount, size, alignment, offset);180}181182mi_decl_nodiscard void* mi_aligned_recalloc(void* p, size_t newcount, size_t size, size_t alignment) mi_attr_noexcept { // Microsoft183return mi_recalloc_aligned(p, newcount, size, alignment);184}185186187