arkcompiler_ets_runtime/ecmascript/runtime.cpp
xiongluo f88552ff52 adjust qos priority for partial gc
issue:https://gitee.com/openharmony/arkcompiler_ets_runtime/issues/I9V0YF

Signed-off-by: xiongluo <xiongluo@huawei.com>
Change-Id: I4285021ebc9ff8b38dfb8d5c07cd83fa2dc0e416
2024-06-04 22:02:13 +08:00

355 lines
14 KiB
C++

/*
* Copyright (c) 2024 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/runtime.h"
#include <memory>
#include "ecmascript/checkpoint/thread_state_transition.h"
#include "ecmascript/compiler/aot_file/an_file_data_manager.h"
#include "ecmascript/daemon/daemon_thread.h"
#include "ecmascript/ecma_string_table.h"
#include "ecmascript/ecma_vm.h"
#include "ecmascript/jit/jit.h"
#include "ecmascript/jspandafile/js_pandafile_manager.h"
#include "ecmascript/jspandafile/program_object.h"
#include "ecmascript/log_wrapper.h"
#include "ecmascript/mem/mem_map_allocator.h"
#include "ecmascript/module/js_module_manager.h"
#include "ecmascript/napi/include/jsnapi_expo.h"
#include "ecmascript/pgo_profiler/pgo_profiler_manager.h"
#include "libpandafile/index_accessor.h"
namespace panda::ecmascript {
using PGOProfilerManager = pgo::PGOProfilerManager;
int32_t Runtime::vmCount_ = 0;
int32_t Runtime::destroyCount_ = 0;
bool Runtime::firstVmCreated_ = false;
Mutex *Runtime::vmCreationLock_ = new Mutex();
Runtime *Runtime::instance_ = nullptr;
Runtime *Runtime::GetInstance()
{
ASSERT(instance_ != nullptr);
return instance_;
}
Runtime::~Runtime()
{
LockHolder lock(constpoolLock_);
auto iter = globalSharedConstpools_.begin();
while (iter != globalSharedConstpools_.end()) {
LOG_ECMA(INFO) << "remove js pandafile by vm destruct, file:" << iter->first->GetJSPandaFileDesc();
JSPandaFileManager::GetInstance()->RemoveJSPandaFile(iter->first);
iter->second.clear();
iter++;
}
globalSharedConstpools_.clear();
delete externalRegisteredStringTable_;
externalRegisteredStringTable_ = nullptr;
}
void Runtime::CreateIfFirstVm(const JSRuntimeOptions &options)
{
LockHolder lock(*vmCreationLock_);
if (!firstVmCreated_) {
Log::Initialize(options);
EcmaVM::InitializeIcuData(options);
MemMapAllocator::GetInstance()->Initialize(ecmascript::DEFAULT_REGION_SIZE);
PGOProfilerManager::GetInstance()->Initialize(options.GetPGOProfilerPath(),
options.GetPGOHotnessThreshold());
ASSERT(instance_ == nullptr);
instance_ = new Runtime();
SharedHeap::CreateNewInstance();
DaemonThread::CreateNewInstance();
firstVmCreated_ = true;
}
}
void Runtime::InitializeIfFirstVm(EcmaVM *vm)
{
{
LockHolder lock(*vmCreationLock_);
if (++vmCount_ == 1) {
ThreadManagedScope managedScope(vm->GetAssociatedJSThread());
PreInitialization(vm);
bool isEnableFastJit = vm->GetJSOptions().IsEnableJIT() && vm->GetJSOptions().GetEnableAsmInterpreter();
bool isEnableBaselineJit =
vm->GetJSOptions().IsEnableBaselineJIT() && vm->GetJSOptions().GetEnableAsmInterpreter();
Jit::GetInstance()->SetEnableOrDisable(vm->GetJSOptions(), isEnableFastJit, isEnableBaselineJit);
vm->Initialize();
PostInitialization(vm);
}
}
if (!vm->IsInitialized()) {
ThreadManagedScope managedScope(vm->GetAssociatedJSThread());
vm->Initialize();
}
}
void Runtime::PreInitialization(const EcmaVM *vm)
{
mainThread_ = vm->GetAssociatedJSThread();
mainThread_->SetMainThread();
nativeAreaAllocator_ = std::make_unique<NativeAreaAllocator>();
heapRegionAllocator_ = std::make_unique<HeapRegionAllocator>();
stringTable_ = std::make_unique<EcmaStringTable>();
SharedHeap::GetInstance()->Initialize(nativeAreaAllocator_.get(), heapRegionAllocator_.get(),
const_cast<EcmaVM*>(vm)->GetJSOptions(), DaemonThread::GetInstance());
}
void Runtime::PostInitialization(const EcmaVM *vm)
{
// Use the main thread's globalconst after it has initialized,
// and copy shared parts to other thread's later.
globalConstants_ = mainThread_->GlobalConstants();
globalEnv_ = vm->GetGlobalEnv().GetTaggedValue();
SharedHeap::GetInstance()->PostInitialization(globalConstants_, const_cast<EcmaVM*>(vm)->GetJSOptions());
SharedModuleManager::GetInstance()->Initialize(vm);
}
void Runtime::DestroyIfLastVm()
{
LockHolder lock(*vmCreationLock_);
if (--vmCount_ <= 0) {
SharedHeap::GetInstance()->WaitAllTasksFinishedAfterAllJSThreadEliminated();
DaemonThread::DestroyInstance();
SharedHeap::DestroyInstance();
AnFileDataManager::GetInstance()->SafeDestroyAllData();
MemMapAllocator::GetInstance()->Finalize();
PGOProfilerManager::GetInstance()->Destroy();
SharedModuleManager::GetInstance()->Destroy();
ASSERT(instance_ != nullptr);
delete instance_;
instance_ = nullptr;
firstVmCreated_ = false;
}
}
void Runtime::RegisterThread(JSThread* newThread)
{
LockHolder lock(threadsLock_);
ASSERT(std::find(threads_.begin(), threads_.end(), newThread) == threads_.end());
threads_.emplace_back(newThread);
// send all current suspended requests to the new thread
for (uint32_t i = 0; i < suspendNewCount_; i++) {
newThread->SuspendThread(true);
}
}
// Note: currently only called when thread is to be destroyed.
void Runtime::UnregisterThread(JSThread* thread)
{
LockHolder lock(threadsLock_);
ASSERT(std::find(threads_.begin(), threads_.end(), thread) != threads_.end());
ASSERT(!thread->IsInRunningState());
threads_.remove(thread);
}
void Runtime::SuspendAll(JSThread *current)
{
ASSERT(current != nullptr);
ASSERT(!current->IsInRunningState());
#ifndef NDEBUG
ASSERT(!current->HasLaunchedSuspendAll());
current->LaunchSuspendAll();
#endif
SuspendAllThreadsImpl(current);
}
void Runtime::ResumeAll(JSThread *current)
{
ASSERT(current != nullptr);
ASSERT(!current->IsInRunningState());
#ifndef NDEBUG
ASSERT(current->HasLaunchedSuspendAll());
current->CompleteSuspendAll();
#endif
ResumeAllThreadsImpl(current);
}
void Runtime::SuspendAllThreadsImpl(JSThread *current)
{
SuspendBarrier barrier;
{
LockHolder lock(threadsLock_);
while (suspendNewCount_ != 0) {
// Someone has already suspended all threads.
// Wait until it finishes.
threadSuspendCondVar_.Wait(&threadsLock_);
}
suspendNewCount_++;
if (threads_.size() == 1) {
ASSERT(current == mainThread_ || current->IsDaemonThread());
return;
}
barrier.Initialize(threads_.size() - 1);
for (auto i: threads_) {
if (i == current) {
continue;
}
i->SuspendThread(+1, &barrier);
// The two flags, SUSPEND_REQUEST and ACTIVE_BARRIER, are set by Suspend-Thread guarded by suspendLock_, so
// the target thread-I may do not see these two flags in time. As a result, it can switch its state freely
// without responding to the ACTIVE_BARRIER flag and the suspend-thread will always wait it. However,
// as long as it sees the flags, the actions of passing barrier will be triggered. When the thread-I
// switches from NON_RUNNING to RUNNING, it will firstly pass the barrier and then be blocked by the
// SUSPEND_REQUEST flag. If the thread-I switches from RUNNING to NON_RUNNING, it will switch the state and
// then act on the barrier. If the thread-I go to checkpoint in RUNNING state, it will act on the barrier
// and be blocked by SUSPEND_REQUEST flag.
if (i->IsSuspended()) {
// Because of the multi-threads situation, currently thread-I may be in RUNNING state or is goding to
// be RUNNING state even inside this branch. In both scenarios, for instance of RUNNING state,
// according to the modification order of atomic-variable stateAndFlags_, thread-I will see the
// SUSPEND_REQUEST and ACTIVE_BARRIER and act on them before switching to RUNNING. Besides, notice the
// using of suspendLock_ inside PassSuspendBarrier(), there is not data-race for passing barrier.
i->PassSuspendBarrier();
}
}
}
barrier.Wait();
}
void Runtime::ResumeAllThreadsImpl(JSThread *current)
{
LockHolder lock(threadsLock_);
if (suspendNewCount_ > 0) {
suspendNewCount_--;
}
if (suspendNewCount_ == 0) {
// Signal to waiting to suspend threads
threadSuspendCondVar_.Signal();
}
for (auto i : threads_) {
if (i != current) {
i->ResumeThread(true);
}
}
}
void Runtime::IterateSerializeRoot(const RootVisitor &v)
{
for (auto &it : serializeRootMap_) {
for (auto &rootObj : it.second) {
v(Root::ROOT_VM, ObjectSlot(reinterpret_cast<uintptr_t>(&rootObj)));
}
}
}
bool Runtime::HasCachedConstpool(const JSPandaFile *jsPandaFile)
{
LockHolder lock(constpoolLock_);
return globalSharedConstpools_.find(jsPandaFile) != globalSharedConstpools_.end();
}
JSTaggedValue Runtime::FindConstpool(const JSPandaFile *jsPandaFile, int32_t index)
{
LockHolder lock(constpoolLock_);
return FindConstpoolUnlocked(jsPandaFile, index);
}
JSTaggedValue Runtime::FindConstpoolUnlocked(const JSPandaFile *jsPandaFile, int32_t index)
{
auto iter = globalSharedConstpools_.find(jsPandaFile);
if (iter == globalSharedConstpools_.end()) {
return JSTaggedValue::Hole();
}
auto constpoolIter = iter->second.find(index);
if (constpoolIter == iter->second.end()) {
return JSTaggedValue::Hole();
}
return constpoolIter->second;
}
JSHandle<ConstantPool> Runtime::AddOrUpdateConstpool(const JSPandaFile *jsPandaFile,
JSHandle<ConstantPool> constpool,
int32_t index)
{
LockHolder lock(constpoolLock_);
if (globalSharedConstpools_.find(jsPandaFile) == globalSharedConstpools_.end()) {
globalSharedConstpools_[jsPandaFile] = CMap<int32_t, JSTaggedValue>();
}
auto &constpoolMap = globalSharedConstpools_[jsPandaFile];
auto iter = constpoolMap.find(index);
if (iter == constpoolMap.end()) {
int32_t constpoolIndex = GetAndIncreaseSharedConstpoolCount();
constpool->SetUnsharedConstpoolIndex(JSTaggedValue(constpoolIndex));
constpoolMap.insert({index, constpool.GetTaggedValue()});
return constpool;
}
return JSHandle<ConstantPool>(reinterpret_cast<uintptr_t>(&iter->second));
}
std::optional<std::reference_wrapper<CMap<int32_t, JSTaggedValue>>> Runtime::FindConstpools(
const JSPandaFile *jsPandaFile)
{
LockHolder lock(constpoolLock_);
auto iter = globalSharedConstpools_.find(jsPandaFile);
if (iter == globalSharedConstpools_.end()) {
return std::nullopt;
}
return iter->second;
}
void Runtime::ProcessNativeDeleteInSharedGC(const WeakRootVisitor &visitor)
{
// No need lock here, only shared gc will sweep shared constpool, meanwhile other threads are suspended.
auto iterator = globalSharedConstpools_.begin();
ECMA_BYTRACE_NAME(HITRACE_TAG_ARK, "Constpools:" + std::to_string(globalSharedConstpools_.size()));
while (iterator != globalSharedConstpools_.end()) {
auto &constpools = iterator->second;
auto constpoolIter = constpools.begin();
while (constpoolIter != constpools.end()) {
JSTaggedValue constpoolVal = constpoolIter->second;
if (constpoolVal.IsHeapObject()) {
TaggedObject *obj = constpoolVal.GetTaggedObject();
auto fwd = visitor(obj);
if (fwd == nullptr) {
int32_t constpoolIndex =
ConstantPool::Cast(constpoolVal.GetTaggedObject())->GetUnsharedConstpoolIndex();
ASSERT(0 <= constpoolIndex && constpoolIndex != ConstantPool::CONSTPOOL_TYPE_FLAG &&
constpoolIndex < UNSHARED_CONSTANTPOOL_COUNT);
EraseUnusedConstpool(iterator->first, constpoolIter->first, constpoolIndex);
constpoolIter = constpools.erase(constpoolIter);
// when shared constpool is not referenced by any objects,
// global unshared constpool count can be reuse.
freeSharedConstpoolIndex_.insert(constpoolIndex);
continue;
}
}
++constpoolIter;
}
if (constpools.size() == 0) {
LOG_ECMA(INFO) << "remove js pandafile by gc, file:" << iterator->first->GetJSPandaFileDesc();
JSPandaFileManager::GetInstance()->RemoveJSPandaFile(iterator->first);
iterator = globalSharedConstpools_.erase(iterator);
} else {
++iterator;
}
}
}
void Runtime::EraseUnusedConstpool(const JSPandaFile *jsPandaFile, int32_t index, int32_t constpoolIndex)
{
GCIterateThreadList([jsPandaFile, index, constpoolIndex](JSThread* thread) {
ASSERT(!thread->IsInRunningState());
auto context = thread->GetCurrentEcmaContext();
context->EraseUnusedConstpool(jsPandaFile, index, constpoolIndex);
});
}
} // namespace panda::ecmascript