2016-01-05 19:08:11 +00:00
|
|
|
// 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 {}
|
|
|
|
|
2016-02-26 10:14:49 +00:00
|
|
|
static const size_t k8BitCapacity = ConstantArrayBuilder::k8BitCapacity;
|
|
|
|
static const size_t k16BitCapacity = ConstantArrayBuilder::k16BitCapacity;
|
2016-01-05 19:08:11 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
STATIC_CONST_MEMBER_DEFINITION const size_t
|
2016-02-26 10:14:49 +00:00
|
|
|
ConstantArrayBuilderTest::k16BitCapacity;
|
2016-01-05 19:08:11 +00:00
|
|
|
STATIC_CONST_MEMBER_DEFINITION const size_t
|
2016-02-26 10:14:49 +00:00
|
|
|
ConstantArrayBuilderTest::k8BitCapacity;
|
2016-01-05 19:08:11 +00:00
|
|
|
|
|
|
|
TEST_F(ConstantArrayBuilderTest, AllocateAllEntries) {
|
|
|
|
ConstantArrayBuilder builder(isolate(), zone());
|
2016-02-26 10:14:49 +00:00
|
|
|
for (size_t i = 0; i < k16BitCapacity; i++) {
|
2016-02-16 14:08:45 +00:00
|
|
|
builder.Insert(handle(Smi::FromInt(static_cast<int>(i)), isolate()));
|
2016-01-05 19:08:11 +00:00
|
|
|
}
|
2016-02-26 10:14:49 +00:00
|
|
|
CHECK_EQ(builder.size(), k16BitCapacity);
|
|
|
|
for (size_t i = 0; i < k16BitCapacity; i++) {
|
2016-02-16 14:08:45 +00:00
|
|
|
CHECK_EQ(Handle<Smi>::cast(builder.At(i))->value(), i);
|
2016-01-05 19:08:11 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
TEST_F(ConstantArrayBuilderTest, AllocateEntriesWithIdx8Reservations) {
|
2016-02-26 10:14:49 +00:00
|
|
|
for (size_t reserved = 1; reserved < k8BitCapacity; reserved *= 3) {
|
2016-01-05 19:08:11 +00:00
|
|
|
ConstantArrayBuilder builder(isolate(), zone());
|
|
|
|
for (size_t i = 0; i < reserved; i++) {
|
|
|
|
OperandSize operand_size = builder.CreateReservedEntry();
|
|
|
|
CHECK(operand_size == OperandSize::kByte);
|
|
|
|
}
|
2016-02-26 10:14:49 +00:00
|
|
|
for (size_t i = 0; i < 2 * k8BitCapacity; i++) {
|
2016-01-05 19:08:11 +00:00
|
|
|
Handle<Object> object = isolate()->factory()->NewNumberFromSize(i);
|
|
|
|
builder.Insert(object);
|
2016-02-26 10:14:49 +00:00
|
|
|
if (i + reserved < k8BitCapacity) {
|
|
|
|
CHECK_LE(builder.size(), k8BitCapacity);
|
2016-01-05 19:08:11 +00:00
|
|
|
CHECK_EQ(builder.size(), i + 1);
|
|
|
|
CHECK(builder.At(i)->SameValue(*object));
|
|
|
|
} else {
|
2016-02-26 10:14:49 +00:00
|
|
|
CHECK_GE(builder.size(), k8BitCapacity);
|
2016-01-05 19:08:11 +00:00
|
|
|
CHECK_EQ(builder.size(), i + reserved + 1);
|
|
|
|
CHECK(builder.At(i + reserved)->SameValue(*object));
|
|
|
|
}
|
|
|
|
}
|
2016-02-26 10:14:49 +00:00
|
|
|
CHECK_EQ(builder.size(), 2 * k8BitCapacity + reserved);
|
2016-01-05 19:08:11 +00:00
|
|
|
|
|
|
|
// Check reserved values represented by the hole.
|
|
|
|
for (size_t i = 0; i < reserved; i++) {
|
2016-02-26 10:14:49 +00:00
|
|
|
Handle<Object> empty = builder.At(k8BitCapacity - reserved + i);
|
2016-01-05 19:08:11 +00:00
|
|
|
CHECK(empty->SameValue(isolate()->heap()->the_hole_value()));
|
|
|
|
}
|
|
|
|
|
|
|
|
// Commmit reserved entries with duplicates and check size does not change.
|
2016-02-26 10:14:49 +00:00
|
|
|
DCHECK_EQ(reserved + 2 * k8BitCapacity, builder.size());
|
2016-01-05 19:08:11 +00:00
|
|
|
size_t duplicates_in_idx8_space =
|
2016-02-26 10:14:49 +00:00
|
|
|
std::min(reserved, k8BitCapacity - reserved);
|
2016-01-05 19:08:11 +00:00
|
|
|
for (size_t i = 0; i < duplicates_in_idx8_space; i++) {
|
|
|
|
builder.CommitReservedEntry(OperandSize::kByte,
|
|
|
|
isolate()->factory()->NewNumberFromSize(i));
|
2016-02-26 10:14:49 +00:00
|
|
|
DCHECK_EQ(reserved + 2 * k8BitCapacity, builder.size());
|
2016-01-05 19:08:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Check all committed values match expected (holes where
|
|
|
|
// duplicates_in_idx8_space allocated).
|
2016-02-26 10:14:49 +00:00
|
|
|
for (size_t i = 0; i < k8BitCapacity - reserved; i++) {
|
2016-01-05 19:08:11 +00:00
|
|
|
Smi* smi = Smi::FromInt(static_cast<int>(i));
|
|
|
|
CHECK(Handle<Smi>::cast(builder.At(i))->SameValue(smi));
|
|
|
|
}
|
2016-02-26 10:14:49 +00:00
|
|
|
for (size_t i = k8BitCapacity; i < 2 * k8BitCapacity + reserved; i++) {
|
2016-01-05 19:08:11 +00:00
|
|
|
Smi* smi = Smi::FromInt(static_cast<int>(i - reserved));
|
|
|
|
CHECK(Handle<Smi>::cast(builder.At(i))->SameValue(smi));
|
|
|
|
}
|
|
|
|
for (size_t i = 0; i < reserved; i++) {
|
2016-02-26 10:14:49 +00:00
|
|
|
size_t index = k8BitCapacity - reserved + i;
|
2016-01-05 19:08:11 +00:00
|
|
|
CHECK(builder.At(index)->IsTheHole());
|
|
|
|
}
|
|
|
|
|
|
|
|
// 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> object =
|
2016-02-26 10:14:49 +00:00
|
|
|
isolate()->factory()->NewNumberFromSize(2 * k8BitCapacity + i);
|
2016-01-05 19:08:11 +00:00
|
|
|
size_t index = builder.CommitReservedEntry(OperandSize::kByte, object);
|
2016-02-26 10:14:49 +00:00
|
|
|
CHECK_EQ(static_cast<int>(index), k8BitCapacity - reserved + i);
|
2016-01-05 19:08:11 +00:00
|
|
|
CHECK(builder.At(static_cast<int>(index))->SameValue(*object));
|
|
|
|
}
|
2016-02-26 10:14:49 +00:00
|
|
|
CHECK_EQ(builder.size(), 2 * k8BitCapacity + reserved);
|
2016-01-05 19:08:11 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
TEST_F(ConstantArrayBuilderTest, AllocateEntriesWithIdx16Reservations) {
|
2016-02-26 10:14:49 +00:00
|
|
|
for (size_t reserved = 1; reserved < k8BitCapacity; reserved *= 3) {
|
2016-01-05 19:08:11 +00:00
|
|
|
ConstantArrayBuilder builder(isolate(), zone());
|
2016-02-26 10:14:49 +00:00
|
|
|
for (size_t i = 0; i < k8BitCapacity; i++) {
|
2016-01-05 19:08:11 +00:00
|
|
|
Handle<Object> 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);
|
2016-02-26 10:14:49 +00:00
|
|
|
CHECK_EQ(builder.size(), k8BitCapacity);
|
2016-01-05 19:08:11 +00:00
|
|
|
}
|
|
|
|
for (size_t i = 0; i < reserved; i++) {
|
|
|
|
builder.DiscardReservedEntry(OperandSize::kShort);
|
2016-02-26 10:14:49 +00:00
|
|
|
CHECK_EQ(builder.size(), k8BitCapacity);
|
2016-01-05 19:08:11 +00:00
|
|
|
}
|
|
|
|
for (size_t i = 0; i < reserved; i++) {
|
|
|
|
OperandSize operand_size = builder.CreateReservedEntry();
|
|
|
|
CHECK(operand_size == OperandSize::kShort);
|
|
|
|
Handle<Object> object = isolate()->factory()->NewNumberFromSize(i);
|
|
|
|
builder.CommitReservedEntry(operand_size, object);
|
2016-02-26 10:14:49 +00:00
|
|
|
CHECK_EQ(builder.size(), k8BitCapacity);
|
2016-01-05 19:08:11 +00:00
|
|
|
}
|
2016-02-26 10:14:49 +00:00
|
|
|
for (size_t i = k8BitCapacity; i < k8BitCapacity + reserved; i++) {
|
2016-01-05 19:08:11 +00:00
|
|
|
OperandSize operand_size = builder.CreateReservedEntry();
|
|
|
|
CHECK(operand_size == OperandSize::kShort);
|
|
|
|
Handle<Object> 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> object = isolate()->factory()->NewNumberFromSize(i);
|
|
|
|
builder.Insert(object);
|
|
|
|
CHECK(builder.At(i)->SameValue(*object));
|
|
|
|
}
|
2016-01-20 11:31:20 +00:00
|
|
|
Handle<FixedArray> constant_array = builder.ToFixedArray();
|
2016-01-05 19:08:11 +00:00
|
|
|
CHECK_EQ(constant_array->length(), kNumberOfElements);
|
|
|
|
for (size_t i = 0; i < kNumberOfElements; i++) {
|
|
|
|
CHECK(constant_array->get(static_cast<int>(i))->SameValue(*builder.At(i)));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
TEST_F(ConstantArrayBuilderTest, GapFilledWhenLowReservationCommitted) {
|
|
|
|
ConstantArrayBuilder builder(isolate(), zone());
|
2016-02-26 10:14:49 +00:00
|
|
|
for (size_t i = 0; i < k8BitCapacity; i++) {
|
2016-01-05 19:08:11 +00:00
|
|
|
OperandSize operand_size = builder.CreateReservedEntry();
|
|
|
|
CHECK(OperandSize::kByte == operand_size);
|
|
|
|
CHECK_EQ(builder.size(), 0);
|
|
|
|
}
|
2016-02-26 10:14:49 +00:00
|
|
|
for (size_t i = 0; i < k8BitCapacity; i++) {
|
2016-01-05 19:08:11 +00:00
|
|
|
Handle<Object> object = isolate()->factory()->NewNumberFromSize(i);
|
|
|
|
builder.Insert(object);
|
2016-02-26 10:14:49 +00:00
|
|
|
CHECK_EQ(builder.size(), i + k8BitCapacity + 1);
|
2016-01-05 19:08:11 +00:00
|
|
|
}
|
2016-02-26 10:14:49 +00:00
|
|
|
for (size_t i = 0; i < k8BitCapacity; i++) {
|
2016-01-05 19:08:11 +00:00
|
|
|
builder.CommitReservedEntry(OperandSize::kByte,
|
2016-02-26 10:14:49 +00:00
|
|
|
builder.At(i + k8BitCapacity));
|
|
|
|
CHECK_EQ(builder.size(), 2 * k8BitCapacity);
|
2016-01-05 19:08:11 +00:00
|
|
|
}
|
2016-02-26 10:14:49 +00:00
|
|
|
for (size_t i = 0; i < k8BitCapacity; i++) {
|
|
|
|
Handle<Object> original = builder.At(k8BitCapacity + i);
|
2016-01-05 19:08:11 +00:00
|
|
|
Handle<Object> duplicate = builder.At(i);
|
|
|
|
CHECK(original->SameValue(*duplicate));
|
|
|
|
Handle<Object> reference = isolate()->factory()->NewNumberFromSize(i);
|
|
|
|
CHECK(original->SameValue(*reference));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
TEST_F(ConstantArrayBuilderTest, GapNotFilledWhenLowReservationDiscarded) {
|
|
|
|
ConstantArrayBuilder builder(isolate(), zone());
|
2016-02-26 10:14:49 +00:00
|
|
|
for (size_t i = 0; i < k8BitCapacity; i++) {
|
2016-01-05 19:08:11 +00:00
|
|
|
OperandSize operand_size = builder.CreateReservedEntry();
|
|
|
|
CHECK(OperandSize::kByte == operand_size);
|
|
|
|
CHECK_EQ(builder.size(), 0);
|
|
|
|
}
|
2016-02-26 10:14:49 +00:00
|
|
|
for (size_t i = 0; i < k8BitCapacity; i++) {
|
2016-01-05 19:08:11 +00:00
|
|
|
Handle<Object> object = isolate()->factory()->NewNumberFromSize(i);
|
|
|
|
builder.Insert(object);
|
2016-02-26 10:14:49 +00:00
|
|
|
CHECK_EQ(builder.size(), i + k8BitCapacity + 1);
|
2016-01-05 19:08:11 +00:00
|
|
|
}
|
2016-02-26 10:14:49 +00:00
|
|
|
for (size_t i = 0; i < k8BitCapacity; i++) {
|
2016-01-05 19:08:11 +00:00
|
|
|
builder.DiscardReservedEntry(OperandSize::kByte);
|
2016-02-26 10:14:49 +00:00
|
|
|
builder.Insert(builder.At(i + k8BitCapacity));
|
|
|
|
CHECK_EQ(builder.size(), 2 * k8BitCapacity);
|
2016-01-05 19:08:11 +00:00
|
|
|
}
|
2016-02-26 10:14:49 +00:00
|
|
|
for (size_t i = 0; i < k8BitCapacity; i++) {
|
2016-01-05 19:08:11 +00:00
|
|
|
Handle<Object> reference = isolate()->factory()->NewNumberFromSize(i);
|
2016-02-26 10:14:49 +00:00
|
|
|
Handle<Object> original = builder.At(k8BitCapacity + i);
|
2016-01-05 19:08:11 +00:00
|
|
|
CHECK(original->SameValue(*reference));
|
|
|
|
Handle<Object> duplicate = builder.At(i);
|
|
|
|
CHECK(duplicate->SameValue(*isolate()->factory()->the_hole_value()));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace interpreter
|
|
|
|
} // namespace internal
|
|
|
|
} // namespace v8
|