// Copyright 2020 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_COMMON_EXTERNAL_POINTER_INL_H_ #define V8_COMMON_EXTERNAL_POINTER_INL_H_ #include "include/v8-internal.h" #include "src/common/external-pointer.h" #include "src/execution/isolate.h" namespace v8 { namespace internal { V8_INLINE Address DecodeExternalPointer(const Isolate* isolate, ExternalPointer_t encoded_pointer, ExternalPointerTag tag) { STATIC_ASSERT(kExternalPointerSize == kSystemPointerSize); #ifdef V8_HEAP_SANDBOX uint32_t index = static_cast(encoded_pointer); return isolate->external_pointer_table().get(index) & ~tag; #else return encoded_pointer; #endif } V8_INLINE void InitExternalPointerField(Address field_address, Isolate* isolate) { #ifdef V8_HEAP_SANDBOX static_assert(kExternalPointerSize == kSystemPointerSize, "Review the code below, once kExternalPointerSize is 4-byte " "the address of the field will always be aligned"); ExternalPointer_t index = isolate->external_pointer_table().allocate(); base::WriteUnalignedValue(field_address, index); #else // Nothing to do. #endif // V8_HEAP_SANDBOX } V8_INLINE void InitExternalPointerField(Address field_address, Isolate* isolate, Address value, ExternalPointerTag tag) { #ifdef V8_HEAP_SANDBOX DCHECK_EQ(value & kExternalPointerTagMask, 0); ExternalPointer_t index = isolate->external_pointer_table().allocate(); isolate->external_pointer_table().set(static_cast(index), value | tag); static_assert(kExternalPointerSize == kSystemPointerSize, "Review the code below, once kExternalPointerSize is 4-byte " "the address of the field will always be aligned"); base::WriteUnalignedValue(field_address, index); #else // Pointer compression causes types larger than kTaggedSize to be unaligned. constexpr bool v8_pointer_compression_unaligned = kExternalPointerSize > kTaggedSize; ExternalPointer_t encoded_value = static_cast(value); if (v8_pointer_compression_unaligned) { base::WriteUnalignedValue(field_address, encoded_value); } else { base::Memory(field_address) = encoded_value; } #endif // V8_HEAP_SANDBOX } V8_INLINE Address ReadExternalPointerField(Address field_address, const Isolate* isolate, ExternalPointerTag tag) { // Pointer compression causes types larger than kTaggedSize to be unaligned. constexpr bool v8_pointer_compression_unaligned = kExternalPointerSize > kTaggedSize; ExternalPointer_t encoded_value; if (v8_pointer_compression_unaligned) { encoded_value = base::ReadUnalignedValue(field_address); } else { encoded_value = base::Memory(field_address); } return DecodeExternalPointer(isolate, encoded_value, tag); } V8_INLINE void WriteExternalPointerField(Address field_address, Isolate* isolate, Address value, ExternalPointerTag tag) { #ifdef V8_HEAP_SANDBOX static_assert(kExternalPointerSize == kSystemPointerSize, "Review the code below, once kExternalPointerSize is 4-byte " "the address of the field will always be aligned"); DCHECK_EQ(value & kExternalPointerTagMask, 0); ExternalPointer_t index = base::ReadUnalignedValue(field_address); isolate->external_pointer_table().set(static_cast(index), value | tag); #else // Pointer compression causes types larger than kTaggedSize to be unaligned. constexpr bool v8_pointer_compression_unaligned = kExternalPointerSize > kTaggedSize; ExternalPointer_t encoded_value = static_cast(value); if (v8_pointer_compression_unaligned) { base::WriteUnalignedValue(field_address, encoded_value); } else { base::Memory(field_address) = encoded_value; } #endif // V8_HEAP_SANDBOX } } // namespace internal } // namespace v8 #endif // V8_COMMON_EXTERNAL_POINTER_INL_H_