* Copyright (c) 2022 Huawei Device Co., Ltd.
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#include "ecmascript/mem/heap_region_allocator.h"
#include "ecmascript/jit/jit.h"
#include "ecmascript/mem/mem_map_allocator.h"
#include "ecmascript/runtime.h"
#include "ecmascript/runtime_lock.h"
namespace panda::ecmascript {
HeapRegionAllocator::HeapRegionAllocator(JSRuntimeOptions &option)
{
enablePageTagThreadId_ = option.EnablePageTagThreadId();
}
Region *HeapRegionAllocator::AllocateAlignedRegion(Space *space, size_t capacity, JSThread* thread, BaseHeap *heap,
bool isFresh, size_t slotSize, Mutex *allocateLock)
{
if (capacity == 0) {
LOG_ECMA_MEM(FATAL) << "capacity must have a size bigger than 0";
UNREACHABLE();
}
RegionSpaceFlag flags = space->GetRegionFlag();
RegionTypeFlag typeFlag = isFresh ? RegionTypeFlag::FRESH : RegionTypeFlag::DEFAULT;
bool isRegular = (flags != RegionSpaceFlag::IN_HUGE_OBJECT_SPACE &&
flags != RegionSpaceFlag::IN_HUGE_MACHINE_CODE_SPACE &&
flags != RegionSpaceFlag::IN_SHARED_HUGE_OBJECT_SPACE);
bool isCompress = flags == RegionSpaceFlag::IN_NON_MOVABLE_SPACE ||
flags == RegionSpaceFlag::IN_SHARED_NON_MOVABLE || flags == RegionSpaceFlag::IN_READ_ONLY_SPACE ||
flags == RegionSpaceFlag::IN_SHARED_READ_ONLY_SPACE;
bool isMachineCode = (flags == RegionSpaceFlag::IN_MACHINE_CODE_SPACE ||
flags == RegionSpaceFlag::IN_HUGE_MACHINE_CODE_SPACE);
JSThread::ThreadId tid = 0;
bool shouldPageTag = AllocateRegionShouldPageTag(space);
if (enablePageTagThreadId_) {
tid = thread ? thread->GetThreadId() : JSThread::GetCurrentThreadId();
}
auto pool = MemMapAllocator::GetInstance()->Allocate(tid, capacity, DEFAULT_REGION_SIZE,
ToSpaceTypeName(space->GetSpaceType()), isRegular, isCompress, isMachineCode,
Jit::GetInstance()->IsEnableJitFort(), shouldPageTag, false);
void *mapMem = pool.GetMem();
if (mapMem == nullptr) {
if (heap->InGC()) {
heap->ShouldForceThrowOOMError();
pool = MemMapAllocator::GetInstance()->Allocate(tid, capacity, DEFAULT_REGION_SIZE,
ToSpaceTypeName(space->GetSpaceType()), isRegular, isCompress, isMachineCode,
Jit::GetInstance()->IsEnableJitFort(), shouldPageTag, true);
mapMem = pool.GetMem();
if (mapMem == nullptr) {
LOG_ECMA_MEM(FATAL) << "pool is empty in GC unexpectedly, process out of memory";
UNREACHABLE();
}
} else {
if (allocateLock != nullptr) {
allocateLock->Unlock();
}
if (thread != nullptr && thread->GetEcmaVM()->IsInitialized()) {
Heap *localHeap = const_cast<Heap *>(thread->GetEcmaVM()->GetHeap());
{
static bool needDumpHeapSnapshot = true;
static Mutex dumpMutex;
RuntimeLockHolder lock(thread, dumpMutex);
if (needDumpHeapSnapshot) {
localHeap->DumpHeapSnapshotBeforeOOM(Runtime::GetInstance()->IsEnableProcDumpInSharedOOM(),
"", capacity, PROCESS_HEAP_STR);
needDumpHeapSnapshot = false;
}
}
heap->ThrowOutOfMemoryErrorForDefault(thread, DEFAULT_REGION_SIZE,
"HeapRegionAllocator::AllocateAlignedRegion", false);
}
LOG_ECMA_MEM(FATAL) << "pool is empty " << annoMemoryUsage_.load(std::memory_order_relaxed)
<< ", process out of memory";
UNREACHABLE();
}
}
#if ECMASCRIPT_ENABLE_ZAP_MEM
if (memset_s(mapMem, capacity, 0, capacity) != EOK) {
LOG_FULL(FATAL) << "memset_s failed";
UNREACHABLE();
}
#endif
IncreaseAnnoMemoryUsage(capacity);
uintptr_t mem = ToUintPtr(mapMem);
uintptr_t end = mem + capacity;
Region *region = nullptr;
if (slotSize == 0) {
uintptr_t begin = AlignUp(mem + sizeof(DefaultRegion), static_cast<size_t>(MemAlignment::MEM_ALIGN_REGION));
region = new (ToVoidPtr(mem)) DefaultRegion(heap->GetNativeAreaAllocator(), mem, begin, end, flags, typeFlag);
} else {
ASSERT(G_USE_CMS_GC);
uintptr_t begin = AlignUp(mem + sizeof(CMSRegion), static_cast<size_t>(MemAlignment::MEM_ALIGN_REGION));
region = new (ToVoidPtr(mem)) CMSRegion(heap->GetNativeAreaAllocator(), mem, begin, end, flags, typeFlag,
slotSize);
}
uintptr_t addr = ToUintPtr(region);
LOG_ECMA_IF(AlignUp(addr, DEFAULT_REGION_SIZE) != addr, FATAL) << "region not align by " << DEFAULT_REGION_SIZE;
std::atomic_thread_fence(std::memory_order_seq_cst);
return region;
}
template<bool asyncFreeMem>
void HeapRegionAllocator::FreeRegion(Region *region, size_t cachedSize, bool skipCache)
{
auto size = region->GetCapacity();
bool isRegular = !region->InHugeObjectSpace() && !region->InHugeMachineCodeSpace() &&
!region->InSharedHugeObjectSpace();
bool isCompress = region->InNonMovableSpace() || region->InSharedNonMovableSpace() ||
region->InReadOnlySpace() || region->InSharedReadOnlySpace();
auto allocateBase = region->GetAllocateBase();
bool shouldPageTag = FreeRegionShouldPageTag(region);
DecreaseAnnoMemoryUsage(size);
region->Invalidate();
#if ECMASCRIPT_ENABLE_ZAP_MEM
if (memset_s(ToVoidPtr(allocateBase), size, INVALID_VALUE, size) != EOK) {
LOG_FULL(FATAL) << "memset_s failed";
UNREACHABLE();
}
#endif
if constexpr (asyncFreeMem) {
MemMapAllocator::GetInstance()->AsyncFree(ToVoidPtr(allocateBase), size, isRegular, isCompress, shouldPageTag);
} else {
MemMapAllocator::GetInstance()->CacheOrFree(ToVoidPtr(allocateBase), size, isRegular, isCompress, cachedSize,
shouldPageTag, skipCache);
}
MEMORY_TRACE_FREEREGION(allocateBase, size);
}
void HeapRegionAllocator::DecreaseMemMapUsage(Region *region)
{
bool isRegular = !region->InHugeObjectSpace() && !region->InHugeMachineCodeSpace() &&
!region->InSharedHugeObjectSpace();
MemMapAllocator::GetInstance()->DecreaseMemUsage(region->GetCapacity(), isRegular);
}
bool HeapRegionAllocator::AllocateRegionShouldPageTag(Space *space) const
{
if (enablePageTagThreadId_) {
return true;
}
MemSpaceType type = space->GetSpaceType();
if constexpr (G_USE_CMS_GC) {
return type != MemSpaceType::SLOT_SPACE;
} else {
return type != MemSpaceType::OLD_SPACE && type != MemSpaceType::LOCAL_SPACE;
}
}
bool HeapRegionAllocator::FreeRegionShouldPageTag(Region *region) const
{
if (enablePageTagThreadId_) {
return true;
}
if constexpr (G_USE_CMS_GC) {
return !region->InSlotSpace();
} else {
return !region->InOldSpace();
}
}
template void HeapRegionAllocator::FreeRegion<true>(Region *region, size_t cachedSize, bool skipCache);
template void HeapRegionAllocator::FreeRegion<false>(Region *region, size_t cachedSize, bool skipCache);
}