fixed-array.h 22.7 KB
Newer Older
1 2 3 4 5 6 7
// 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_

8
#include "src/handles/maybe-handles.h"
9
#include "src/objects/instance-type.h"
10
#include "src/objects/objects.h"
11
#include "src/objects/smi.h"
12 13 14 15 16 17

// Has to be the last include (doesn't have include guards):
#include "src/objects/object-macros.h"

namespace v8 {
namespace internal {
18

19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50
#define FIXED_ARRAY_SUB_INSTANCE_TYPE_LIST(V)    \
  V(BYTECODE_ARRAY_CONSTANT_POOL_SUB_TYPE)       \
  V(BYTECODE_ARRAY_HANDLER_TABLE_SUB_TYPE)       \
  V(CODE_STUBS_TABLE_SUB_TYPE)                   \
  V(COMPILATION_CACHE_TABLE_SUB_TYPE)            \
  V(CONTEXT_SUB_TYPE)                            \
  V(COPY_ON_WRITE_SUB_TYPE)                      \
  V(DEOPTIMIZATION_DATA_SUB_TYPE)                \
  V(DESCRIPTOR_ARRAY_SUB_TYPE)                   \
  V(EMBEDDED_OBJECT_SUB_TYPE)                    \
  V(ENUM_CACHE_SUB_TYPE)                         \
  V(ENUM_INDICES_CACHE_SUB_TYPE)                 \
  V(DEPENDENT_CODE_SUB_TYPE)                     \
  V(DICTIONARY_ELEMENTS_SUB_TYPE)                \
  V(DICTIONARY_PROPERTIES_SUB_TYPE)              \
  V(EMPTY_PROPERTIES_DICTIONARY_SUB_TYPE)        \
  V(PACKED_ELEMENTS_SUB_TYPE)                    \
  V(FAST_PROPERTIES_SUB_TYPE)                    \
  V(FAST_TEMPLATE_INSTANTIATIONS_CACHE_SUB_TYPE) \
  V(HANDLER_TABLE_SUB_TYPE)                      \
  V(JS_COLLECTION_SUB_TYPE)                      \
  V(JS_WEAK_COLLECTION_SUB_TYPE)                 \
  V(NOSCRIPT_SHARED_FUNCTION_INFOS_SUB_TYPE)     \
  V(NUMBER_STRING_CACHE_SUB_TYPE)                \
  V(OBJECT_TO_CODE_SUB_TYPE)                     \
  V(OPTIMIZED_CODE_LITERALS_SUB_TYPE)            \
  V(OPTIMIZED_CODE_MAP_SUB_TYPE)                 \
  V(PROTOTYPE_USERS_SUB_TYPE)                    \
  V(REGEXP_MULTIPLE_CACHE_SUB_TYPE)              \
  V(RETAINED_MAPS_SUB_TYPE)                      \
  V(SCOPE_INFO_SUB_TYPE)                         \
  V(SCRIPT_LIST_SUB_TYPE)                        \
51
  V(SERIALIZED_OBJECTS_SUB_TYPE)                 \
52 53 54 55 56 57 58 59 60 61 62 63 64 65 66
  V(SHARED_FUNCTION_INFOS_SUB_TYPE)              \
  V(SINGLE_CHARACTER_STRING_CACHE_SUB_TYPE)      \
  V(SLOW_TEMPLATE_INSTANTIATIONS_CACHE_SUB_TYPE) \
  V(STRING_SPLIT_CACHE_SUB_TYPE)                 \
  V(TEMPLATE_INFO_SUB_TYPE)                      \
  V(FEEDBACK_METADATA_SUB_TYPE)                  \
  V(WEAK_NEW_SPACE_OBJECT_TO_CODE_SUB_TYPE)

enum FixedArraySubInstanceType {
#define DEFINE_FIXED_ARRAY_SUB_INSTANCE_TYPE(name) name,
  FIXED_ARRAY_SUB_INSTANCE_TYPE_LIST(DEFINE_FIXED_ARRAY_SUB_INSTANCE_TYPE)
#undef DEFINE_FIXED_ARRAY_SUB_INSTANCE_TYPE
      LAST_FIXED_ARRAY_SUB_TYPE = WEAK_NEW_SPACE_OBJECT_TO_CODE_SUB_TYPE
};

67 68
#include "torque-generated/src/objects/fixed-array-tq.inc"

69 70
// Common superclass for FixedArrays that allow implementations to share
// common accessors and some code paths.
71 72
class FixedArrayBase
    : public TorqueGeneratedFixedArrayBase<FixedArrayBase, HeapObject> {
73
 public:
74 75 76 77
  // Forward declare the non-atomic (set_)length defined in torque.
  using TorqueGeneratedFixedArrayBase::length;
  using TorqueGeneratedFixedArrayBase::set_length;
  DECL_RELEASE_ACQUIRE_INT_ACCESSORS(length)
78

79
  inline Object unchecked_length(AcquireLoadTag) const;
80

81 82
  static int GetMaxLengthForNewSpaceAllocation(ElementsKind kind);

83
  V8_EXPORT_PRIVATE bool IsCowArray() const;
84

85 86 87 88 89 90 91 92
  // Maximal allowed size, in bytes, of a single FixedArrayBase.
  // Prevents overflowing size computations, as well as extreme memory
  // consumption. It's either (512Mb - kTaggedSize) or (1024Mb - kTaggedSize).
  // -kTaggedSize is here to ensure that this max size always fits into Smi
  // which is necessary for being able to create a free space filler for the
  // whole array of kMaxSize.
  static const int kMaxSize = 128 * kTaggedSize * MB - kTaggedSize;
  STATIC_ASSERT(Smi::IsValid(kMaxSize));
93

94
 protected:
95 96 97
  TQ_OBJECT_CONSTRUCTORS(FixedArrayBase)
  inline FixedArrayBase(Address ptr,
                        HeapObject::AllowInlineSmiStorage allow_smi);
98 99
};

100
// FixedArray describes fixed-sized arrays with element type Object.
101 102
class FixedArray
    : public TorqueGeneratedFixedArray<FixedArray, FixedArrayBase> {
103 104
 public:
  // Setter and getter for elements.
105
  inline Object get(int index) const;
106
  inline Object get(PtrComprCageBase cage_base, int index) const;
107

108
  static inline Handle<Object> get(FixedArray array, int index,
109 110 111
                                   Isolate* isolate);

  // Return a grown copy if the index is bigger than the array's length.
112
  V8_EXPORT_PRIVATE static Handle<FixedArray> SetAndGrow(
113
      Isolate* isolate, Handle<FixedArray> array, int index,
114
      Handle<Object> value);
115

116 117
  // Relaxed accessors.
  inline Object get(int index, RelaxedLoadTag) const;
118 119
  inline Object get(PtrComprCageBase cage_base, int index,
                    RelaxedLoadTag) const;
120 121 122 123 124 125
  inline void set(int index, Object value, RelaxedStoreTag,
                  WriteBarrierMode mode = UPDATE_WRITE_BARRIER);
  inline void set(int index, Smi value, RelaxedStoreTag);

  // Acquire/release accessors.
  inline Object get(int index, AcquireLoadTag) const;
126 127
  inline Object get(PtrComprCageBase cage_base, int index,
                    AcquireLoadTag) const;
128 129 130
  inline void set(int index, Object value, ReleaseStoreTag,
                  WriteBarrierMode mode = UPDATE_WRITE_BARRIER);
  inline void set(int index, Smi value, ReleaseStoreTag);
131

132
  // Setter that uses write barrier.
133
  inline void set(int index, Object value);
134 135 136
  inline bool is_the_hole(Isolate* isolate, int index);

  // Setter that doesn't need write barrier.
137
  inline void set(int index, Smi value);
138
  // Setter with explicit barrier mode.
139
  inline void set(int index, Object value, WriteBarrierMode mode);
140 141 142 143 144 145 146 147 148

  // Setters for frequently used oddballs located in old space.
  inline void set_undefined(int index);
  inline void set_undefined(Isolate* isolate, int index);
  inline void set_null(int index);
  inline void set_null(Isolate* isolate, int index);
  inline void set_the_hole(int index);
  inline void set_the_hole(Isolate* isolate, int index);

149
  inline ObjectSlot GetFirstElementAddress();
150 151 152
  inline bool ContainsOnlySmisOrHoles();

  // Gives access to raw memory which stores the array's data.
153 154
  inline ObjectSlot data_start();

155 156
  inline void MoveElements(Isolate* isolate, int dst_index, int src_index,
                           int len, WriteBarrierMode mode);
157

158
  inline void CopyElements(Isolate* isolate, int dst_index, FixedArray src,
159 160
                           int src_index, int len, WriteBarrierMode mode);

161 162
  inline void FillWithHoles(int from, int to);

163
  // Shrink the array and insert filler objects. {new_length} must be > 0.
164
  V8_EXPORT_PRIVATE void Shrink(Isolate* isolate, int new_length);
165 166
  // If {new_length} is 0, return the canonical empty FixedArray. Otherwise
  // like above.
167 168
  static Handle<FixedArray> ShrinkOrEmpty(Isolate* isolate,
                                          Handle<FixedArray> array,
169
                                          int new_length);
170 171

  // Copy a sub array from the receiver to dest.
172 173
  V8_EXPORT_PRIVATE void CopyTo(int pos, FixedArray dest, int dest_pos,
                                int len) const;
174 175 176

  // Garbage collection support.
  static constexpr int SizeFor(int length) {
177
    return kHeaderSize + length * kTaggedSize;
178 179 180
  }

  // Code Generation support.
181 182 183 184
  static constexpr int OffsetOfElementAt(int index) {
    STATIC_ASSERT(kObjectsOffset == SizeFor(0));
    return SizeFor(index);
  }
185 186

  // Garbage collection support.
187
  inline ObjectSlot RawFieldOfElementAt(int index);
188 189

  // Maximally allowed length of a FixedArray.
190
  static const int kMaxLength = (kMaxSize - kHeaderSize) / kTaggedSize;
191 192 193
  static_assert(Internals::IsValidSmi(kMaxLength),
                "FixedArray maxLength not a Smi");

194 195 196
  // Maximally allowed length for regular (non large object space) object.
  STATIC_ASSERT(kMaxRegularHeapObjectSize < kMaxSize);
  static const int kMaxRegularLength =
197
      (kMaxRegularHeapObjectSize - kHeaderSize) / kTaggedSize;
198 199 200 201

  // Dispatched behavior.
  DECL_PRINTER(FixedArray)

202 203 204
  int AllocatedSize();

  class BodyDescriptor;
205

206 207
  static constexpr int kObjectsOffset = kHeaderSize;

208 209 210
 protected:
  // Set operation on FixedArray without using write barriers. Can
  // only be used for storing old space objects or smis.
211
  static inline void NoWriteBarrierSet(FixedArray array, int index,
212
                                       Object value);
213 214 215 216

 private:
  STATIC_ASSERT(kHeaderSize == Internals::kFixedArrayHeaderSize);

217 218 219 220
  inline void set_undefined(ReadOnlyRoots ro_roots, int index);
  inline void set_null(ReadOnlyRoots ro_roots, int index);
  inline void set_the_hole(ReadOnlyRoots ro_roots, int index);

221
  TQ_OBJECT_CONSTRUCTORS(FixedArray)
222 223
};

224 225 226
// 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].
227
class FixedArrayExact final : public FixedArray {};
228 229

// FixedDoubleArray describes fixed-sized arrays with element type double.
230 231
class FixedDoubleArray
    : public TorqueGeneratedFixedDoubleArray<FixedDoubleArray, FixedArrayBase> {
232 233 234 235
 public:
  // Setter and getter for elements.
  inline double get_scalar(int index);
  inline uint64_t get_representation(int index);
236
  static inline Handle<Object> get(FixedDoubleArray array, int index,
237 238 239 240 241 242 243 244 245 246 247 248 249 250
                                   Isolate* isolate);
  inline void set(int index, double value);
  inline void set_the_hole(Isolate* isolate, int index);
  inline void set_the_hole(int index);

  // Checking for the hole.
  inline bool is_the_hole(Isolate* isolate, int index);
  inline bool is_the_hole(int index);

  // Garbage collection support.
  inline static int SizeFor(int length) {
    return kHeaderSize + length * kDoubleSize;
  }

251 252
  inline void MoveElements(Isolate* isolate, int dst_index, int src_index,
                           int len, WriteBarrierMode mode);
253 254 255 256 257 258

  inline void FillWithHoles(int from, int to);

  // Code Generation support.
  static int OffsetOfElementAt(int index) { return SizeFor(index); }

259 260 261
  // Start offset of elements.
  static constexpr int kFloatsOffset = kHeaderSize;

262
  // Maximally allowed length of a FixedDoubleArray.
263
  static const int kMaxLength = (kMaxSize - kHeaderSize) / kDoubleSize;
264 265
  static_assert(Internals::IsValidSmi(kMaxLength),
                "FixedDoubleArray maxLength not a Smi");
266 267 268 269 270 271 272

  // Dispatched behavior.
  DECL_PRINTER(FixedDoubleArray)
  DECL_VERIFIER(FixedDoubleArray)

  class BodyDescriptor;

273
  TQ_OBJECT_CONSTRUCTORS(FixedDoubleArray)
274 275
};

276
// WeakFixedArray describes fixed-sized arrays with element type
277
// MaybeObject.
278 279
class WeakFixedArray
    : public TorqueGeneratedWeakFixedArray<WeakFixedArray, HeapObject> {
280
 public:
281
  inline MaybeObject Get(int index) const;
282
  inline MaybeObject Get(PtrComprCageBase cage_base, int index) const;
283

284 285 286
  inline void Set(
      int index, MaybeObject value,
      WriteBarrierMode mode = WriteBarrierMode::UPDATE_WRITE_BARRIER);
287

288 289 290 291
  // Forward declare the non-atomic (set_)length defined in torque.
  using TorqueGeneratedWeakFixedArray::length;
  using TorqueGeneratedWeakFixedArray::set_length;
  DECL_RELEASE_ACQUIRE_INT_ACCESSORS(length)
292 293

  // Gives access to raw memory which stores the array's data.
294
  inline MaybeObjectSlot data_start();
295

296
  inline MaybeObjectSlot RawFieldOfElementAt(int index);
297

298 299 300
  inline void CopyElements(Isolate* isolate, int dst_index, WeakFixedArray src,
                           int src_index, int len, WriteBarrierMode mode);

301
  DECL_PRINTER(WeakFixedArray)
302 303
  DECL_VERIFIER(WeakFixedArray)

304
  class BodyDescriptor;
305 306

  static const int kMaxLength =
307
      (FixedArray::kMaxSize - kHeaderSize) / kTaggedSize;
308 309
  static_assert(Internals::IsValidSmi(kMaxLength),
                "WeakFixedArray maxLength not a Smi");
310

311 312
  int AllocatedSize();

313
  static int OffsetOfElementAt(int index) {
314 315
    STATIC_ASSERT(kObjectsOffset == SizeFor(0));
    return SizeFor(index);
316 317
  }

318
 private:
319 320 321 322
  friend class Heap;

  static const int kFirstIndex = 1;

323
  TQ_OBJECT_CONSTRUCTORS(WeakFixedArray)
324 325
};

326 327 328 329 330
// 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.
331 332
class WeakArrayList
    : public TorqueGeneratedWeakArrayList<WeakArrayList, HeapObject> {
333
 public:
334
  NEVER_READ_ONLY_SPACE
335 336
  DECL_PRINTER(WeakArrayList)

337 338 339
  V8_EXPORT_PRIVATE static Handle<WeakArrayList> AddToEnd(
      Isolate* isolate, Handle<WeakArrayList> array,
      const MaybeObjectHandle& value);
340

341 342 343 344 345 346
  // A version that adds to 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,
      const MaybeObjectHandle& value1, const MaybeObjectHandle& value2);

347 348 349 350 351 352 353 354 355 356 357
  // 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 Handle<WeakArrayList> Append(
      Isolate* isolate, Handle<WeakArrayList> array,
      const MaybeObjectHandle& value,
      AllocationType allocation = AllocationType::kYoung);

