blob: 2cd9092241cc7a6814739d106bb5a69745ed0eda [file] [log] [blame] [edit]
// 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_