[map] Make field representation updates work with elements kind transitions.
Generalize the existing work-around in the method `Map::GeneralizeIfCanHaveTransitionableFastElementsKind()` to also go to the most general field representation (in addition to going to the most field type) for objects with transitionable fast elements kinds. That means that we essentially disable field representation tracking for arrays, arguments objects and value wrappers (for which the field type tracking is already disabled). Drive-by-fix: Remove the `constness` parameter to the above mentioned helper method. And fix the printing of the descriptor expectations to properly print the field type. Change-Id: I1bba9415f4bdd2c916f9d105d9120c7071d2c498 Cq-Include-Trybots: luci.chromium.try:linux-rel,win7-rel Doc: http://bit.ly/v8-in-place-field-representation-changes Bug: v8:8749, v8:8865, v8:9114, chromium:959645, chromium:952682 Reviewed-on: https://chromium-review.googlesource.com/c/v8/v8/+/1598756 Commit-Queue: Benedikt Meurer <bmeurer@chromium.org> Reviewed-by: Igor Sheludko <ishell@chromium.org> Cr-Commit-Position: refs/heads/master@{#61284}
This commit is contained in:
parent
25af5db375
commit
6564c6dfc4
@ -41,7 +41,7 @@ class FieldType : public Object {
|
|||||||
bool NowIs(FieldType other) const;
|
bool NowIs(FieldType other) const;
|
||||||
bool NowIs(Handle<FieldType> other) const;
|
bool NowIs(Handle<FieldType> other) const;
|
||||||
|
|
||||||
void PrintTo(std::ostream& os) const;
|
V8_EXPORT_PRIVATE void PrintTo(std::ostream& os) const;
|
||||||
|
|
||||||
FieldType* operator->() { return this; }
|
FieldType* operator->() { return this; }
|
||||||
const FieldType* operator->() const { return this; }
|
const FieldType* operator->() const { return this; }
|
||||||
|
@ -147,8 +147,8 @@ Handle<Map> MapUpdater::ReconfigureToDataField(int descriptor,
|
|||||||
}
|
}
|
||||||
|
|
||||||
Map::GeneralizeIfCanHaveTransitionableFastElementsKind(
|
Map::GeneralizeIfCanHaveTransitionableFastElementsKind(
|
||||||
isolate_, old_map_->instance_type(), &new_constness_,
|
isolate_, old_map_->instance_type(), &new_representation_,
|
||||||
&new_representation_, &new_field_type_);
|
&new_field_type_);
|
||||||
|
|
||||||
if (TryReconfigureToDataFieldInplace() == kEnd) return result_map_;
|
if (TryReconfigureToDataFieldInplace() == kEnd) return result_map_;
|
||||||
if (FindRootMap() == kEnd) return result_map_;
|
if (FindRootMap() == kEnd) return result_map_;
|
||||||
@ -586,8 +586,7 @@ Handle<DescriptorArray> MapUpdater::BuildDescriptorArray() {
|
|||||||
target_field_type, isolate_);
|
target_field_type, isolate_);
|
||||||
|
|
||||||
Map::GeneralizeIfCanHaveTransitionableFastElementsKind(
|
Map::GeneralizeIfCanHaveTransitionableFastElementsKind(
|
||||||
isolate_, instance_type, &next_constness, &next_representation,
|
isolate_, instance_type, &next_representation, &next_field_type);
|
||||||
&next_field_type);
|
|
||||||
|
|
||||||
MaybeObjectHandle wrapped_type(
|
MaybeObjectHandle wrapped_type(
|
||||||
Map::WrapFieldType(isolate_, next_field_type));
|
Map::WrapFieldType(isolate_, next_field_type));
|
||||||
|
@ -122,19 +122,16 @@ bool Map::CanHaveFastTransitionableElementsKind() const {
|
|||||||
|
|
||||||
// static
|
// static
|
||||||
void Map::GeneralizeIfCanHaveTransitionableFastElementsKind(
|
void Map::GeneralizeIfCanHaveTransitionableFastElementsKind(
|
||||||
Isolate* isolate, InstanceType instance_type, PropertyConstness* constness,
|
Isolate* isolate, InstanceType instance_type,
|
||||||
Representation* representation, Handle<FieldType>* field_type) {
|
Representation* representation, Handle<FieldType>* field_type) {
|
||||||
if (CanHaveFastTransitionableElementsKind(instance_type)) {
|
if (CanHaveFastTransitionableElementsKind(instance_type)) {
|
||||||
// We don't support propagation of field generalization through elements
|
// We don't support propagation of field generalization through elements
|
||||||
// kind transitions because they are inserted into the transition tree
|
// kind transitions because they are inserted into the transition tree
|
||||||
// before field transitions. In order to avoid complexity of handling
|
// before field transitions. In order to avoid complexity of handling
|
||||||
// such a case we ensure that all maps with transitionable elements kinds
|
// such a case we ensure that all maps with transitionable elements kinds
|
||||||
// have the most general field type.
|
// have the most general field representation and type.
|
||||||
if (representation->IsHeapObject()) {
|
|
||||||
// The field type is either already Any or should become Any if it was
|
|
||||||
// something else.
|
|
||||||
*field_type = FieldType::Any(isolate);
|
*field_type = FieldType::Any(isolate);
|
||||||
}
|
*representation = Representation::Tagged();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -447,7 +447,7 @@ MaybeHandle<Map> Map::CopyWithField(Isolate* isolate, Handle<Map> map,
|
|||||||
type = FieldType::Any(isolate);
|
type = FieldType::Any(isolate);
|
||||||
} else {
|
} else {
|
||||||
Map::GeneralizeIfCanHaveTransitionableFastElementsKind(
|
Map::GeneralizeIfCanHaveTransitionableFastElementsKind(
|
||||||
isolate, map->instance_type(), &constness, &representation, &type);
|
isolate, map->instance_type(), &representation, &type);
|
||||||
}
|
}
|
||||||
|
|
||||||
MaybeObjectHandle wrapped_type = WrapFieldType(isolate, type);
|
MaybeObjectHandle wrapped_type = WrapFieldType(isolate, type);
|
||||||
|
@ -518,17 +518,16 @@ class Map : public HeapObject {
|
|||||||
static inline bool IsMostGeneralFieldType(Representation representation,
|
static inline bool IsMostGeneralFieldType(Representation representation,
|
||||||
FieldType field_type);
|
FieldType field_type);
|
||||||
|
|
||||||
// Generalizes constness, representation and field_type if objects with given
|
// Generalizes representation and field_type if objects with given
|
||||||
// instance type can have fast elements that can be transitioned by stubs or
|
// instance type can have fast elements that can be transitioned by
|
||||||
// optimized code to more general elements kind.
|
// stubs or optimized code to more general elements kind.
|
||||||
// This generalization is necessary in order to ensure that elements kind
|
// This generalization is necessary in order to ensure that elements kind
|
||||||
// transitions performed by stubs / optimized code don't silently transition
|
// transitions performed by stubs / optimized code don't silently transition
|
||||||
// PropertyConstness::kMutable fields back to VariableMode::kConst state or
|
// fields with representation "Tagged" back to "Smi" or "HeapObject" or
|
||||||
// fields with HeapObject representation and "Any" type back to "Class" type.
|
// fields with HeapObject representation and "Any" type back to "Class" type.
|
||||||
static inline void GeneralizeIfCanHaveTransitionableFastElementsKind(
|
static inline void GeneralizeIfCanHaveTransitionableFastElementsKind(
|
||||||
Isolate* isolate, InstanceType instance_type,
|
Isolate* isolate, InstanceType instance_type,
|
||||||
PropertyConstness* constness, Representation* representation,
|
Representation* representation, Handle<FieldType>* field_type);
|
||||||
Handle<FieldType>* field_type);
|
|
||||||
|
|
||||||
V8_EXPORT_PRIVATE static Handle<Map> ReconfigureProperty(
|
V8_EXPORT_PRIVATE static Handle<Map> ReconfigureProperty(
|
||||||
Isolate* isolate, Handle<Map> map, int modify_index,
|
Isolate* isolate, Handle<Map> map, int modify_index,
|
||||||
|
@ -123,6 +123,7 @@ class Expectations {
|
|||||||
// Maps with transitionable elements kinds must have the most general
|
// Maps with transitionable elements kinds must have the most general
|
||||||
// field type.
|
// field type.
|
||||||
value = FieldType::Any(isolate_);
|
value = FieldType::Any(isolate_);
|
||||||
|
representation = Representation::Tagged();
|
||||||
}
|
}
|
||||||
constnesses_[index] = constness;
|
constnesses_[index] = constness;
|
||||||
attributes_[index] = attributes;
|
attributes_[index] = attributes;
|
||||||
@ -138,7 +139,7 @@ class Expectations {
|
|||||||
os << "Descriptor @ ";
|
os << "Descriptor @ ";
|
||||||
|
|
||||||
if (kinds_[i] == kData) {
|
if (kinds_[i] == kData) {
|
||||||
os << Brief(*values_[i]);
|
Handle<FieldType>::cast(values_[i])->PrintTo(os);
|
||||||
} else {
|
} else {
|
||||||
// kAccessor
|
// kAccessor
|
||||||
os << "(get: " << Brief(*values_[i])
|
os << "(get: " << Brief(*values_[i])
|
||||||
@ -1727,104 +1728,6 @@ TEST(ReconfigureDataFieldAttribute_AccConstantToDataFieldAfterTargetMap) {
|
|||||||
|
|
||||||
namespace {
|
namespace {
|
||||||
|
|
||||||
// This test ensures that field generalization is correctly propagated from one
|
|
||||||
// branch of transition tree (|map2) to another (|map|).
|
|
||||||
//
|
|
||||||
// + - p0 - p1 - p2A - p3 - p4: |map|
|
|
||||||
// |
|
|
||||||
// ek
|
|
||||||
// |
|
|
||||||
// {} - p0 - p1 - p2B - p3 - p4: |map2|
|
|
||||||
//
|
|
||||||
// where "p2A" and "p2B" differ only in the representation/field type.
|
|
||||||
//
|
|
||||||
static void TestReconfigureElementsKind_GeneralizeField(
|
|
||||||
const CRFTData& from, const CRFTData& to, const CRFTData& expected) {
|
|
||||||
Isolate* isolate = CcTest::i_isolate();
|
|
||||||
|
|
||||||
Expectations expectations(isolate, PACKED_SMI_ELEMENTS);
|
|
||||||
|
|
||||||
// Create a map, add required properties to it and initialize expectations.
|
|
||||||
Handle<Map> initial_map = Map::Create(isolate, 0);
|
|
||||||
initial_map->set_instance_type(JS_ARRAY_TYPE);
|
|
||||||
initial_map->set_elements_kind(PACKED_SMI_ELEMENTS);
|
|
||||||
|
|
||||||
Handle<Map> map = initial_map;
|
|
||||||
map = expectations.AsElementsKind(map, PACKED_ELEMENTS);
|
|
||||||
for (int i = 0; i < kPropCount; i++) {
|
|
||||||
map = expectations.AddDataField(map, NONE, from.constness,
|
|
||||||
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 |kDiffProp|
|
|
||||||
// has different representatio/field type), initialize expectations.
|
|
||||||
const int kDiffProp = kPropCount / 2;
|
|
||||||
Expectations expectations2(isolate, PACKED_SMI_ELEMENTS);
|
|
||||||
|
|
||||||
Handle<Map> map2 = initial_map;
|
|
||||||
for (int i = 0; i < kPropCount; i++) {
|
|
||||||
if (i == kDiffProp) {
|
|
||||||
map2 = expectations2.AddDataField(map2, NONE, to.constness,
|
|
||||||
to.representation, to.type);
|
|
||||||
} else {
|
|
||||||
map2 = expectations2.AddDataField(map2, NONE, from.constness,
|
|
||||||
from.representation, from.type);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
CHECK(!map2->is_deprecated());
|
|
||||||
CHECK(map2->is_stable());
|
|
||||||
CHECK(expectations2.Check(*map2));
|
|
||||||
|
|
||||||
// Create dummy optimized code object to test correct dependencies
|
|
||||||
// on the field owner.
|
|
||||||
Handle<Code> code = CreateDummyOptimizedCode(isolate);
|
|
||||||
Handle<Map> field_owner(map->FindFieldOwner(isolate, kDiffProp), isolate);
|
|
||||||
DependentCode::InstallDependency(isolate, MaybeObjectHandle::Weak(code),
|
|
||||||
field_owner,
|
|
||||||
DependentCode::kFieldOwnerGroup);
|
|
||||||
CHECK(!code->marked_for_deoptimization());
|
|
||||||
|
|
||||||
// Reconfigure elements kinds of |map2|, which should generalize
|
|
||||||
// representations in |map|.
|
|
||||||
Handle<Map> new_map =
|
|
||||||
Map::ReconfigureElementsKind(isolate, map2, PACKED_ELEMENTS);
|
|
||||||
|
|
||||||
// |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.
|
|
||||||
expectations.SetDataField(kDiffProp, expected.constness,
|
|
||||||
expected.representation, expected.type);
|
|
||||||
|
|
||||||
CHECK(map->is_deprecated());
|
|
||||||
CHECK(!code->marked_for_deoptimization());
|
|
||||||
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(isolate, map);
|
|
||||||
CHECK_EQ(*new_map, *updated_map);
|
|
||||||
CheckMigrationTarget(isolate, *map, *updated_map);
|
|
||||||
|
|
||||||
// Ensure Map::FindElementsKindTransitionedMap() is able to find the
|
|
||||||
// transitioned map.
|
|
||||||
{
|
|
||||||
MapHandles map_list;
|
|
||||||
map_list.push_back(updated_map);
|
|
||||||
Map transitioned_map =
|
|
||||||
map2->FindElementsKindTransitionedMap(isolate, map_list);
|
|
||||||
CHECK_EQ(*updated_map, transitioned_map);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// This test ensures that trivial field generalization (from HeapObject to
|
// This test ensures that trivial field generalization (from HeapObject to
|
||||||
// HeapObject) is correctly propagated from one branch of transition tree
|
// HeapObject) is correctly propagated from one branch of transition tree
|
||||||
// (|map2|) to another (|map|).
|
// (|map2|) to another (|map|).
|
||||||
@ -1934,22 +1837,22 @@ TEST(ReconfigureElementsKind_GeneralizeSmiFieldToDouble) {
|
|||||||
|
|
||||||
Handle<FieldType> any_type = FieldType::Any(isolate);
|
Handle<FieldType> any_type = FieldType::Any(isolate);
|
||||||
|
|
||||||
TestReconfigureElementsKind_GeneralizeField(
|
TestReconfigureElementsKind_GeneralizeFieldTrivial(
|
||||||
{PropertyConstness::kConst, Representation::Smi(), any_type},
|
{PropertyConstness::kConst, Representation::Smi(), any_type},
|
||||||
{PropertyConstness::kConst, Representation::Double(), any_type},
|
{PropertyConstness::kConst, Representation::Double(), any_type},
|
||||||
{PropertyConstness::kConst, Representation::Double(), any_type});
|
{PropertyConstness::kConst, Representation::Double(), any_type});
|
||||||
|
|
||||||
TestReconfigureElementsKind_GeneralizeField(
|
TestReconfigureElementsKind_GeneralizeFieldTrivial(
|
||||||
{PropertyConstness::kConst, Representation::Smi(), any_type},
|
{PropertyConstness::kConst, Representation::Smi(), any_type},
|
||||||
{PropertyConstness::kMutable, Representation::Double(), any_type},
|
{PropertyConstness::kMutable, Representation::Double(), any_type},
|
||||||
{PropertyConstness::kMutable, Representation::Double(), any_type});
|
{PropertyConstness::kMutable, Representation::Double(), any_type});
|
||||||
|
|
||||||
TestReconfigureElementsKind_GeneralizeField(
|
TestReconfigureElementsKind_GeneralizeFieldTrivial(
|
||||||
{PropertyConstness::kMutable, Representation::Smi(), any_type},
|
{PropertyConstness::kMutable, Representation::Smi(), any_type},
|
||||||
{PropertyConstness::kConst, Representation::Double(), any_type},
|
{PropertyConstness::kConst, Representation::Double(), any_type},
|
||||||
{PropertyConstness::kMutable, Representation::Double(), any_type});
|
{PropertyConstness::kMutable, Representation::Double(), any_type});
|
||||||
|
|
||||||
TestReconfigureElementsKind_GeneralizeField(
|
TestReconfigureElementsKind_GeneralizeFieldTrivial(
|
||||||
{PropertyConstness::kMutable, Representation::Smi(), any_type},
|
{PropertyConstness::kMutable, Representation::Smi(), any_type},
|
||||||
{PropertyConstness::kMutable, Representation::Double(), any_type},
|
{PropertyConstness::kMutable, Representation::Double(), any_type},
|
||||||
{PropertyConstness::kMutable, Representation::Double(), any_type});
|
{PropertyConstness::kMutable, Representation::Double(), any_type});
|
||||||
@ -1964,22 +1867,22 @@ TEST(ReconfigureElementsKind_GeneralizeSmiFieldToTagged) {
|
|||||||
Handle<FieldType> value_type =
|
Handle<FieldType> value_type =
|
||||||
FieldType::Class(Map::Create(isolate, 0), isolate);
|
FieldType::Class(Map::Create(isolate, 0), isolate);
|
||||||
|
|
||||||
TestReconfigureElementsKind_GeneralizeField(
|
TestReconfigureElementsKind_GeneralizeFieldTrivial(
|
||||||
{PropertyConstness::kConst, Representation::Smi(), any_type},
|
{PropertyConstness::kConst, Representation::Smi(), any_type},
|
||||||
{PropertyConstness::kConst, Representation::HeapObject(), value_type},
|
{PropertyConstness::kConst, Representation::HeapObject(), value_type},
|
||||||
{PropertyConstness::kConst, Representation::Tagged(), any_type});
|
{PropertyConstness::kConst, Representation::Tagged(), any_type});
|
||||||
|
|
||||||
TestReconfigureElementsKind_GeneralizeField(
|
TestReconfigureElementsKind_GeneralizeFieldTrivial(
|
||||||
{PropertyConstness::kConst, Representation::Smi(), any_type},
|
{PropertyConstness::kConst, Representation::Smi(), any_type},
|
||||||
{PropertyConstness::kMutable, Representation::HeapObject(), value_type},
|
{PropertyConstness::kMutable, Representation::HeapObject(), value_type},
|
||||||
{PropertyConstness::kMutable, Representation::Tagged(), any_type});
|
{PropertyConstness::kMutable, Representation::Tagged(), any_type});
|
||||||
|
|
||||||
TestReconfigureElementsKind_GeneralizeField(
|
TestReconfigureElementsKind_GeneralizeFieldTrivial(
|
||||||
{PropertyConstness::kMutable, Representation::Smi(), any_type},
|
{PropertyConstness::kMutable, Representation::Smi(), any_type},
|
||||||
{PropertyConstness::kConst, Representation::HeapObject(), value_type},
|
{PropertyConstness::kConst, Representation::HeapObject(), value_type},
|
||||||
{PropertyConstness::kMutable, Representation::Tagged(), any_type});
|
{PropertyConstness::kMutable, Representation::Tagged(), any_type});
|
||||||
|
|
||||||
TestReconfigureElementsKind_GeneralizeField(
|
TestReconfigureElementsKind_GeneralizeFieldTrivial(
|
||||||
{PropertyConstness::kMutable, Representation::Smi(), any_type},
|
{PropertyConstness::kMutable, Representation::Smi(), any_type},
|
||||||
{PropertyConstness::kMutable, Representation::HeapObject(), value_type},
|
{PropertyConstness::kMutable, Representation::HeapObject(), value_type},
|
||||||
{PropertyConstness::kMutable, Representation::Tagged(), any_type});
|
{PropertyConstness::kMutable, Representation::Tagged(), any_type});
|
||||||
@ -1994,22 +1897,22 @@ TEST(ReconfigureElementsKind_GeneralizeDoubleFieldToTagged) {
|
|||||||
Handle<FieldType> value_type =
|
Handle<FieldType> value_type =
|
||||||
FieldType::Class(Map::Create(isolate, 0), isolate);
|
FieldType::Class(Map::Create(isolate, 0), isolate);
|
||||||
|
|
||||||
TestReconfigureElementsKind_GeneralizeField(
|
TestReconfigureElementsKind_GeneralizeFieldTrivial(
|
||||||
{PropertyConstness::kConst, Representation::Double(), any_type},
|
{PropertyConstness::kConst, Representation::Double(), any_type},
|
||||||
{PropertyConstness::kConst, Representation::HeapObject(), value_type},
|
{PropertyConstness::kConst, Representation::HeapObject(), value_type},
|
||||||
{PropertyConstness::kConst, Representation::Tagged(), any_type});
|
{PropertyConstness::kConst, Representation::Tagged(), any_type});
|
||||||
|
|
||||||
TestReconfigureElementsKind_GeneralizeField(
|
TestReconfigureElementsKind_GeneralizeFieldTrivial(
|
||||||
{PropertyConstness::kConst, Representation::Double(), any_type},
|
{PropertyConstness::kConst, Representation::Double(), any_type},
|
||||||
{PropertyConstness::kMutable, Representation::HeapObject(), value_type},
|
{PropertyConstness::kMutable, Representation::HeapObject(), value_type},
|
||||||
{PropertyConstness::kMutable, Representation::Tagged(), any_type});
|
{PropertyConstness::kMutable, Representation::Tagged(), any_type});
|
||||||
|
|
||||||
TestReconfigureElementsKind_GeneralizeField(
|
TestReconfigureElementsKind_GeneralizeFieldTrivial(
|
||||||
{PropertyConstness::kMutable, Representation::Double(), any_type},
|
{PropertyConstness::kMutable, Representation::Double(), any_type},
|
||||||
{PropertyConstness::kConst, Representation::HeapObject(), value_type},
|
{PropertyConstness::kConst, Representation::HeapObject(), value_type},
|
||||||
{PropertyConstness::kMutable, Representation::Tagged(), any_type});
|
{PropertyConstness::kMutable, Representation::Tagged(), any_type});
|
||||||
|
|
||||||
TestReconfigureElementsKind_GeneralizeField(
|
TestReconfigureElementsKind_GeneralizeFieldTrivial(
|
||||||
{PropertyConstness::kMutable, Representation::Double(), any_type},
|
{PropertyConstness::kMutable, Representation::Double(), any_type},
|
||||||
{PropertyConstness::kMutable, Representation::HeapObject(), value_type},
|
{PropertyConstness::kMutable, Representation::HeapObject(), value_type},
|
||||||
{PropertyConstness::kMutable, Representation::Tagged(), any_type});
|
{PropertyConstness::kMutable, Representation::Tagged(), any_type});
|
||||||
@ -2092,22 +1995,22 @@ TEST(ReconfigureElementsKind_GeneralizeHeapObjectFieldToTagged) {
|
|||||||
Handle<FieldType> value_type =
|
Handle<FieldType> value_type =
|
||||||
FieldType::Class(Map::Create(isolate, 0), isolate);
|
FieldType::Class(Map::Create(isolate, 0), isolate);
|
||||||
|
|
||||||
TestReconfigureElementsKind_GeneralizeField(
|
TestReconfigureElementsKind_GeneralizeFieldTrivial(
|
||||||
{PropertyConstness::kConst, Representation::HeapObject(), value_type},
|
{PropertyConstness::kConst, Representation::HeapObject(), value_type},
|
||||||
{PropertyConstness::kConst, Representation::Smi(), any_type},
|
{PropertyConstness::kConst, Representation::Smi(), any_type},
|
||||||
{PropertyConstness::kConst, Representation::Tagged(), any_type});
|
{PropertyConstness::kConst, Representation::Tagged(), any_type});
|
||||||
|
|
||||||
TestReconfigureElementsKind_GeneralizeField(
|
TestReconfigureElementsKind_GeneralizeFieldTrivial(
|
||||||
{PropertyConstness::kConst, Representation::HeapObject(), value_type},
|
{PropertyConstness::kConst, Representation::HeapObject(), value_type},
|
||||||
{PropertyConstness::kMutable, Representation::Smi(), any_type},
|
{PropertyConstness::kMutable, Representation::Smi(), any_type},
|
||||||
{PropertyConstness::kMutable, Representation::Tagged(), any_type});
|
{PropertyConstness::kMutable, Representation::Tagged(), any_type});
|
||||||
|
|
||||||
TestReconfigureElementsKind_GeneralizeField(
|
TestReconfigureElementsKind_GeneralizeFieldTrivial(
|
||||||
{PropertyConstness::kMutable, Representation::HeapObject(), value_type},
|
{PropertyConstness::kMutable, Representation::HeapObject(), value_type},
|
||||||
{PropertyConstness::kConst, Representation::Smi(), any_type},
|
{PropertyConstness::kConst, Representation::Smi(), any_type},
|
||||||
{PropertyConstness::kMutable, Representation::Tagged(), any_type});
|
{PropertyConstness::kMutable, Representation::Tagged(), any_type});
|
||||||
|
|
||||||
TestReconfigureElementsKind_GeneralizeField(
|
TestReconfigureElementsKind_GeneralizeFieldTrivial(
|
||||||
{PropertyConstness::kMutable, Representation::HeapObject(), value_type},
|
{PropertyConstness::kMutable, Representation::HeapObject(), value_type},
|
||||||
{PropertyConstness::kMutable, Representation::Smi(), any_type},
|
{PropertyConstness::kMutable, Representation::Smi(), any_type},
|
||||||
{PropertyConstness::kMutable, Representation::Tagged(), any_type});
|
{PropertyConstness::kMutable, Representation::Tagged(), any_type});
|
||||||
|
@ -85,8 +85,11 @@ static double GetDoubleFieldValue(JSObject obj, FieldIndex field_index) {
|
|||||||
return obj->RawFastDoublePropertyAt(field_index);
|
return obj->RawFastDoublePropertyAt(field_index);
|
||||||
} else {
|
} else {
|
||||||
Object value = obj->RawFastPropertyAt(field_index);
|
Object value = obj->RawFastPropertyAt(field_index);
|
||||||
CHECK(value->IsMutableHeapNumber());
|
if (value->IsMutableHeapNumber()) {
|
||||||
return MutableHeapNumber::cast(value)->value();
|
return MutableHeapNumber::cast(value)->value();
|
||||||
|
} else {
|
||||||
|
return value->Number();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
15
test/mjsunit/regress/regress-crbug-959645-1.js
Normal file
15
test/mjsunit/regress/regress-crbug-959645-1.js
Normal file
@ -0,0 +1,15 @@
|
|||||||
|
// Copyright 2019 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.
|
||||||
|
|
||||||
|
// Flags: --allow-natives-syntax --modify-field-representations-inplace
|
||||||
|
|
||||||
|
function f(array, x) {
|
||||||
|
array.x = x;
|
||||||
|
array[0] = 1.1;
|
||||||
|
return array;
|
||||||
|
}
|
||||||
|
|
||||||
|
f([1], 1);
|
||||||
|
f([2], 1);
|
||||||
|
%HeapObjectVerify(f([3], undefined));
|
15
test/mjsunit/regress/regress-crbug-959645-2.js
Normal file
15
test/mjsunit/regress/regress-crbug-959645-2.js
Normal file
@ -0,0 +1,15 @@
|
|||||||
|
// Copyright 2019 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.
|
||||||
|
|
||||||
|
// Flags: --allow-natives-syntax --modify-field-representations-inplace
|
||||||
|
|
||||||
|
function f(array, x) {
|
||||||
|
array.x = x;
|
||||||
|
array[0] = undefined;
|
||||||
|
return array;
|
||||||
|
}
|
||||||
|
|
||||||
|
f([1.1], 1);
|
||||||
|
f([2.2], 1);
|
||||||
|
%HeapObjectVerify(f([3.3], undefined));
|
Loading…
Reference in New Issue
Block a user