  // Compact weak references to the beginning of the array.
  V8_EXPORT_PRIVATE void Compact(Isolate* isolate);

358
  inline MaybeObject Get(int index) const;
359
  inline MaybeObject Get(PtrComprCageBase cage_base, int index) const;
360 361

  // Set the element at index to obj. The underlying array must be large enough.
362
  // If you need to grow the WeakArrayList, use the static AddToEnd() method
363
  // instead.
364
  inline void Set(int index, MaybeObject value,
365 366 367
                  WriteBarrierMode mode = UPDATE_WRITE_BARRIER);

  static constexpr int SizeForCapacity(int capacity) {
368
    return SizeFor(capacity);
369 370
  }

371
  static constexpr int CapacityForLength(int length) {
372
    return length + std::max(length / 2, 2);
373 374
  }

375
  // Gives access to raw memory which stores the array's data.
376
  inline MaybeObjectSlot data_start();
377

378 379 380
  inline void CopyElements(Isolate* isolate, int dst_index, WeakArrayList src,
                           int src_index, int len, WriteBarrierMode mode);

381
  V8_EXPORT_PRIVATE bool IsFull();
382

383
  int AllocatedSize();
384

385
  class BodyDescriptor;
386

387
  static const int kMaxCapacity =
388
      (FixedArray::kMaxSize - kHeaderSize) / kTaggedSize;
389

390 391
  static Handle<WeakArrayList> EnsureSpace(
      Isolate* isolate, Handle<WeakArrayList> array, int length,
392
      AllocationType allocation = AllocationType::kYoung);
393

394 395 396
  // Returns the number of non-cleaned weak references in the array.
  int CountLiveWeakReferences() const;

397 398 399
  // Returns the number of non-cleaned elements in the array.
  int CountLiveElements() const;

400 401 402 403
  // 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.
404
  V8_EXPORT_PRIVATE bool RemoveOne(const MaybeObjectHandle& value);
405

406
  class Iterator;
407

408 409
 private:
  static int OffsetOfElementAt(int index) {
410
    return kHeaderSize + index * kTaggedSize;
411 412
  }

413
  TQ_OBJECT_CONSTRUCTORS(WeakArrayList)
414 415 416 417 418
};

