[turbofan] Serialize descriptor arrays.

- Provide MapData::SerializeDescriptors method for serializing the whole
  descriptor array.
- Trigger this in JSObjectData::SerializeAsBoilerplate.
- Further make things more consistent across the broker.

Bug: v8:7790
Change-Id: Ie6499da8857f7c6561f7c44922aeffcea4876be7
Reviewed-on: https://chromium-review.googlesource.com/1199102
Commit-Queue: Georg Neis <neis@chromium.org>
Reviewed-by: Jaroslav Sevcik <jarin@chromium.org>
Reviewed-by: Maya Lekova <mslekova@chromium.org>
Cr-Commit-Position: refs/heads/master@{#55756}
This commit is contained in:
Georg Neis 2018-09-10 15:22:19 +02:00 committed by Commit Bot
parent 5365cc1e40
commit 9ed348e65f
6 changed files with 200 additions and 101 deletions

View File

@ -335,7 +335,7 @@ bool AccessInfoFactory::ComputeElementAccessInfos(
return true; return true;
} }
// TODO(mslekova): Refactor this function to make it easier to read.
bool AccessInfoFactory::ComputePropertyAccessInfo( bool AccessInfoFactory::ComputePropertyAccessInfo(
Handle<Map> map, Handle<Name> name, AccessMode access_mode, Handle<Map> map, Handle<Name> name, AccessMode access_mode,
PropertyAccessInfo* access_info) { PropertyAccessInfo* access_info) {
@ -404,8 +404,9 @@ bool AccessInfoFactory::ComputePropertyAccessInfo(
// The field type was cleared by the GC, so we don't know anything // The field type was cleared by the GC, so we don't know anything
// about the contents now. // about the contents now.
} else if (descriptors_field_type->IsClass()) { } else if (descriptors_field_type->IsClass()) {
dependencies()->DependOnFieldType(MapRef(js_heap_broker(), map), MapRef map_ref(js_heap_broker(), map);
number); map_ref.SerializeDescriptors(); // TODO(neis): Remove later.
dependencies()->DependOnFieldType(map_ref, number);
// Remember the field map, and try to infer a useful type. // Remember the field map, and try to infer a useful type.
Handle<Map> map(descriptors_field_type->AsClass(), isolate()); Handle<Map> map(descriptors_field_type->AsClass(), isolate());
field_type = Type::For(js_heap_broker(), map); field_type = Type::For(js_heap_broker(), map);
@ -708,8 +709,9 @@ bool AccessInfoFactory::LookupTransition(Handle<Map> map, Handle<Name> name,
// Store is not safe if the field type was cleared. // Store is not safe if the field type was cleared.
return false; return false;
} else if (descriptors_field_type->IsClass()) { } else if (descriptors_field_type->IsClass()) {
dependencies()->DependOnFieldType( MapRef transition_map_ref(js_heap_broker(), transition_map);
MapRef(js_heap_broker(), transition_map), number); transition_map_ref.SerializeDescriptors(); // TODO(neis): Remove later.
dependencies()->DependOnFieldType(transition_map_ref, number);
// Remember the field map, and try to infer a useful type. // Remember the field map, and try to infer a useful type.
Handle<Map> map(descriptors_field_type->AsClass(), isolate()); Handle<Map> map(descriptors_field_type->AsClass(), isolate());
field_type = Type::For(js_heap_broker(), map); field_type = Type::For(js_heap_broker(), map);

View File

@ -57,12 +57,7 @@ class HeapObjectData : public ObjectData {
MapData* map() const { return map_; } MapData* map() const { return map_; }
HeapObjectData(JSHeapBroker* broker, Handle<HeapObject> object, HeapObjectData(JSHeapBroker* broker, Handle<HeapObject> object,
HeapObjectType type) HeapObjectType type);
: ObjectData(broker, object, false),
type_(type),
map_(broker->GetOrCreateData(object->map())->AsMap()) {
CHECK(broker->SerializingAllowed());
}
private: private:
HeapObjectType const type_; HeapObjectType const type_;
@ -356,36 +351,8 @@ bool IsInlinableFastLiteral(Handle<JSObject> boilerplate) {
class AllocationSiteData : public HeapObjectData { class AllocationSiteData : public HeapObjectData {
public: public:
AllocationSiteData(JSHeapBroker* broker, Handle<AllocationSite> object, AllocationSiteData(JSHeapBroker* broker, Handle<AllocationSite> object,
HeapObjectType type) HeapObjectType type);
: HeapObjectData(broker, object, type), void SerializeBoilerplate();
PointsToLiteral_(object->PointsToLiteral()),
GetPretenureMode_(object->GetPretenureMode()) {
if (PointsToLiteral_) {
IsFastLiteral_ = IsInlinableFastLiteral(
handle(object->boilerplate(), broker->isolate()));
} else {
GetElementsKind_ = object->GetElementsKind();
CanInlineCall_ = object->CanInlineCall();
}
}
void SerializeBoilerplate() {
if (boilerplate_ != nullptr || !IsFastLiteral_) return;
Handle<AllocationSite> site = Handle<AllocationSite>::cast(object());
Handle<JSObject> boilerplate_object(site->boilerplate(),
broker()->isolate());
boilerplate_ = broker()->GetOrCreateData(boilerplate_object)->AsJSObject();
boilerplate_->SerializeAsBoilerplate();
DCHECK_NULL(nested_site_);
Handle<Object> nested_site_object =
handle(site->nested_site(), broker()->isolate());
nested_site_ = broker()->GetOrCreateData(nested_site_object);
if (nested_site_->IsAllocationSite()) {
nested_site_->AsAllocationSite()->SerializeBoilerplate();
}
}
bool PointsToLiteral() const { return PointsToLiteral_; } bool PointsToLiteral() const { return PointsToLiteral_; }
PretenureFlag GetPretenureMode() const { return GetPretenureMode_; } PretenureFlag GetPretenureMode() const { return GetPretenureMode_; }
@ -405,6 +372,7 @@ class AllocationSiteData : public HeapObjectData {
JSObjectData* boilerplate_ = nullptr; JSObjectData* boilerplate_ = nullptr;
ElementsKind GetElementsKind_ = NO_ELEMENTS; ElementsKind GetElementsKind_ = NO_ELEMENTS;
bool CanInlineCall_ = false; bool CanInlineCall_ = false;
bool serialized_boilerplate_ = false;
}; };
// Only used in JSNativeContextSpecialization. // Only used in JSNativeContextSpecialization.
@ -415,6 +383,14 @@ class ScriptContextTableData : public HeapObjectData {
: HeapObjectData(broker, object, type) {} : HeapObjectData(broker, object, type) {}
}; };
struct PropertyDescriptor {
NameData* key = nullptr;
PropertyDetails details = PropertyDetails::Empty();
FieldIndex field_index;
MapData* field_owner = nullptr;
ObjectData* field_type = nullptr;
};
class MapData : public HeapObjectData { class MapData : public HeapObjectData {
public: public:
InstanceType instance_type() const { return instance_type_; } InstanceType instance_type() const { return instance_type_; }
@ -426,11 +402,20 @@ class MapData : public HeapObjectData {
MapData(JSHeapBroker* broker, Handle<Map> object, HeapObjectType type); MapData(JSHeapBroker* broker, Handle<Map> object, HeapObjectType type);
// Extra information. // Extra information.
void SerializeElementsKindGeneralizations(); void SerializeElementsKindGeneralizations();
const ZoneVector<MapData*>& elements_kind_generalizations() { const ZoneVector<MapData*>& elements_kind_generalizations() const {
CHECK(serialized_elements_kind_generalizations_);
return elements_kind_generalizations_; return elements_kind_generalizations_;
} }
// Serialize the descriptor array and, recursively, that of any field owner.
void SerializeDescriptors();
const ZoneVector<PropertyDescriptor>& descriptors() const {
CHECK(serialized_descriptors_);
return descriptors_;
}
private: private:
InstanceType const instance_type_; InstanceType const instance_type_;
int const instance_size_; int const instance_size_;
@ -440,8 +425,52 @@ class MapData : public HeapObjectData {
bool serialized_elements_kind_generalizations_ = false; bool serialized_elements_kind_generalizations_ = false;
ZoneVector<MapData*> elements_kind_generalizations_; ZoneVector<MapData*> elements_kind_generalizations_;
bool serialized_descriptors_ = false;
ZoneVector<PropertyDescriptor> descriptors_;
}; };
AllocationSiteData::AllocationSiteData(JSHeapBroker* broker,
Handle<AllocationSite> object,
HeapObjectType type)
: HeapObjectData(broker, object, type),
PointsToLiteral_(object->PointsToLiteral()),
GetPretenureMode_(object->GetPretenureMode()) {
if (PointsToLiteral_) {
IsFastLiteral_ = IsInlinableFastLiteral(
handle(object->boilerplate(), broker->isolate()));
} else {
GetElementsKind_ = object->GetElementsKind();
CanInlineCall_ = object->CanInlineCall();
}
}
void AllocationSiteData::SerializeBoilerplate() {
if (serialized_boilerplate_) return;
serialized_boilerplate_ = true;
Handle<AllocationSite> site = Handle<AllocationSite>::cast(object());
CHECK(IsFastLiteral_);
DCHECK_NULL(boilerplate_);
boilerplate_ = broker()->GetOrCreateData(site->boilerplate())->AsJSObject();
boilerplate_->SerializeAsBoilerplate();
DCHECK_NULL(nested_site_);
nested_site_ = broker()->GetOrCreateData(site->nested_site());
if (nested_site_->IsAllocationSite()) {
nested_site_->AsAllocationSite()->SerializeBoilerplate();
}
}
HeapObjectData::HeapObjectData(JSHeapBroker* broker, Handle<HeapObject> object,
HeapObjectType type)
: ObjectData(broker, object, false),
type_(type),
map_(broker->GetOrCreateData(object->map())->AsMap()) {
CHECK(broker->SerializingAllowed());
}
MapData::MapData(JSHeapBroker* broker, Handle<Map> object, HeapObjectType type) MapData::MapData(JSHeapBroker* broker, Handle<Map> object, HeapObjectType type)
: HeapObjectData(broker, object, type), : HeapObjectData(broker, object, type),
instance_type_(object->instance_type()), instance_type_(object->instance_type()),
@ -449,7 +478,8 @@ MapData::MapData(JSHeapBroker* broker, Handle<Map> object, HeapObjectType type)
bit_field_(object->bit_field()), bit_field_(object->bit_field()),
bit_field2_(object->bit_field2()), bit_field2_(object->bit_field2()),
bit_field3_(object->bit_field3()), bit_field3_(object->bit_field3()),
elements_kind_generalizations_(broker->zone()) {} elements_kind_generalizations_(broker->zone()),
descriptors_(broker->zone()) {}
JSFunctionData::JSFunctionData(JSHeapBroker* broker, Handle<JSFunction> object, JSFunctionData::JSFunctionData(JSHeapBroker* broker, Handle<JSFunction> object,
HeapObjectType type) HeapObjectType type)
@ -466,10 +496,10 @@ void JSFunctionData::Serialize() {
Handle<JSFunction> function = Handle<JSFunction>::cast(object()); Handle<JSFunction> function = Handle<JSFunction>::cast(object());
CHECK_NULL(global_proxy_); DCHECK_NULL(global_proxy_);
CHECK_NULL(initial_map_); DCHECK_NULL(initial_map_);
CHECK_NULL(prototype_); DCHECK_NULL(prototype_);
CHECK_NULL(shared_); DCHECK_NULL(shared_);
global_proxy_ = global_proxy_ =
broker()->GetOrCreateData(function->global_proxy())->AsJSGlobalProxy(); broker()->GetOrCreateData(function->global_proxy())->AsJSGlobalProxy();
@ -532,26 +562,25 @@ void FeedbackVectorData::SerializeSlots() {
if (serialized_) return; if (serialized_) return;
serialized_ = true; serialized_ = true;
Handle<FeedbackVector> vector = Handle<FeedbackVector>::cast(object());
DCHECK(feedback_.empty()); DCHECK(feedback_.empty());
feedback_.reserve(vector->length());
Handle<FeedbackVector> feedback_vector = for (int i = 0; i < vector->length(); ++i) {
Handle<FeedbackVector>::cast(object()); MaybeObject* value = vector->get(i);
feedback_.reserve(feedback_vector->length());
for (int i = 0; i < feedback_vector->length(); ++i) {
MaybeObject* value = feedback_vector->get(i);
ObjectData* slot_value = value->IsObject() ObjectData* slot_value = value->IsObject()
? broker()->GetOrCreateData(value->ToObject()) ? broker()->GetOrCreateData(value->ToObject())
: nullptr; : nullptr;
feedback_.push_back(slot_value); feedback_.push_back(slot_value);
if (slot_value == nullptr) continue; if (slot_value == nullptr) continue;
if (slot_value->IsAllocationSite()) { if (slot_value->IsAllocationSite() &&
slot_value->AsAllocationSite()->IsFastLiteral()) {
slot_value->AsAllocationSite()->SerializeBoilerplate(); slot_value->AsAllocationSite()->SerializeBoilerplate();
} else if (slot_value->IsJSRegExp()) { } else if (slot_value->IsJSRegExp()) {
slot_value->AsJSRegExp()->SerializeAsRegExpBoilerplate(); slot_value->AsJSRegExp()->SerializeAsRegExpBoilerplate();
} }
} }
DCHECK_EQ(feedback_vector->length(), feedback_.size()); DCHECK_EQ(vector->length(), feedback_.size());
} }
class FixedArrayBaseData : public HeapObjectData { class FixedArrayBaseData : public HeapObjectData {
@ -589,13 +618,13 @@ void FixedArrayData::SerializeContents() {
if (serialized_contents_) return; if (serialized_contents_) return;
serialized_contents_ = true; serialized_contents_ = true;
Handle<FixedArray> fixed_array = Handle<FixedArray>::cast(object()); Handle<FixedArray> array = Handle<FixedArray>::cast(object());
CHECK_EQ(fixed_array->length(), length()); CHECK_EQ(array->length(), length());
CHECK(contents_.empty()); CHECK(contents_.empty());
contents_.reserve(static_cast<size_t>(length())); contents_.reserve(static_cast<size_t>(length()));
for (int i = 0; i < length(); i++) { for (int i = 0; i < length(); i++) {
Handle<Object> value = handle(fixed_array->get(i), broker()->isolate()); Handle<Object> value = handle(array->get(i), broker()->isolate());
contents_.push_back(broker()->GetOrCreateData(value)); contents_.push_back(broker()->GetOrCreateData(value));
} }
} }
@ -765,16 +794,43 @@ void JSObjectData::SerializeElements() {
Handle<JSObject> boilerplate = Handle<JSObject>::cast(object()); Handle<JSObject> boilerplate = Handle<JSObject>::cast(object());
Handle<FixedArrayBase> elements_object(boilerplate->elements(), Handle<FixedArrayBase> elements_object(boilerplate->elements(),
broker()->isolate()); broker()->isolate());
CHECK_NULL(elements_); DCHECK_NULL(elements_);
elements_ = broker()->GetOrCreateData(elements_object)->AsFixedArrayBase(); elements_ = broker()->GetOrCreateData(elements_object)->AsFixedArrayBase();
} }
void MapData::SerializeDescriptors() {
if (serialized_descriptors_) return;
serialized_descriptors_ = true;
Handle<Map> map = Handle<Map>::cast(object());
Isolate* const isolate = broker()->isolate();
Handle<DescriptorArray> descriptors(map->instance_descriptors(), isolate);
// We copy all descriptors (not only the own) in order to support
// FindFieldOwner, which is used by the FieldType compilation dependency.
int const number_of_descriptors = descriptors->number_of_descriptors();
DCHECK(descriptors_.empty());
descriptors_.reserve(number_of_descriptors);
for (int i = 0; i < number_of_descriptors; ++i) {
PropertyDescriptor d;
d.key = broker()->GetOrCreateData(descriptors->GetKey(i))->AsName();
d.details = descriptors->GetDetails(i);
if (d.details.location() == kField) {
d.field_index = FieldIndex::ForDescriptor(*map, i);
d.field_owner =
broker()->GetOrCreateData(map->FindFieldOwner(isolate, i))->AsMap();
d.field_type = broker()->GetOrCreateData(descriptors->GetFieldType(i));
d.field_owner->SerializeDescriptors();
}
descriptors_.push_back(d);
}
}
void JSObjectData::SerializeRecursive(int depth) { void JSObjectData::SerializeRecursive(int depth) {
if (serialized_as_boilerplate_) return; if (serialized_as_boilerplate_) return;
serialized_as_boilerplate_ = true; serialized_as_boilerplate_ = true;
Handle<JSObject> boilerplate = Handle<JSObject>::cast(object()); Handle<JSObject> boilerplate = Handle<JSObject>::cast(object());
Isolate* const isolate = boilerplate->GetIsolate();
// We only serialize boilerplates that pass the IsInlinableFastLiteral // We only serialize boilerplates that pass the IsInlinableFastLiteral
// check, so we only do a sanity check on the depth here. // check, so we only do a sanity check on the depth here.
@ -782,6 +838,7 @@ void JSObjectData::SerializeRecursive(int depth) {
CHECK(!boilerplate->map()->is_deprecated()); CHECK(!boilerplate->map()->is_deprecated());
// Serialize the elements. // Serialize the elements.
Isolate* const isolate = broker()->isolate();
Handle<FixedArrayBase> elements_object(boilerplate->elements(), isolate); Handle<FixedArrayBase> elements_object(boilerplate->elements(), isolate);
// Boilerplates need special serialization - we need to make sure COW arrays // Boilerplates need special serialization - we need to make sure COW arrays
@ -803,7 +860,7 @@ void JSObjectData::SerializeRecursive(int depth) {
cow_or_empty_elements_tenured_ = true; cow_or_empty_elements_tenured_ = true;
} }
CHECK_NULL(elements_); DCHECK_NULL(elements_);
elements_ = broker()->GetOrCreateData(elements_object)->AsFixedArrayBase(); elements_ = broker()->GetOrCreateData(elements_object)->AsFixedArrayBase();
if (empty_or_cow) { if (empty_or_cow) {
@ -860,6 +917,8 @@ void JSObjectData::SerializeRecursive(int depth) {
inobject_fields_.push_back(JSObjectField{value_data}); inobject_fields_.push_back(JSObjectField{value_data});
} }
} }
map()->SerializeDescriptors();
} }
void JSRegExpData::SerializeAsRegExpBoilerplate() { void JSRegExpData::SerializeAsRegExpBoilerplate() {
@ -952,6 +1011,8 @@ bool JSHeapBroker::SerializingAllowed() const {
} }
void JSHeapBroker::SerializeStandardObjects() { void JSHeapBroker::SerializeStandardObjects() {
if (mode() == kDisabled) return;
Trace("Serializing standard objects.\n"); Trace("Serializing standard objects.\n");
Builtins* const b = isolate()->builtins(); Builtins* const b = isolate()->builtins();
@ -1050,6 +1111,8 @@ ObjectData* JSHeapBroker::GetOrCreateData(Handle<Object> object) {
// TODO(neis): Remove these Allow* once we serialize everything upfront. // TODO(neis): Remove these Allow* once we serialize everything upfront.
AllowHandleAllocation handle_allocation; AllowHandleAllocation handle_allocation;
AllowHandleDereference handle_dereference; AllowHandleDereference handle_dereference;
// TODO(neis): Inline Serializehere, now that we have subclass-specific
// Serialize methods.
data = ObjectData::Serialize(this, object); data = ObjectData::Serialize(this, object);
} }
CHECK_NOT_NULL(data); CHECK_NOT_NULL(data);
@ -1233,9 +1296,12 @@ void JSObjectRef::EnsureElementsTenured() {
} }
} }
FieldIndex MapRef::GetFieldIndexFor(int i) const { FieldIndex MapRef::GetFieldIndexFor(int descriptor_index) const {
AllowHandleDereference allow_handle_dereference; if (broker()->mode() == JSHeapBroker::kDisabled) {
return FieldIndex::ForDescriptor(*object<Map>(), i); AllowHandleDereference allow_handle_dereference;
return FieldIndex::ForDescriptor(*object<Map>(), descriptor_index);
}
return data()->AsMap()->descriptors().at(descriptor_index).field_index;
} }
int MapRef::GetInObjectPropertyOffset(int i) const { int MapRef::GetInObjectPropertyOffset(int i) const {
@ -1243,17 +1309,24 @@ int MapRef::GetInObjectPropertyOffset(int i) const {
return object<Map>()->GetInObjectPropertyOffset(i); return object<Map>()->GetInObjectPropertyOffset(i);
} }
PropertyDetails MapRef::GetPropertyDetails(int i) const { PropertyDetails MapRef::GetPropertyDetails(int descriptor_index) const {
AllowHandleDereference allow_handle_dereference; if (broker()->mode() == JSHeapBroker::kDisabled) {
return object<Map>()->instance_descriptors()->GetDetails(i); AllowHandleDereference allow_handle_dereference;
return object<Map>()->instance_descriptors()->GetDetails(descriptor_index);
}
return data()->AsMap()->descriptors().at(descriptor_index).details;
} }
NameRef MapRef::GetPropertyKey(int i) const { NameRef MapRef::GetPropertyKey(int descriptor_index) const {
AllowHandleAllocation handle_allocation; if (broker()->mode() == JSHeapBroker::kDisabled) {
AllowHandleDereference allow_handle_dereference; AllowHandleAllocation handle_allocation;
return NameRef(broker(), AllowHandleDereference allow_handle_dereference;
handle(object<Map>()->instance_descriptors()->GetKey(i), return NameRef(
broker()->isolate())); broker(),
handle(object<Map>()->instance_descriptors()->GetKey(descriptor_index),
broker()->isolate()));
}
return NameRef(data()->AsMap()->descriptors().at(descriptor_index).key);
} }
bool MapRef::IsFixedCowArrayMap() const { bool MapRef::IsFixedCowArrayMap() const {
@ -1262,22 +1335,30 @@ bool MapRef::IsFixedCowArrayMap() const {
ReadOnlyRoots(broker()->isolate()).fixed_cow_array_map(); ReadOnlyRoots(broker()->isolate()).fixed_cow_array_map();
} }
MapRef MapRef::FindFieldOwner(int descriptor) const { MapRef MapRef::FindFieldOwner(int descriptor_index) const {
AllowHandleAllocation handle_allocation; if (broker()->mode() == JSHeapBroker::kDisabled) {
AllowHandleDereference allow_handle_dereference; AllowHandleAllocation handle_allocation;
Handle<Map> owner( AllowHandleDereference allow_handle_dereference;
object<Map>()->FindFieldOwner(broker()->isolate(), descriptor), Handle<Map> owner(
broker()->isolate()); object<Map>()->FindFieldOwner(broker()->isolate(), descriptor_index),
return MapRef(broker(), owner); broker()->isolate());
return MapRef(broker(), owner);
}
return MapRef(
data()->AsMap()->descriptors().at(descriptor_index).field_owner);
} }
ObjectRef MapRef::GetFieldType(int descriptor) const { ObjectRef MapRef::GetFieldType(int descriptor_index) const {
AllowHandleAllocation handle_allocation; if (broker()->mode() == JSHeapBroker::kDisabled) {
AllowHandleDereference allow_handle_dereference; AllowHandleAllocation handle_allocation;
Handle<FieldType> field_type( AllowHandleDereference allow_handle_dereference;
object<Map>()->instance_descriptors()->GetFieldType(descriptor), Handle<FieldType> field_type(
broker()->isolate()); object<Map>()->instance_descriptors()->GetFieldType(descriptor_index),
return ObjectRef(broker(), field_type); broker()->isolate());
return ObjectRef(broker(), field_type);
}
return ObjectRef(
data()->AsMap()->descriptors().at(descriptor_index).field_type);
} }
bool MapRef::IsUnboxedDoubleField(FieldIndex index) const { bool MapRef::IsUnboxedDoubleField(FieldIndex index) const {
@ -1652,7 +1733,7 @@ void NativeContextData::Serialize() {
Handle<NativeContext> context = Handle<NativeContext>::cast(object()); Handle<NativeContext> context = Handle<NativeContext>::cast(object());
#define SERIALIZE_MEMBER(type, name) \ #define SERIALIZE_MEMBER(type, name) \
CHECK_NULL(name##_); \ DCHECK_NULL(name##_); \
name##_ = broker()->GetOrCreateData(context->name())->As##type(); \ name##_ = broker()->GetOrCreateData(context->name())->As##type(); \
if (name##_->IsJSFunction()) name##_->AsJSFunction()->Serialize(); if (name##_->IsJSFunction()) name##_->AsJSFunction()->Serialize();
BROKER_NATIVE_CONTEXT_FIELDS(SERIALIZE_MEMBER) BROKER_NATIVE_CONTEXT_FIELDS(SERIALIZE_MEMBER)
@ -1665,6 +1746,12 @@ void JSFunctionRef::Serialize() {
data()->AsJSFunction()->Serialize(); data()->AsJSFunction()->Serialize();
} }
void MapRef::SerializeDescriptors() {
if (broker()->mode() == JSHeapBroker::kDisabled) return;
CHECK_EQ(broker()->mode(), JSHeapBroker::kSerializing);
data()->AsMap()->SerializeDescriptors();
}
#undef BIMODAL_ACCESSOR #undef BIMODAL_ACCESSOR
#undef BIMODAL_ACCESSOR_B #undef BIMODAL_ACCESSOR_B
#undef BIMODAL_ACCESSOR_C #undef BIMODAL_ACCESSOR_C

View File

@ -332,11 +332,12 @@ class MapRef : public HeapObjectRef {
base::Optional<MapRef> AsElementsKind(ElementsKind kind) const; base::Optional<MapRef> AsElementsKind(ElementsKind kind) const;
// Concerning the underlying instance_descriptors: // Concerning the underlying instance_descriptors:
MapRef FindFieldOwner(int descriptor) const; void SerializeDescriptors();
PropertyDetails GetPropertyDetails(int i) const; MapRef FindFieldOwner(int descriptor_index) const;
NameRef GetPropertyKey(int i) const; PropertyDetails GetPropertyDetails(int descriptor_index) const;
FieldIndex GetFieldIndexFor(int i) const; NameRef GetPropertyKey(int descriptor_index) const;
ObjectRef GetFieldType(int descriptor) const; FieldIndex GetFieldIndexFor(int descriptor_index) const;
ObjectRef GetFieldType(int descriptor_index) const;
bool IsUnboxedDoubleField(FieldIndex index) const; bool IsUnboxedDoubleField(FieldIndex index) const;
}; };

View File

@ -1997,9 +1997,7 @@ bool PipelineImpl::CreateGraph() {
data->node_origins()->AddDecorator(); data->node_origins()->AddDecorator();
} }
if (FLAG_concurrent_compiler_frontend) { data->js_heap_broker()->SerializeStandardObjects();
data->js_heap_broker()->SerializeStandardObjects();
}
Run<GraphBuilderPhase>(); Run<GraphBuilderPhase>();
RunPrintAndVerify(GraphBuilderPhase::phase_name(), true); RunPrintAndVerify(GraphBuilderPhase::phase_name(), true);

View File

@ -209,6 +209,7 @@ Node* PropertyAccessBuilder::TryBuildLoadConstantDataField(
DCHECK(!it.is_dictionary_holder()); DCHECK(!it.is_dictionary_holder());
MapRef map(js_heap_broker(), MapRef map(js_heap_broker(),
handle(it.GetHolder<HeapObject>()->map(), isolate())); handle(it.GetHolder<HeapObject>()->map(), isolate()));
map.SerializeDescriptors(); // TODO(neis): Remove later.
dependencies()->DependOnFieldType(map, it.GetFieldDescriptorIndex()); dependencies()->DependOnFieldType(map, it.GetFieldDescriptorIndex());
} }
return value; return value;

View File

@ -657,7 +657,9 @@ static void TestGeneralizeField(int detach_property_at_index,
CanonicalHandleScope canonical(isolate); CanonicalHandleScope canonical(isolate);
JSHeapBroker broker(isolate, &zone); JSHeapBroker broker(isolate, &zone);
CompilationDependencies dependencies(isolate, &zone); CompilationDependencies dependencies(isolate, &zone);
dependencies.DependOnFieldType(MapRef(&broker, map), property_index); MapRef map_ref(&broker, map);
map_ref.SerializeDescriptors();
dependencies.DependOnFieldType(map_ref, property_index);
Handle<Map> field_owner(map->FindFieldOwner(isolate, property_index), Handle<Map> field_owner(map->FindFieldOwner(isolate, property_index),
isolate); isolate);
@ -1029,7 +1031,9 @@ static void TestReconfigureDataFieldAttribute_GeneralizeField(
CanonicalHandleScope canonical(isolate); CanonicalHandleScope canonical(isolate);
JSHeapBroker broker(isolate, &zone); JSHeapBroker broker(isolate, &zone);
CompilationDependencies dependencies(isolate, &zone); CompilationDependencies dependencies(isolate, &zone);
dependencies.DependOnFieldType(MapRef(&broker, map), kSplitProp); MapRef map_ref(&broker, map);
map_ref.SerializeDescriptors();
dependencies.DependOnFieldType(map_ref, kSplitProp);
// Reconfigure attributes of property |kSplitProp| of |map2| to NONE, which // Reconfigure attributes of property |kSplitProp| of |map2| to NONE, which
// should generalize representations in |map1|. // should generalize representations in |map1|.
@ -1113,7 +1117,9 @@ static void TestReconfigureDataFieldAttribute_GeneralizeFieldTrivial(
CanonicalHandleScope canonical(isolate); CanonicalHandleScope canonical(isolate);
JSHeapBroker broker(isolate, &zone); JSHeapBroker broker(isolate, &zone);
CompilationDependencies dependencies(isolate, &zone); CompilationDependencies dependencies(isolate, &zone);
dependencies.DependOnFieldType(MapRef(&broker, map), kSplitProp); MapRef map_ref(&broker, map);
map_ref.SerializeDescriptors();
dependencies.DependOnFieldType(map_ref, kSplitProp);
// Reconfigure attributes of property |kSplitProp| of |map2| to NONE, which // Reconfigure attributes of property |kSplitProp| of |map2| to NONE, which
// should generalize representations in |map1|. // should generalize representations in |map1|.
@ -1794,7 +1800,9 @@ static void TestReconfigureElementsKind_GeneralizeField(
CanonicalHandleScope canonical(isolate); CanonicalHandleScope canonical(isolate);
JSHeapBroker broker(isolate, &zone); JSHeapBroker broker(isolate, &zone);
CompilationDependencies dependencies(isolate, &zone); CompilationDependencies dependencies(isolate, &zone);
dependencies.DependOnFieldType(MapRef(&broker, map), kDiffProp); MapRef map_ref(&broker, map);
map_ref.SerializeDescriptors();
dependencies.DependOnFieldType(map_ref, kDiffProp);
// Reconfigure elements kinds of |map2|, which should generalize // Reconfigure elements kinds of |map2|, which should generalize
// representations in |map|. // representations in |map|.
@ -1889,7 +1897,9 @@ static void TestReconfigureElementsKind_GeneralizeFieldTrivial(
CanonicalHandleScope canonical(isolate); CanonicalHandleScope canonical(isolate);
JSHeapBroker broker(isolate, &zone); JSHeapBroker broker(isolate, &zone);
CompilationDependencies dependencies(isolate, &zone); CompilationDependencies dependencies(isolate, &zone);
dependencies.DependOnFieldType(MapRef(&broker, map), kDiffProp); MapRef map_ref(&broker, map);
map_ref.SerializeDescriptors();
dependencies.DependOnFieldType(map_ref, kDiffProp);
// Reconfigure elements kinds of |map2|, which should generalize // Reconfigure elements kinds of |map2|, which should generalize
// representations in |map|. // representations in |map|.