2021-06-27 21:25:29 +00:00
|
|
|
/***
|
|
|
|
Copyright (C) 2021 J Reece Wilson (a/k/a "Reece"). All rights reserved.
|
|
|
|
|
2024-03-19 15:47:42 +00:00
|
|
|
File: AuMemory.cpp
|
2021-06-27 21:25:29 +00:00
|
|
|
Date: 2021-6-12
|
|
|
|
Author: Reece
|
|
|
|
***/
|
2021-09-30 14:57:41 +00:00
|
|
|
#include <Source/RuntimeInternal.hpp>
|
2024-03-19 15:47:42 +00:00
|
|
|
#include "AuMemory.hpp"
|
2021-06-27 21:25:29 +00:00
|
|
|
#include <mimalloc.h>
|
2022-11-17 07:46:07 +00:00
|
|
|
#include <Source/HWInfo/AuHWInfo.hpp>
|
2022-12-07 01:29:09 +00:00
|
|
|
#include <Source/Debug/MemoryCrunch.hpp>
|
2024-02-14 00:32:30 +00:00
|
|
|
#include <Source/Debug/ErrorStack.hpp>
|
2022-12-07 01:29:09 +00:00
|
|
|
|
2021-06-27 21:25:29 +00:00
|
|
|
namespace Aurora::Memory
|
|
|
|
{
|
2022-12-08 19:34:15 +00:00
|
|
|
AuUInt gBytesCounterAllocated {};
|
|
|
|
AuUInt gBytesCounterPeak {};
|
|
|
|
|
2023-01-15 06:05:22 +00:00
|
|
|
thread_local AuInt64 tlsLastOutOfMemory {};
|
2024-03-19 15:47:42 +00:00
|
|
|
thread_local MemoryLowNotification_f tlsMemoryLowNotification;
|
2023-10-17 07:45:44 +00:00
|
|
|
|
|
|
|
static LeakFinderAlloc_f gLeakFinderAlloc;
|
2024-03-19 15:47:42 +00:00
|
|
|
static LeakFinderFree_f gLeakFinderFree;
|
2023-01-15 06:05:22 +00:00
|
|
|
|
2022-12-08 19:34:15 +00:00
|
|
|
static void AddBytesToCounter(AuUInt uBytes)
|
|
|
|
{
|
|
|
|
gBytesCounterPeak = AuMax(gBytesCounterPeak, AuAtomicAdd(&gBytesCounterAllocated, uBytes));
|
|
|
|
}
|
|
|
|
|
|
|
|
static void RemoveBytesFromCounter(AuUInt uBytes)
|
|
|
|
{
|
|
|
|
AuAtomicSub(&gBytesCounterAllocated, uBytes);
|
|
|
|
}
|
2023-10-17 07:45:44 +00:00
|
|
|
|
2023-10-24 17:00:09 +00:00
|
|
|
AUKN_SYM void SetMemoryLowNotification(MemoryLowNotification_f pFunc)
|
|
|
|
{
|
|
|
|
tlsMemoryLowNotification = pFunc;
|
|
|
|
}
|
|
|
|
|
2023-10-17 07:45:44 +00:00
|
|
|
AUKN_SYM void SetLeakFinder(LeakFinderAlloc_f pAlloc,
|
|
|
|
LeakFinderFree_f pFree)
|
|
|
|
{
|
|
|
|
gLeakFinderAlloc = pAlloc;
|
|
|
|
gLeakFinderFree = pFree;
|
|
|
|
}
|
2022-12-08 19:34:15 +00:00
|
|
|
|
2023-10-26 16:05:44 +00:00
|
|
|
AUKN_SYM void ReserveHeapMemory(AuUInt uHeapSize, bool bCommit)
|
|
|
|
{
|
|
|
|
mi_reserve_os_memory(uHeapSize, bCommit, true);
|
|
|
|
}
|
|
|
|
|
2022-03-21 05:20:19 +00:00
|
|
|
AUKN_SYM AuUInt GetChunkSize(const void *head)
|
2021-06-27 21:25:29 +00:00
|
|
|
{
|
2022-12-07 01:29:09 +00:00
|
|
|
if (AuDebug::IsPointerReserveRange((void *)head))
|
|
|
|
{
|
|
|
|
return AuDebug::gReserveHeap->GetChunkSize(head);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return ::mi_malloc_size(head);
|
|
|
|
}
|
2021-06-27 21:25:29 +00:00
|
|
|
}
|
|
|
|
|
2024-02-14 00:32:30 +00:00
|
|
|
static void OnOOM(AuUInt uLength)
|
|
|
|
{
|
|
|
|
tlsLastOutOfMemory = AuTime::CurrentClockNS();
|
|
|
|
|
|
|
|
AuDebug::OnOutOfMemory();
|
|
|
|
|
|
|
|
if (auto pLowNotification = tlsMemoryLowNotification)
|
|
|
|
{
|
|
|
|
pLowNotification(uLength, 4);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-12-07 16:20:13 +00:00
|
|
|
#define CHECK_WRAP_RETURN(exp, exp2, string) \
|
|
|
|
auto pRet = exp; \
|
|
|
|
if (!pRet) \
|
|
|
|
{ \
|
2024-02-14 00:32:30 +00:00
|
|
|
OnOOM(length); \
|
2023-01-15 06:05:22 +00:00
|
|
|
\
|
2022-12-07 16:20:13 +00:00
|
|
|
bool bCrunchFlag {}; \
|
|
|
|
if (((bCrunchFlag = AuDebug::IsTlsMemoryCrunchActive()) || \
|
|
|
|
(gRuntimeConfig.debug.bIsApplicationClientSoftwareOnJitteryMemorySystem)) && \
|
|
|
|
AuDebug::gReserveHeap) \
|
|
|
|
{ \
|
2023-10-24 17:00:09 +00:00
|
|
|
if (auto pLowNotification = tlsMemoryLowNotification) \
|
|
|
|
{ \
|
|
|
|
pLowNotification(length, 2); \
|
|
|
|
} \
|
2022-12-07 16:20:13 +00:00
|
|
|
if (!(pRet = AuDebug::gReserveHeap-> AU_WHAT exp2)) \
|
|
|
|
{ \
|
|
|
|
if (bCrunchFlag || gRuntimeConfig.debug.bIsMemoryErrorFatal) \
|
|
|
|
{ \
|
|
|
|
SysPanic("out of contingency memory"); \
|
|
|
|
} \
|
|
|
|
} \
|
|
|
|
} \
|
|
|
|
else if (gRuntimeConfig.debug.bIsMemoryErrorFatal) \
|
|
|
|
{ \
|
|
|
|
SysPanic(string); \
|
|
|
|
} \
|
2023-10-24 17:00:09 +00:00
|
|
|
if (auto pLowNotification = tlsMemoryLowNotification) \
|
|
|
|
{ \
|
|
|
|
pLowNotification(length, !pRet ? 0 : 1); \
|
|
|
|
} \
|
2022-12-07 16:20:13 +00:00
|
|
|
} \
|
2022-12-08 19:34:15 +00:00
|
|
|
else \
|
|
|
|
{ \
|
2024-01-17 20:55:59 +00:00
|
|
|
auto uCount = ::mi_malloc_size(pRet); \
|
2023-10-24 17:00:09 +00:00
|
|
|
if (gLeakFinderAlloc) \
|
|
|
|
{ \
|
2024-01-17 20:55:59 +00:00
|
|
|
gLeakFinderAlloc(pRet, uCount); \
|
2023-10-24 17:00:09 +00:00
|
|
|
} \
|
2024-01-17 20:55:59 +00:00
|
|
|
AddBytesToCounter(uCount); \
|
2022-12-08 19:34:15 +00:00
|
|
|
} \
|
2022-09-12 22:38:44 +00:00
|
|
|
return pRet;
|
|
|
|
|
2021-06-27 21:25:29 +00:00
|
|
|
AUKN_SYM void *_ZAlloc(Types::size_t length)
|
|
|
|
{
|
2023-01-15 06:05:22 +00:00
|
|
|
SysAssertDbg(length);
|
2022-12-07 01:29:09 +00:00
|
|
|
CHECK_WRAP_RETURN(::mi_zalloc(length), (ZAlloc(length)), "ZAlloc out of memory");
|
2021-06-27 21:25:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
AUKN_SYM void *_ZAlloc(Types::size_t length, Types::size_t align)
|
|
|
|
{
|
2023-01-15 06:05:22 +00:00
|
|
|
SysAssertDbg(length);
|
2022-12-07 01:29:09 +00:00
|
|
|
CHECK_WRAP_RETURN(::mi_zalloc_aligned(length, align), (ZAlloc(length, align)), "ZAlloc out of memory");
|
2021-06-27 21:25:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
AUKN_SYM void *_FAlloc(Types::size_t length)
|
|
|
|
{
|
2023-01-15 06:05:22 +00:00
|
|
|
SysAssertDbg(length);
|
2022-12-07 01:29:09 +00:00
|
|
|
CHECK_WRAP_RETURN(::mi_malloc(length), (FAlloc(length)), "FAlloc out of memory");
|
2021-06-27 21:25:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
AUKN_SYM void *_FAlloc(Types::size_t length, Types::size_t align)
|
|
|
|
{
|
2023-01-15 06:05:22 +00:00
|
|
|
SysAssertDbg(length);
|
2022-12-07 01:29:09 +00:00
|
|
|
CHECK_WRAP_RETURN(::mi_malloc_aligned(length, align), (FAlloc(length, align)), "FAllocEx out of memory");
|
2021-06-27 21:25:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
AUKN_SYM void *_ZRealloc(void *buffer, Types::size_t length, Types::size_t align)
|
|
|
|
{
|
2024-03-19 15:47:42 +00:00
|
|
|
void *pRet = nullptr;
|
2022-12-12 23:50:05 +00:00
|
|
|
|
2022-12-07 01:29:09 +00:00
|
|
|
if (AuDebug::IsPointerReserveRange(buffer))
|
|
|
|
{
|
2022-12-12 23:50:05 +00:00
|
|
|
pRet = AuDebug::gReserveHeap->ZRealloc(buffer, length, align);
|
2022-12-07 01:29:09 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2023-10-17 07:45:44 +00:00
|
|
|
auto oldLen = ::mi_malloc_size(buffer);
|
|
|
|
|
2024-03-19 15:47:42 +00:00
|
|
|
if (buffer && !length)
|
|
|
|
{
|
|
|
|
::mi_free(buffer);
|
|
|
|
}
|
|
|
|
else if (!buffer && length)
|
|
|
|
{
|
|
|
|
return _ZAlloc(length, align);
|
|
|
|
}
|
|
|
|
else if ((pRet = ::mi_rezalloc_aligned(buffer, length, align)))
|
2023-10-17 07:45:44 +00:00
|
|
|
{
|
|
|
|
auto uNewSize = ::mi_malloc_size(pRet);
|
|
|
|
|
|
|
|
RemoveBytesFromCounter(oldLen);
|
|
|
|
AddBytesToCounter(uNewSize);
|
|
|
|
|
|
|
|
if (gLeakFinderAlloc)
|
|
|
|
{
|
|
|
|
if (gLeakFinderFree)
|
|
|
|
{
|
2024-01-17 20:55:59 +00:00
|
|
|
gLeakFinderFree(buffer, oldLen);
|
2023-10-17 07:45:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
gLeakFinderAlloc(pRet, uNewSize);
|
|
|
|
}
|
|
|
|
}
|
2022-12-12 23:50:05 +00:00
|
|
|
}
|
|
|
|
|
2024-03-19 15:47:42 +00:00
|
|
|
if (!pRet && buffer && length)
|
2022-12-12 23:50:05 +00:00
|
|
|
{
|
2024-03-19 15:47:42 +00:00
|
|
|
OnOOM(length);
|
|
|
|
|
2022-12-12 23:50:05 +00:00
|
|
|
if (gRuntimeConfig.debug.bIsMemoryErrorFatal)
|
2022-12-07 01:29:09 +00:00
|
|
|
{
|
2022-12-12 23:50:05 +00:00
|
|
|
SysPanic("ZAllocEx out of memory");
|
2022-12-07 01:29:09 +00:00
|
|
|
}
|
|
|
|
}
|
2022-12-12 23:50:05 +00:00
|
|
|
|
|
|
|
return pRet;
|
2021-06-27 21:25:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
AUKN_SYM void *_ZRealloc(void *buffer, Types::size_t length)
|
|
|
|
{
|
2024-03-19 15:47:42 +00:00
|
|
|
void *pRet = nullptr;
|
2022-12-12 23:50:05 +00:00
|
|
|
|
2022-12-07 01:29:09 +00:00
|
|
|
if (AuDebug::IsPointerReserveRange(buffer))
|
|
|
|
{
|
2022-12-12 23:50:05 +00:00
|
|
|
pRet = AuDebug::gReserveHeap->ZRealloc(buffer, length);
|
2022-12-07 01:29:09 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2023-10-17 07:45:44 +00:00
|
|
|
auto oldLen = ::mi_malloc_size(buffer);
|
|
|
|
|
2024-03-19 15:47:42 +00:00
|
|
|
if (buffer && !length)
|
|
|
|
{
|
|
|
|
::mi_free(buffer);
|
|
|
|
}
|
|
|
|
else if (!buffer && length)
|
|
|
|
{
|
|
|
|
return _ZAlloc(length);
|
|
|
|
}
|
|
|
|
else if ((pRet = ::mi_rezalloc(buffer, length)))
|
2023-10-17 07:45:44 +00:00
|
|
|
{
|
|
|
|
auto uNewSize = ::mi_malloc_size(pRet);
|
|
|
|
|
|
|
|
RemoveBytesFromCounter(oldLen);
|
|
|
|
AddBytesToCounter(uNewSize);
|
|
|
|
|
|
|
|
if (gLeakFinderAlloc)
|
|
|
|
{
|
|
|
|
if (gLeakFinderFree)
|
|
|
|
{
|
2024-01-17 20:55:59 +00:00
|
|
|
gLeakFinderFree(buffer, oldLen);
|
2023-10-17 07:45:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
gLeakFinderAlloc(pRet, uNewSize);
|
|
|
|
}
|
|
|
|
}
|
2022-12-12 23:50:05 +00:00
|
|
|
}
|
|
|
|
|
2024-03-19 15:47:42 +00:00
|
|
|
if (!pRet && buffer && length)
|
2022-12-12 23:50:05 +00:00
|
|
|
{
|
2024-03-19 15:47:42 +00:00
|
|
|
OnOOM(length);
|
|
|
|
|
2022-12-12 23:50:05 +00:00
|
|
|
if (gRuntimeConfig.debug.bIsMemoryErrorFatal)
|
2022-12-07 01:29:09 +00:00
|
|
|
{
|
2022-12-12 23:50:05 +00:00
|
|
|
SysPanic("ZAlloc out of memory");
|
2022-12-07 01:29:09 +00:00
|
|
|
}
|
|
|
|
}
|
2022-12-12 23:50:05 +00:00
|
|
|
|
|
|
|
return pRet;
|
2021-06-27 21:25:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
AUKN_SYM void *_FRealloc(void *buffer, Types::size_t length, Types::size_t align)
|
|
|
|
{
|
2024-03-19 15:47:42 +00:00
|
|
|
void *pRet = nullptr;
|
2022-12-12 23:50:05 +00:00
|
|
|
|
2022-12-07 01:29:09 +00:00
|
|
|
if (AuDebug::IsPointerReserveRange(buffer))
|
|
|
|
{
|
2022-12-12 23:50:05 +00:00
|
|
|
pRet = AuDebug::gReserveHeap->FRealloc(buffer, length, align);
|
2022-12-07 01:29:09 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2023-10-17 07:45:44 +00:00
|
|
|
auto oldLen = ::mi_malloc_size(buffer);
|
|
|
|
|
2024-03-19 15:47:42 +00:00
|
|
|
if (buffer && !length)
|
|
|
|
{
|
|
|
|
::mi_free(buffer);
|
|
|
|
}
|
|
|
|
else if (!buffer && length)
|
|
|
|
{
|
|
|
|
return _FAlloc(length, align);
|
|
|
|
}
|
|
|
|
else if ((pRet = ::mi_realloc_aligned(buffer, length, align)))
|
2023-10-17 07:45:44 +00:00
|
|
|
{
|
|
|
|
auto uNewSize = ::mi_malloc_size(pRet);
|
|
|
|
|
|
|
|
RemoveBytesFromCounter(oldLen);
|
|
|
|
AddBytesToCounter(uNewSize);
|
|
|
|
|
|
|
|
if (gLeakFinderAlloc)
|
|
|
|
{
|
|
|
|
if (gLeakFinderFree)
|
|
|
|
{
|
2024-01-17 20:55:59 +00:00
|
|
|
gLeakFinderFree(buffer, oldLen);
|
2023-10-17 07:45:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
gLeakFinderAlloc(pRet, uNewSize);
|
|
|
|
}
|
|
|
|
}
|
2022-12-12 23:50:05 +00:00
|
|
|
}
|
|
|
|
|
2024-03-19 15:47:42 +00:00
|
|
|
if (!pRet && buffer && length)
|
2022-12-12 23:50:05 +00:00
|
|
|
{
|
2024-03-19 15:47:42 +00:00
|
|
|
OnOOM(length);
|
|
|
|
|
2022-12-12 23:50:05 +00:00
|
|
|
if (gRuntimeConfig.debug.bIsMemoryErrorFatal)
|
2022-12-07 01:29:09 +00:00
|
|
|
{
|
2022-12-12 23:50:05 +00:00
|
|
|
SysPanic("FReallocEx out of memory");
|
2022-12-07 01:29:09 +00:00
|
|
|
}
|
|
|
|
}
|
2022-12-12 23:50:05 +00:00
|
|
|
|
|
|
|
return pRet;
|
2021-06-27 21:25:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
AUKN_SYM void *_FRealloc(void *buffer, Types::size_t length)
|
|
|
|
{
|
2024-03-19 15:47:42 +00:00
|
|
|
void *pRet = nullptr;
|
2022-12-12 23:50:05 +00:00
|
|
|
|
2022-12-07 01:29:09 +00:00
|
|
|
if (AuDebug::IsPointerReserveRange(buffer))
|
|
|
|
{
|
2022-12-12 23:50:05 +00:00
|
|
|
pRet = AuDebug::gReserveHeap->FRealloc(buffer, length);
|
2022-12-07 01:29:09 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2023-10-17 07:45:44 +00:00
|
|
|
auto oldLen = ::mi_malloc_size(buffer);
|
|
|
|
|
2024-03-19 15:47:42 +00:00
|
|
|
if (buffer && !length)
|
|
|
|
{
|
|
|
|
::mi_free(buffer);
|
|
|
|
}
|
|
|
|
else if (!buffer && length)
|
|
|
|
{
|
|
|
|
return _FAlloc(length);
|
|
|
|
}
|
|
|
|
else if ((pRet = ::mi_realloc(buffer, length)))
|
2023-10-17 07:45:44 +00:00
|
|
|
{
|
|
|
|
auto uNewSize = ::mi_malloc_size(pRet);
|
|
|
|
|
|
|
|
RemoveBytesFromCounter(oldLen);
|
|
|
|
AddBytesToCounter(uNewSize);
|
|
|
|
|
|
|
|
if (gLeakFinderAlloc)
|
|
|
|
{
|
|
|
|
if (gLeakFinderFree)
|
|
|
|
{
|
2024-01-17 20:55:59 +00:00
|
|
|
gLeakFinderFree(buffer, oldLen);
|
2023-10-17 07:45:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
gLeakFinderAlloc(pRet, uNewSize);
|
|
|
|
}
|
|
|
|
}
|
2022-12-12 23:50:05 +00:00
|
|
|
}
|
|
|
|
|
2024-03-19 15:47:42 +00:00
|
|
|
if (!pRet && buffer && length)
|
2022-12-12 23:50:05 +00:00
|
|
|
{
|
2024-03-19 15:47:42 +00:00
|
|
|
OnOOM(length);
|
|
|
|
|
2022-12-12 23:50:05 +00:00
|
|
|
if (gRuntimeConfig.debug.bIsMemoryErrorFatal)
|
2022-12-07 01:29:09 +00:00
|
|
|
{
|
2022-12-12 23:50:05 +00:00
|
|
|
SysPanic("FRealloc out of memory");
|
2022-12-07 01:29:09 +00:00
|
|
|
}
|
|
|
|
}
|
2022-12-12 23:50:05 +00:00
|
|
|
|
|
|
|
return pRet;
|
2021-06-27 21:25:29 +00:00
|
|
|
}
|
|
|
|
|
2022-12-08 19:34:15 +00:00
|
|
|
AUKN_SYM void _Free(void *pHead)
|
2021-06-27 21:25:29 +00:00
|
|
|
{
|
2022-12-12 23:50:05 +00:00
|
|
|
if (!pHead)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2022-12-08 19:34:15 +00:00
|
|
|
if (AuDebug::IsPointerReserveRange(pHead))
|
2022-12-07 01:29:09 +00:00
|
|
|
{
|
2022-12-08 19:34:15 +00:00
|
|
|
AuDebug::gReserveHeap->Free(pHead);
|
2022-12-07 01:29:09 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2024-01-17 20:55:59 +00:00
|
|
|
auto uCount = ::mi_malloc_size(pHead);
|
|
|
|
|
|
|
|
RemoveBytesFromCounter(uCount);
|
2022-12-08 19:34:15 +00:00
|
|
|
::mi_free(pHead);
|
2023-10-17 07:45:44 +00:00
|
|
|
|
2024-01-17 20:55:59 +00:00
|
|
|
if (gLeakFinderFree)
|
|
|
|
{
|
|
|
|
gLeakFinderFree(pHead, uCount);
|
|
|
|
}
|
2023-10-17 07:45:44 +00:00
|
|
|
}
|
2021-06-27 21:25:29 +00:00
|
|
|
}
|
2022-03-21 05:20:19 +00:00
|
|
|
|
2023-01-15 06:05:22 +00:00
|
|
|
AUKN_SYM AuInt64 GetLastOutOfMemoryTimeNS()
|
|
|
|
{
|
|
|
|
return tlsLastOutOfMemory;
|
|
|
|
}
|
|
|
|
|
2022-03-21 05:20:19 +00:00
|
|
|
AUKN_SYM AuUInt GetPageSize()
|
|
|
|
{
|
|
|
|
return AuHwInfo::GetPageSize();
|
|
|
|
}
|
2021-06-27 21:25:29 +00:00
|
|
|
}
|