class WeakArrayList::Iterator {
 public:
  explicit Iterator(WeakArrayList array) : index_(0), array_(array) {}
419 420
  Iterator(const Iterator&) = delete;
  Iterator& operator=(const Iterator&) = delete;
421

422
  inline HeapObject Next();
423 424 425 426

 private:
  int index_;
  WeakArrayList array_;
427
  DISALLOW_GARBAGE_COLLECTION(no_gc_)
428 429
};

430 431 432 433 434 435 436
// Generic array grows dynamically with O(1) amortized insertion.
//
// ArrayList is a FixedArray with static convenience methods for adding more
// elements. The Length() method returns the number of elements in the list, not
// the allocated size. The number of elements is stored at kLengthIndex and is
// updated with every insertion. The elements of the ArrayList are stored in the
// underlying FixedArray starting at kFirstIndex.
437
class ArrayList : public TorqueGeneratedArrayList<ArrayList, FixedArray> {
438
 public:
439 440 441 442 443 444 445
  V8_EXPORT_PRIVATE static Handle<ArrayList> Add(Isolate* isolate,
                                                 Handle<ArrayList> array,
                                                 Handle<Object> obj);
  V8_EXPORT_PRIVATE static Handle<ArrayList> Add(Isolate* isolate,
                                                 Handle<ArrayList> array,
                                                 Handle<Object> obj1,
                                                 Handle<Object> obj2);
446 447 448
  V8_EXPORT_PRIVATE static Handle<ArrayList> Add(Isolate* isolate,
                                                 Handle<ArrayList> array,
                                                 Handle<Object> obj1, Smi obj2,
449
                                                 Smi obj3, Smi obj4);
450 451 452 453 454 455 456 457 458
  static Handle<ArrayList> New(Isolate* isolate, int size);

