| // Copyright 2018 the V8 project authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #ifndef V8_HEAP_HEAP_WRITE_BARRIER_INL_H_ |
| #define V8_HEAP_HEAP_WRITE_BARRIER_INL_H_ |
| |
| // Clients of this interface shouldn't depend on lots of heap internals. |
| // Do not include anything from src/heap here! |
| |
| #include "src/heap/heap-write-barrier.h" |
| |
| #include "src/common/globals.h" |
| #include "src/objects/code.h" |
| #include "src/objects/compressed-slots-inl.h" |
| #include "src/objects/fixed-array.h" |
| #include "src/objects/heap-object.h" |
| #include "src/objects/maybe-object-inl.h" |
| #include "src/objects/slots-inl.h" |
| |
| namespace v8 { |
| namespace internal { |
| |
| // Defined in heap.cc. |
| V8_EXPORT_PRIVATE bool Heap_PageFlagsAreConsistent(HeapObject object); |
| V8_EXPORT_PRIVATE bool Heap_ValueMightRequireGenerationalWriteBarrier( |
| HeapObject value); |
| V8_EXPORT_PRIVATE void Heap_GenerationalBarrierSlow(HeapObject object, |
| Address slot, |
| HeapObject value); |
| V8_EXPORT_PRIVATE void Heap_WriteBarrierForCodeSlow(Code host); |
| |
| V8_EXPORT_PRIVATE void Heap_GenerationalBarrierForCodeSlow(Code host, |
| RelocInfo* rinfo, |
| HeapObject object); |
| |
| V8_EXPORT_PRIVATE void Heap_GenerationalEphemeronKeyBarrierSlow( |
| Heap* heap, EphemeronHashTable table, Address slot); |
| |
| // Do not use these internal details anywhere outside of this file. These |
| // internals are only intended to shortcut write barrier checks. |
| namespace heap_internals { |
| |
| struct MemoryChunk { |
| static constexpr uintptr_t kFlagsOffset = kSizetSize; |
| static constexpr uintptr_t kHeapOffset = kSizetSize + kUIntptrSize; |
| static constexpr uintptr_t kIsExecutableBit = uintptr_t{1} << 0; |
| static constexpr uintptr_t kMarkingBit = uintptr_t{1} << 17; |
| static constexpr uintptr_t kFromPageBit = uintptr_t{1} << 3; |
| static constexpr uintptr_t kToPageBit = uintptr_t{1} << 4; |
| static constexpr uintptr_t kReadOnlySpaceBit = uintptr_t{1} << 20; |
| |
| V8_INLINE static heap_internals::MemoryChunk* FromHeapObject( |
| HeapObject object) { |
| DCHECK(!V8_ENABLE_THIRD_PARTY_HEAP_BOOL); |
| return reinterpret_cast<MemoryChunk*>(object.ptr() & ~kPageAlignmentMask); |
| } |
| |
| V8_INLINE bool IsMarking() const { return GetFlags() & kMarkingBit; } |
| |
| V8_INLINE bool InYoungGeneration() const { |
| if (V8_ENABLE_THIRD_PARTY_HEAP_BOOL) return false; |
| constexpr uintptr_t kYoungGenerationMask = kFromPageBit | kToPageBit; |
| return GetFlags() & kYoungGenerationMask; |
| } |
| |
| V8_INLINE uintptr_t GetFlags() const { |
| return *reinterpret_cast<const uintptr_t*>(reinterpret_cast<Address>(this) + |
| kFlagsOffset); |
| } |
| |
| V8_INLINE Heap* GetHeap() { |
| Heap* heap = *reinterpret_cast<Heap**>(reinterpret_cast<Address>(this) + |
| kHeapOffset); |
| DCHECK_NOT_NULL(heap); |
| return heap; |
| } |
| |
| V8_INLINE bool InReadOnlySpace() const { |
| return GetFlags() & kReadOnlySpaceBit; |
| } |
| |
| V8_INLINE bool InCodeSpace() const { return GetFlags() & kIsExecutableBit; } |
| }; |
| |
| inline void GenerationalBarrierInternal(HeapObject object, Address slot, |
| HeapObject value) { |
| DCHECK(Heap_PageFlagsAreConsistent(object)); |
| heap_internals::MemoryChunk* value_chunk = |
| heap_internals::MemoryChunk::FromHeapObject(value); |
| heap_internals::MemoryChunk* object_chunk = |
| heap_internals::MemoryChunk::FromHeapObject(object); |
| |
| if (!value_chunk->InYoungGeneration() || object_chunk->InYoungGeneration()) { |
| return; |
| } |
| |
| Heap_GenerationalBarrierSlow(object, slot, value); |
| } |
| |
| inline void GenerationalEphemeronKeyBarrierInternal(EphemeronHashTable table, |
| Address slot, |
| HeapObject value) { |
| DCHECK(Heap::PageFlagsAreConsistent(table)); |
| heap_internals::MemoryChunk* value_chunk = |
| heap_internals::MemoryChunk::FromHeapObject(value); |
| heap_internals::MemoryChunk* table_chunk = |
| heap_internals::MemoryChunk::FromHeapObject(table); |
| |
| if (!value_chunk->InYoungGeneration() || table_chunk->InYoungGeneration()) { |
| return; |
| } |
| |
| Heap_GenerationalEphemeronKeyBarrierSlow(table_chunk->GetHeap(), table, slot); |
| } |
| |
| } // namespace heap_internals |
| |
| inline void WriteBarrierForCode(Code host, RelocInfo* rinfo, Object value) { |
| DCHECK(!HasWeakHeapObjectTag(value)); |
| if (!value.IsHeapObject()) return; |
| WriteBarrierForCode(host, rinfo, HeapObject::cast(value)); |
| } |
| |
| inline void WriteBarrierForCode(Code host, RelocInfo* rinfo, HeapObject value) { |
| GenerationalBarrierForCode(host, rinfo, value); |
| WriteBarrier::Marking(host, rinfo, value); |
| } |
| |
| inline void WriteBarrierForCode(Code host) { |
| Heap_WriteBarrierForCodeSlow(host); |
| } |
| |
| inline void GenerationalBarrier(HeapObject object, ObjectSlot slot, |
| Object value) { |
| if (V8_ENABLE_THIRD_PARTY_HEAP_BOOL) return; |
| DCHECK(!HasWeakHeapObjectTag(value)); |
| if (!value.IsHeapObject()) return; |
| GenerationalBarrier(object, slot, HeapObject::cast(value)); |
| } |
| |
| inline void GenerationalBarrier(HeapObject object, ObjectSlot slot, |
| Code value) { |
| if (V8_ENABLE_THIRD_PARTY_HEAP_BOOL) return; |
| DCHECK(!Heap_ValueMightRequireGenerationalWriteBarrier(value)); |
| } |
| |
| inline void GenerationalBarrier(HeapObject object, ObjectSlot slot, |
| HeapObject value) { |
| if (V8_ENABLE_THIRD_PARTY_HEAP_BOOL) return; |
| DCHECK(!HasWeakHeapObjectTag(*slot)); |
| heap_internals::GenerationalBarrierInternal(object, slot.address(), value); |
| } |
| |
| inline void GenerationalEphemeronKeyBarrier(EphemeronHashTable table, |
| ObjectSlot slot, Object value) { |
| if (V8_ENABLE_THIRD_PARTY_HEAP_BOOL) return; |
| DCHECK(!HasWeakHeapObjectTag(*slot)); |
| DCHECK(!HasWeakHeapObjectTag(value)); |
| DCHECK(value.IsHeapObject()); |
| heap_internals::GenerationalEphemeronKeyBarrierInternal( |
| table, slot.address(), HeapObject::cast(value)); |
| } |
| |
| inline void GenerationalBarrier(HeapObject object, MaybeObjectSlot slot, |
| MaybeObject value) { |
| if (V8_ENABLE_THIRD_PARTY_HEAP_BOOL) return; |
| HeapObject value_heap_object; |
| if (!value->GetHeapObject(&value_heap_object)) return; |
| heap_internals::GenerationalBarrierInternal(object, slot.address(), |
| value_heap_object); |
| } |
| |
| inline void GenerationalBarrierForCode(Code host, RelocInfo* rinfo, |
| HeapObject object) { |
| if (V8_ENABLE_THIRD_PARTY_HEAP_BOOL) return; |
| heap_internals::MemoryChunk* object_chunk = |
| heap_internals::MemoryChunk::FromHeapObject(object); |
| if (!object_chunk->InYoungGeneration()) return; |
| Heap_GenerationalBarrierForCodeSlow(host, rinfo, object); |
| } |
| |
| inline WriteBarrierMode GetWriteBarrierModeForObject( |
| HeapObject object, const DisallowGarbageCollection* promise) { |
| if (FLAG_disable_write_barriers) return SKIP_WRITE_BARRIER; |
| DCHECK(Heap_PageFlagsAreConsistent(object)); |
| heap_internals::MemoryChunk* chunk = |
| heap_internals::MemoryChunk::FromHeapObject(object); |
| if (chunk->IsMarking()) return UPDATE_WRITE_BARRIER; |
| if (chunk->InYoungGeneration()) return SKIP_WRITE_BARRIER; |
| return UPDATE_WRITE_BARRIER; |
| } |
| |
| inline bool ObjectInYoungGeneration(Object object) { |
| // TODO(rong): Fix caller of this function when we deploy |
| // v8_use_third_party_heap. |
| if (FLAG_single_generation) return false; |
| if (object.IsSmi()) return false; |
| return heap_internals::MemoryChunk::FromHeapObject(HeapObject::cast(object)) |
| ->InYoungGeneration(); |
| } |
| |
| inline bool IsReadOnlyHeapObject(HeapObject object) { |
| if (V8_ENABLE_THIRD_PARTY_HEAP_BOOL) return ReadOnlyHeap::Contains(object); |
| heap_internals::MemoryChunk* chunk = |
| heap_internals::MemoryChunk::FromHeapObject(object); |
| return chunk->InReadOnlySpace(); |
| } |
| |
| inline bool IsCodeSpaceObject(HeapObject object) { |
| heap_internals::MemoryChunk* chunk = |
| heap_internals::MemoryChunk::FromHeapObject(object); |
| return chunk->InCodeSpace(); |
| } |
| |
| base::Optional<Heap*> WriteBarrier::GetHeapIfMarking(HeapObject object) { |
| if (V8_ENABLE_THIRD_PARTY_HEAP_BOOL) return {}; |
| heap_internals::MemoryChunk* chunk = |
| heap_internals::MemoryChunk::FromHeapObject(object); |
| if (!chunk->IsMarking()) return {}; |
| return chunk->GetHeap(); |
| } |
| |
| void WriteBarrier::Marking(HeapObject host, ObjectSlot slot, Object value) { |
| DCHECK(!HasWeakHeapObjectTag(value)); |
| if (!value.IsHeapObject()) return; |
| Marking(host, HeapObjectSlot(slot), HeapObject::cast(value)); |
| } |
| |
| void WriteBarrier::Marking(HeapObject host, MaybeObjectSlot slot, |
| MaybeObject value) { |
| HeapObject value_heap_object; |
| if (!value->GetHeapObject(&value_heap_object)) return; |
| Marking(host, HeapObjectSlot(slot), value_heap_object); |
| } |
| |
| void WriteBarrier::Marking(HeapObject host, HeapObjectSlot slot, |
| HeapObject value) { |
| auto heap = GetHeapIfMarking(host); |
| if (!heap) return; |
| MarkingSlow(*heap, host, slot, value); |
| } |
| |
| void WriteBarrier::Marking(Code host, RelocInfo* reloc_info, HeapObject value) { |
| auto heap = GetHeapIfMarking(host); |
| if (!heap) return; |
| MarkingSlow(*heap, host, reloc_info, value); |
| } |
| |
| void WriteBarrier::Marking(JSArrayBuffer host, |
| ArrayBufferExtension* extension) { |
| if (!extension) return; |
| auto heap = GetHeapIfMarking(host); |
| if (!heap) return; |
| MarkingSlow(*heap, host, extension); |
| } |
| |
| void WriteBarrier::Marking(DescriptorArray descriptor_array, |
| int number_of_own_descriptors) { |
| auto heap = GetHeapIfMarking(descriptor_array); |
| if (!heap) return; |
| MarkingSlow(*heap, descriptor_array, number_of_own_descriptors); |
| } |
| |
| // static |
| void WriteBarrier::MarkingFromGlobalHandle(Object value) { |
| if (V8_ENABLE_THIRD_PARTY_HEAP_BOOL) return; |
| if (!value.IsHeapObject()) return; |
| |
| HeapObject heap_value = HeapObject::cast(value); |
| // Value may be in read only space but the chunk should never be marked |
| // as marking which would result in a bail out. |
| auto heap = GetHeapIfMarking(heap_value); |
| if (!heap) return; |
| MarkingSlowFromGlobalHandle(*heap, heap_value); |
| } |
| |
| // static |
| void WriteBarrier::MarkingFromInternalFields(JSObject host) { |
| if (V8_ENABLE_THIRD_PARTY_HEAP_BOOL) return; |
| auto heap = GetHeapIfMarking(host); |
| if (!heap) return; |
| MarkingSlowFromInternalFields(*heap, host); |
| } |
| |
| #ifdef ENABLE_SLOW_DCHECKS |
| // static |
| template <typename T> |
| bool WriteBarrier::IsRequired(HeapObject host, T value) { |
| if (BasicMemoryChunk::FromHeapObject(host)->InYoungGeneration()) return false; |
| if (value.IsSmi()) return false; |
| if (value.IsCleared()) return false; |
| HeapObject target = value.GetHeapObject(); |
| if (ReadOnlyHeap::Contains(target)) return false; |
| return !IsImmortalImmovableHeapObject(target); |
| } |
| #endif |
| |
| } // namespace internal |
| } // namespace v8 |
| |
| #endif // V8_HEAP_HEAP_WRITE_BARRIER_INL_H_ |