2018-05-28 12:40:22 +02:00
|
|
|
#include "wut_newlib.h"
|
|
|
|
|
|
|
|
#include <coreinit/atomic.h>
|
2018-06-14 10:32:37 +02:00
|
|
|
#include <coreinit/memheap.h>
|
2018-06-13 19:38:53 +02:00
|
|
|
#include <coreinit/memdefaultheap.h>
|
2018-06-14 10:32:37 +02:00
|
|
|
#include <coreinit/memexpheap.h>
|
2018-05-28 12:40:22 +02:00
|
|
|
|
2018-06-14 10:32:37 +02:00
|
|
|
extern uint32_t __attribute__((weak)) __wut_heap_max_size;
|
|
|
|
|
|
|
|
static MEMHeapHandle sHeapHandle = NULL;
|
|
|
|
static void *sHeapBase = NULL;
|
|
|
|
static uint32_t sHeapMaxSize = 0;
|
2018-05-28 12:40:22 +02:00
|
|
|
static volatile uint32_t sHeapSize = 0;
|
|
|
|
|
|
|
|
void *
|
|
|
|
__wut_sbrk_r(struct _reent *r,
|
|
|
|
ptrdiff_t incr)
|
|
|
|
{
|
|
|
|
uint32_t oldSize, newSize;
|
|
|
|
|
|
|
|
do {
|
|
|
|
oldSize = sHeapSize;
|
|
|
|
newSize = oldSize + incr;
|
|
|
|
|
|
|
|
if (newSize > sHeapMaxSize) {
|
|
|
|
r->_errno = ENOMEM;
|
|
|
|
return (void *)-1;
|
|
|
|
}
|
|
|
|
} while (!OSCompareAndSwapAtomic(&sHeapSize, oldSize, newSize));
|
|
|
|
|
2018-06-14 10:32:37 +02:00
|
|
|
return ((uint8_t *)sHeapBase) + oldSize;
|
2018-05-28 12:40:22 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
__init_wut_sbrk_heap()
|
|
|
|
{
|
2018-06-14 10:32:37 +02:00
|
|
|
if (sHeapBase) {
|
|
|
|
// Already initialised
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-06-15 14:59:30 +02:00
|
|
|
if (&__wut_heap_max_size) {
|
2018-06-14 10:32:37 +02:00
|
|
|
// Use default heap
|
|
|
|
sHeapBase = MEMAllocFromDefaultHeap(__wut_heap_max_size);
|
|
|
|
sHeapMaxSize = __wut_heap_max_size;
|
|
|
|
} else {
|
|
|
|
// No max size specified, use 90% of base MEM2 heap
|
|
|
|
sHeapHandle = MEMGetBaseHeapHandle(MEM_BASE_HEAP_MEM2);
|
|
|
|
|
|
|
|
uint32_t freeSize = MEMGetAllocatableSizeForExpHeapEx(sHeapHandle, 4);
|
|
|
|
sHeapMaxSize = (uint32_t)(0.9f * (float)freeSize) & ~0xFFF;
|
|
|
|
|
|
|
|
sHeapBase = MEMAllocFromExpHeapEx(sHeapHandle, sHeapMaxSize, 4);
|
|
|
|
}
|
|
|
|
|
2018-05-28 12:40:22 +02:00
|
|
|
sHeapSize = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
__fini_wut_sbrk_heap()
|
|
|
|
{
|
2018-06-14 10:32:37 +02:00
|
|
|
if (!sHeapBase) {
|
|
|
|
// Already finalised
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (sHeapHandle) {
|
|
|
|
MEMFreeToExpHeap(sHeapHandle, sHeapBase);
|
|
|
|
} else {
|
|
|
|
MEMFreeToDefaultHeap(sHeapBase);
|
|
|
|
}
|
|
|
|
|
2018-06-13 19:38:53 +02:00
|
|
|
sHeapBase = NULL;
|
|
|
|
sHeapSize = 0;
|
|
|
|
sHeapMaxSize = 0;
|
2018-05-28 12:40:22 +02:00
|
|
|
}
|