  // Returns the number of elements in the list, not the allocated size, which
  // is length(). Lower and upper case length() return different results!
  inline int Length() const;

  // Sets the Length() as used by Elements(). Does not change the underlying
  // storage capacity, i.e., length().
  inline void SetLength(int length);
459
  inline Object Get(int index) const;
460
  inline Object Get(PtrComprCageBase cage_base, int index) const;
461
  inline ObjectSlot Slot(int index);
462 463 464

  // Set the element at index to obj. The underlying array must be large enough.
  // If you need to grow the ArrayList, use the static Add() methods instead.
465
  inline void Set(int index, Object obj,
466 467
                  WriteBarrierMode mode = UPDATE_WRITE_BARRIER);

468 469
  inline void Set(int index, Smi obj);

470
  // Set the element at index to undefined. This does not change the Length().
471
  inline void Clear(int index, Object undefined);
472 473 474

  // Return a copy of the list of size Length() without the first entry. The
  // number returned by Length() is stored in the first entry.
475
  static Handle<FixedArray> Elements(Isolate* isolate, Handle<ArrayList> array);
476

477 478
  static const int kHeaderFields = 1;

479
 private:
480 481
  static Handle<ArrayList> EnsureSpace(Isolate* isolate,
                                       Handle<ArrayList> array, int length);
482 483
  static const int kLengthIndex = 0;
  static const int kFirstIndex = 1;
484
  STATIC_ASSERT(kHeaderFields == kFirstIndex);
485
  TQ_OBJECT_CONSTRUCTORS(ArrayList)
486 487 488 489 490
};

