2018-03-04 01:30:16 +01:00
|
|
|
/**
|
|
|
|
* @file Arena Memory Allocator
|
|
|
|
*
|
|
|
|
* Provides a `reset_memory` function to reset the heap to its initial state. A user has to make
|
|
|
|
* sure that there are no more references to cleared memory afterwards. Always aligns to 8 bytes.
|
|
|
|
*/
|
2018-01-28 15:30:49 +01:00
|
|
|
|
2018-03-12 14:06:39 +01:00
|
|
|
import { MASK as AL_MASK } from "./common/alignment";
|
2018-01-28 15:30:49 +01:00
|
|
|
|
2018-03-12 14:06:39 +01:00
|
|
|
var offset: usize = (HEAP_BASE + AL_MASK) & ~AL_MASK;
|
2018-01-28 15:30:49 +01:00
|
|
|
|
2018-02-17 11:09:22 +01:00
|
|
|
@global
|
2018-01-28 15:30:49 +01:00
|
|
|
export function allocate_memory(size: usize): usize {
|
|
|
|
if (!size) return 0;
|
2018-03-12 14:06:39 +01:00
|
|
|
var ptr = offset;
|
2018-03-04 17:25:32 +01:00
|
|
|
var newPtr = (ptr + size + AL_MASK) & ~AL_MASK;
|
|
|
|
var pagesBefore = current_memory();
|
|
|
|
if (newPtr > <usize>pagesBefore << 16) {
|
2018-03-13 02:32:10 +01:00
|
|
|
let pagesNeeded = ((newPtr - ptr + 0xffff) & ~0xffff) >>> 16;
|
|
|
|
let pagesWanted = max(pagesBefore, pagesNeeded); // double memory
|
2018-03-04 17:25:32 +01:00
|
|
|
if (grow_memory(pagesWanted) < 0) {
|
|
|
|
if (grow_memory(pagesNeeded) < 0) {
|
|
|
|
unreachable(); // out of memory
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2018-03-12 14:06:39 +01:00
|
|
|
offset = newPtr;
|
2018-01-28 15:30:49 +01:00
|
|
|
return ptr;
|
|
|
|
}
|
|
|
|
|
2018-02-17 11:09:22 +01:00
|
|
|
@global
|
2018-01-28 15:30:49 +01:00
|
|
|
export function free_memory(ptr: usize): void {
|
|
|
|
// nop
|
|
|
|
}
|
|
|
|
|
2018-02-17 11:09:22 +01:00
|
|
|
@global
|
2018-02-02 03:07:54 +01:00
|
|
|
export function reset_memory(): void {
|
2018-03-12 14:06:39 +01:00
|
|
|
offset = (HEAP_BASE + AL_MASK) & ~AL_MASK;
|
2018-01-28 15:30:49 +01:00
|
|
|
}
|