// 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_OBJECTS_EMBEDDER_DATA_SLOT_INL_H_ #define V8_OBJECTS_EMBEDDER_DATA_SLOT_INL_H_ #include "src/objects/embedder-data-slot.h" #include "src/base/memory.h" #include "src/heap/heap-write-barrier-inl.h" #include "src/objects/embedder-data-array.h" #include "src/objects/js-objects-inl.h" #include "src/objects/objects-inl.h" // Has to be the last include (doesn't have include guards): #include "src/objects/object-macros.h" namespace v8 { namespace internal { EmbedderDataSlot::EmbedderDataSlot(EmbedderDataArray array, int entry_index) : SlotBase(FIELD_ADDR(array, EmbedderDataArray::OffsetOfElementAt(entry_index))) {} EmbedderDataSlot::EmbedderDataSlot(JSObject object, int embedder_field_index) : SlotBase(FIELD_ADDR( object, object.GetEmbedderFieldOffset(embedder_field_index))) {} Object EmbedderDataSlot::load_tagged() const { return ObjectSlot(address() + kTaggedPayloadOffset).Relaxed_Load(); } void EmbedderDataSlot::store_smi(Smi value) { ObjectSlot(address() + kTaggedPayloadOffset).Relaxed_Store(value); #ifdef V8_COMPRESS_POINTERS // See gc_safe_store() for the reasons behind two stores. ObjectSlot(address() + kRawPayloadOffset).Relaxed_Store(Smi::zero()); #endif } // static void EmbedderDataSlot::store_tagged(EmbedderDataArray array, int entry_index, Object value) { int slot_offset = EmbedderDataArray::OffsetOfElementAt(entry_index); ObjectSlot(FIELD_ADDR(array, slot_offset + kTaggedPayloadOffset)) .Relaxed_Store(value); WRITE_BARRIER(array, slot_offset + kTaggedPayloadOffset, value); #ifdef V8_COMPRESS_POINTERS // See gc_safe_store() for the reasons behind two stores. ObjectSlot(FIELD_ADDR(array, slot_offset + kRawPayloadOffset)) .Relaxed_Store(Smi::zero()); #endif } // static void EmbedderDataSlot::store_tagged(JSObject object, int embedder_field_index, Object value) { int slot_offset = object.GetEmbedderFieldOffset(embedder_field_index); ObjectSlot(FIELD_ADDR(object, slot_offset + kTaggedPayloadOffset)) .Relaxed_Store(value); WRITE_BARRIER(object, slot_offset + kTaggedPayloadOffset, value); #ifdef V8_COMPRESS_POINTERS // See gc_safe_store() for the reasons behind two stores. ObjectSlot(FIELD_ADDR(object, slot_offset + kRawPayloadOffset)) .Relaxed_Store(Smi::zero()); #endif } bool EmbedderDataSlot::ToAlignedPointer(const Isolate* isolate, void** out_pointer) const { // We don't care about atomicity of access here because embedder slots // are accessed this way only from the main thread via API during "mutator" // phase which is propely synched with GC (concurrent marker may still look // at the tagged part of the embedder slot but read-only access is ok). #ifdef V8_COMPRESS_POINTERS // TODO(ishell, v8:8875): When pointer compression is enabled 8-byte size // fields (external pointers, doubles and BigInt data) are only kTaggedSize // aligned so we have to use unaligned pointer friendly way of accessing them // in order to avoid undefined behavior in C++ code. Address raw_value = base::ReadUnalignedValue
(address()); // We currently have to treat zero as nullptr in embedder slots. if (raw_value) raw_value = DecodeExternalPointer(isolate, raw_value); #else Address raw_value = *location(); #endif *out_pointer = reinterpret_cast