enum SearchMode { ALL_ENTRIES, VALID_ENTRIES };

template <SearchMode search_mode, typename T>
491
inline int Search(T* array, Name name, int valid_entries = 0,
492 493
                  int* out_insertion_index = nullptr,
                  bool concurrent_search = false);
494 495 496

// ByteArray represents fixed sized byte arrays.  Used for the relocation info
// that is attached to code objects.
497
class ByteArray : public TorqueGeneratedByteArray<ByteArray, FixedArrayBase> {
498 499 500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515
 public:
  inline int Size();

  // Setter and getter.
  inline byte get(int index) const;
  inline void set(int index, byte value);

  // Copy in / copy out whole byte slices.
  inline void copy_out(int index, byte* buffer, int length);
  inline void copy_in(int index, const byte* buffer, int length);

  // Treat contents as an int array.
  inline int get_int(int index) const;
  inline void set_int(int index, int value);

  inline uint32_t get_uint32(int index) const;
  inline void set_uint32(int index, uint32_t value);

516 517 518
  inline uint32_t get_uint32_relaxed(int index) const;
  inline void set_uint32_relaxed(int index, uint32_t value);

519 520 521 522 523 524 525 526 527 528 529 530
  // Clear uninitialized padding space. This ensures that the snapshot content
  // is deterministic.
  inline void clear_padding();

