/* * 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/runtime.h" namespace panda::ecmascript { void Barriers::UpdateWithoutEden(const JSThread *thread, uintptr_t slotAddr, Region *objectRegion, TaggedObject *value, Region *valueRegion, WriteBarrierType writeType) { ASSERT(!valueRegion->InSharedHeap()); auto heap = thread->GetEcmaVM()->GetHeap(); if (heap->IsConcurrentFullMark()) { if (valueRegion->InCollectSet() && !objectRegion->InGeneralNewSpaceOrCSet()) { objectRegion->AtomicInsertCrossRegionRSet(slotAddr); } } else { if (!valueRegion->InYoungSpace()) { return; } } // Weak ref record and concurrent mark record maybe conflict. // This conflict is solved by keeping alive weak reference. A small amount of floating garbage may be added. TaggedObject *heapValue = JSTaggedValue(value).GetHeapObject(); if (valueRegion->IsFreshRegion()) { valueRegion->NonAtomicMark(heapValue); } else if (writeType != WriteBarrierType::DESERIALIZE && valueRegion->AtomicMark(heapValue)) { heap->GetWorkManager()->Push(MAIN_THREAD_INDEX, heapValue); } } void Barriers::Update(const JSThread *thread, uintptr_t slotAddr, Region *objectRegion, TaggedObject *value, Region *valueRegion, WriteBarrierType writeType) { if (valueRegion->InSharedHeap()) { return; } auto heap = thread->GetEcmaVM()->GetHeap(); if (heap->IsConcurrentFullMark()) { if (valueRegion->InCollectSet() && !objectRegion->InGeneralNewSpaceOrCSet()) { objectRegion->AtomicInsertCrossRegionRSet(slotAddr); } } else if (heap->IsYoungMark()) { if (!valueRegion->InGeneralNewSpace()) { return; } } else { if (!valueRegion->InEdenSpace()) { return; } } // Weak ref record and concurrent mark record maybe conflict. // This conflict is solved by keeping alive weak reference. A small amount of floating garbage may be added. TaggedObject *heapValue = JSTaggedValue(value).GetHeapObject(); if (valueRegion->IsFreshRegion()) { valueRegion->NonAtomicMark(heapValue); } else if (writeType != WriteBarrierType::DESERIALIZE && valueRegion->AtomicMark(heapValue)) { heap->GetWorkManager()->Push(MAIN_THREAD_INDEX, heapValue); } } void Barriers::UpdateShared(const JSThread *thread, TaggedObject *value, Region *valueRegion) { ASSERT(DaemonThread::GetInstance()->IsConcurrentMarkingOrFinished()); ASSERT(valueRegion->InSharedSweepableSpace()); // Weak ref record and concurrent mark record maybe conflict. // This conflict is solved by keeping alive weak reference. A small amount of floating garbage may be added. TaggedObject *heapValue = JSTaggedValue(value).GetHeapObject(); if (valueRegion->AtomicMark(heapValue)) { Heap *heap = const_cast(thread->GetEcmaVM()->GetHeap()); WorkNode *&localBuffer = heap->GetMarkingObjectLocalBuffer(); SharedHeap::GetInstance()->GetWorkManager()->PushToLocalMarkingBuffer(localBuffer, heapValue); } } template ARK_NOINLINE bool BatchBitSet([[maybe_unused]] const JSThread* thread, Region* objectRegion, JSTaggedValue* dst, size_t count) { bool allValueNotHeap = true; Region::Updater updater = objectRegion->GetBatchRSetUpdater(ToUintPtr(dst)); for (size_t i = 0; i < count; i++, updater.Next()) { JSTaggedValue taggedValue = dst[i]; if (!taggedValue.IsHeapObject()) { continue; } allValueNotHeap = false; const Region* valueRegion = Region::ObjectAddressToRange(taggedValue.GetTaggedObject()); #if ECMASCRIPT_ENABLE_BARRIER_CHECK ASSERT(taggedValue.GetRawData() != JSTaggedValue::VALUE_UNDEFINED); if (!thread->GetEcmaVM()->GetHeap()->IsAlive(taggedValue.GetHeapObject())) { LOG_FULL(FATAL) << "WriteBarrier checked value:" << taggedValue.GetRawData() << " is invalid!"; } #endif if (valueRegion->InSharedSweepableSpace()) { #ifndef NDEBUG if (UNLIKELY(taggedValue.IsWeakForHeapObject())) { CHECK_NO_LOCAL_TO_SHARE_WEAK_REF_HANDLE; } #endif updater.UpdateLocalToShare(); continue; } if constexpr (kind == Region::InYoung) { if (valueRegion->InEdenSpace()) { updater.UpdateNewToEden(); continue; } } else if constexpr (kind == Region::InGeneralOld) { if (valueRegion->InGeneralNewSpace()) { updater.UpdateOldToNew(); continue; } } } return allValueNotHeap; } template bool BatchBitSet(const JSThread*, Region*, JSTaggedValue*, size_t); template bool BatchBitSet(const JSThread*, Region*, JSTaggedValue*, size_t); template bool BatchBitSet(const JSThread*, Region*, JSTaggedValue*, size_t); } // namespace panda::ecmascript