| // Copyright 2017 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_FIXED_ARRAY_H_ |
| #define V8_OBJECTS_FIXED_ARRAY_H_ |
| |
| #include <optional> |
| |
| #include "src/common/globals.h" |
| #include "src/handles/maybe-handles.h" |
| #include "src/objects/heap-object.h" |
| #include "src/objects/instance-type.h" |
| #include "src/objects/maybe-object.h" |
| #include "src/objects/objects.h" |
| #include "src/objects/smi.h" |
| #include "src/objects/tagged.h" |
| #include "src/objects/trusted-object.h" |
| #include "src/roots/roots.h" |
| #include "src/utils/memcopy.h" |
| |
| // Has to be the last include (doesn't have include guards): |
| #include "src/objects/object-macros.h" |
| |
| namespace v8::internal { |
| |
| #include "torque-generated/src/objects/fixed-array-tq.inc" |
| |
| // Limit all fixed arrays to the same max capacity, so that non-resizing |
| // transitions between different elements kinds (like Smi to Double) will not |
| // error. |
| static constexpr int kMaxFixedArrayCapacity = |
| V8_LOWER_LIMITS_MODE_BOOL ? (16 * 1024 * 1024) : (64 * 1024 * 1024); |
| |
| namespace detail { |
| template <class Super, bool kLengthEqualsCapacity> |
| class ArrayHeaderBase; |
| |
| V8_OBJECT template <class Super> |
| class ArrayHeaderBase<Super, false> : public Super { |
| public: |
| inline int capacity() const; |
| inline int capacity(AcquireLoadTag tag) const; |
| inline void set_capacity(int value); |
| inline void set_capacity(int value, ReleaseStoreTag tag); |
| |
| // TODO(leszeks): Make this private. |
| public: |
| TaggedMember<Smi> capacity_; |
| } V8_OBJECT_END; |
| |
| V8_OBJECT template <class Super> |
| class ArrayHeaderBase<Super, true> : public Super { |
| public: |
| inline int length() const; |
| inline int length(AcquireLoadTag tag) const; |
| inline void set_length(int value); |
| inline void set_length(int value, ReleaseStoreTag tag); |
| |
| inline int capacity() const; |
| inline int capacity(AcquireLoadTag tag) const; |
| inline void set_capacity(int value); |
| inline void set_capacity(int value, ReleaseStoreTag tag); |
| |
| // TODO(leszeks): Make this private. |
| public: |
| TaggedMember<Smi> length_; |
| } V8_OBJECT_END; |
| |
| template <class Shape, class Super, typename = void> |
| struct TaggedArrayHeaderHelper { |
| using type = ArrayHeaderBase<Super, Shape::kLengthEqualsCapacity>; |
| }; |
| template <class Shape, class Super> |
| struct TaggedArrayHeaderHelper< |
| Shape, Super, std::void_t<typename Shape::template ExtraFields<Super>>> { |
| using BaseHeader = ArrayHeaderBase<Super, Shape::kLengthEqualsCapacity>; |
| using type = typename Shape::template ExtraFields<BaseHeader>; |
| static_assert(std::is_base_of_v<BaseHeader, type>); |
| }; |
| template <class Shape, class Super> |
| using TaggedArrayHeader = typename TaggedArrayHeaderHelper<Shape, Super>::type; |
| } // namespace detail |
| |
| #define V8_ARRAY_EXTRA_FIELDS(...) \ |
| V8_OBJECT template <typename Super> \ |
| struct ExtraFields : public Super __VA_ARGS__ V8_OBJECT_END |
| |
| // Derived: must not have any fields - extra fields can be specified in the |
| // Shap using V8_ARRAY_EXTRA_FIELDS. |
| V8_OBJECT template <class Derived, class ShapeT, class Super = HeapObjectLayout> |
| class TaggedArrayBase : public detail::TaggedArrayHeader<ShapeT, Super> { |
| static_assert(std::is_base_of_v<HeapObjectLayout, Super>); |
| using ElementT = typename ShapeT::ElementT; |
| |
| static_assert(sizeof(TaggedMember<ElementT>) == kTaggedSize); |
| static_assert(is_subtype_v<ElementT, MaybeObject>); |
| |
| using ElementMemberT = |
| TaggedMember<ElementT, typename ShapeT::CompressionScheme>; |
| |
| template <typename ElementT> |
| static constexpr bool kSupportsSmiElements = |
| std::is_convertible_v<Smi, ElementT>; |
| |
| static constexpr WriteBarrierMode kDefaultMode = |
| std::is_same_v<ElementT, Smi> ? SKIP_WRITE_BARRIER : UPDATE_WRITE_BARRIER; |
| |
| public: |
| using Header = detail::TaggedArrayHeader<ShapeT, Super>; |
| static constexpr bool kElementsAreMaybeObject = is_maybe_weak_v<ElementT>; |
| static constexpr int kElementSize = kTaggedSize; |
| |
| private: |
| using SlotType = |
| std::conditional_t<kElementsAreMaybeObject, MaybeObjectSlot, ObjectSlot>; |
| |
| public: |
| using Shape = ShapeT; |
| |
| inline Tagged<ElementT> get(int index) const; |
| inline Tagged<ElementT> get(int index, RelaxedLoadTag) const; |
| inline Tagged<ElementT> get(int index, AcquireLoadTag) const; |
| inline Tagged<ElementT> get(int index, SeqCstAccessTag) const; |
| |
| inline void set(int index, Tagged<ElementT> value, |
| WriteBarrierMode mode = kDefaultMode); |
| template <typename T = ElementT, |
| typename = std::enable_if<kSupportsSmiElements<T>>> |
| inline void set(int index, Tagged<Smi> value); |
| inline void set(int index, Tagged<ElementT> value, RelaxedStoreTag, |
| WriteBarrierMode mode = kDefaultMode); |
| template <typename T = ElementT, |
| typename = std::enable_if<kSupportsSmiElements<T>>> |
| inline void set(int index, Tagged<Smi> value, RelaxedStoreTag); |
| inline void set(int index, Tagged<ElementT> value, ReleaseStoreTag, |
| WriteBarrierMode mode = kDefaultMode); |
| template <typename T = ElementT, |
| typename = std::enable_if<kSupportsSmiElements<T>>> |
| inline void set(int index, Tagged<Smi> value, ReleaseStoreTag); |
| inline void set(int index, Tagged<ElementT> value, SeqCstAccessTag, |
| WriteBarrierMode mode = kDefaultMode); |
| template <typename T = ElementT, |
| typename = std::enable_if<kSupportsSmiElements<T>>> |
| inline void set(int index, Tagged<Smi> value, SeqCstAccessTag); |
| |
| inline Tagged<ElementT> swap(int index, Tagged<ElementT> value, |
| SeqCstAccessTag, |
| WriteBarrierMode mode = kDefaultMode); |
| inline Tagged<ElementT> compare_and_swap( |
| int index, Tagged<ElementT> expected, Tagged<ElementT> value, |
| SeqCstAccessTag, WriteBarrierMode mode = kDefaultMode); |
| |
| // Move vs. Copy behaves like memmove vs. memcpy: for Move, the memory |
| // regions may overlap, for Copy they must not overlap. |
| inline static void MoveElements(Isolate* isolate, Tagged<Derived> dst, |
| int dst_index, Tagged<Derived> src, |
| int src_index, int len, |
| WriteBarrierMode mode = kDefaultMode); |
| inline static void CopyElements(Isolate* isolate, Tagged<Derived> dst, |
| int dst_index, Tagged<Derived> src, |
| int src_index, int len, |
| WriteBarrierMode mode = kDefaultMode); |
| |
| // Right-trim the array. |
| // Invariant: 0 < new_length <= length() |
| inline void RightTrim(Isolate* isolate, int new_capacity); |
| |
| inline int AllocatedSize() const; |
| static inline constexpr int SizeFor(int capacity) { |
| return sizeof(Header) + capacity * kElementSize; |
| } |
| static inline constexpr int OffsetOfElementAt(int index) { |
| return SizeFor(index); |
| } |
| |
| // Gives access to raw memory which stores the array's data. |
| inline SlotType RawFieldOfFirstElement() const; |
| inline SlotType RawFieldOfElementAt(int index) const; |
| |
| // Maximal allowed capacity, in number of elements. Chosen s.t. the byte size |
| // fits into a Smi which is necessary for being able to create a free space |
| // filler. |
| // TODO(jgruber): The kMaxCapacity could be larger (`(Smi::kMaxValue - |
| // Shape::kHeaderSize) / kElementSize`), but our tests rely on a |
| // smaller maximum to avoid timeouts. |
| static constexpr int kMaxCapacity = kMaxFixedArrayCapacity; |
| static_assert(Smi::IsValid(SizeFor(kMaxCapacity))); |
| |
| // Maximally allowed length for regular (non large object space) object. |
| static constexpr int kMaxRegularCapacity = |
| (kMaxRegularHeapObjectSize - sizeof(Header)) / kElementSize; |
| static_assert(kMaxRegularCapacity < kMaxCapacity); |
| |
| protected: |
| template <class IsolateT> |
| static Handle<Derived> Allocate( |
| IsolateT* isolate, int capacity, |
| std::optional<DisallowGarbageCollection>* no_gc_out, |
| AllocationType allocation = AllocationType::kYoung); |
| |
| static constexpr int NewCapacityForIndex(int index, int old_capacity); |
| |
| inline bool IsInBounds(int index) const; |
| inline bool IsCowArray() const; |
| |
| FLEXIBLE_ARRAY_MEMBER(ElementMemberT, objects); |
| } V8_OBJECT_END; |
| |
| class TaggedArrayShape final : public AllStatic { |
| public: |
| using ElementT = Object; |
| using CompressionScheme = V8HeapCompressionScheme; |
| static constexpr RootIndex kMapRootIndex = RootIndex::kFixedArrayMap; |
| static constexpr bool kLengthEqualsCapacity = true; |
| }; |
| |
| // FixedArray describes fixed-sized arrays with element type Object. |
| V8_OBJECT class FixedArray |
| : public TaggedArrayBase<FixedArray, TaggedArrayShape> { |
| using Super = TaggedArrayBase<FixedArray, TaggedArrayShape>; |
| |
| public: |
| template <class IsolateT> |
| static inline Handle<FixedArray> New( |
| IsolateT* isolate, int capacity, |
| AllocationType allocation = AllocationType::kYoung); |
| |
| using Super::CopyElements; |
| using Super::MoveElements; |
| |
| // TODO(jgruber): Only needed for FixedArrays used as JSObject elements. |
| inline void MoveElements(Isolate* isolate, int dst_index, int src_index, |
| int len, WriteBarrierMode mode); |
| inline void CopyElements(Isolate* isolate, int dst_index, |
| Tagged<FixedArray> src, int src_index, int len, |
| WriteBarrierMode mode); |
| |
| // Return a grown copy if the index is bigger than the array's length. |
| template <template <typename> typename HandleType> |
| requires( |
| std::is_convertible_v<HandleType<FixedArray>, DirectHandle<FixedArray>>) |
| V8_EXPORT_PRIVATE static HandleType<FixedArray> SetAndGrow( |
| Isolate* isolate, HandleType<FixedArray> array, int index, |
| DirectHandle<Object> value); |
| |
| // Right-trim the array. |
| // Invariant: 0 < new_length <= length() |
| V8_EXPORT_PRIVATE void RightTrim(Isolate* isolate, int new_capacity); |
| // Right-trims the array, and canonicalizes length 0 to empty_fixed_array. |
| template <template <typename> typename HandleType> |
| requires( |
| std::is_convertible_v<HandleType<FixedArray>, DirectHandle<FixedArray>>) |
| static HandleType<FixedArray> RightTrimOrEmpty(Isolate* isolate, |
| HandleType<FixedArray> array, |
| int new_length); |
| |
| // TODO(jgruber): Only needed for FixedArrays used as JSObject elements. |
| inline void FillWithHoles(int from, int to); |
| |
| // For compatibility with FixedDoubleArray: |
| // TODO(jgruber): Only needed for FixedArrays used as JSObject elements. |
| inline bool is_the_hole(Isolate* isolate, int index); |
| inline void set_the_hole(Isolate* isolate, int index); |
| inline void set_the_hole(ReadOnlyRoots ro_roots, int index); |
| |
| DECL_PRINTER(FixedArray) |
| DECL_VERIFIER(FixedArray) |
| |
| class BodyDescriptor; |
| |
| static constexpr int kMaxLength = FixedArray::kMaxCapacity; |
| static constexpr int kMaxRegularLength = FixedArray::kMaxRegularCapacity; |
| |
| private: |
| inline static Handle<FixedArray> Resize( |
| Isolate* isolate, DirectHandle<FixedArray> xs, int new_capacity, |
| AllocationType allocation = AllocationType::kYoung, |
| WriteBarrierMode mode = UPDATE_WRITE_BARRIER); |
| } V8_OBJECT_END; |
| |
| static_assert(sizeof(FixedArray) == Internals::kFixedArrayHeaderSize); |
| |
| class TrustedArrayShape final : public AllStatic { |
| public: |
| using ElementT = Object; |
| // The elements in a TrustedFixedArray are pointers into the main cage! |
| using CompressionScheme = V8HeapCompressionScheme; |
| static constexpr RootIndex kMapRootIndex = RootIndex::kTrustedFixedArrayMap; |
| static constexpr bool kLengthEqualsCapacity = true; |
| }; |
| |
| // A FixedArray in trusted space and with a unique instance type. |
| // |
| // Note: while the array itself is trusted, it contains tagged pointers into |
| // the main pointer compression heap and therefore to _untrusted_ objects. |
| // If you are storing references to other trusted object (i.e. protected |
| // pointers), use ProtectedFixedArray. |
| V8_OBJECT class TrustedFixedArray |
| : public TaggedArrayBase<TrustedFixedArray, TrustedArrayShape, |
| TrustedObjectLayout> { |
| using Super = TaggedArrayBase<TrustedFixedArray, TrustedArrayShape, |
| TrustedObjectLayout>; |
| |
| public: |
| template <class IsolateT> |
| static inline Handle<TrustedFixedArray> New( |
| IsolateT* isolate, int capacity, |
| AllocationType allocation = AllocationType::kTrusted); |
| |
| DECL_PRINTER(TrustedFixedArray) |
| DECL_VERIFIER(TrustedFixedArray) |
| |
| class BodyDescriptor; |
| |
| static constexpr int kMaxLength = TrustedFixedArray::kMaxCapacity; |
| static constexpr int kMaxRegularLength = |
| TrustedFixedArray::kMaxRegularCapacity; |
| } V8_OBJECT_END; |
| |
| class ProtectedArrayShape final : public AllStatic { |
| public: |
| using ElementT = Union<TrustedObject, Smi>; |
| using CompressionScheme = TrustedSpaceCompressionScheme; |
| static constexpr RootIndex kMapRootIndex = RootIndex::kProtectedFixedArrayMap; |
| static constexpr bool kLengthEqualsCapacity = true; |
| }; |
| |
| // A FixedArray in trusted space, holding protected pointers (to other trusted |
| // objects). If you want to store JS-heap references, use TrustedFixedArray. |
| // ProtectedFixedArray has a unique instance type. |
| V8_OBJECT class ProtectedFixedArray |
| : public TaggedArrayBase<ProtectedFixedArray, ProtectedArrayShape, |
| TrustedObjectLayout> { |
| using Super = TaggedArrayBase<ProtectedFixedArray, ProtectedArrayShape, |
| TrustedObjectLayout>; |
| |
| public: |
| // Allocate a new ProtectedFixedArray of the given capacity, initialized with |
| // Smi::zero(). |
| template <class IsolateT> |
| static inline Handle<ProtectedFixedArray> New(IsolateT* isolate, int capacity, |
| bool shared = false); |
| |
| DECL_PRINTER(ProtectedFixedArray) |
| DECL_VERIFIER(ProtectedFixedArray) |
| |
| class BodyDescriptor; |
| |
| static constexpr int kMaxLength = Super::kMaxCapacity; |
| static constexpr int kMaxRegularLength = |
| ProtectedFixedArray::kMaxRegularCapacity; |
| } V8_OBJECT_END; |
| |
| // FixedArray alias added only because of IsFixedArrayExact() predicate, which |
| // checks for the exact instance type FIXED_ARRAY_TYPE instead of a range |
| // check: [FIRST_FIXED_ARRAY_TYPE, LAST_FIXED_ARRAY_TYPE]. |
| V8_OBJECT |
| class FixedArrayExact final : public FixedArray { |
| } V8_OBJECT_END; |
| |
| // Common superclass for FixedArrays that allow implementations to share common |
| // accessors and some code paths. Note that due to single-inheritance |
| // restrictions, it is not part of the actual type hierarchy. Instead, we slot |
| // it in with manual is_subtype specializations in tagged.h. |
| // TODO(jgruber): This class is really specific to FixedArrays used as |
| // elements backing stores and should not be part of the common FixedArray |
| // hierarchy. |
| V8_OBJECT |
| class FixedArrayBase : public detail::ArrayHeaderBase<HeapObjectLayout, true> { |
| public: |
| static constexpr int kLengthOffset = HeapObject::kHeaderSize; |
| static constexpr int kHeaderSize = kLengthOffset + kTaggedSize; |
| static constexpr int kMaxLength = FixedArray::kMaxCapacity; |
| static constexpr int kMaxRegularLength = FixedArray::kMaxRegularCapacity; |
| |
| static int GetMaxLengthForNewSpaceAllocation(ElementsKind kind); |
| |
| V8_EXPORT_PRIVATE bool IsCowArray() const; |
| |
| DECL_VERIFIER(FixedArrayBase) |
| } V8_OBJECT_END; |
| |
| V8_OBJECT |
| template <class Derived, class ShapeT, class Super = HeapObjectLayout> |
| class PrimitiveArrayBase : public detail::ArrayHeaderBase<Super, true> { |
| static_assert(std::is_base_of_v<HeapObjectLayout, Super>); |
| |
| using ElementT = typename ShapeT::ElementT; |
| static_assert(!is_subtype_v<ElementT, Object>); |
| |
| // Bug(v8:8875): Doubles may be unaligned. |
| using ElementMemberT = std::conditional_t<std::is_same_v<ElementT, double>, |
| UnalignedDoubleMember, ElementT>; |
| static_assert(alignof(ElementMemberT) <= alignof(Tagged_t)); |
| |
| public: |
| using Shape = ShapeT; |
| static constexpr bool kElementsAreMaybeObject = false; |
| static constexpr int kElementSize = sizeof(ElementMemberT); |
| using Header = detail::ArrayHeaderBase<Super, true>; |
| |
| inline ElementMemberT get(int index) const; |
| inline void set(int index, ElementMemberT value); |
| |
| inline int AllocatedSize() const; |
| static inline constexpr int SizeFor(int length) { |
| return OBJECT_POINTER_ALIGN(OffsetOfElementAt(length)); |
| } |
| static inline constexpr int OffsetOfElementAt(int index) { |
| return sizeof(Header) + index * kElementSize; |
| } |
| |
| // Gives access to raw memory which stores the array's data. |
| // Note that on 32-bit archs and on 64-bit platforms with pointer compression |
| // the pointers to 8-byte size elements are not guaranteed to be aligned. |
| inline ElementMemberT* begin(); |
| inline const ElementMemberT* begin() const; |
| inline ElementMemberT* end(); |
| inline const ElementMemberT* end() const; |
| inline int DataSize() const; |
| |
| static inline Tagged<Derived> FromAddressOfFirstElement(Address address); |
| |
| // Maximal allowed length, in number of elements. Chosen s.t. the byte size |
| // fits into a Smi which is necessary for being able to create a free space |
| // filler. |
| // TODO(jgruber): The kMaxLength could be larger (`(Smi::kMaxValue - |
| // sizeof(Header)) / kElementSize`), but our tests rely on a |
| // smaller maximum to avoid timeouts. |
| static constexpr int kMaxLength = kMaxFixedArrayCapacity; |
| static_assert(Smi::IsValid(SizeFor(kMaxLength))); |
| |
| // Maximally allowed length for regular (non large object space) object. |
| static constexpr int kMaxRegularLength = |
| (kMaxRegularHeapObjectSize - sizeof(Header)) / kElementSize; |
| static_assert(kMaxRegularLength < kMaxLength); |
| |
| protected: |
| template <class IsolateT> |
| static Handle<Derived> Allocate( |
| IsolateT* isolate, int length, |
| std::optional<DisallowGarbageCollection>* no_gc_out, |
| AllocationType allocation = AllocationType::kYoung); |
| |
| inline bool IsInBounds(int index) const; |
| |
| FLEXIBLE_ARRAY_MEMBER(ElementMemberT, values); |
| } V8_OBJECT_END; |
| |
| class FixedDoubleArrayShape final : public AllStatic { |
| public: |
| using ElementT = double; |
| static constexpr RootIndex kMapRootIndex = RootIndex::kFixedDoubleArrayMap; |
| }; |
| |
| // FixedDoubleArray describes fixed-sized arrays with element type double. |
| V8_OBJECT class FixedDoubleArray |
| : public PrimitiveArrayBase<FixedDoubleArray, FixedDoubleArrayShape> { |
| using Super = PrimitiveArrayBase<FixedDoubleArray, FixedDoubleArrayShape>; |
| |
| public: |
| // Note this returns FixedArrayBase due to canonicalization to |
| // empty_fixed_array. |
| template <class IsolateT> |
| static inline Handle<FixedArrayBase> New( |
| IsolateT* isolate, int capacity, |
| AllocationType allocation = AllocationType::kYoung); |
| |
| // Setter and getter for elements. |
| inline double get_scalar(int index); |
| inline uint64_t get_representation(int index); |
| static inline Handle<Object> get(Tagged<FixedDoubleArray> array, int index, |
| Isolate* isolate); |
| inline void set(int index, double value); |
| #ifdef V8_ENABLE_EXPERIMENTAL_UNDEFINED_DOUBLE |
| inline void set_undefined(int index); |
| inline bool is_undefined(int index); |
| #endif // V8_ENABLE_EXPERIMENTAL_UNDEFINED_DOUBLE |
| |
| inline void set_the_hole(Isolate* isolate, int index); |
| inline void set_the_hole(int index); |
| inline bool is_the_hole(Isolate* isolate, int index); |
| inline bool is_the_hole(int index); |
| |
| inline void MoveElements(Isolate* isolate, int dst_index, int src_index, |
| int len, WriteBarrierMode /* unused */); |
| |
| inline void FillWithHoles(int from, int to); |
| |
| DECL_PRINTER(FixedDoubleArray) |
| DECL_VERIFIER(FixedDoubleArray) |
| |
| class BodyDescriptor; |
| } V8_OBJECT_END; |
| |
| static_assert(FixedDoubleArray::kMaxLength == FixedArray::kMaxLength); |
| |
| class WeakFixedArrayShape final : public AllStatic { |
| public: |
| using ElementT = MaybeObject; |
| using CompressionScheme = V8HeapCompressionScheme; |
| static constexpr RootIndex kMapRootIndex = RootIndex::kWeakFixedArrayMap; |
| static constexpr bool kLengthEqualsCapacity = true; |
| }; |
| |
| // WeakFixedArray describes fixed-sized arrays with element type |
| // Tagged<MaybeObject>. |
| V8_OBJECT class WeakFixedArray |
| : public TaggedArrayBase<WeakFixedArray, WeakFixedArrayShape> { |
| using Super = TaggedArrayBase<WeakFixedArray, WeakFixedArrayShape>; |
| |
| public: |
| template <class IsolateT> |
| static inline Handle<WeakFixedArray> New( |
| IsolateT* isolate, int capacity, |
| AllocationType allocation = AllocationType::kYoung, |
| MaybeDirectHandle<Object> initial_value = {}); |
| |
| DECL_PRINTER(WeakFixedArray) |
| DECL_VERIFIER(WeakFixedArray) |
| |
| class BodyDescriptor; |
| } V8_OBJECT_END; |
| |
| class TrustedWeakFixedArrayShape final : public AllStatic { |
| public: |
| using ElementT = MaybeObject; |
| using CompressionScheme = V8HeapCompressionScheme; |
| static constexpr RootIndex kMapRootIndex = |
| RootIndex::kTrustedWeakFixedArrayMap; |
| static constexpr bool kLengthEqualsCapacity = true; |
| }; |
| |
| // A WeakFixedArray in trusted space holding pointers into the main cage. |
| V8_OBJECT class TrustedWeakFixedArray |
| : public TaggedArrayBase<TrustedWeakFixedArray, TrustedWeakFixedArrayShape, |
| TrustedObjectLayout> { |
| using Super = |
| TaggedArrayBase<TrustedWeakFixedArray, TrustedWeakFixedArrayShape>; |
| |
| public: |
| template <class IsolateT> |
| static inline Handle<TrustedWeakFixedArray> New(IsolateT* isolate, |
| int capacity); |
| |
| DECL_PRINTER(TrustedWeakFixedArray) |
| DECL_VERIFIER(TrustedWeakFixedArray) |
| |
| class BodyDescriptor; |
| } V8_OBJECT_END; |
| |
| class ProtectedWeakFixedArrayShape final : public AllStatic { |
| public: |
| using ElementT = Union<MaybeWeak<TrustedObject>, Smi>; |
| using CompressionScheme = TrustedSpaceCompressionScheme; |
| static constexpr RootIndex kMapRootIndex = |
| RootIndex::kProtectedWeakFixedArrayMap; |
| static constexpr bool kLengthEqualsCapacity = true; |
| }; |
| |
| // A WeakFixedArray in trusted space, containing weak pointers to other |
| // trusted objects (or smis). |
| V8_OBJECT class ProtectedWeakFixedArray |
| : public TaggedArrayBase<ProtectedWeakFixedArray, |
| ProtectedWeakFixedArrayShape, |
| TrustedObjectLayout> { |
| using Super = |
| TaggedArrayBase<ProtectedWeakFixedArray, ProtectedWeakFixedArrayShape, |
| TrustedObjectLayout>; |
| |
| public: |
| template <class IsolateT> |
| static inline Handle<ProtectedWeakFixedArray> New(IsolateT* isolate, |
| int capacity); |
| DECL_PRINTER(ProtectedWeakFixedArray) |
| DECL_VERIFIER(ProtectedWeakFixedArray) |
| |
| class BodyDescriptor; |
| } V8_OBJECT_END; |
| |
| // WeakArrayList is like a WeakFixedArray with static convenience methods for |
| // adding more elements. length() returns the number of elements in the list and |
| // capacity() returns the allocated size. The number of elements is stored at |
| // kLengthOffset and is updated with every insertion. The array grows |
| // dynamically with O(1) amortized insertion. |
| class WeakArrayList |
| : public TorqueGeneratedWeakArrayList<WeakArrayList, HeapObject> { |
| public: |
| NEVER_READ_ONLY_SPACE |
| DECL_PRINTER(WeakArrayList) |
| |
| V8_EXPORT_PRIVATE static Handle<WeakArrayList> AddToEnd( |
| Isolate* isolate, Handle<WeakArrayList> array, |
| MaybeObjectDirectHandle value); |
| |
| // A version that adds two elements. This ensures that the elements are |
| // inserted atomically w.r.t GC. |
| V8_EXPORT_PRIVATE static Handle<WeakArrayList> AddToEnd( |
| Isolate* isolate, Handle<WeakArrayList> array, |
| MaybeObjectDirectHandle value1, Tagged<Smi> value2); |
| |
| // Appends an element to the array and possibly compacts and shrinks live weak |
| // references to the start of the collection. Only use this method when |
| // indices to elements can change. |
| static V8_WARN_UNUSED_RESULT DirectHandle<WeakArrayList> Append( |
| Isolate* isolate, DirectHandle<WeakArrayList> array, |
| MaybeObjectDirectHandle value, |
| AllocationType allocation = AllocationType::kYoung); |
| |
| // Compact weak references to the beginning of the array. |
| V8_EXPORT_PRIVATE void Compact(Isolate* isolate); |
| |
| inline Tagged<MaybeObject> Get(int index) const; |
| inline Tagged<MaybeObject> Get(PtrComprCageBase cage_base, int index) const; |
| // TODO(jgruber): Remove this once it's no longer needed for compatibility |
| // with WeakFixedArray. |
| inline Tagged<MaybeObject> get(int index) const; |
| |
| // Set the element at index to obj. The underlying array must be large enough. |
| // If you need to grow the WeakArrayList, use the static AddToEnd() method |
| // instead. |
| inline void Set(int index, Tagged<MaybeObject> value, |
| WriteBarrierMode mode = UPDATE_WRITE_BARRIER); |
| inline void Set(int index, Tagged<Smi> value); |
| |
| static constexpr int SizeForCapacity(int capacity) { |
| return SizeFor(capacity); |
| } |
| |
| static constexpr int CapacityForLength(int length) { |
| return length + std::max(length / 2, 2); |
| } |
| |
| // Gives access to raw memory which stores the array's data. |
| inline MaybeObjectSlot data_start(); |
| |
| inline void CopyElements(Isolate* isolate, int dst_index, |
| Tagged<WeakArrayList> src, int src_index, int len, |
| WriteBarrierMode mode); |
| |
| V8_EXPORT_PRIVATE bool IsFull() const; |
| |
| inline int AllocatedSize() const; |
| |
| class BodyDescriptor; |
| |
| // Maximal allowed length, in number of elements. Chosen s.t. the byte size |
| // fits into a Smi which is necessary for being able to create a free space |
| // filler. |
| // TODO(jgruber): The kMaxLength could be larger (`(Smi::kMaxValue - |
| // sizeof(Header)) / kElementSize`), but our tests rely on a |
| // smaller maximum to avoid timeouts. |
| static constexpr int kMaxCapacity = kMaxFixedArrayCapacity; |
| static_assert(Smi::IsValid(SizeFor(kMaxCapacity))); |
| |
| static Handle<WeakArrayList> EnsureSpace( |
| Isolate* isolate, Handle<WeakArrayList> array, int length, |
| AllocationType allocation = AllocationType::kYoung); |
| |
| // Returns the number of non-cleaned weak references in the array. |
| int CountLiveWeakReferences() const; |
| |
| // Returns the number of non-cleaned elements in the array. |
| int CountLiveElements() const; |
| |
| // Returns whether an entry was found and removed. Will move the elements |
| // around in the array - this method can only be used in cases where the user |
| // doesn't care about the indices! Users should make sure there are no |
| // duplicates. |
| V8_EXPORT_PRIVATE bool RemoveOne(MaybeObjectDirectHandle value); |
| |
| // Searches the array (linear time) and returns whether it contains the value. |
| V8_EXPORT_PRIVATE bool Contains(Tagged<MaybeObject> value); |
| |
| class Iterator; |
| |
| private: |
| static int OffsetOfElementAt(int index) { |
| return kHeaderSize + index * kTaggedSize; |
| } |
| |
| TQ_OBJECT_CONSTRUCTORS(WeakArrayList) |
| }; |
| |
| class WeakArrayList::Iterator { |
| public: |
| explicit Iterator(Tagged<WeakArrayList> array) : index_(0), array_(array) {} |
| Iterator(const Iterator&) = delete; |
| Iterator& operator=(const Iterator&) = delete; |
| |
| inline Tagged<HeapObject> Next(); |
| |
| private: |
| int index_; |
| Tagged<WeakArrayList> array_; |
| DISALLOW_GARBAGE_COLLECTION(no_gc_) |
| }; |
| |
| class ArrayListShape final : public AllStatic { |
| public: |
| using ElementT = Object; |
| using CompressionScheme = V8HeapCompressionScheme; |
| static constexpr RootIndex kMapRootIndex = RootIndex::kArrayListMap; |
| static constexpr bool kLengthEqualsCapacity = false; |
| |
| V8_ARRAY_EXTRA_FIELDS({ TaggedMember<Smi> length_; }); |
| }; |
| |
| // A generic array that grows dynamically with O(1) amortized insertion. |
| V8_OBJECT class ArrayList : public TaggedArrayBase<ArrayList, ArrayListShape> { |
| using Super = TaggedArrayBase<ArrayList, ArrayListShape>; |
| |
| public: |
| using Shape = ArrayListShape; |
| |
| template <class IsolateT> |
| static inline DirectHandle<ArrayList> New( |
| IsolateT* isolate, int capacity, |
| AllocationType allocation = AllocationType::kYoung); |
| |
| inline int length() const; |
| inline void set_length(int value); |
| |
| V8_EXPORT_PRIVATE static DirectHandle<ArrayList> Add( |
| Isolate* isolate, DirectHandle<ArrayList> array, Tagged<Smi> obj, |
| AllocationType allocation = AllocationType::kYoung); |
| V8_EXPORT_PRIVATE static DirectHandle<ArrayList> Add( |
| Isolate* isolate, DirectHandle<ArrayList> array, DirectHandle<Object> obj, |
| AllocationType allocation = AllocationType::kYoung); |
| V8_EXPORT_PRIVATE static DirectHandle<ArrayList> Add( |
| Isolate* isolate, DirectHandle<ArrayList> array, |
| DirectHandle<Object> obj0, DirectHandle<Object> obj1, |
| AllocationType allocation = AllocationType::kYoung); |
| |
| V8_EXPORT_PRIVATE static DirectHandle<FixedArray> ToFixedArray( |
| Isolate* isolate, DirectHandle<ArrayList> array, |
| AllocationType allocation = AllocationType::kYoung); |
| |
| // Right-trim the array. |
| // Invariant: 0 < new_length <= length() |
| void RightTrim(Isolate* isolate, int new_capacity); |
| |
| DECL_PRINTER(ArrayList) |
| DECL_VERIFIER(ArrayList) |
| |
| class BodyDescriptor; |
| |
| private: |
| static DirectHandle<ArrayList> EnsureSpace( |
| Isolate* isolate, DirectHandle<ArrayList> array, int length, |
| AllocationType allocation = AllocationType::kYoung); |
| } V8_OBJECT_END; |
| |
| class ByteArrayShape final : public AllStatic { |
| public: |
| static constexpr int kElementSize = kUInt8Size; |
| using ElementT = uint8_t; |
| static constexpr RootIndex kMapRootIndex = RootIndex::kByteArrayMap; |
| static constexpr bool kLengthEqualsCapacity = true; |
| }; |
| |
| // ByteArray represents fixed sized arrays containing raw bytes that will not |
| // be scanned by the garbage collector. |
| V8_OBJECT class ByteArray |
| : public PrimitiveArrayBase<ByteArray, ByteArrayShape> { |
| using Super = PrimitiveArrayBase<ByteArray, ByteArrayShape>; |
| |
| public: |
| using Shape = ByteArrayShape; |
| |
| template <class IsolateT> |
| static inline Handle<ByteArray> New( |
| IsolateT* isolate, int capacity, |
| AllocationType allocation = AllocationType::kYoung); |
| |
| inline uint32_t get_int(int offset) const; |
| inline void set_int(int offset, uint32_t value); |
| |
| // Given the full object size in bytes, return the length that should be |
| // passed to New s.t. an object of the same size is created. |
| static constexpr int LengthFor(int size_in_bytes) { |
| DCHECK(IsAligned(size_in_bytes, kTaggedSize)); |
| DCHECK_GE(size_in_bytes, sizeof(Header)); |
| return size_in_bytes - sizeof(Header); |
| } |
| |
| DECL_PRINTER(ByteArray) |
| DECL_VERIFIER(ByteArray) |
| |
| class BodyDescriptor; |
| } V8_OBJECT_END; |
| |
| class TrustedByteArrayShape final : public AllStatic { |
| public: |
| static constexpr int kElementSize = kUInt8Size; |
| using ElementT = uint8_t; |
| static constexpr RootIndex kMapRootIndex = RootIndex::kTrustedByteArrayMap; |
| static constexpr bool kLengthEqualsCapacity = true; |
| }; |
| |
| // A ByteArray in trusted space. |
| V8_OBJECT |
| class TrustedByteArray |
| : public PrimitiveArrayBase<TrustedByteArray, TrustedByteArrayShape, |
| TrustedObjectLayout> { |
| using Super = PrimitiveArrayBase<TrustedByteArray, TrustedByteArrayShape, |
| TrustedObjectLayout>; |
| |
| public: |
| using Shape = TrustedByteArrayShape; |
| |
| template <class IsolateT> |
| static inline Handle<TrustedByteArray> New( |
| IsolateT* isolate, int capacity, |
| AllocationType allocation_type = AllocationType::kTrusted); |
| |
| inline uint32_t get_int(int offset) const; |
| inline void set_int(int offset, uint32_t value); |
| |
| // Given the full object size in bytes, return the length that should be |
| // passed to New s.t. an object of the same size is created. |
| static constexpr int LengthFor(int size_in_bytes) { |
| DCHECK(IsAligned(size_in_bytes, kTaggedSize)); |
| DCHECK_GE(size_in_bytes, sizeof(Header)); |
| return size_in_bytes - sizeof(Header); |
| } |
| |
| DECL_PRINTER(TrustedByteArray) |
| DECL_VERIFIER(TrustedByteArray) |
| |
| class BodyDescriptor; |
| } V8_OBJECT_END; |
| |
| // Convenience class for treating a ByteArray / TrustedByteArray as array of |
| // fixed-size integers. |
| V8_OBJECT |
| template <typename T, typename Base> |
| class FixedIntegerArrayBase : public Base { |
| static_assert(std::is_integral_v<T>); |
| |
| public: |
| // {MoreArgs...} allows passing the `AllocationType` if `Base` is `ByteArray`. |
| template <typename... MoreArgs> |
| static Handle<FixedIntegerArrayBase<T, Base>> New(Isolate* isolate, |
| int length, |
| MoreArgs&&... more_args); |
| |
| // Get/set the contents of this array. |
| T get(int index) const; |
| void set(int index, T value); |
| |
| // Code Generation support. |
| static constexpr int OffsetOfElementAt(int index) { |
| return sizeof(typename Base::Header) + index * sizeof(T); |
| } |
| |
| inline int length() const; |
| |
| protected: |
| Address get_element_address(int index) const; |
| } V8_OBJECT_END; |
| |
| using FixedInt8Array = FixedIntegerArrayBase<int8_t, ByteArray>; |
| using FixedUInt8Array = FixedIntegerArrayBase<uint8_t, ByteArray>; |
| using FixedInt16Array = FixedIntegerArrayBase<int16_t, ByteArray>; |
| using FixedUInt16Array = FixedIntegerArrayBase<uint16_t, ByteArray>; |
| using FixedInt32Array = FixedIntegerArrayBase<int32_t, ByteArray>; |
| using FixedUInt32Array = FixedIntegerArrayBase<uint32_t, ByteArray>; |
| using FixedInt64Array = FixedIntegerArrayBase<int64_t, ByteArray>; |
| using FixedUInt64Array = FixedIntegerArrayBase<uint64_t, ByteArray>; |
| |
| // Use with care! Raw addresses on the heap are not safe in combination with |
| // the sandbox. However, this can for example be used to store sandboxed |
| // pointers, which is safe. |
| V8_OBJECT |
| template <typename Base> |
| class FixedAddressArrayBase : public FixedIntegerArrayBase<Address, Base> { |
| using Underlying = FixedIntegerArrayBase<Address, Base>; |
| |
| public: |
| // Get/set a sandboxed pointer from this array. |
| inline Address get_sandboxed_pointer(int index) const; |
| inline void set_sandboxed_pointer(int index, Address value); |
| |
| // {MoreArgs...} allows passing the `AllocationType` if `Base` is `ByteArray`. |
| template <typename... MoreArgs> |
| static inline DirectHandle<FixedAddressArrayBase> New( |
| Isolate* isolate, int length, MoreArgs&&... more_args); |
| } V8_OBJECT_END; |
| |
| using FixedAddressArray = FixedAddressArrayBase<ByteArray>; |
| using TrustedFixedAddressArray = FixedAddressArrayBase<TrustedByteArray>; |
| |
| V8_OBJECT |
| template <class T, class Super> |
| class PodArrayBase : public Super { |
| public: |
| void copy_out(int index, T* result, int length) { |
| MemCopy(result, &this->values()[index * sizeof(T)], length * sizeof(T)); |
| } |
| |
| void copy_in(int index, const T* buffer, int length) { |
| MemCopy(&this->values()[index * sizeof(T)], buffer, length * sizeof(T)); |
| } |
| |
| bool matches(const T* buffer, int length) { |
| DCHECK_LE(length, this->length()); |
| return memcmp(this->begin(), buffer, length * sizeof(T)) == 0; |
| } |
| |
| bool matches(int offset, const T* buffer, int length) { |
| DCHECK_LE(offset, this->length()); |
| DCHECK_LE(offset + length, this->length()); |
| return memcmp(this->begin() + sizeof(T) * offset, buffer, |
| length * sizeof(T)) == 0; |
| } |
| |
| T get(int index) { |
| T result; |
| copy_out(index, &result, 1); |
| return result; |
| } |
| |
| void set(int index, const T& value) { copy_in(index, &value, 1); } |
| |
| inline int length() const; |
| } V8_OBJECT_END; |
| |
| // Wrapper class for ByteArray which can store arbitrary C++ classes, as long |
| // as they can be copied with memcpy. |
| V8_OBJECT |
| template <class T> |
| class PodArray : public PodArrayBase<T, ByteArray> { |
| public: |
| static Handle<PodArray<T>> New( |
| Isolate* isolate, int length, |
| AllocationType allocation = AllocationType::kYoung); |
| static Handle<PodArray<T>> New( |
| LocalIsolate* isolate, int length, |
| AllocationType allocation = AllocationType::kOld); |
| } V8_OBJECT_END; |
| |
| V8_OBJECT |
| template <class T> |
| class TrustedPodArray : public PodArrayBase<T, TrustedByteArray> { |
| public: |
| static DirectHandle<TrustedPodArray<T>> New(Isolate* isolate, int length); |
| static DirectHandle<TrustedPodArray<T>> New(LocalIsolate* isolate, |
| int length); |
| } V8_OBJECT_END; |
| |
| } // namespace v8::internal |
| |
| #include "src/objects/object-macros-undef.h" |
| |
| #endif // V8_OBJECTS_FIXED_ARRAY_H_ |