  static int SizeFor(int length) {
    return OBJECT_POINTER_ALIGN(kHeaderSize + length);
  }
  // We use byte arrays for free blocks in the heap.  Given a desired size in
  // bytes that is a multiple of the word size and big enough to hold a byte
  // array, this function returns the number of elements a byte array should
  // have.
  static int LengthFor(int size_in_bytes) {
531
    DCHECK(IsAligned(size_in_bytes, kTaggedSize));
532 533 534 535 536
    DCHECK_GE(size_in_bytes, kHeaderSize);
    return size_in_bytes - kHeaderSize;
  }

  // Returns data start address.
537
  inline byte* GetDataStartAddress();
538 539
  // Returns address of the past-the-end element.
  inline byte* GetDataEndAddress();
540 541 542 543

  inline int DataSize() const;

  // Returns a pointer to the ByteArray object for a given data start address.
544
  static inline ByteArray FromDataStartAddress(Address address);
545 546 547 548 549 550 551 552 553 554

  // Dispatched behavior.
  inline int ByteArraySize();
  DECL_PRINTER(ByteArray)

  // Layout description.
  static const int kAlignedSize = OBJECT_POINTER_ALIGN(kHeaderSize);

  // Maximal length of a single ByteArray.
  static const int kMaxLength = kMaxSize - kHeaderSize;
555 556
  static_assert(Internals::IsValidSmi(kMaxLength),
                "ByteArray maxLength not a Smi");
557 558 559

