blob: 0bc2d3f5ef3bf6b5c378298b1a2f882ba43647c0 [file] [log] [blame] [edit]
/*
* Copyright 2019 The Emscripten Authors. All rights reserved.
* Emscripten is available under two separate licenses, the MIT license and the
* University of Illinois/NCSA Open Source License. Both these licenses can be
* found in the LICENSE file.
*
*/
#ifndef EMSCRIPTEN_NO_ERRNO
#include <errno.h>
#endif
#include <limits.h>
#include <stddef.h>
#include <stdint.h>
#if __EMSCRIPTEN_PTHREADS__ // for error handling, see below
#include <stdio.h>
#include <stdlib.h>
#endif
#ifdef __EMSCRIPTEN_TRACING__
#include <emscripten/em_asm.h>
#endif
#include <emscripten/heap.h>
#ifndef EMSCRIPTEN_NO_ERRNO
#define SET_ERRNO() { errno = ENOMEM; }
#else
#define SET_ERRNO()
#endif
#define RETURN_ERROR() { \
SET_ERRNO() \
return (void*)-1; \
}
void *sbrk(intptr_t increment) {
// Enforce preserving a minimal 4-byte alignment for sbrk.
increment = (increment + 3) & ~3;
#if __EMSCRIPTEN_PTHREADS__
// Our default dlmalloc uses locks around each malloc/free, so no additional
// work is necessary to keep things threadsafe, but we also make sure sbrk
// itself is threadsafe so alternative allocators work. We do that by looping
// and retrying if we hit interference with another thread.
while (1) {
#endif // __EMSCRIPTEN_PTHREADS__
intptr_t* sbrk_ptr = emscripten_get_sbrk_ptr();
#if __EMSCRIPTEN_PTHREADS__
intptr_t old_brk = __c11_atomic_load((_Atomic(intptr_t)*)sbrk_ptr, __ATOMIC_SEQ_CST);
#else
intptr_t old_brk = *sbrk_ptr;
#endif
intptr_t new_brk = old_brk + increment;
#ifdef __wasm__
uintptr_t old_size = __builtin_wasm_memory_size(0) * WASM_PAGE_SIZE;
#else
uintptr_t old_size = emscripten_get_heap_size();
#endif
if (new_brk > old_size) {
// Try to grow memory.
intptr_t diff = new_brk - old_size;
if (!emscripten_resize_heap(new_brk)) {
RETURN_ERROR();
}
}
#if __EMSCRIPTEN_PTHREADS__
// Attempt to update the dynamic top to new value. Another thread may have
// beat this one to the update, in which case we will need to start over
// by iterating the loop body again.
intptr_t expected = old_brk;
__c11_atomic_compare_exchange_strong(
(_Atomic(intptr_t)*)sbrk_ptr,
&expected, new_brk,
__ATOMIC_SEQ_CST, __ATOMIC_SEQ_CST);
if (expected != old_brk) {
continue;
}
#else // __EMSCRIPTEN_PTHREADS__
*sbrk_ptr = new_brk;
#endif // __EMSCRIPTEN_PTHREADS__
#ifdef __EMSCRIPTEN_TRACING__
EM_ASM({if (typeof emscriptenMemoryProfiler !== 'undefined') emscriptenMemoryProfiler.onSbrkGrow($0, $1)}, old_brk, old_brk + increment );
#endif
return (void*)old_brk;
#if __EMSCRIPTEN_PTHREADS__
}
#endif // __EMSCRIPTEN_PTHREADS__
}
int brk(intptr_t ptr) {
#if __EMSCRIPTEN_PTHREADS__
// FIXME
printf("brk() is not theadsafe yet, https://github.com/emscripten-core/emscripten/issues/10006");
abort();
#endif
intptr_t last = (intptr_t)sbrk(0);
if (sbrk(ptr - last) == (void*)-1) {
return -1;
}
return 0;
}