| Index: test/cctest/test-migrations.cc
|
| diff --git a/test/cctest/test-migrations.cc b/test/cctest/test-migrations.cc
|
| deleted file mode 100644
|
| index d441990e3c5b17ef4d07b5cda25a5968b8605f5b..0000000000000000000000000000000000000000
|
| --- a/test/cctest/test-migrations.cc
|
| +++ /dev/null
|
| @@ -1,2182 +0,0 @@
|
| -// 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.
|
| -
|
| -#include <stdlib.h>
|
| -#include <utility>
|
| -
|
| -#include "src/v8.h"
|
| -
|
| -#include "src/compilation-cache.h"
|
| -#include "src/execution.h"
|
| -#include "src/factory.h"
|
| -#include "src/global-handles.h"
|
| -#include "src/ic/stub-cache.h"
|
| -#include "src/macro-assembler.h"
|
| -#include "test/cctest/cctest.h"
|
| -
|
| -using namespace v8::internal;
|
| -
|
| -
|
| -// TODO(ishell): fix this once TransitionToPrototype stops generalizing
|
| -// all field representations (similar to crbug/448711 where elements kind
|
| -// and observed transitions caused generalization of all field representations).
|
| -const bool IS_PROTO_TRANS_ISSUE_FIXED = false;
|
| -
|
| -
|
| -// TODO(ishell): fix this once TransitionToAccessorProperty is able to always
|
| -// keep map in fast mode.
|
| -const bool IS_ACCESSOR_FIELD_SUPPORTED = false;
|
| -
|
| -
|
| -// Number of properties used in the tests.
|
| -const int kPropCount = 7;
|
| -
|
| -
|
| -//
|
| -// Helper functions.
|
| -//
|
| -
|
| -static Handle<String> MakeString(const char* str) {
|
| - Isolate* isolate = CcTest::i_isolate();
|
| - Factory* factory = isolate->factory();
|
| - return factory->InternalizeUtf8String(str);
|
| -}
|
| -
|
| -
|
| -static Handle<String> MakeName(const char* str, int suffix) {
|
| - EmbeddedVector<char, 128> buffer;
|
| - SNPrintF(buffer, "%s%d", str, suffix);
|
| - return MakeString(buffer.start());
|
| -}
|
| -
|
| -
|
| -static Handle<AccessorPair> CreateAccessorPair(bool with_getter,
|
| - bool with_setter) {
|
| - Isolate* isolate = CcTest::i_isolate();
|
| - Factory* factory = isolate->factory();
|
| - Handle<AccessorPair> pair = factory->NewAccessorPair();
|
| - Handle<String> empty_string = factory->empty_string();
|
| - if (with_getter) {
|
| - Handle<JSFunction> func = factory->NewFunction(empty_string);
|
| - pair->set_getter(*func);
|
| - }
|
| - if (with_setter) {
|
| - Handle<JSFunction> func = factory->NewFunction(empty_string);
|
| - pair->set_setter(*func);
|
| - }
|
| - return pair;
|
| -}
|
| -
|
| -
|
| -static bool EqualDetails(DescriptorArray* descriptors, int descriptor,
|
| - PropertyType type, PropertyAttributes attributes,
|
| - Representation representation, int field_index = -1) {
|
| - PropertyDetails details = descriptors->GetDetails(descriptor);
|
| - if (details.type() != type) return false;
|
| - if (details.attributes() != attributes) return false;
|
| - if (!details.representation().Equals(representation)) return false;
|
| - if (field_index >= 0 && details.field_index() != field_index) return false;
|
| - return true;
|
| -}
|
| -
|
| -
|
| -class Expectations {
|
| - static const int MAX_PROPERTIES = 10;
|
| - Isolate* isolate_;
|
| - PropertyType types_[MAX_PROPERTIES];
|
| - PropertyAttributes attributes_[MAX_PROPERTIES];
|
| - Representation representations_[MAX_PROPERTIES];
|
| - // HeapType for kField, value for DATA_CONSTANT and getter for
|
| - // ACCESSOR_CONSTANT.
|
| - Handle<Object> values_[MAX_PROPERTIES];
|
| - // Setter for ACCESSOR_CONSTANT.
|
| - Handle<Object> setter_values_[MAX_PROPERTIES];
|
| - int number_of_properties_;
|
| -
|
| - public:
|
| - explicit Expectations(Isolate* isolate)
|
| - : isolate_(isolate), number_of_properties_(0) {}
|
| -
|
| - void Init(int index, PropertyType type, PropertyAttributes attributes,
|
| - Representation representation, Handle<Object> value) {
|
| - DCHECK(index < MAX_PROPERTIES);
|
| - types_[index] = type;
|
| - attributes_[index] = attributes;
|
| - representations_[index] = representation;
|
| - values_[index] = value;
|
| - }
|
| -
|
| - void Print() const {
|
| - OFStream os(stdout);
|
| - os << "Expectations: #" << number_of_properties_ << "\n";
|
| - for (int i = 0; i < number_of_properties_; i++) {
|
| - os << " " << i << ": ";
|
| - os << "Descriptor @ ";
|
| - if (types_[i] == ACCESSOR_CONSTANT) {
|
| - os << "(get: " << Brief(*values_[i])
|
| - << ", set: " << Brief(*setter_values_[i]) << ") ";
|
| - } else {
|
| - os << Brief(*values_[i]);
|
| - }
|
| - os << " (";
|
| - switch (types_[i]) {
|
| - case DATA_CONSTANT:
|
| - os << "immutable ";
|
| - // Fall through.
|
| - case DATA:
|
| - os << "data";
|
| - break;
|
| -
|
| - case ACCESSOR_CONSTANT:
|
| - os << "immutable ";
|
| - // Fall through.
|
| - case ACCESSOR:
|
| - os << "accessor";
|
| - break;
|
| - }
|
| - os << ": " << representations_[i].Mnemonic();
|
| - os << ", attrs: " << attributes_[i] << ")\n";
|
| - }
|
| - os << "\n";
|
| - }
|
| -
|
| - Handle<HeapType> GetFieldType(int index) {
|
| - CHECK(index < MAX_PROPERTIES);
|
| - CHECK(types_[index] == DATA || types_[index] == ACCESSOR);
|
| - return Handle<HeapType>::cast(values_[index]);
|
| - }
|
| -
|
| - void SetDataField(int index, PropertyAttributes attrs,
|
| - Representation representation, Handle<HeapType> value) {
|
| - Init(index, DATA, attrs, representation, value);
|
| - }
|
| -
|
| - void SetDataField(int index, Representation representation,
|
| - Handle<HeapType> value) {
|
| - SetDataField(index, attributes_[index], representation, value);
|
| - }
|
| -
|
| - void SetAccessorField(int index, PropertyAttributes attrs) {
|
| - Init(index, ACCESSOR, attrs, Representation::Tagged(),
|
| - HeapType::Any(isolate_));
|
| - }
|
| -
|
| - void SetAccessorField(int index) {
|
| - SetAccessorField(index, attributes_[index]);
|
| - }
|
| -
|
| - void SetDataConstant(int index, PropertyAttributes attrs,
|
| - Handle<JSFunction> value) {
|
| - Init(index, DATA_CONSTANT, attrs, Representation::HeapObject(), value);
|
| - }
|
| -
|
| - void SetDataConstant(int index, Handle<JSFunction> value) {
|
| - SetDataConstant(index, attributes_[index], value);
|
| - }
|
| -
|
| - void SetAccessorConstant(int index, PropertyAttributes attrs,
|
| - Handle<Object> getter, Handle<Object> setter) {
|
| - Init(index, ACCESSOR_CONSTANT, attrs, Representation::Tagged(), getter);
|
| - setter_values_[index] = setter;
|
| - }
|
| -
|
| - void SetAccessorConstantComponent(int index, PropertyAttributes attrs,
|
| - AccessorComponent component,
|
| - Handle<Object> accessor) {
|
| - CHECK_EQ(ACCESSOR_CONSTANT, types_[index]);
|
| - CHECK(index < number_of_properties_);
|
| - if (component == ACCESSOR_GETTER) {
|
| - values_[index] = accessor;
|
| - } else {
|
| - setter_values_[index] = accessor;
|
| - }
|
| - }
|
| -
|
| - void SetAccessorConstant(int index, PropertyAttributes attrs,
|
| - Handle<AccessorPair> pair) {
|
| - Handle<Object> getter = handle(pair->getter(), isolate_);
|
| - Handle<Object> setter = handle(pair->setter(), isolate_);
|
| - SetAccessorConstant(index, attrs, getter, setter);
|
| - }
|
| -
|
| - void SetAccessorConstant(int index, Handle<Object> getter,
|
| - Handle<Object> setter) {
|
| - SetAccessorConstant(index, attributes_[index], getter, setter);
|
| - }
|
| -
|
| - void SetAccessorConstant(int index, Handle<AccessorPair> pair) {
|
| - Handle<Object> getter = handle(pair->getter(), isolate_);
|
| - Handle<Object> setter = handle(pair->setter(), isolate_);
|
| - SetAccessorConstant(index, getter, setter);
|
| - }
|
| -
|
| - void GeneralizeRepresentation(int index) {
|
| - CHECK(index < number_of_properties_);
|
| - representations_[index] = Representation::Tagged();
|
| - if (types_[index] == DATA || types_[index] == ACCESSOR) {
|
| - values_[index] = HeapType::Any(isolate_);
|
| - }
|
| - }
|
| -
|
| -
|
| - bool Check(DescriptorArray* descriptors, int descriptor) const {
|
| - PropertyType type = types_[descriptor];
|
| - if (!EqualDetails(descriptors, descriptor, type, attributes_[descriptor],
|
| - representations_[descriptor])) {
|
| - return false;
|
| - }
|
| - Object* value = descriptors->GetValue(descriptor);
|
| - Object* expected_value = *values_[descriptor];
|
| - switch (type) {
|
| - case DATA:
|
| - case ACCESSOR: {
|
| - HeapType* type = descriptors->GetFieldType(descriptor);
|
| - return HeapType::cast(expected_value)->Equals(type);
|
| - }
|
| -
|
| - case DATA_CONSTANT:
|
| - return value == expected_value;
|
| -
|
| - case ACCESSOR_CONSTANT: {
|
| - if (value == expected_value) return true;
|
| - if (!value->IsAccessorPair()) return false;
|
| - AccessorPair* pair = AccessorPair::cast(value);
|
| - return pair->Equals(expected_value, *setter_values_[descriptor]);
|
| - }
|
| - }
|
| - UNREACHABLE();
|
| - return false;
|
| - }
|
| -
|
| - bool Check(Map* map, int expected_nof) const {
|
| - CHECK(number_of_properties_ <= MAX_PROPERTIES);
|
| - CHECK_EQ(expected_nof, map->NumberOfOwnDescriptors());
|
| - CHECK(!map->is_dictionary_map());
|
| -
|
| - DescriptorArray* descriptors = map->instance_descriptors();
|
| - CHECK(expected_nof <= number_of_properties_);
|
| - for (int i = 0; i < expected_nof; i++) {
|
| - if (!Check(descriptors, i)) {
|
| - Print();
|
| -#ifdef OBJECT_PRINT
|
| - descriptors->Print();
|
| -#endif
|
| - Check(descriptors, i);
|
| - return false;
|
| - }
|
| - }
|
| - return true;
|
| - }
|
| -
|
| - bool Check(Map* map) const { return Check(map, number_of_properties_); }
|
| -
|
| -
|
| - //
|
| - // Helper methods for initializing expectations and adding properties to
|
| - // given |map|.
|
| - //
|
| -
|
| - Handle<Map> AddDataField(Handle<Map> map, PropertyAttributes attributes,
|
| - Representation representation,
|
| - Handle<HeapType> heap_type) {
|
| - CHECK_EQ(number_of_properties_, map->NumberOfOwnDescriptors());
|
| - int property_index = number_of_properties_++;
|
| - SetDataField(property_index, attributes, representation, heap_type);
|
| -
|
| - Handle<String> name = MakeName("prop", property_index);
|
| - return Map::CopyWithField(map, name, heap_type, attributes, representation,
|
| - INSERT_TRANSITION).ToHandleChecked();
|
| - }
|
| -
|
| - Handle<Map> AddDataConstant(Handle<Map> map, PropertyAttributes attributes,
|
| - Handle<JSFunction> value) {
|
| - CHECK_EQ(number_of_properties_, map->NumberOfOwnDescriptors());
|
| - int property_index = number_of_properties_++;
|
| - SetDataConstant(property_index, attributes, value);
|
| -
|
| - Handle<String> name = MakeName("prop", property_index);
|
| - return Map::CopyWithConstant(map, name, value, attributes,
|
| - INSERT_TRANSITION).ToHandleChecked();
|
| - }
|
| -
|
| - Handle<Map> TransitionToDataField(Handle<Map> map,
|
| - PropertyAttributes attributes,
|
| - Representation representation,
|
| - Handle<HeapType> heap_type,
|
| - Handle<Object> value) {
|
| - CHECK_EQ(number_of_properties_, map->NumberOfOwnDescriptors());
|
| - int property_index = number_of_properties_++;
|
| - SetDataField(property_index, attributes, representation, heap_type);
|
| -
|
| - Handle<String> name = MakeName("prop", property_index);
|
| - return Map::TransitionToDataProperty(
|
| - map, name, value, attributes, Object::CERTAINLY_NOT_STORE_FROM_KEYED);
|
| - }
|
| -
|
| - Handle<Map> TransitionToDataConstant(Handle<Map> map,
|
| - PropertyAttributes attributes,
|
| - Handle<JSFunction> value) {
|
| - CHECK_EQ(number_of_properties_, map->NumberOfOwnDescriptors());
|
| - int property_index = number_of_properties_++;
|
| - SetDataConstant(property_index, attributes, value);
|
| -
|
| - Handle<String> name = MakeName("prop", property_index);
|
| - return Map::TransitionToDataProperty(
|
| - map, name, value, attributes, Object::CERTAINLY_NOT_STORE_FROM_KEYED);
|
| - }
|
| -
|
| - Handle<Map> FollowDataTransition(Handle<Map> map,
|
| - PropertyAttributes attributes,
|
| - Representation representation,
|
| - Handle<HeapType> heap_type) {
|
| - CHECK_EQ(number_of_properties_, map->NumberOfOwnDescriptors());
|
| - int property_index = number_of_properties_++;
|
| - SetDataField(property_index, attributes, representation, heap_type);
|
| -
|
| - Handle<String> name = MakeName("prop", property_index);
|
| - Map* target =
|
| - TransitionArray::SearchTransition(*map, kData, *name, attributes);
|
| - CHECK(target != NULL);
|
| - return handle(target);
|
| - }
|
| -
|
| - Handle<Map> AddAccessorConstant(Handle<Map> map,
|
| - PropertyAttributes attributes,
|
| - Handle<AccessorPair> pair) {
|
| - CHECK_EQ(number_of_properties_, map->NumberOfOwnDescriptors());
|
| - int property_index = number_of_properties_++;
|
| - SetAccessorConstant(property_index, attributes, pair);
|
| -
|
| - Handle<String> name = MakeName("prop", property_index);
|
| -
|
| - AccessorConstantDescriptor new_desc(name, pair, attributes);
|
| - return Map::CopyInsertDescriptor(map, &new_desc, INSERT_TRANSITION);
|
| - }
|
| -
|
| - Handle<Map> AddAccessorConstant(Handle<Map> map,
|
| - PropertyAttributes attributes,
|
| - Handle<Object> getter,
|
| - Handle<Object> setter) {
|
| - CHECK_EQ(number_of_properties_, map->NumberOfOwnDescriptors());
|
| - int property_index = number_of_properties_++;
|
| - SetAccessorConstant(property_index, attributes, getter, setter);
|
| -
|
| - Handle<String> name = MakeName("prop", property_index);
|
| -
|
| - CHECK(!getter->IsNull() || !setter->IsNull());
|
| - Factory* factory = isolate_->factory();
|
| -
|
| - if (!getter->IsNull()) {
|
| - Handle<AccessorPair> pair = factory->NewAccessorPair();
|
| - pair->SetComponents(*getter, *factory->null_value());
|
| - AccessorConstantDescriptor new_desc(name, pair, attributes);
|
| - map = Map::CopyInsertDescriptor(map, &new_desc, INSERT_TRANSITION);
|
| - }
|
| - if (!setter->IsNull()) {
|
| - Handle<AccessorPair> pair = factory->NewAccessorPair();
|
| - pair->SetComponents(*getter, *setter);
|
| - AccessorConstantDescriptor new_desc(name, pair, attributes);
|
| - map = Map::CopyInsertDescriptor(map, &new_desc, INSERT_TRANSITION);
|
| - }
|
| - return map;
|
| - }
|
| -
|
| - Handle<Map> TransitionToAccessorConstant(Handle<Map> map,
|
| - PropertyAttributes attributes,
|
| - Handle<AccessorPair> pair) {
|
| - CHECK_EQ(number_of_properties_, map->NumberOfOwnDescriptors());
|
| - int property_index = number_of_properties_++;
|
| - SetAccessorConstant(property_index, attributes, pair);
|
| -
|
| - Handle<String> name = MakeName("prop", property_index);
|
| -
|
| - Isolate* isolate = CcTest::i_isolate();
|
| - Handle<Object> getter(pair->getter(), isolate);
|
| - Handle<Object> setter(pair->setter(), isolate);
|
| -
|
| - map = Map::TransitionToAccessorProperty(map, name, ACCESSOR_GETTER, getter,
|
| - attributes);
|
| - CHECK(!map->is_deprecated());
|
| - CHECK(!map->is_dictionary_map());
|
| -
|
| - map = Map::TransitionToAccessorProperty(map, name, ACCESSOR_SETTER, setter,
|
| - attributes);
|
| - CHECK(!map->is_deprecated());
|
| - CHECK(!map->is_dictionary_map());
|
| - return map;
|
| - }
|
| -};
|
| -
|
| -
|
| -////////////////////////////////////////////////////////////////////////////////
|
| -// A set of tests for property reconfiguration that makes new transition tree
|
| -// branch.
|
| -//
|
| -
|
| -TEST(ReconfigureAccessorToNonExistingDataField) {
|
| - CcTest::InitializeVM();
|
| - v8::HandleScope scope(CcTest::isolate());
|
| - Isolate* isolate = CcTest::i_isolate();
|
| - Handle<HeapType> any_type = HeapType::Any(isolate);
|
| - Handle<HeapType> none_type = HeapType::None(isolate);
|
| - Handle<AccessorPair> pair = CreateAccessorPair(true, true);
|
| -
|
| - Expectations expectations(isolate);
|
| -
|
| - // Create a map, add required properties to it and initialize expectations.
|
| - Handle<Map> initial_map = Map::Create(isolate, 0);
|
| - Handle<Map> map = initial_map;
|
| - map = expectations.AddAccessorConstant(map, NONE, pair);
|
| -
|
| - CHECK(!map->is_deprecated());
|
| - CHECK(map->is_stable());
|
| - CHECK(expectations.Check(*map));
|
| -
|
| - Handle<Map> new_map = Map::ReconfigureProperty(
|
| - map, 0, kData, NONE, Representation::None(), none_type, FORCE_FIELD);
|
| - // |map| did not change except marked unstable.
|
| - CHECK(!map->is_deprecated());
|
| - CHECK(!map->is_stable());
|
| - CHECK(expectations.Check(*map));
|
| -
|
| - expectations.SetDataField(0, NONE, Representation::None(), none_type);
|
| -
|
| - CHECK(!new_map->is_deprecated());
|
| - CHECK(new_map->is_stable());
|
| - CHECK(expectations.Check(*new_map));
|
| -
|
| - Handle<Map> new_map2 = Map::ReconfigureProperty(
|
| - map, 0, kData, NONE, Representation::None(), none_type, FORCE_FIELD);
|
| - CHECK_EQ(*new_map, *new_map2);
|
| -
|
| - Handle<Object> value(Smi::FromInt(0), isolate);
|
| - Handle<Map> prepared_map = Map::PrepareForDataProperty(new_map, 0, value);
|
| - // None to Smi generalization is trivial, map does not change.
|
| - CHECK_EQ(*new_map, *prepared_map);
|
| -
|
| - expectations.SetDataField(0, NONE, Representation::Smi(), any_type);
|
| - CHECK(prepared_map->is_stable());
|
| - CHECK(expectations.Check(*prepared_map));
|
| -
|
| - // Now create an object with |map|, migrate it to |prepared_map| and ensure
|
| - // that the data property is uninitialized.
|
| - Factory* factory = isolate->factory();
|
| - Handle<JSObject> obj = factory->NewJSObjectFromMap(map);
|
| - JSObject::MigrateToMap(obj, prepared_map);
|
| - FieldIndex index = FieldIndex::ForDescriptor(*prepared_map, 0);
|
| - CHECK(obj->RawFastPropertyAt(index)->IsUninitialized());
|
| -#ifdef VERIFY_HEAP
|
| - obj->ObjectVerify();
|
| -#endif
|
| -}
|
| -
|
| -
|
| -// This test checks that the LookupIterator machinery involved in
|
| -// JSObject::SetOwnPropertyIgnoreAttributes() does not try to migrate object
|
| -// to a map with a property with None representation.
|
| -TEST(ReconfigureAccessorToNonExistingDataFieldHeavy) {
|
| - CcTest::InitializeVM();
|
| - Isolate* isolate = CcTest::i_isolate();
|
| - Factory* factory = isolate->factory();
|
| - v8::HandleScope scope(CcTest::isolate());
|
| -
|
| - CompileRun(
|
| - "function getter() { return 1; };"
|
| - "function setter() {};"
|
| - "var o = {};"
|
| - "Object.defineProperty(o, 'foo', "
|
| - " { get: getter, set: setter, "
|
| - " configurable: true, enumerable: true});");
|
| -
|
| - Handle<String> foo_str = factory->InternalizeUtf8String("foo");
|
| - Handle<String> obj_name = factory->InternalizeUtf8String("o");
|
| -
|
| - Handle<Object> obj_value =
|
| - Object::GetProperty(isolate->global_object(), obj_name).ToHandleChecked();
|
| - CHECK(obj_value->IsJSObject());
|
| - Handle<JSObject> obj = Handle<JSObject>::cast(obj_value);
|
| -
|
| - CHECK_EQ(1, obj->map()->NumberOfOwnDescriptors());
|
| - CHECK(obj->map()->instance_descriptors()->GetValue(0)->IsAccessorPair());
|
| -
|
| - Handle<Object> value(Smi::FromInt(42), isolate);
|
| - JSObject::SetOwnPropertyIgnoreAttributes(obj, foo_str, value, NONE).Check();
|
| -
|
| - // Check that the property contains |value|.
|
| - CHECK_EQ(1, obj->map()->NumberOfOwnDescriptors());
|
| - FieldIndex index = FieldIndex::ForDescriptor(obj->map(), 0);
|
| - Object* the_value = obj->RawFastPropertyAt(index);
|
| - CHECK(the_value->IsSmi());
|
| - CHECK_EQ(42, Smi::cast(the_value)->value());
|
| -}
|
| -
|
| -
|
| -////////////////////////////////////////////////////////////////////////////////
|
| -// A set of tests for representation generalization case.
|
| -//
|
| -
|
| -// This test ensures that representation/field type generalization at
|
| -// |property_index| is done correctly independently of the fact that the |map|
|
| -// is detached from transition tree or not.
|
| -//
|
| -// {} - p0 - p1 - p2: |detach_point_map|
|
| -// |
|
| -// X - detached at |detach_property_at_index|
|
| -// |
|
| -// + - p3 - p4: |map|
|
| -//
|
| -// Detaching does not happen if |detach_property_at_index| is -1.
|
| -//
|
| -static void TestGeneralizeRepresentation(
|
| - int detach_property_at_index, int property_index,
|
| - Representation from_representation, Handle<HeapType> from_type,
|
| - Representation to_representation, Handle<HeapType> to_type,
|
| - Representation expected_representation, Handle<HeapType> expected_type,
|
| - bool expected_deprecation, bool expected_field_type_dependency) {
|
| - Isolate* isolate = CcTest::i_isolate();
|
| - Handle<HeapType> any_type = HeapType::Any(isolate);
|
| -
|
| - CHECK(detach_property_at_index >= -1 &&
|
| - detach_property_at_index < kPropCount);
|
| - CHECK(property_index < kPropCount);
|
| - CHECK_NE(detach_property_at_index, property_index);
|
| -
|
| - const bool is_detached_map = detach_property_at_index >= 0;
|
| -
|
| - Expectations expectations(isolate);
|
| -
|
| - // Create a map, add required properties to it and initialize expectations.
|
| - Handle<Map> initial_map = Map::Create(isolate, 0);
|
| - Handle<Map> map = initial_map;
|
| - Handle<Map> detach_point_map;
|
| - for (int i = 0; i < kPropCount; i++) {
|
| - if (i == property_index) {
|
| - map =
|
| - expectations.AddDataField(map, NONE, from_representation, from_type);
|
| - } else {
|
| - map =
|
| - expectations.AddDataField(map, NONE, Representation::Smi(), any_type);
|
| - if (i == detach_property_at_index) {
|
| - detach_point_map = map;
|
| - }
|
| - }
|
| - }
|
| - CHECK(!map->is_deprecated());
|
| - CHECK(map->is_stable());
|
| - CHECK(expectations.Check(*map));
|
| -
|
| - Zone zone;
|
| -
|
| - if (is_detached_map) {
|
| - detach_point_map = Map::ReconfigureProperty(
|
| - detach_point_map, detach_property_at_index, kData, NONE,
|
| - Representation::Tagged(), any_type, FORCE_FIELD);
|
| - expectations.SetDataField(detach_property_at_index,
|
| - Representation::Tagged(), any_type);
|
| - CHECK(map->is_deprecated());
|
| - CHECK(expectations.Check(*detach_point_map,
|
| - detach_point_map->NumberOfOwnDescriptors()));
|
| - }
|
| -
|
| - // Create new maps by generalizing representation of propX field.
|
| - Handle<Map> field_owner(map->FindFieldOwner(property_index), isolate);
|
| - CompilationInfo info("testing", isolate, &zone);
|
| - CHECK(!info.dependencies()->HasAborted());
|
| -
|
| - info.dependencies()->AssumeFieldType(field_owner);
|
| -
|
| - Handle<Map> new_map =
|
| - Map::ReconfigureProperty(map, property_index, kData, NONE,
|
| - to_representation, to_type, FORCE_FIELD);
|
| -
|
| - expectations.SetDataField(property_index, expected_representation,
|
| - expected_type);
|
| -
|
| - CHECK(!new_map->is_deprecated());
|
| - CHECK(expectations.Check(*new_map));
|
| -
|
| - if (is_detached_map) {
|
| - CHECK(!map->is_stable());
|
| - CHECK(map->is_deprecated());
|
| - CHECK_NE(*map, *new_map);
|
| - CHECK_EQ(expected_field_type_dependency && !field_owner->is_deprecated(),
|
| - info.dependencies()->HasAborted());
|
| -
|
| - } else if (expected_deprecation) {
|
| - CHECK(!map->is_stable());
|
| - CHECK(map->is_deprecated());
|
| - CHECK(field_owner->is_deprecated());
|
| - CHECK_NE(*map, *new_map);
|
| - CHECK(!info.dependencies()->HasAborted());
|
| -
|
| - } else {
|
| - CHECK(!field_owner->is_deprecated());
|
| - CHECK(map->is_stable()); // Map did not change, must be left stable.
|
| - CHECK_EQ(*map, *new_map);
|
| -
|
| - CHECK_EQ(expected_field_type_dependency, info.dependencies()->HasAborted());
|
| - }
|
| -
|
| - info.dependencies()->Rollback(); // Properly cleanup compilation info.
|
| -
|
| - // Update all deprecated maps and check that they are now the same.
|
| - Handle<Map> updated_map = Map::Update(map);
|
| - CHECK_EQ(*new_map, *updated_map);
|
| -}
|
| -
|
| -
|
| -static void TestGeneralizeRepresentation(
|
| - Representation from_representation, Handle<HeapType> from_type,
|
| - Representation to_representation, Handle<HeapType> to_type,
|
| - Representation expected_representation, Handle<HeapType> expected_type,
|
| - bool expected_deprecation, bool expected_field_type_dependency) {
|
| - // Check the cases when the map being reconfigured is a part of the
|
| - // transition tree.
|
| - STATIC_ASSERT(kPropCount > 4);
|
| - int indices[] = {0, 2, kPropCount - 1};
|
| - for (int i = 0; i < static_cast<int>(arraysize(indices)); i++) {
|
| - TestGeneralizeRepresentation(
|
| - -1, indices[i], from_representation, from_type, to_representation,
|
| - to_type, expected_representation, expected_type, expected_deprecation,
|
| - expected_field_type_dependency);
|
| - }
|
| -
|
| - if (!from_representation.IsNone()) {
|
| - // Check the cases when the map being reconfigured is NOT a part of the
|
| - // transition tree. "None -> anything" representation changes make sense
|
| - // only for "attached" maps.
|
| - int indices[] = {0, kPropCount - 1};
|
| - for (int i = 0; i < static_cast<int>(arraysize(indices)); i++) {
|
| - TestGeneralizeRepresentation(
|
| - indices[i], 2, from_representation, from_type, to_representation,
|
| - to_type, expected_representation, expected_type, expected_deprecation,
|
| - expected_field_type_dependency);
|
| - }
|
| -
|
| - // Check that reconfiguration to the very same field works correctly.
|
| - Representation representation = from_representation;
|
| - Handle<HeapType> type = from_type;
|
| - TestGeneralizeRepresentation(-1, 2, representation, type, representation,
|
| - type, representation, type, false, false);
|
| - }
|
| -}
|
| -
|
| -
|
| -static void TestGeneralizeRepresentation(Representation from_representation,
|
| - Handle<HeapType> from_type,
|
| - Representation to_representation,
|
| - Handle<HeapType> to_type,
|
| - Representation expected_representation,
|
| - Handle<HeapType> expected_type) {
|
| - const bool expected_deprecation = true;
|
| - const bool expected_field_type_dependency = false;
|
| -
|
| - TestGeneralizeRepresentation(
|
| - from_representation, from_type, to_representation, to_type,
|
| - expected_representation, expected_type, expected_deprecation,
|
| - expected_field_type_dependency);
|
| -}
|
| -
|
| -
|
| -static void TestGeneralizeRepresentationTrivial(
|
| - Representation from_representation, Handle<HeapType> from_type,
|
| - Representation to_representation, Handle<HeapType> to_type,
|
| - Representation expected_representation, Handle<HeapType> expected_type,
|
| - bool expected_field_type_dependency = true) {
|
| - const bool expected_deprecation = false;
|
| -
|
| - TestGeneralizeRepresentation(
|
| - from_representation, from_type, to_representation, to_type,
|
| - expected_representation, expected_type, expected_deprecation,
|
| - expected_field_type_dependency);
|
| -}
|
| -
|
| -
|
| -TEST(GeneralizeRepresentationSmiToDouble) {
|
| - CcTest::InitializeVM();
|
| - v8::HandleScope scope(CcTest::isolate());
|
| - Isolate* isolate = CcTest::i_isolate();
|
| - Handle<HeapType> any_type = HeapType::Any(isolate);
|
| -
|
| - TestGeneralizeRepresentation(Representation::Smi(), any_type,
|
| - Representation::Double(), any_type,
|
| - Representation::Double(), any_type);
|
| -}
|
| -
|
| -
|
| -TEST(GeneralizeRepresentationSmiToTagged) {
|
| - CcTest::InitializeVM();
|
| - v8::HandleScope scope(CcTest::isolate());
|
| - Isolate* isolate = CcTest::i_isolate();
|
| - Handle<HeapType> any_type = HeapType::Any(isolate);
|
| - Handle<HeapType> value_type =
|
| - HeapType::Class(Map::Create(isolate, 0), isolate);
|
| -
|
| - TestGeneralizeRepresentation(Representation::Smi(), any_type,
|
| - Representation::HeapObject(), value_type,
|
| - Representation::Tagged(), any_type);
|
| -}
|
| -
|
| -
|
| -TEST(GeneralizeRepresentationDoubleToTagged) {
|
| - CcTest::InitializeVM();
|
| - v8::HandleScope scope(CcTest::isolate());
|
| - Isolate* isolate = CcTest::i_isolate();
|
| - Handle<HeapType> any_type = HeapType::Any(isolate);
|
| - Handle<HeapType> value_type =
|
| - HeapType::Class(Map::Create(isolate, 0), isolate);
|
| -
|
| - TestGeneralizeRepresentation(Representation::Double(), any_type,
|
| - Representation::HeapObject(), value_type,
|
| - Representation::Tagged(), any_type);
|
| -}
|
| -
|
| -
|
| -TEST(GeneralizeRepresentationHeapObjectToTagged) {
|
| - CcTest::InitializeVM();
|
| - v8::HandleScope scope(CcTest::isolate());
|
| - Isolate* isolate = CcTest::i_isolate();
|
| - Handle<HeapType> any_type = HeapType::Any(isolate);
|
| - Handle<HeapType> value_type =
|
| - HeapType::Class(Map::Create(isolate, 0), isolate);
|
| -
|
| - TestGeneralizeRepresentation(Representation::HeapObject(), value_type,
|
| - Representation::Smi(), any_type,
|
| - Representation::Tagged(), any_type);
|
| -}
|
| -
|
| -
|
| -TEST(GeneralizeRepresentationHeapObjectToHeapObject) {
|
| - CcTest::InitializeVM();
|
| - v8::HandleScope scope(CcTest::isolate());
|
| - Isolate* isolate = CcTest::i_isolate();
|
| - Handle<HeapType> any_type = HeapType::Any(isolate);
|
| -
|
| - const int kMaxClassesPerFieldType = 1;
|
| - Handle<HeapType> current_type =
|
| - HeapType::Class(Map::Create(isolate, 0), isolate);
|
| -
|
| - for (int i = 0; i < kMaxClassesPerFieldType; i++) {
|
| - Handle<HeapType> new_type =
|
| - HeapType::Class(Map::Create(isolate, 0), isolate);
|
| -
|
| - Handle<HeapType> expected_type =
|
| - (i < kMaxClassesPerFieldType - 1)
|
| - ? HeapType::Union(current_type, new_type, isolate)
|
| - : any_type;
|
| -
|
| - TestGeneralizeRepresentationTrivial(
|
| - Representation::HeapObject(), current_type,
|
| - Representation::HeapObject(), new_type, Representation::HeapObject(),
|
| - expected_type);
|
| - current_type = expected_type;
|
| - }
|
| -
|
| - Handle<HeapType> new_type = HeapType::Class(Map::Create(isolate, 0), isolate);
|
| -
|
| - TestGeneralizeRepresentationTrivial(
|
| - Representation::HeapObject(), any_type, Representation::HeapObject(),
|
| - new_type, Representation::HeapObject(), any_type, false);
|
| -}
|
| -
|
| -
|
| -TEST(GeneralizeRepresentationNoneToSmi) {
|
| - CcTest::InitializeVM();
|
| - v8::HandleScope scope(CcTest::isolate());
|
| - Isolate* isolate = CcTest::i_isolate();
|
| - Handle<HeapType> none_type = HeapType::None(isolate);
|
| - Handle<HeapType> any_type = HeapType::Any(isolate);
|
| -
|
| - // None -> Smi representation change is trivial.
|
| - TestGeneralizeRepresentationTrivial(Representation::None(), none_type,
|
| - Representation::Smi(), any_type,
|
| - Representation::Smi(), any_type);
|
| -}
|
| -
|
| -
|
| -TEST(GeneralizeRepresentationNoneToDouble) {
|
| - CcTest::InitializeVM();
|
| - v8::HandleScope scope(CcTest::isolate());
|
| - Isolate* isolate = CcTest::i_isolate();
|
| - Handle<HeapType> none_type = HeapType::None(isolate);
|
| - Handle<HeapType> any_type = HeapType::Any(isolate);
|
| -
|
| - // None -> Double representation change is NOT trivial.
|
| - TestGeneralizeRepresentation(Representation::None(), none_type,
|
| - Representation::Double(), any_type,
|
| - Representation::Double(), any_type);
|
| -}
|
| -
|
| -
|
| -TEST(GeneralizeRepresentationNoneToHeapObject) {
|
| - CcTest::InitializeVM();
|
| - v8::HandleScope scope(CcTest::isolate());
|
| - Isolate* isolate = CcTest::i_isolate();
|
| - Handle<HeapType> none_type = HeapType::None(isolate);
|
| - Handle<HeapType> value_type =
|
| - HeapType::Class(Map::Create(isolate, 0), isolate);
|
| -
|
| - // None -> HeapObject representation change is trivial.
|
| - TestGeneralizeRepresentationTrivial(Representation::None(), none_type,
|
| - Representation::HeapObject(), value_type,
|
| - Representation::HeapObject(), value_type);
|
| -}
|
| -
|
| -
|
| -TEST(GeneralizeRepresentationNoneToTagged) {
|
| - CcTest::InitializeVM();
|
| - v8::HandleScope scope(CcTest::isolate());
|
| - Isolate* isolate = CcTest::i_isolate();
|
| - Handle<HeapType> none_type = HeapType::None(isolate);
|
| - Handle<HeapType> any_type = HeapType::Any(isolate);
|
| -
|
| - // None -> HeapObject representation change is trivial.
|
| - TestGeneralizeRepresentationTrivial(Representation::None(), none_type,
|
| - Representation::Tagged(), any_type,
|
| - Representation::Tagged(), any_type);
|
| -}
|
| -
|
| -
|
| -////////////////////////////////////////////////////////////////////////////////
|
| -// A set of tests for representation generalization case with kAccessor
|
| -// properties.
|
| -//
|
| -
|
| -TEST(GeneralizeRepresentationWithAccessorProperties) {
|
| - CcTest::InitializeVM();
|
| - v8::HandleScope scope(CcTest::isolate());
|
| - Isolate* isolate = CcTest::i_isolate();
|
| - Handle<HeapType> any_type = HeapType::Any(isolate);
|
| - Handle<AccessorPair> pair = CreateAccessorPair(true, true);
|
| -
|
| - const int kAccessorProp = kPropCount / 2;
|
| - Expectations expectations(isolate);
|
| -
|
| - // Create a map, add required properties to it and initialize expectations.
|
| - Handle<Map> initial_map = Map::Create(isolate, 0);
|
| - Handle<Map> map = initial_map;
|
| - for (int i = 0; i < kPropCount; i++) {
|
| - if (i == kAccessorProp) {
|
| - map = expectations.AddAccessorConstant(map, NONE, pair);
|
| - } else {
|
| - map =
|
| - expectations.AddDataField(map, NONE, Representation::Smi(), any_type);
|
| - }
|
| - }
|
| - CHECK(!map->is_deprecated());
|
| - CHECK(map->is_stable());
|
| - CHECK(expectations.Check(*map));
|
| -
|
| - // Create new maps by generalizing representation of propX field.
|
| - Handle<Map> maps[kPropCount];
|
| - for (int i = 0; i < kPropCount; i++) {
|
| - if (i == kAccessorProp) {
|
| - // Skip accessor property reconfiguration.
|
| - maps[i] = maps[i - 1];
|
| - continue;
|
| - }
|
| - Handle<Map> new_map = Map::ReconfigureProperty(
|
| - map, i, kData, NONE, Representation::Double(), any_type, FORCE_FIELD);
|
| - maps[i] = new_map;
|
| -
|
| - expectations.SetDataField(i, Representation::Double(), any_type);
|
| -
|
| - CHECK(!map->is_stable());
|
| - CHECK(map->is_deprecated());
|
| - CHECK_NE(*map, *new_map);
|
| - CHECK(i == 0 || maps[i - 1]->is_deprecated());
|
| -
|
| - CHECK(!new_map->is_deprecated());
|
| - CHECK(expectations.Check(*new_map));
|
| - }
|
| -
|
| - Handle<Map> active_map = maps[kPropCount - 1];
|
| - CHECK(!active_map->is_deprecated());
|
| -
|
| - // Update all deprecated maps and check that they are now the same.
|
| - Handle<Map> updated_map = Map::Update(map);
|
| - CHECK_EQ(*active_map, *updated_map);
|
| - for (int i = 0; i < kPropCount; i++) {
|
| - updated_map = Map::Update(maps[i]);
|
| - CHECK_EQ(*active_map, *updated_map);
|
| - }
|
| -}
|
| -
|
| -
|
| -////////////////////////////////////////////////////////////////////////////////
|
| -// A set of tests for attribute reconfiguration case.
|
| -//
|
| -
|
| -// This test ensures that representation/field type generalization is correctly
|
| -// propagated from one branch of transition tree (|map2|) to another (|map|).
|
| -//
|
| -// + - p2B - p3 - p4: |map2|
|
| -// |
|
| -// {} - p0 - p1 - p2A - p3 - p4: |map|
|
| -//
|
| -// where "p2A" and "p2B" differ only in the attributes.
|
| -//
|
| -static void TestReconfigureDataFieldAttribute_GeneralizeRepresentation(
|
| - Representation from_representation, Handle<HeapType> from_type,
|
| - Representation to_representation, Handle<HeapType> to_type,
|
| - Representation expected_representation, Handle<HeapType> expected_type) {
|
| - Isolate* isolate = CcTest::i_isolate();
|
| -
|
| - Expectations expectations(isolate);
|
| -
|
| - // Create a map, add required properties to it and initialize expectations.
|
| - Handle<Map> initial_map = Map::Create(isolate, 0);
|
| - Handle<Map> map = initial_map;
|
| - for (int i = 0; i < kPropCount; i++) {
|
| - map = expectations.AddDataField(map, NONE, from_representation, from_type);
|
| - }
|
| - CHECK(!map->is_deprecated());
|
| - CHECK(map->is_stable());
|
| - CHECK(expectations.Check(*map));
|
| -
|
| -
|
| - // Create another branch in transition tree (property at index |kSplitProp|
|
| - // has different attributes), initialize expectations.
|
| - const int kSplitProp = kPropCount / 2;
|
| - Expectations expectations2(isolate);
|
| -
|
| - Handle<Map> map2 = initial_map;
|
| - for (int i = 0; i < kSplitProp; i++) {
|
| - map2 = expectations2.FollowDataTransition(map2, NONE, from_representation,
|
| - from_type);
|
| - }
|
| - map2 =
|
| - expectations2.AddDataField(map2, READ_ONLY, to_representation, to_type);
|
| -
|
| - for (int i = kSplitProp + 1; i < kPropCount; i++) {
|
| - map2 = expectations2.AddDataField(map2, NONE, to_representation, to_type);
|
| - }
|
| - CHECK(!map2->is_deprecated());
|
| - CHECK(map2->is_stable());
|
| - CHECK(expectations2.Check(*map2));
|
| -
|
| - Zone zone;
|
| - Handle<Map> field_owner(map->FindFieldOwner(kSplitProp), isolate);
|
| - CompilationInfo info("testing", isolate, &zone);
|
| - CHECK(!info.dependencies()->HasAborted());
|
| - info.dependencies()->AssumeFieldType(field_owner);
|
| -
|
| - // Reconfigure attributes of property |kSplitProp| of |map2| to NONE, which
|
| - // should generalize representations in |map1|.
|
| - Handle<Map> new_map =
|
| - Map::ReconfigureExistingProperty(map2, kSplitProp, kData, NONE);
|
| -
|
| - // |map2| should be left unchanged but marked unstable.
|
| - CHECK(!map2->is_stable());
|
| - CHECK(!map2->is_deprecated());
|
| - CHECK_NE(*map2, *new_map);
|
| - CHECK(expectations2.Check(*map2));
|
| -
|
| - // |map| should be deprecated and |new_map| should match new expectations.
|
| - for (int i = kSplitProp; i < kPropCount; i++) {
|
| - expectations.SetDataField(i, expected_representation, expected_type);
|
| - }
|
| - CHECK(map->is_deprecated());
|
| - CHECK(!info.dependencies()->HasAborted());
|
| - info.dependencies()->Rollback(); // Properly cleanup compilation info.
|
| - CHECK_NE(*map, *new_map);
|
| -
|
| - CHECK(!new_map->is_deprecated());
|
| - CHECK(expectations.Check(*new_map));
|
| -
|
| - // Update deprecated |map|, it should become |new_map|.
|
| - Handle<Map> updated_map = Map::Update(map);
|
| - CHECK_EQ(*new_map, *updated_map);
|
| -}
|
| -
|
| -
|
| -// This test ensures that trivial representation/field type generalization
|
| -// (from HeapObject to HeapObject) is correctly propagated from one branch of
|
| -// transition tree (|map2|) to another (|map|).
|
| -//
|
| -// + - p2B - p3 - p4: |map2|
|
| -// |
|
| -// {} - p0 - p1 - p2A - p3 - p4: |map|
|
| -//
|
| -// where "p2A" and "p2B" differ only in the attributes.
|
| -//
|
| -static void TestReconfigureDataFieldAttribute_GeneralizeRepresentationTrivial(
|
| - Representation from_representation, Handle<HeapType> from_type,
|
| - Representation to_representation, Handle<HeapType> to_type,
|
| - Representation expected_representation, Handle<HeapType> expected_type,
|
| - bool expected_field_type_dependency = true) {
|
| - Isolate* isolate = CcTest::i_isolate();
|
| -
|
| - Expectations expectations(isolate);
|
| -
|
| - // Create a map, add required properties to it and initialize expectations.
|
| - Handle<Map> initial_map = Map::Create(isolate, 0);
|
| - Handle<Map> map = initial_map;
|
| - for (int i = 0; i < kPropCount; i++) {
|
| - map = expectations.AddDataField(map, NONE, from_representation, from_type);
|
| - }
|
| - CHECK(!map->is_deprecated());
|
| - CHECK(map->is_stable());
|
| - CHECK(expectations.Check(*map));
|
| -
|
| -
|
| - // Create another branch in transition tree (property at index |kSplitProp|
|
| - // has different attributes), initialize expectations.
|
| - const int kSplitProp = kPropCount / 2;
|
| - Expectations expectations2(isolate);
|
| -
|
| - Handle<Map> map2 = initial_map;
|
| - for (int i = 0; i < kSplitProp; i++) {
|
| - map2 = expectations2.FollowDataTransition(map2, NONE, from_representation,
|
| - from_type);
|
| - }
|
| - map2 =
|
| - expectations2.AddDataField(map2, READ_ONLY, to_representation, to_type);
|
| -
|
| - for (int i = kSplitProp + 1; i < kPropCount; i++) {
|
| - map2 = expectations2.AddDataField(map2, NONE, to_representation, to_type);
|
| - }
|
| - CHECK(!map2->is_deprecated());
|
| - CHECK(map2->is_stable());
|
| - CHECK(expectations2.Check(*map2));
|
| -
|
| - Zone zone;
|
| - Handle<Map> field_owner(map->FindFieldOwner(kSplitProp), isolate);
|
| - CompilationInfo info("testing", isolate, &zone);
|
| - CHECK(!info.dependencies()->HasAborted());
|
| - info.dependencies()->AssumeFieldType(field_owner);
|
| -
|
| - // Reconfigure attributes of property |kSplitProp| of |map2| to NONE, which
|
| - // should generalize representations in |map1|.
|
| - Handle<Map> new_map =
|
| - Map::ReconfigureExistingProperty(map2, kSplitProp, kData, NONE);
|
| -
|
| - // |map2| should be left unchanged but marked unstable.
|
| - CHECK(!map2->is_stable());
|
| - CHECK(!map2->is_deprecated());
|
| - CHECK_NE(*map2, *new_map);
|
| - CHECK(expectations2.Check(*map2));
|
| -
|
| - // In trivial case |map| should be returned as a result of the property
|
| - // reconfiguration, respective field types should be generalized and
|
| - // respective code dependencies should be invalidated. |map| should be NOT
|
| - // deprecated and it should match new expectations.
|
| - for (int i = kSplitProp; i < kPropCount; i++) {
|
| - expectations.SetDataField(i, expected_representation, expected_type);
|
| - }
|
| - CHECK(!map->is_deprecated());
|
| - CHECK_EQ(*map, *new_map);
|
| - CHECK_EQ(expected_field_type_dependency, info.dependencies()->HasAborted());
|
| - info.dependencies()->Rollback(); // Properly cleanup compilation info.
|
| -
|
| - CHECK(!new_map->is_deprecated());
|
| - CHECK(expectations.Check(*new_map));
|
| -
|
| - Handle<Map> updated_map = Map::Update(map);
|
| - CHECK_EQ(*new_map, *updated_map);
|
| -}
|
| -
|
| -
|
| -TEST(ReconfigureDataFieldAttribute_GeneralizeRepresentationSmiToDouble) {
|
| - CcTest::InitializeVM();
|
| - v8::HandleScope scope(CcTest::isolate());
|
| - Isolate* isolate = CcTest::i_isolate();
|
| - Handle<HeapType> any_type = HeapType::Any(isolate);
|
| -
|
| - TestReconfigureDataFieldAttribute_GeneralizeRepresentation(
|
| - Representation::Smi(), any_type, Representation::Double(), any_type,
|
| - Representation::Double(), any_type);
|
| -}
|
| -
|
| -
|
| -TEST(ReconfigureDataFieldAttribute_GeneralizeRepresentationSmiToTagged) {
|
| - CcTest::InitializeVM();
|
| - v8::HandleScope scope(CcTest::isolate());
|
| - Isolate* isolate = CcTest::i_isolate();
|
| - Handle<HeapType> any_type = HeapType::Any(isolate);
|
| - Handle<HeapType> value_type =
|
| - HeapType::Class(Map::Create(isolate, 0), isolate);
|
| -
|
| - TestReconfigureDataFieldAttribute_GeneralizeRepresentation(
|
| - Representation::Smi(), any_type, Representation::HeapObject(), value_type,
|
| - Representation::Tagged(), any_type);
|
| -}
|
| -
|
| -
|
| -TEST(ReconfigureDataFieldAttribute_GeneralizeRepresentationDoubleToTagged) {
|
| - CcTest::InitializeVM();
|
| - v8::HandleScope scope(CcTest::isolate());
|
| - Isolate* isolate = CcTest::i_isolate();
|
| - Handle<HeapType> any_type = HeapType::Any(isolate);
|
| - Handle<HeapType> value_type =
|
| - HeapType::Class(Map::Create(isolate, 0), isolate);
|
| -
|
| - TestReconfigureDataFieldAttribute_GeneralizeRepresentation(
|
| - Representation::Double(), any_type, Representation::HeapObject(),
|
| - value_type, Representation::Tagged(), any_type);
|
| -}
|
| -
|
| -
|
| -TEST(ReconfigureDataFieldAttribute_GeneralizeRepresentationHeapObjToHeapObj) {
|
| - CcTest::InitializeVM();
|
| - v8::HandleScope scope(CcTest::isolate());
|
| - Isolate* isolate = CcTest::i_isolate();
|
| - Handle<HeapType> any_type = HeapType::Any(isolate);
|
| -
|
| - const int kMaxClassesPerFieldType = 1;
|
| - Handle<HeapType> current_type =
|
| - HeapType::Class(Map::Create(isolate, 0), isolate);
|
| -
|
| - for (int i = 0; i < kMaxClassesPerFieldType; i++) {
|
| - Handle<HeapType> new_type =
|
| - HeapType::Class(Map::Create(isolate, 0), isolate);
|
| -
|
| - Handle<HeapType> expected_type =
|
| - (i < kMaxClassesPerFieldType - 1)
|
| - ? HeapType::Union(current_type, new_type, isolate)
|
| - : any_type;
|
| -
|
| - TestReconfigureDataFieldAttribute_GeneralizeRepresentationTrivial(
|
| - Representation::HeapObject(), current_type,
|
| - Representation::HeapObject(), new_type, Representation::HeapObject(),
|
| - expected_type);
|
| - current_type = expected_type;
|
| - }
|
| -
|
| - Handle<HeapType> new_type = HeapType::Class(Map::Create(isolate, 0), isolate);
|
| -
|
| - TestReconfigureDataFieldAttribute_GeneralizeRepresentationTrivial(
|
| - Representation::HeapObject(), any_type, Representation::HeapObject(),
|
| - new_type, Representation::HeapObject(), any_type, false);
|
| -}
|
| -
|
| -
|
| -TEST(ReconfigureDataFieldAttribute_GeneralizeRepresentationHeapObjectToTagged) {
|
| - CcTest::InitializeVM();
|
| - v8::HandleScope scope(CcTest::isolate());
|
| - Isolate* isolate = CcTest::i_isolate();
|
| - Handle<HeapType> any_type = HeapType::Any(isolate);
|
| - Handle<HeapType> value_type =
|
| - HeapType::Class(Map::Create(isolate, 0), isolate);
|
| -
|
| - TestReconfigureDataFieldAttribute_GeneralizeRepresentation(
|
| - Representation::HeapObject(), value_type, Representation::Smi(), any_type,
|
| - Representation::Tagged(), any_type);
|
| -}
|
| -
|
| -
|
| -// Checks that given |map| is deprecated and that it updates to given |new_map|
|
| -// which in turn should match expectations.
|
| -struct CheckDeprecated {
|
| - void Check(Handle<Map> map, Handle<Map> new_map,
|
| - const Expectations& expectations) {
|
| - CHECK(map->is_deprecated());
|
| - CHECK_NE(*map, *new_map);
|
| -
|
| - CHECK(!new_map->is_deprecated());
|
| - CHECK(expectations.Check(*new_map));
|
| -
|
| - // Update deprecated |map|, it should become |new_map|.
|
| - Handle<Map> updated_map = Map::Update(map);
|
| - CHECK_EQ(*new_map, *updated_map);
|
| - }
|
| -};
|
| -
|
| -
|
| -// Checks that given |map| is NOT deprecated, equals to given |new_map| and
|
| -// matches expectations.
|
| -struct CheckSameMap {
|
| - void Check(Handle<Map> map, Handle<Map> new_map,
|
| - const Expectations& expectations) {
|
| - // |map| was not reconfigured, therefore it should stay stable.
|
| - CHECK(map->is_stable());
|
| - CHECK(!map->is_deprecated());
|
| - CHECK_EQ(*map, *new_map);
|
| -
|
| - CHECK(!new_map->is_deprecated());
|
| - CHECK(expectations.Check(*new_map));
|
| -
|
| - // Update deprecated |map|, it should become |new_map|.
|
| - Handle<Map> updated_map = Map::Update(map);
|
| - CHECK_EQ(*new_map, *updated_map);
|
| - }
|
| -};
|
| -
|
| -
|
| -// Checks that given |map| is NOT deprecated and matches expectations.
|
| -// |new_map| is unrelated to |map|.
|
| -struct CheckUnrelated {
|
| - void Check(Handle<Map> map, Handle<Map> new_map,
|
| - const Expectations& expectations) {
|
| - CHECK(!map->is_deprecated());
|
| - CHECK_NE(*map, *new_map);
|
| - CHECK(expectations.Check(*map));
|
| -
|
| - CHECK(new_map->is_stable());
|
| - CHECK(!new_map->is_deprecated());
|
| - }
|
| -};
|
| -
|
| -
|
| -// Checks that given |map| is NOT deprecated, and |new_map| is a result of
|
| -// copy-generalize-all-representations.
|
| -struct CheckCopyGeneralizeAllRepresentations {
|
| - void Check(Handle<Map> map, Handle<Map> new_map, Expectations& expectations) {
|
| - CHECK(!map->is_deprecated());
|
| - CHECK_NE(*map, *new_map);
|
| -
|
| - CHECK(new_map->GetBackPointer()->IsUndefined());
|
| - for (int i = 0; i < kPropCount; i++) {
|
| - expectations.GeneralizeRepresentation(i);
|
| - }
|
| -
|
| - CHECK(!new_map->is_deprecated());
|
| - CHECK(expectations.Check(*new_map));
|
| - }
|
| -};
|
| -
|
| -
|
| -// This test ensures that representation/field type generalization is correctly
|
| -// propagated from one branch of transition tree (|map2|) to another (|map1|).
|
| -//
|
| -// + - p2B - p3 - p4: |map2|
|
| -// |
|
| -// {} - p0 - p1: |map|
|
| -// |
|
| -// + - p2A - p3 - p4: |map1|
|
| -// |
|
| -// + - the property customized by the TestConfig provided
|
| -//
|
| -// where "p2A" and "p2B" differ only in the attributes.
|
| -//
|
| -template <typename TestConfig, typename Checker>
|
| -static void TestReconfigureProperty_CustomPropertyAfterTargetMap(
|
| - TestConfig& config, Checker& checker) {
|
| - Isolate* isolate = CcTest::i_isolate();
|
| - Handle<HeapType> any_type = HeapType::Any(isolate);
|
| -
|
| - const int kCustomPropIndex = kPropCount - 2;
|
| - Expectations expectations(isolate);
|
| -
|
| - const int kSplitProp = 2;
|
| - CHECK(kSplitProp < kCustomPropIndex);
|
| -
|
| - const Representation representation = Representation::Smi();
|
| -
|
| - // Create common part of transition tree.
|
| - Handle<Map> initial_map = Map::Create(isolate, 0);
|
| - Handle<Map> map = initial_map;
|
| - for (int i = 0; i < kSplitProp; i++) {
|
| - map = expectations.AddDataField(map, NONE, representation, any_type);
|
| - }
|
| - CHECK(!map->is_deprecated());
|
| - CHECK(map->is_stable());
|
| - CHECK(expectations.Check(*map));
|
| -
|
| -
|
| - // Create branch to |map1|.
|
| - Handle<Map> map1 = map;
|
| - Expectations expectations1 = expectations;
|
| - for (int i = kSplitProp; i < kCustomPropIndex; i++) {
|
| - map1 = expectations1.AddDataField(map1, NONE, representation, any_type);
|
| - }
|
| - map1 = config.AddPropertyAtBranch(1, expectations1, map1);
|
| - for (int i = kCustomPropIndex + 1; i < kPropCount; i++) {
|
| - map1 = expectations1.AddDataField(map1, NONE, representation, any_type);
|
| - }
|
| - CHECK(!map1->is_deprecated());
|
| - CHECK(map1->is_stable());
|
| - CHECK(expectations1.Check(*map1));
|
| -
|
| -
|
| - // Create another branch in transition tree (property at index |kSplitProp|
|
| - // has different attributes), initialize expectations.
|
| - Handle<Map> map2 = map;
|
| - Expectations expectations2 = expectations;
|
| - map2 = expectations2.AddDataField(map2, READ_ONLY, representation, any_type);
|
| - for (int i = kSplitProp + 1; i < kCustomPropIndex; i++) {
|
| - map2 = expectations2.AddDataField(map2, NONE, representation, any_type);
|
| - }
|
| - map2 = config.AddPropertyAtBranch(2, expectations2, map2);
|
| - for (int i = kCustomPropIndex + 1; i < kPropCount; i++) {
|
| - map2 = expectations2.AddDataField(map2, NONE, representation, any_type);
|
| - }
|
| - CHECK(!map2->is_deprecated());
|
| - CHECK(map2->is_stable());
|
| - CHECK(expectations2.Check(*map2));
|
| -
|
| -
|
| - // Reconfigure attributes of property |kSplitProp| of |map2| to NONE, which
|
| - // should generalize representations in |map1|.
|
| - Handle<Map> new_map =
|
| - Map::ReconfigureExistingProperty(map2, kSplitProp, kData, NONE);
|
| -
|
| - // |map2| should be left unchanged but marked unstable.
|
| - CHECK(!map2->is_stable());
|
| - CHECK(!map2->is_deprecated());
|
| - CHECK_NE(*map2, *new_map);
|
| - CHECK(expectations2.Check(*map2));
|
| -
|
| - config.UpdateExpectations(kCustomPropIndex, expectations1);
|
| - checker.Check(map1, new_map, expectations1);
|
| -}
|
| -
|
| -
|
| -TEST(ReconfigureDataFieldAttribute_SameDataConstantAfterTargetMap) {
|
| - CcTest::InitializeVM();
|
| - v8::HandleScope scope(CcTest::isolate());
|
| -
|
| - struct TestConfig {
|
| - Handle<JSFunction> js_func_;
|
| - TestConfig() {
|
| - Isolate* isolate = CcTest::i_isolate();
|
| - Factory* factory = isolate->factory();
|
| - js_func_ = factory->NewFunction(factory->empty_string());
|
| - }
|
| -
|
| - Handle<Map> AddPropertyAtBranch(int branch_id, Expectations& expectations,
|
| - Handle<Map> map) {
|
| - CHECK(branch_id == 1 || branch_id == 2);
|
| - // Add the same data constant property at both transition tree branches.
|
| - return expectations.AddDataConstant(map, NONE, js_func_);
|
| - }
|
| -
|
| - void UpdateExpectations(int property_index, Expectations& expectations) {
|
| - // Expectations stay the same.
|
| - }
|
| - };
|
| -
|
| - TestConfig config;
|
| - // Two branches are "compatible" so the |map1| should NOT be deprecated.
|
| - CheckSameMap checker;
|
| - TestReconfigureProperty_CustomPropertyAfterTargetMap(config, checker);
|
| -}
|
| -
|
| -
|
| -TEST(ReconfigureDataFieldAttribute_DataConstantToDataFieldAfterTargetMap) {
|
| - CcTest::InitializeVM();
|
| - v8::HandleScope scope(CcTest::isolate());
|
| -
|
| - struct TestConfig {
|
| - Handle<JSFunction> js_func1_;
|
| - Handle<JSFunction> js_func2_;
|
| - TestConfig() {
|
| - Isolate* isolate = CcTest::i_isolate();
|
| - Factory* factory = isolate->factory();
|
| - js_func1_ = factory->NewFunction(factory->empty_string());
|
| - js_func2_ = factory->NewFunction(factory->empty_string());
|
| - }
|
| -
|
| - Handle<Map> AddPropertyAtBranch(int branch_id, Expectations& expectations,
|
| - Handle<Map> map) {
|
| - CHECK(branch_id == 1 || branch_id == 2);
|
| - Handle<JSFunction> js_func = branch_id == 1 ? js_func1_ : js_func2_;
|
| - return expectations.AddDataConstant(map, NONE, js_func);
|
| - }
|
| -
|
| - void UpdateExpectations(int property_index, Expectations& expectations) {
|
| - Isolate* isolate = CcTest::i_isolate();
|
| - Handle<HeapType> any_type = HeapType::Any(isolate);
|
| - expectations.SetDataField(property_index, Representation::HeapObject(),
|
| - any_type);
|
| - }
|
| - };
|
| -
|
| - TestConfig config;
|
| - // Two branches are "incompatible" so the |map1| should be deprecated.
|
| - CheckDeprecated checker;
|
| - TestReconfigureProperty_CustomPropertyAfterTargetMap(config, checker);
|
| -}
|
| -
|
| -
|
| -TEST(ReconfigureDataFieldAttribute_DataConstantToAccConstantAfterTargetMap) {
|
| - CcTest::InitializeVM();
|
| - v8::HandleScope scope(CcTest::isolate());
|
| -
|
| - struct TestConfig {
|
| - Handle<JSFunction> js_func_;
|
| - Handle<AccessorPair> pair_;
|
| - TestConfig() {
|
| - Isolate* isolate = CcTest::i_isolate();
|
| - Factory* factory = isolate->factory();
|
| - js_func_ = factory->NewFunction(factory->empty_string());
|
| - pair_ = CreateAccessorPair(true, true);
|
| - }
|
| -
|
| - Handle<Map> AddPropertyAtBranch(int branch_id, Expectations& expectations,
|
| - Handle<Map> map) {
|
| - CHECK(branch_id == 1 || branch_id == 2);
|
| - if (branch_id == 1) {
|
| - return expectations.AddDataConstant(map, NONE, js_func_);
|
| - } else {
|
| - return expectations.AddAccessorConstant(map, NONE, pair_);
|
| - }
|
| - }
|
| -
|
| - void UpdateExpectations(int property_index, Expectations& expectations) {}
|
| - };
|
| -
|
| - TestConfig config;
|
| - // These are completely separate branches in transition tree.
|
| - CheckUnrelated checker;
|
| - TestReconfigureProperty_CustomPropertyAfterTargetMap(config, checker);
|
| -}
|
| -
|
| -
|
| -TEST(ReconfigureDataFieldAttribute_SameAccessorConstantAfterTargetMap) {
|
| - CcTest::InitializeVM();
|
| - v8::HandleScope scope(CcTest::isolate());
|
| -
|
| - struct TestConfig {
|
| - Handle<AccessorPair> pair_;
|
| - TestConfig() { pair_ = CreateAccessorPair(true, true); }
|
| -
|
| - Handle<Map> AddPropertyAtBranch(int branch_id, Expectations& expectations,
|
| - Handle<Map> map) {
|
| - CHECK(branch_id == 1 || branch_id == 2);
|
| - // Add the same accessor constant property at both transition tree
|
| - // branches.
|
| - return expectations.AddAccessorConstant(map, NONE, pair_);
|
| - }
|
| -
|
| - void UpdateExpectations(int property_index, Expectations& expectations) {
|
| - // Two branches are "compatible" so the |map1| should NOT be deprecated.
|
| - }
|
| - };
|
| -
|
| - TestConfig config;
|
| - CheckSameMap checker;
|
| - TestReconfigureProperty_CustomPropertyAfterTargetMap(config, checker);
|
| -}
|
| -
|
| -
|
| -TEST(ReconfigureDataFieldAttribute_AccConstantToAccFieldAfterTargetMap) {
|
| - CcTest::InitializeVM();
|
| - v8::HandleScope scope(CcTest::isolate());
|
| -
|
| - struct TestConfig {
|
| - Handle<AccessorPair> pair1_;
|
| - Handle<AccessorPair> pair2_;
|
| - TestConfig() {
|
| - pair1_ = CreateAccessorPair(true, true);
|
| - pair2_ = CreateAccessorPair(true, true);
|
| - }
|
| -
|
| - Handle<Map> AddPropertyAtBranch(int branch_id, Expectations& expectations,
|
| - Handle<Map> map) {
|
| - CHECK(branch_id == 1 || branch_id == 2);
|
| - Handle<AccessorPair> pair = branch_id == 1 ? pair1_ : pair2_;
|
| - return expectations.AddAccessorConstant(map, NONE, pair);
|
| - }
|
| -
|
| - void UpdateExpectations(int property_index, Expectations& expectations) {
|
| - if (IS_ACCESSOR_FIELD_SUPPORTED) {
|
| - expectations.SetAccessorField(property_index);
|
| - } else {
|
| - // Currently we have a copy-generalize-all-representations case and
|
| - // ACCESSOR property becomes ACCESSOR_CONSTANT.
|
| - expectations.SetAccessorConstant(property_index, pair2_);
|
| - }
|
| - }
|
| - };
|
| -
|
| - TestConfig config;
|
| - if (IS_ACCESSOR_FIELD_SUPPORTED) {
|
| - CheckCopyGeneralizeAllRepresentations checker;
|
| - TestReconfigureProperty_CustomPropertyAfterTargetMap(config, checker);
|
| - } else {
|
| - // Currently we have a copy-generalize-all-representations case.
|
| - CheckCopyGeneralizeAllRepresentations checker;
|
| - TestReconfigureProperty_CustomPropertyAfterTargetMap(config, checker);
|
| - }
|
| -}
|
| -
|
| -
|
| -TEST(ReconfigureDataFieldAttribute_AccConstantToDataFieldAfterTargetMap) {
|
| - CcTest::InitializeVM();
|
| - v8::HandleScope scope(CcTest::isolate());
|
| -
|
| - struct TestConfig {
|
| - Handle<AccessorPair> pair_;
|
| - TestConfig() { pair_ = CreateAccessorPair(true, true); }
|
| -
|
| - Handle<Map> AddPropertyAtBranch(int branch_id, Expectations& expectations,
|
| - Handle<Map> map) {
|
| - CHECK(branch_id == 1 || branch_id == 2);
|
| - if (branch_id == 1) {
|
| - return expectations.AddAccessorConstant(map, NONE, pair_);
|
| - } else {
|
| - Isolate* isolate = CcTest::i_isolate();
|
| - Handle<HeapType> any_type = HeapType::Any(isolate);
|
| - return expectations.AddDataField(map, NONE, Representation::Smi(),
|
| - any_type);
|
| - }
|
| - }
|
| -
|
| - void UpdateExpectations(int property_index, Expectations& expectations) {}
|
| - };
|
| -
|
| - TestConfig config;
|
| - // These are completely separate branches in transition tree.
|
| - CheckUnrelated checker;
|
| - TestReconfigureProperty_CustomPropertyAfterTargetMap(config, checker);
|
| -}
|
| -
|
| -
|
| -////////////////////////////////////////////////////////////////////////////////
|
| -// A set of tests checking split map deprecation.
|
| -//
|
| -
|
| -TEST(ReconfigurePropertySplitMapTransitionsOverflow) {
|
| - CcTest::InitializeVM();
|
| - v8::HandleScope scope(CcTest::isolate());
|
| - Isolate* isolate = CcTest::i_isolate();
|
| - Handle<HeapType> any_type = HeapType::Any(isolate);
|
| -
|
| - Expectations expectations(isolate);
|
| -
|
| - // Create a map, add required properties to it and initialize expectations.
|
| - Handle<Map> initial_map = Map::Create(isolate, 0);
|
| - Handle<Map> map = initial_map;
|
| - for (int i = 0; i < kPropCount; i++) {
|
| - map = expectations.AddDataField(map, NONE, Representation::Smi(), any_type);
|
| - }
|
| - CHECK(!map->is_deprecated());
|
| - CHECK(map->is_stable());
|
| -
|
| - // Generalize representation of property at index |kSplitProp|.
|
| - const int kSplitProp = kPropCount / 2;
|
| - Handle<Map> split_map;
|
| - Handle<Map> map2 = initial_map;
|
| - {
|
| - for (int i = 0; i < kSplitProp + 1; i++) {
|
| - if (i == kSplitProp) {
|
| - split_map = map2;
|
| - }
|
| -
|
| - Handle<String> name = MakeName("prop", i);
|
| - Map* target =
|
| - TransitionArray::SearchTransition(*map2, kData, *name, NONE);
|
| - CHECK(target != NULL);
|
| - map2 = handle(target);
|
| - }
|
| -
|
| - map2 = Map::ReconfigureProperty(map2, kSplitProp, kData, NONE,
|
| - Representation::Double(), any_type,
|
| - FORCE_FIELD);
|
| - expectations.SetDataField(kSplitProp, Representation::Double(), any_type);
|
| -
|
| - CHECK(expectations.Check(*split_map, kSplitProp));
|
| - CHECK(expectations.Check(*map2, kSplitProp + 1));
|
| - }
|
| -
|
| - // At this point |map| should be deprecated and disconnected from the
|
| - // transition tree.
|
| - CHECK(map->is_deprecated());
|
| - CHECK(!split_map->is_deprecated());
|
| - CHECK(map2->is_stable());
|
| - CHECK(!map2->is_deprecated());
|
| -
|
| - // Fill in transition tree of |map2| so that it can't have more transitions.
|
| - for (int i = 0; i < TransitionArray::kMaxNumberOfTransitions; i++) {
|
| - CHECK(TransitionArray::CanHaveMoreTransitions(map2));
|
| - Handle<String> name = MakeName("foo", i);
|
| - Map::CopyWithField(map2, name, any_type, NONE, Representation::Smi(),
|
| - INSERT_TRANSITION).ToHandleChecked();
|
| - }
|
| - CHECK(!TransitionArray::CanHaveMoreTransitions(map2));
|
| -
|
| - // Try to update |map|, since there is no place for propX transition at |map2|
|
| - // |map| should become "copy-generalized".
|
| - Handle<Map> updated_map = Map::Update(map);
|
| - CHECK(updated_map->GetBackPointer()->IsUndefined());
|
| -
|
| - for (int i = 0; i < kPropCount; i++) {
|
| - expectations.SetDataField(i, Representation::Tagged(), any_type);
|
| - }
|
| - CHECK(expectations.Check(*updated_map));
|
| -}
|
| -
|
| -
|
| -////////////////////////////////////////////////////////////////////////////////
|
| -// A set of tests involving special transitions (such as elements kind
|
| -// transition, observed transition or prototype transition).
|
| -//
|
| -
|
| -// This test ensures that representation/field type generalization is correctly
|
| -// propagated from one branch of transition tree (|map2|) to another (|map|).
|
| -//
|
| -// p4B: |map2|
|
| -// |
|
| -// * - special transition
|
| -// |
|
| -// {} - p0 - p1 - p2A - p3 - p4A: |map|
|
| -//
|
| -// where "p4A" and "p4B" are exactly the same properties.
|
| -//
|
| -// TODO(ishell): unify this test template with
|
| -// TestReconfigureDataFieldAttribute_GeneralizeRepresentation once
|
| -// IS_PROTO_TRANS_ISSUE_FIXED and IS_NON_EQUIVALENT_TRANSITION_SUPPORTED are
|
| -// fixed.
|
| -template <typename TestConfig>
|
| -static void TestGeneralizeRepresentationWithSpecialTransition(
|
| - TestConfig& config, Representation from_representation,
|
| - Handle<HeapType> from_type, Representation to_representation,
|
| - Handle<HeapType> to_type, Representation expected_representation,
|
| - Handle<HeapType> expected_type) {
|
| - Isolate* isolate = CcTest::i_isolate();
|
| -
|
| - Expectations expectations(isolate);
|
| -
|
| - // Create a map, add required properties to it and initialize expectations.
|
| - Handle<Map> initial_map = Map::Create(isolate, 0);
|
| - Handle<Map> map = initial_map;
|
| - for (int i = 0; i < kPropCount; i++) {
|
| - map = expectations.AddDataField(map, NONE, from_representation, from_type);
|
| - }
|
| - CHECK(!map->is_deprecated());
|
| - CHECK(map->is_stable());
|
| - CHECK(expectations.Check(*map));
|
| -
|
| - // Apply some special transition to |map|.
|
| - CHECK(map->owns_descriptors());
|
| - Handle<Map> map2 = config.Transition(map);
|
| -
|
| - // |map| should still match expectations.
|
| - CHECK(!map->is_deprecated());
|
| - CHECK(expectations.Check(*map));
|
| -
|
| - Expectations expectations2 = expectations;
|
| - if (config.generalizes_representations()) {
|
| - for (int i = 0; i < kPropCount; i++) {
|
| - expectations2.GeneralizeRepresentation(i);
|
| - }
|
| - }
|
| -
|
| - CHECK(!map2->is_deprecated());
|
| - CHECK(map2->is_stable());
|
| - CHECK(expectations2.Check(*map2));
|
| -
|
| - // Create new maps by generalizing representation of propX field.
|
| - Handle<Map> maps[kPropCount];
|
| - for (int i = 0; i < kPropCount; i++) {
|
| - Handle<Map> new_map = Map::ReconfigureProperty(
|
| - map, i, kData, NONE, to_representation, to_type, FORCE_FIELD);
|
| - maps[i] = new_map;
|
| -
|
| - expectations.SetDataField(i, expected_representation, expected_type);
|
| -
|
| - CHECK(map->is_deprecated());
|
| - CHECK_NE(*map, *new_map);
|
| - CHECK(i == 0 || maps[i - 1]->is_deprecated());
|
| - CHECK(expectations.Check(*new_map));
|
| -
|
| - Handle<Map> new_map2 = Map::Update(map2);
|
| - CHECK(!new_map2->is_deprecated());
|
| - CHECK(!new_map2->is_dictionary_map());
|
| -
|
| - Handle<Map> tmp_map;
|
| - if (Map::TryUpdate(map2).ToHandle(&tmp_map)) {
|
| - // If Map::TryUpdate() manages to succeed the result must match the result
|
| - // of Map::Update().
|
| - CHECK_EQ(*new_map2, *tmp_map);
|
| - }
|
| -
|
| - if (config.is_non_equevalent_transition()) {
|
| - // In case of non-equivalent transition currently we generalize all
|
| - // representations.
|
| - for (int i = 0; i < kPropCount; i++) {
|
| - expectations2.GeneralizeRepresentation(i);
|
| - }
|
| - CHECK(new_map2->GetBackPointer()->IsUndefined());
|
| - CHECK(expectations2.Check(*new_map2));
|
| - } else {
|
| - CHECK(!new_map2->GetBackPointer()->IsUndefined());
|
| - CHECK(expectations2.Check(*new_map2));
|
| - }
|
| - }
|
| -
|
| - Handle<Map> active_map = maps[kPropCount - 1];
|
| - CHECK(!active_map->is_deprecated());
|
| -
|
| - // Update all deprecated maps and check that they are now the same.
|
| - Handle<Map> updated_map = Map::Update(map);
|
| - CHECK_EQ(*active_map, *updated_map);
|
| - for (int i = 0; i < kPropCount; i++) {
|
| - updated_map = Map::Update(maps[i]);
|
| - CHECK_EQ(*active_map, *updated_map);
|
| - }
|
| -}
|
| -
|
| -
|
| -TEST(ElementsKindTransitionFromMapOwningDescriptor) {
|
| - CcTest::InitializeVM();
|
| - v8::HandleScope scope(CcTest::isolate());
|
| - Isolate* isolate = CcTest::i_isolate();
|
| - Handle<HeapType> any_type = HeapType::Any(isolate);
|
| - Handle<HeapType> value_type =
|
| - HeapType::Class(Map::Create(isolate, 0), isolate);
|
| -
|
| - struct TestConfig {
|
| - Handle<Map> Transition(Handle<Map> map) {
|
| - return Map::CopyAsElementsKind(map, DICTIONARY_ELEMENTS,
|
| - INSERT_TRANSITION);
|
| - }
|
| - // TODO(ishell): remove once IS_PROTO_TRANS_ISSUE_FIXED is removed.
|
| - bool generalizes_representations() const { return false; }
|
| - bool is_non_equevalent_transition() const { return false; }
|
| - };
|
| - TestConfig config;
|
| - TestGeneralizeRepresentationWithSpecialTransition(
|
| - config, Representation::Smi(), any_type, Representation::HeapObject(),
|
| - value_type, Representation::Tagged(), any_type);
|
| -}
|
| -
|
| -
|
| -TEST(ElementsKindTransitionFromMapNotOwningDescriptor) {
|
| - CcTest::InitializeVM();
|
| - v8::HandleScope scope(CcTest::isolate());
|
| - Isolate* isolate = CcTest::i_isolate();
|
| - Handle<HeapType> any_type = HeapType::Any(isolate);
|
| - Handle<HeapType> value_type =
|
| - HeapType::Class(Map::Create(isolate, 0), isolate);
|
| -
|
| - struct TestConfig {
|
| - Handle<Map> Transition(Handle<Map> map) {
|
| - Isolate* isolate = CcTest::i_isolate();
|
| - Handle<HeapType> any_type = HeapType::Any(isolate);
|
| -
|
| - // Add one more transition to |map| in order to prevent descriptors
|
| - // ownership.
|
| - CHECK(map->owns_descriptors());
|
| - Map::CopyWithField(map, MakeString("foo"), any_type, NONE,
|
| - Representation::Smi(),
|
| - INSERT_TRANSITION).ToHandleChecked();
|
| - CHECK(!map->owns_descriptors());
|
| -
|
| - return Map::CopyAsElementsKind(map, DICTIONARY_ELEMENTS,
|
| - INSERT_TRANSITION);
|
| - }
|
| - // TODO(ishell): remove once IS_PROTO_TRANS_ISSUE_FIXED is removed.
|
| - bool generalizes_representations() const { return false; }
|
| - bool is_non_equevalent_transition() const { return false; }
|
| - };
|
| - TestConfig config;
|
| - TestGeneralizeRepresentationWithSpecialTransition(
|
| - config, Representation::Smi(), any_type, Representation::HeapObject(),
|
| - value_type, Representation::Tagged(), any_type);
|
| -}
|
| -
|
| -
|
| -TEST(ForObservedTransitionFromMapOwningDescriptor) {
|
| - CcTest::InitializeVM();
|
| - v8::HandleScope scope(CcTest::isolate());
|
| - Isolate* isolate = CcTest::i_isolate();
|
| - Handle<HeapType> any_type = HeapType::Any(isolate);
|
| - Handle<HeapType> value_type =
|
| - HeapType::Class(Map::Create(isolate, 0), isolate);
|
| -
|
| - struct TestConfig {
|
| - Handle<Map> Transition(Handle<Map> map) {
|
| - return Map::CopyForObserved(map);
|
| - }
|
| - // TODO(ishell): remove once IS_PROTO_TRANS_ISSUE_FIXED is removed.
|
| - bool generalizes_representations() const { return false; }
|
| - bool is_non_equevalent_transition() const { return true; }
|
| - };
|
| - TestConfig config;
|
| - TestGeneralizeRepresentationWithSpecialTransition(
|
| - config, Representation::Smi(), any_type, Representation::HeapObject(),
|
| - value_type, Representation::Tagged(), any_type);
|
| -}
|
| -
|
| -
|
| -TEST(ForObservedTransitionFromMapNotOwningDescriptor) {
|
| - CcTest::InitializeVM();
|
| - v8::HandleScope scope(CcTest::isolate());
|
| - Isolate* isolate = CcTest::i_isolate();
|
| - Handle<HeapType> any_type = HeapType::Any(isolate);
|
| - Handle<HeapType> value_type =
|
| - HeapType::Class(Map::Create(isolate, 0), isolate);
|
| -
|
| - struct TestConfig {
|
| - Handle<Map> Transition(Handle<Map> map) {
|
| - Isolate* isolate = CcTest::i_isolate();
|
| - Handle<HeapType> any_type = HeapType::Any(isolate);
|
| -
|
| - // Add one more transition to |map| in order to prevent descriptors
|
| - // ownership.
|
| - CHECK(map->owns_descriptors());
|
| - Map::CopyWithField(map, MakeString("foo"), any_type, NONE,
|
| - Representation::Smi(),
|
| - INSERT_TRANSITION).ToHandleChecked();
|
| - CHECK(!map->owns_descriptors());
|
| -
|
| - return Map::CopyForObserved(map);
|
| - }
|
| - // TODO(ishell): remove once IS_PROTO_TRANS_ISSUE_FIXED is removed.
|
| - bool generalizes_representations() const { return false; }
|
| - bool is_non_equevalent_transition() const { return true; }
|
| - };
|
| - TestConfig config;
|
| - TestGeneralizeRepresentationWithSpecialTransition(
|
| - config, Representation::Smi(), any_type, Representation::HeapObject(),
|
| - value_type, Representation::Tagged(), any_type);
|
| -}
|
| -
|
| -
|
| -TEST(PrototypeTransitionFromMapOwningDescriptor) {
|
| - CcTest::InitializeVM();
|
| - v8::HandleScope scope(CcTest::isolate());
|
| - Isolate* isolate = CcTest::i_isolate();
|
| -
|
| - Handle<HeapType> any_type = HeapType::Any(isolate);
|
| - Handle<HeapType> value_type =
|
| - HeapType::Class(Map::Create(isolate, 0), isolate);
|
| -
|
| - struct TestConfig {
|
| - Handle<JSObject> prototype_;
|
| -
|
| - TestConfig() {
|
| - Isolate* isolate = CcTest::i_isolate();
|
| - Factory* factory = isolate->factory();
|
| - prototype_ = factory->NewJSObjectFromMap(Map::Create(isolate, 0));
|
| - }
|
| -
|
| - Handle<Map> Transition(Handle<Map> map) {
|
| - return Map::TransitionToPrototype(map, prototype_, REGULAR_PROTOTYPE);
|
| - }
|
| - // TODO(ishell): remove once IS_PROTO_TRANS_ISSUE_FIXED is removed.
|
| - bool generalizes_representations() const {
|
| - return !IS_PROTO_TRANS_ISSUE_FIXED;
|
| - }
|
| - bool is_non_equevalent_transition() const { return true; }
|
| - };
|
| - TestConfig config;
|
| - TestGeneralizeRepresentationWithSpecialTransition(
|
| - config, Representation::Smi(), any_type, Representation::HeapObject(),
|
| - value_type, Representation::Tagged(), any_type);
|
| -}
|
| -
|
| -
|
| -TEST(PrototypeTransitionFromMapNotOwningDescriptor) {
|
| - CcTest::InitializeVM();
|
| - v8::HandleScope scope(CcTest::isolate());
|
| - Isolate* isolate = CcTest::i_isolate();
|
| -
|
| - Handle<HeapType> any_type = HeapType::Any(isolate);
|
| - Handle<HeapType> value_type =
|
| - HeapType::Class(Map::Create(isolate, 0), isolate);
|
| -
|
| - struct TestConfig {
|
| - Handle<JSObject> prototype_;
|
| -
|
| - TestConfig() {
|
| - Isolate* isolate = CcTest::i_isolate();
|
| - Factory* factory = isolate->factory();
|
| - prototype_ = factory->NewJSObjectFromMap(Map::Create(isolate, 0));
|
| - }
|
| -
|
| - Handle<Map> Transition(Handle<Map> map) {
|
| - Isolate* isolate = CcTest::i_isolate();
|
| - Handle<HeapType> any_type = HeapType::Any(isolate);
|
| -
|
| - // Add one more transition to |map| in order to prevent descriptors
|
| - // ownership.
|
| - CHECK(map->owns_descriptors());
|
| - Map::CopyWithField(map, MakeString("foo"), any_type, NONE,
|
| - Representation::Smi(),
|
| - INSERT_TRANSITION).ToHandleChecked();
|
| - CHECK(!map->owns_descriptors());
|
| -
|
| - return Map::TransitionToPrototype(map, prototype_, REGULAR_PROTOTYPE);
|
| - }
|
| - // TODO(ishell): remove once IS_PROTO_TRANS_ISSUE_FIXED is removed.
|
| - bool generalizes_representations() const {
|
| - return !IS_PROTO_TRANS_ISSUE_FIXED;
|
| - }
|
| - bool is_non_equevalent_transition() const { return true; }
|
| - };
|
| - TestConfig config;
|
| - TestGeneralizeRepresentationWithSpecialTransition(
|
| - config, Representation::Smi(), any_type, Representation::HeapObject(),
|
| - value_type, Representation::Tagged(), any_type);
|
| -}
|
| -
|
| -
|
| -////////////////////////////////////////////////////////////////////////////////
|
| -// A set of tests for higher level transitioning mechanics.
|
| -//
|
| -
|
| -struct TransitionToDataFieldOperator {
|
| - Representation representation_;
|
| - PropertyAttributes attributes_;
|
| - Handle<HeapType> heap_type_;
|
| - Handle<Object> value_;
|
| -
|
| - TransitionToDataFieldOperator(Representation representation,
|
| - Handle<HeapType> heap_type,
|
| - Handle<Object> value,
|
| - PropertyAttributes attributes = NONE)
|
| - : representation_(representation),
|
| - attributes_(attributes),
|
| - heap_type_(heap_type),
|
| - value_(value) {}
|
| -
|
| - Handle<Map> DoTransition(Expectations& expectations, Handle<Map> map) {
|
| - return expectations.TransitionToDataField(map, attributes_, representation_,
|
| - heap_type_, value_);
|
| - }
|
| -};
|
| -
|
| -
|
| -struct TransitionToDataConstantOperator {
|
| - PropertyAttributes attributes_;
|
| - Handle<JSFunction> value_;
|
| -
|
| - TransitionToDataConstantOperator(Handle<JSFunction> value,
|
| - PropertyAttributes attributes = NONE)
|
| - : attributes_(attributes), value_(value) {}
|
| -
|
| - Handle<Map> DoTransition(Expectations& expectations, Handle<Map> map) {
|
| - return expectations.TransitionToDataConstant(map, attributes_, value_);
|
| - }
|
| -};
|
| -
|
| -
|
| -struct TransitionToAccessorConstantOperator {
|
| - PropertyAttributes attributes_;
|
| - Handle<AccessorPair> pair_;
|
| -
|
| - TransitionToAccessorConstantOperator(Handle<AccessorPair> pair,
|
| - PropertyAttributes attributes = NONE)
|
| - : attributes_(attributes), pair_(pair) {}
|
| -
|
| - Handle<Map> DoTransition(Expectations& expectations, Handle<Map> map) {
|
| - return expectations.TransitionToAccessorConstant(map, attributes_, pair_);
|
| - }
|
| -};
|
| -
|
| -
|
| -struct ReconfigureAsDataPropertyOperator {
|
| - int descriptor_;
|
| - Representation representation_;
|
| - PropertyAttributes attributes_;
|
| - Handle<HeapType> heap_type_;
|
| -
|
| - ReconfigureAsDataPropertyOperator(int descriptor,
|
| - Representation representation,
|
| - Handle<HeapType> heap_type,
|
| - PropertyAttributes attributes = NONE)
|
| - : descriptor_(descriptor),
|
| - representation_(representation),
|
| - attributes_(attributes),
|
| - heap_type_(heap_type) {}
|
| -
|
| - Handle<Map> DoTransition(Expectations& expectations, Handle<Map> map) {
|
| - expectations.SetDataField(descriptor_, representation_, heap_type_);
|
| - return Map::ReconfigureExistingProperty(map, descriptor_, kData,
|
| - attributes_);
|
| - }
|
| -};
|
| -
|
| -
|
| -struct ReconfigureAsAccessorPropertyOperator {
|
| - int descriptor_;
|
| - PropertyAttributes attributes_;
|
| -
|
| - ReconfigureAsAccessorPropertyOperator(int descriptor,
|
| - PropertyAttributes attributes = NONE)
|
| - : descriptor_(descriptor), attributes_(attributes) {}
|
| -
|
| - Handle<Map> DoTransition(Expectations& expectations, Handle<Map> map) {
|
| - expectations.SetAccessorField(descriptor_);
|
| - return Map::ReconfigureExistingProperty(map, descriptor_, kAccessor,
|
| - attributes_);
|
| - }
|
| -};
|
| -
|
| -
|
| -// Checks that representation/field type generalization happened.
|
| -struct FieldGeneralizationChecker {
|
| - int descriptor_;
|
| - Representation representation_;
|
| - PropertyAttributes attributes_;
|
| - Handle<HeapType> heap_type_;
|
| -
|
| - FieldGeneralizationChecker(int descriptor, Representation representation,
|
| - Handle<HeapType> heap_type,
|
| - PropertyAttributes attributes = NONE)
|
| - : descriptor_(descriptor),
|
| - representation_(representation),
|
| - attributes_(attributes),
|
| - heap_type_(heap_type) {}
|
| -
|
| - void Check(Expectations& expectations2, Handle<Map> map1, Handle<Map> map2) {
|
| - CHECK(!map2->is_deprecated());
|
| -
|
| - CHECK(map1->is_deprecated());
|
| - CHECK_NE(*map1, *map2);
|
| - Handle<Map> updated_map = Map::Update(map1);
|
| - CHECK_EQ(*map2, *updated_map);
|
| -
|
| - expectations2.SetDataField(descriptor_, attributes_, representation_,
|
| - heap_type_);
|
| - CHECK(expectations2.Check(*map2));
|
| - }
|
| -};
|
| -
|
| -
|
| -// Checks that existing transition was taken as is.
|
| -struct SameMapChecker {
|
| - void Check(Expectations& expectations, Handle<Map> map1, Handle<Map> map2) {
|
| - CHECK(!map2->is_deprecated());
|
| - CHECK_EQ(*map1, *map2);
|
| - CHECK(expectations.Check(*map2));
|
| - }
|
| -};
|
| -
|
| -
|
| -// Checks that both |map1| and |map2| should stays non-deprecated, this is
|
| -// the case when property kind is change.
|
| -struct PropertyKindReconfigurationChecker {
|
| - void Check(Expectations& expectations, Handle<Map> map1, Handle<Map> map2) {
|
| - CHECK(!map1->is_deprecated());
|
| - CHECK(!map2->is_deprecated());
|
| - CHECK_NE(*map1, *map2);
|
| - CHECK(expectations.Check(*map2));
|
| - }
|
| -};
|
| -
|
| -
|
| -// This test transitions to various property types under different
|
| -// circumstances.
|
| -// Plan:
|
| -// 1) create a |map| with p0..p3 properties.
|
| -// 2) create |map1| by adding "p4" to |map0|.
|
| -// 3) create |map2| by transition to "p4" from |map0|.
|
| -//
|
| -// + - p4B: |map2|
|
| -// |
|
| -// {} - p0 - p1 - pA - p3: |map|
|
| -// |
|
| -// + - p4A: |map1|
|
| -//
|
| -// where "p4A" and "p4B" differ only in the attributes.
|
| -//
|
| -template <typename TransitionOp1, typename TransitionOp2, typename Checker>
|
| -static void TestTransitionTo(TransitionOp1& transition_op1,
|
| - TransitionOp2& transition_op2, Checker& checker) {
|
| - Isolate* isolate = CcTest::i_isolate();
|
| - Handle<HeapType> any_type = HeapType::Any(isolate);
|
| -
|
| - Expectations expectations(isolate);
|
| -
|
| - // Create a map, add required properties to it and initialize expectations.
|
| - Handle<Map> initial_map = Map::Create(isolate, 0);
|
| - Handle<Map> map = initial_map;
|
| - for (int i = 0; i < kPropCount - 1; i++) {
|
| - map = expectations.AddDataField(map, NONE, Representation::Smi(), any_type);
|
| - }
|
| - CHECK(expectations.Check(*map));
|
| -
|
| - Expectations expectations1 = expectations;
|
| - Handle<Map> map1 = transition_op1.DoTransition(expectations1, map);
|
| - CHECK(expectations1.Check(*map1));
|
| -
|
| - Expectations expectations2 = expectations;
|
| - Handle<Map> map2 = transition_op2.DoTransition(expectations2, map);
|
| -
|
| - // Let the test customization do the check.
|
| - checker.Check(expectations2, map1, map2);
|
| -}
|
| -
|
| -
|
| -TEST(TransitionDataFieldToDataField) {
|
| - CcTest::InitializeVM();
|
| - v8::HandleScope scope(CcTest::isolate());
|
| - Isolate* isolate = CcTest::i_isolate();
|
| - Handle<HeapType> any_type = HeapType::Any(isolate);
|
| -
|
| - Handle<Object> value1 = handle(Smi::FromInt(0), isolate);
|
| - TransitionToDataFieldOperator transition_op1(Representation::Smi(), any_type,
|
| - value1);
|
| -
|
| - Handle<Object> value2 = isolate->factory()->NewHeapNumber(0);
|
| - TransitionToDataFieldOperator transition_op2(Representation::Double(),
|
| - any_type, value2);
|
| -
|
| - FieldGeneralizationChecker checker(kPropCount - 1, Representation::Double(),
|
| - any_type);
|
| - TestTransitionTo(transition_op1, transition_op2, checker);
|
| -}
|
| -
|
| -
|
| -TEST(TransitionDataConstantToSameDataConstant) {
|
| - CcTest::InitializeVM();
|
| - v8::HandleScope scope(CcTest::isolate());
|
| - Isolate* isolate = CcTest::i_isolate();
|
| - Factory* factory = isolate->factory();
|
| -
|
| - Handle<JSFunction> js_func = factory->NewFunction(factory->empty_string());
|
| - TransitionToDataConstantOperator transition_op(js_func);
|
| -
|
| - SameMapChecker checker;
|
| - TestTransitionTo(transition_op, transition_op, checker);
|
| -}
|
| -
|
| -
|
| -TEST(TransitionDataConstantToAnotherDataConstant) {
|
| - CcTest::InitializeVM();
|
| - v8::HandleScope scope(CcTest::isolate());
|
| - Isolate* isolate = CcTest::i_isolate();
|
| - Factory* factory = isolate->factory();
|
| - Handle<HeapType> any_type = HeapType::Any(isolate);
|
| -
|
| - Handle<JSFunction> js_func1 = factory->NewFunction(factory->empty_string());
|
| - TransitionToDataConstantOperator transition_op1(js_func1);
|
| -
|
| - Handle<JSFunction> js_func2 = factory->NewFunction(factory->empty_string());
|
| - TransitionToDataConstantOperator transition_op2(js_func2);
|
| -
|
| - FieldGeneralizationChecker checker(kPropCount - 1,
|
| - Representation::HeapObject(), any_type);
|
| - TestTransitionTo(transition_op1, transition_op2, checker);
|
| -}
|
| -
|
| -
|
| -TEST(TransitionDataConstantToDataField) {
|
| - CcTest::InitializeVM();
|
| - v8::HandleScope scope(CcTest::isolate());
|
| - Isolate* isolate = CcTest::i_isolate();
|
| - Factory* factory = isolate->factory();
|
| - Handle<HeapType> any_type = HeapType::Any(isolate);
|
| -
|
| - Handle<JSFunction> js_func1 = factory->NewFunction(factory->empty_string());
|
| - TransitionToDataConstantOperator transition_op1(js_func1);
|
| -
|
| - Handle<Object> value2 = isolate->factory()->NewHeapNumber(0);
|
| - TransitionToDataFieldOperator transition_op2(Representation::Double(),
|
| - any_type, value2);
|
| -
|
| - FieldGeneralizationChecker checker(kPropCount - 1, Representation::Tagged(),
|
| - any_type);
|
| - TestTransitionTo(transition_op1, transition_op2, checker);
|
| -}
|
| -
|
| -
|
| -TEST(TransitionAccessorConstantToSameAccessorConstant) {
|
| - CcTest::InitializeVM();
|
| - v8::HandleScope scope(CcTest::isolate());
|
| -
|
| - Handle<AccessorPair> pair = CreateAccessorPair(true, true);
|
| - TransitionToAccessorConstantOperator transition_op(pair);
|
| -
|
| - SameMapChecker checker;
|
| - TestTransitionTo(transition_op, transition_op, checker);
|
| -}
|
| -
|
| -
|
| -// TODO(ishell): add this test once IS_ACCESSOR_FIELD_SUPPORTED is supported.
|
| -// TEST(TransitionAccessorConstantToAnotherAccessorConstant)
|
|
|