  class BodyDescriptor;

560
 protected:
561 562
  TQ_OBJECT_CONSTRUCTORS(ByteArray)
  inline ByteArray(Address ptr, HeapObject::AllowInlineSmiStorage allow_smi);
563 564 565 566 567 568 569
};

// Wrapper class for ByteArray which can store arbitrary C++ classes, as long
// as they can be copied with memcpy.
template <class T>
class PodArray : public ByteArray {
 public:
570 571 572
  static Handle<PodArray<T>> New(
      Isolate* isolate, int length,
      AllocationType allocation = AllocationType::kYoung);
573
  void copy_out(int index, T* result, int length) {
574
    ByteArray::copy_out(index * sizeof(T), reinterpret_cast<byte*>(result),
575
                        length * sizeof(T));
576
  }
577 578 579 580 581 582

  void copy_in(int index, const T* buffer, int length) {
    ByteArray::copy_in(index * sizeof(T), reinterpret_cast<const byte*>(buffer),
                       length * sizeof(T));
  }

583 584 585 586 587
  bool matches(const T* buffer, int length) {
    DCHECK_LE(length, this->length());
    return memcmp(GetDataStartAddress(), buffer, length * sizeof(T)) == 0;
  }

588 589
  T get(int index) {
    T result;
590
    copy_out(index, &result, 1);
591 592
    return result;
  }
593 594 595

  void set(int index, const T& value) { copy_in(index, &value, 1); }

596
  inline int length() const;
597
  DECL_CAST(PodArray<T>)
598

599
  OBJECT_CONSTRUCTORS(PodArray<T>, ByteArray);
600 601
};

602 603
class TemplateList
    : public TorqueGeneratedTemplateList<TemplateList, FixedArray> {
604 605 606
 public:
  static Handle<TemplateList> New(Isolate* isolate, int size);
  inline int length() const;
607
  inline Object get(int index) const;
608
  inline Object get(PtrComprCageBase cage_base, int index) const;
609
  inline void set(int index, Object value);
610 611 612 613 614
  static Handle<TemplateList> Add(Isolate* isolate, Handle<TemplateList> list,
                                  Handle<Object> value);
 private:
  static const int kLengthIndex = 0;
  static const int kFirstElementIndex = kLengthIndex + 1;
615

616
  TQ_OBJECT_CONSTRUCTORS(TemplateList)
617 618 619 620 621 622 623 624
};

}  // namespace internal
}  // namespace v8

#include "src/objects/object-macros-undef.h"

#endif  // V8_OBJECTS_FIXED_ARRAY_H_