// Copyright 2014 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. #include "src/v8.h" #include "src/factory.h" #include "src/handles-inl.h" #include "src/interpreter/constant-array-builder.h" #include "src/isolate.h" #include "test/unittests/test-utils.h" namespace v8 { namespace internal { namespace interpreter { class ConstantArrayBuilderTest : public TestWithIsolateAndZone { public: ConstantArrayBuilderTest() {} ~ConstantArrayBuilderTest() override {} static const size_t k8BitCapacity = ConstantArrayBuilder::k8BitCapacity; static const size_t k16BitCapacity = ConstantArrayBuilder::k16BitCapacity; }; STATIC_CONST_MEMBER_DEFINITION const size_t ConstantArrayBuilderTest::k16BitCapacity; STATIC_CONST_MEMBER_DEFINITION const size_t ConstantArrayBuilderTest::k8BitCapacity; TEST_F(ConstantArrayBuilderTest, AllocateAllEntries) { ConstantArrayBuilder builder(isolate(), zone()); for (size_t i = 0; i < k16BitCapacity; i++) { builder.Insert(handle(Smi::FromInt(static_cast(i)), isolate())); } CHECK_EQ(builder.size(), k16BitCapacity); for (size_t i = 0; i < k16BitCapacity; i++) { CHECK_EQ(Handle::cast(builder.At(i))->value(), i); } } TEST_F(ConstantArrayBuilderTest, AllocateEntriesWithIdx8Reservations) { for (size_t reserved = 1; reserved < k8BitCapacity; reserved *= 3) { ConstantArrayBuilder builder(isolate(), zone()); for (size_t i = 0; i < reserved; i++) { OperandSize operand_size = builder.CreateReservedEntry(); CHECK(operand_size == OperandSize::kByte); } for (size_t i = 0; i < 2 * k8BitCapacity; i++) { Handle object = isolate()->factory()->NewNumberFromSize(i); builder.Insert(object); if (i + reserved < k8BitCapacity) { CHECK_LE(builder.size(), k8BitCapacity); CHECK_EQ(builder.size(), i + 1); CHECK(builder.At(i)->SameValue(*object)); } else { CHECK_GE(builder.size(), k8BitCapacity); CHECK_EQ(builder.size(), i + reserved + 1); CHECK(builder.At(i + reserved)->SameValue(*object)); } } CHECK_EQ(builder.size(), 2 * k8BitCapacity + reserved); // Check reserved values represented by the hole. for (size_t i = 0; i < reserved; i++) { Handle empty = builder.At(k8BitCapacity - reserved + i); CHECK(empty->SameValue(isolate()->heap()->the_hole_value())); } // Commmit reserved entries with duplicates and check size does not change. DCHECK_EQ(reserved + 2 * k8BitCapacity, builder.size()); size_t duplicates_in_idx8_space = std::min(reserved, k8BitCapacity - reserved); for (size_t i = 0; i < duplicates_in_idx8_space; i++) { builder.CommitReservedEntry(OperandSize::kByte, isolate()->factory()->NewNumberFromSize(i)); DCHECK_EQ(reserved + 2 * k8BitCapacity, builder.size()); } // Check all committed values match expected (holes where // duplicates_in_idx8_space allocated). for (size_t i = 0; i < k8BitCapacity - reserved; i++) { Smi* smi = Smi::FromInt(static_cast(i)); CHECK(Handle::cast(builder.At(i))->SameValue(smi)); } for (size_t i = k8BitCapacity; i < 2 * k8BitCapacity + reserved; i++) { Smi* smi = Smi::FromInt(static_cast(i - reserved)); CHECK(Handle::cast(builder.At(i))->SameValue(smi)); } for (size_t i = 0; i < reserved; i++) { size_t index = k8BitCapacity - reserved + i; CHECK(builder.At(index)->IsTheHole(isolate())); } // Now make reservations, and commit them with unique entries. for (size_t i = 0; i < duplicates_in_idx8_space; i++) { OperandSize operand_size = builder.CreateReservedEntry(); CHECK(operand_size == OperandSize::kByte); } for (size_t i = 0; i < duplicates_in_idx8_space; i++) { Handle object = isolate()->factory()->NewNumberFromSize(2 * k8BitCapacity + i); size_t index = builder.CommitReservedEntry(OperandSize::kByte, object); CHECK_EQ(static_cast(index), k8BitCapacity - reserved + i); CHECK(builder.At(static_cast(index))->SameValue(*object)); } CHECK_EQ(builder.size(), 2 * k8BitCapacity + reserved); } } TEST_F(ConstantArrayBuilderTest, AllocateEntriesWithWideReservations) { for (size_t reserved = 1; reserved < k8BitCapacity; reserved *= 3) { ConstantArrayBuilder builder(isolate(), zone()); for (size_t i = 0; i < k8BitCapacity; i++) { Handle object = isolate()->factory()->NewNumberFromSize(i); builder.Insert(object); CHECK(builder.At(i)->SameValue(*object)); CHECK_EQ(builder.size(), i + 1); } for (size_t i = 0; i < reserved; i++) { OperandSize operand_size = builder.CreateReservedEntry(); CHECK(operand_size == OperandSize::kShort); CHECK_EQ(builder.size(), k8BitCapacity); } for (size_t i = 0; i < reserved; i++) { builder.DiscardReservedEntry(OperandSize::kShort); CHECK_EQ(builder.size(), k8BitCapacity); } for (size_t i = 0; i < reserved; i++) { OperandSize operand_size = builder.CreateReservedEntry(); CHECK(operand_size == OperandSize::kShort); Handle object = isolate()->factory()->NewNumberFromSize(i); builder.CommitReservedEntry(operand_size, object); CHECK_EQ(builder.size(), k8BitCapacity); } for (size_t i = k8BitCapacity; i < k8BitCapacity + reserved; i++) { OperandSize operand_size = builder.CreateReservedEntry(); CHECK(operand_size == OperandSize::kShort); Handle object = isolate()->factory()->NewNumberFromSize(i); builder.CommitReservedEntry(operand_size, object); CHECK_EQ(builder.size(), i + 1); } } } TEST_F(ConstantArrayBuilderTest, ToFixedArray) { ConstantArrayBuilder builder(isolate(), zone()); static const size_t kNumberOfElements = 37; for (size_t i = 0; i < kNumberOfElements; i++) { Handle object = isolate()->factory()->NewNumberFromSize(i); builder.Insert(object); CHECK(builder.At(i)->SameValue(*object)); } Handle constant_array = builder.ToFixedArray(); CHECK_EQ(constant_array->length(), kNumberOfElements); for (size_t i = 0; i < kNumberOfElements; i++) { CHECK(constant_array->get(static_cast(i))->SameValue(*builder.At(i))); } } TEST_F(ConstantArrayBuilderTest, ToLargeFixedArray) { ConstantArrayBuilder builder(isolate(), zone()); static const size_t kNumberOfElements = 37373; for (size_t i = 0; i < kNumberOfElements; i++) { Handle object = isolate()->factory()->NewNumberFromSize(i); builder.Insert(object); CHECK(builder.At(i)->SameValue(*object)); } Handle constant_array = builder.ToFixedArray(); CHECK_EQ(constant_array->length(), kNumberOfElements); for (size_t i = 0; i < kNumberOfElements; i++) { CHECK(constant_array->get(static_cast(i))->SameValue(*builder.At(i))); } } TEST_F(ConstantArrayBuilderTest, GapFilledWhenLowReservationCommitted) { ConstantArrayBuilder builder(isolate(), zone()); for (size_t i = 0; i < k8BitCapacity; i++) { OperandSize operand_size = builder.CreateReservedEntry(); CHECK(OperandSize::kByte == operand_size); CHECK_EQ(builder.size(), 0); } for (size_t i = 0; i < k8BitCapacity; i++) { Handle object = isolate()->factory()->NewNumberFromSize(i); builder.Insert(object); CHECK_EQ(builder.size(), i + k8BitCapacity + 1); } for (size_t i = 0; i < k8BitCapacity; i++) { builder.CommitReservedEntry(OperandSize::kByte, builder.At(i + k8BitCapacity)); CHECK_EQ(builder.size(), 2 * k8BitCapacity); } for (size_t i = 0; i < k8BitCapacity; i++) { Handle original = builder.At(k8BitCapacity + i); Handle duplicate = builder.At(i); CHECK(original->SameValue(*duplicate)); Handle reference = isolate()->factory()->NewNumberFromSize(i); CHECK(original->SameValue(*reference)); } } TEST_F(ConstantArrayBuilderTest, GapNotFilledWhenLowReservationDiscarded) { ConstantArrayBuilder builder(isolate(), zone()); for (size_t i = 0; i < k8BitCapacity; i++) { OperandSize operand_size = builder.CreateReservedEntry(); CHECK(OperandSize::kByte == operand_size); CHECK_EQ(builder.size(), 0); } for (size_t i = 0; i < k8BitCapacity; i++) { Handle object = isolate()->factory()->NewNumberFromSize(i); builder.Insert(object); CHECK_EQ(builder.size(), i + k8BitCapacity + 1); } for (size_t i = 0; i < k8BitCapacity; i++) { builder.DiscardReservedEntry(OperandSize::kByte); builder.Insert(builder.At(i + k8BitCapacity)); CHECK_EQ(builder.size(), 2 * k8BitCapacity); } for (size_t i = 0; i < k8BitCapacity; i++) { Handle reference = isolate()->factory()->NewNumberFromSize(i); Handle original = builder.At(k8BitCapacity + i); CHECK(original->SameValue(*reference)); Handle duplicate = builder.At(i); CHECK(duplicate->SameValue(*isolate()->factory()->the_hole_value())); } } TEST_F(ConstantArrayBuilderTest, HolesWithUnusedReservations) { static int kNumberOfHoles = 128; ConstantArrayBuilder builder(isolate(), zone()); for (int i = 0; i < kNumberOfHoles; ++i) { CHECK_EQ(builder.CreateReservedEntry(), OperandSize::kByte); } for (int i = 0; i < 128; ++i) { CHECK_EQ(builder.Insert(isolate()->factory()->NewNumber(i)), i); } CHECK_EQ(builder.Insert(isolate()->factory()->NewNumber(256)), 256); Handle constant_array = builder.ToFixedArray(); CHECK_EQ(constant_array->length(), 257); for (int i = 128; i < 256; i++) { CHECK(constant_array->get(i)->SameValue( *isolate()->factory()->the_hole_value())); } CHECK(!constant_array->get(127)->SameValue( *isolate()->factory()->the_hole_value())); CHECK(!constant_array->get(256)->SameValue( *isolate()->factory()->the_hole_value())); } TEST_F(ConstantArrayBuilderTest, ReservationsAtAllScales) { ConstantArrayBuilder builder(isolate(), zone()); for (int i = 0; i < 256; i++) { CHECK_EQ(builder.CreateReservedEntry(), OperandSize::kByte); } for (int i = 256; i < 65536; ++i) { CHECK_EQ(builder.CreateReservedEntry(), OperandSize::kShort); } for (int i = 65536; i < 131072; ++i) { CHECK_EQ(builder.CreateReservedEntry(), OperandSize::kQuad); } CHECK_EQ(builder.CommitReservedEntry(OperandSize::kByte, isolate()->factory()->NewNumber(1)), 0); CHECK_EQ(builder.CommitReservedEntry(OperandSize::kShort, isolate()->factory()->NewNumber(2)), 256); CHECK_EQ(builder.CommitReservedEntry(OperandSize::kQuad, isolate()->factory()->NewNumber(3)), 65536); Handle constant_array = builder.ToFixedArray(); CHECK_EQ(constant_array->length(), 65537); int count = 1; for (int i = 0; i < constant_array->length(); ++i) { Handle expected; if (i == 0 || i == 256 || i == 65536) { expected = isolate()->factory()->NewNumber(count++); } else { expected = isolate()->factory()->the_hole_value(); } CHECK(constant_array->get(i)->SameValue(*expected)); } } } // namespace interpreter } // namespace internal } // namespace v8