[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:
parent
5365cc1e40
commit
9ed348e65f
@ -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);
|
||||||
|
@ -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
|
||||||
|
@ -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;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -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);
|
||||||
|
@ -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;
|
||||||
|
@ -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|.
|
||||||
|
Loading…
Reference in New Issue
Block a user