// Copyright 2015 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_INTERPRETER_CONSTANT_ARRAY_BUILDER_H_ #define V8_INTERPRETER_CONSTANT_ARRAY_BUILDER_H_ #include "src/ast/ast-value-factory.h" #include "src/globals.h" #include "src/identity-map.h" #include "src/interpreter/bytecodes.h" #include "src/zone/zone-containers.h" namespace v8 { namespace internal { class Isolate; class AstRawString; class AstValue; namespace interpreter { // Constant array entries that represent singletons. #define SINGLETON_CONSTANT_ENTRY_TYPES(V) \ V(AsyncIteratorSymbol, async_iterator_symbol) \ V(ClassFieldsSymbol, class_fields_symbol) \ V(EmptyObjectBoilerplateDescription, empty_object_boilerplate_description) \ V(EmptyArrayBoilerplateDescription, empty_array_boilerplate_description) \ V(EmptyFixedArray, empty_fixed_array) \ V(HomeObjectSymbol, home_object_symbol) \ V(IteratorSymbol, iterator_symbol) \ V(InterpreterTrampolineSymbol, interpreter_trampoline_symbol) \ V(NaN, nan_value) // A helper class for constructing constant arrays for the // interpreter. Each instance of this class is intended to be used to // generate exactly one FixedArray of constants via the ToFixedArray // method. class V8_EXPORT_PRIVATE ConstantArrayBuilder final BASE_EMBEDDED { public: // Capacity of the 8-bit operand slice. static const size_t k8BitCapacity = 1u << kBitsPerByte; // Capacity of the 16-bit operand slice. static const size_t k16BitCapacity = (1u << 2 * kBitsPerByte) - k8BitCapacity; // Capacity of the 32-bit operand slice. static const size_t k32BitCapacity = kMaxUInt32 - k16BitCapacity - k8BitCapacity + 1; ConstantArrayBuilder(Zone* zone); // Generate a fixed array of constant handles based on inserted objects. Handle<FixedArray> ToFixedArray(Isolate* isolate); // Returns the object, as a handle in |isolate|, that is in the constant pool // array at index |index|. Returns null if there is no handle at this index. // Only expected to be used in tests. MaybeHandle<Object> At(size_t index, Isolate* isolate) const; // Returns the number of elements in the array. size_t size() const; // Insert an object into the constants array if it is not already present. // Returns the array index associated with the object. size_t Insert(Smi* smi); size_t Insert(double number); size_t Insert(const AstRawString* raw_string); size_t Insert(AstBigInt bigint); size_t Insert(const Scope* scope); #define INSERT_ENTRY(NAME, ...) size_t Insert##NAME(); SINGLETON_CONSTANT_ENTRY_TYPES(INSERT_ENTRY) #undef INSERT_ENTRY // Inserts an empty entry and returns the array index associated with the // reservation. The entry's handle value can be inserted by calling // SetDeferredAt(). size_t InsertDeferred(); // Inserts |size| consecutive empty entries and returns the array index // associated with the first reservation. Each entry's Smi value can be // inserted by calling SetJumpTableSmi(). size_t InsertJumpTable(size_t size); // Sets the deferred value at |index| to |object|. void SetDeferredAt(size_t index, Handle<Object> object); // Sets the jump table entry at |index| to |smi|. Note that |index| is the // constant pool index, not the switch case value. void SetJumpTableSmi(size_t index, Smi* smi); // Creates a reserved entry in the constant pool and returns // the size of the operand that'll be required to hold the entry // when committed. OperandSize CreateReservedEntry(); // Commit reserved entry and returns the constant pool index for the // SMI value. size_t CommitReservedEntry(OperandSize operand_size, Smi* value); // Discards constant pool reservation. void DiscardReservedEntry(OperandSize operand_size); private: typedef uint32_t index_t; struct ConstantArraySlice; class Entry { private: enum class Tag : uint8_t; public: explicit Entry(Smi* smi) : smi_(smi), tag_(Tag::kSmi) {} explicit Entry(double heap_number) : heap_number_(heap_number), tag_(Tag::kHeapNumber) {} explicit Entry(const AstRawString* raw_string) : raw_string_(raw_string), tag_(Tag::kRawString) {} explicit Entry(AstBigInt bigint) : bigint_(bigint), tag_(Tag::kBigInt) {} explicit Entry(const Scope* scope) : scope_(scope), tag_(Tag::kScope) {} #define CONSTRUCT_ENTRY(NAME, LOWER_NAME) \ static Entry NAME() { return Entry(Tag::k##NAME); } SINGLETON_CONSTANT_ENTRY_TYPES(CONSTRUCT_ENTRY) #undef CONSTRUCT_ENTRY static Entry Deferred() { return Entry(Tag::kDeferred); } static Entry UninitializedJumpTableSmi() { return Entry(Tag::kUninitializedJumpTableSmi); } bool IsDeferred() const { return tag_ == Tag::kDeferred; } bool IsJumpTableEntry() const { return tag_ == Tag::kUninitializedJumpTableSmi || tag_ == Tag::kJumpTableSmi; } void SetDeferred(Handle<Object> handle) { DCHECK_EQ(tag_, Tag::kDeferred); tag_ = Tag::kHandle; handle_ = handle; } void SetJumpTableSmi(Smi* smi) { DCHECK_EQ(tag_, Tag::kUninitializedJumpTableSmi); tag_ = Tag::kJumpTableSmi; smi_ = smi; } Handle<Object> ToHandle(Isolate* isolate) const; private: explicit Entry(Tag tag) : tag_(tag) {} union { Handle<Object> handle_; Smi* smi_; double heap_number_; const AstRawString* raw_string_; AstBigInt bigint_; const Scope* scope_; }; enum class Tag : uint8_t { kDeferred, kHandle, kSmi, kRawString, kHeapNumber, kBigInt, kScope, kUninitializedJumpTableSmi, kJumpTableSmi, #define ENTRY_TAG(NAME, ...) k##NAME, SINGLETON_CONSTANT_ENTRY_TYPES(ENTRY_TAG) #undef ENTRY_TAG } tag_; #if DEBUG // Required by CheckAllElementsAreUnique(). friend struct ConstantArraySlice; #endif }; index_t AllocateIndex(Entry constant_entry); index_t AllocateIndexArray(Entry constant_entry, size_t size); index_t AllocateReservedEntry(Smi* value); struct ConstantArraySlice final : public ZoneObject { ConstantArraySlice(Zone* zone, size_t start_index, size_t capacity, OperandSize operand_size); void Reserve(); void Unreserve(); size_t Allocate(Entry entry, size_t count = 1); Entry& At(size_t index); const Entry& At(size_t index) const; #if DEBUG void CheckAllElementsAreUnique(Isolate* isolate) const; #endif inline size_t available() const { return capacity() - reserved() - size(); } inline size_t reserved() const { return reserved_; } inline size_t capacity() const { return capacity_; } inline size_t size() const { return constants_.size(); } inline size_t start_index() const { return start_index_; } inline size_t max_index() const { return start_index_ + capacity() - 1; } inline OperandSize operand_size() const { return operand_size_; } private: const size_t start_index_; const size_t capacity_; size_t reserved_; OperandSize operand_size_; ZoneVector<Entry> constants_; DISALLOW_COPY_AND_ASSIGN(ConstantArraySlice); }; ConstantArraySlice* IndexToSlice(size_t index) const; ConstantArraySlice* OperandSizeToSlice(OperandSize operand_size) const; ConstantArraySlice* idx_slice_[3]; base::TemplateHashMapImpl<intptr_t, index_t, base::KeyEqualityMatcher<intptr_t>, ZoneAllocationPolicy> constants_map_; ZoneMap<Smi*, index_t> smi_map_; ZoneVector<std::pair<Smi*, index_t>> smi_pairs_; ZoneMap<double, index_t> heap_number_map_; #define SINGLETON_ENTRY_FIELD(NAME, LOWER_NAME) int LOWER_NAME##_; SINGLETON_CONSTANT_ENTRY_TYPES(SINGLETON_ENTRY_FIELD) #undef SINGLETON_ENTRY_FIELD Zone* zone_; }; } // namespace interpreter } // namespace internal } // namespace v8 #endif // V8_INTERPRETER_CONSTANT_ARRAY_BUILDER_H_