|
@ -5995,59 +5995,34 @@ void HOptimizedGraphBuilder::VisitArrayLiteral(ArrayLiteral* expr) { |
|
|
Handle<AllocationSite> site; |
|
|
Handle<AllocationSite> site; |
|
|
Handle<LiteralsArray> literals(environment()->closure()->literals(), |
|
|
Handle<LiteralsArray> literals(environment()->closure()->literals(), |
|
|
isolate()); |
|
|
isolate()); |
|
|
bool uninitialized = false; |
|
|
|
|
|
Handle<Object> literals_cell(literals->literal(expr->literal_index()), |
|
|
Handle<Object> literals_cell(literals->literal(expr->literal_index()), |
|
|
isolate()); |
|
|
isolate()); |
|
|
Handle<JSObject> boilerplate_object; |
|
|
Handle<JSObject> boilerplate_object; |
|
|
if (literals_cell->IsUndefined()) { |
|
|
if (!literals_cell->IsUndefined()) { |
|
|
uninitialized = true; |
|
|
|
|
|
Handle<Object> raw_boilerplate; |
|
|
|
|
|
ASSIGN_RETURN_ON_EXCEPTION_VALUE( |
|
|
|
|
|
isolate(), raw_boilerplate, |
|
|
|
|
|
Runtime::CreateArrayLiteralBoilerplate(isolate(), literals, |
|
|
|
|
|
expr->constant_elements()), |
|
|
|
|
|
Bailout(kArrayBoilerplateCreationFailed)); |
|
|
|
|
|
|
|
|
|
|
|
boilerplate_object = Handle<JSObject>::cast(raw_boilerplate); |
|
|
|
|
|
AllocationSiteCreationContext creation_context(isolate()); |
|
|
|
|
|
site = creation_context.EnterNewScope(); |
|
|
|
|
|
if (JSObject::DeepWalk(boilerplate_object, &creation_context).is_null()) { |
|
|
|
|
|
return Bailout(kArrayBoilerplateCreationFailed); |
|
|
|
|
|
} |
|
|
|
|
|
creation_context.ExitScope(site, boilerplate_object); |
|
|
|
|
|
literals->set_literal(expr->literal_index(), *site); |
|
|
|
|
|
|
|
|
|
|
|
if (boilerplate_object->elements()->map() == |
|
|
|
|
|
isolate()->heap()->fixed_cow_array_map()) { |
|
|
|
|
|
isolate()->counters()->cow_arrays_created_runtime()->Increment(); |
|
|
|
|
|
} |
|
|
|
|
|
} else { |
|
|
|
|
|
DCHECK(literals_cell->IsAllocationSite()); |
|
|
DCHECK(literals_cell->IsAllocationSite()); |
|
|
site = Handle<AllocationSite>::cast(literals_cell); |
|
|
site = Handle<AllocationSite>::cast(literals_cell); |
|
|
boilerplate_object = Handle<JSObject>( |
|
|
boilerplate_object = Handle<JSObject>( |
|
|
JSObject::cast(site->transition_info()), isolate()); |
|
|
JSObject::cast(site->transition_info()), isolate()); |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
DCHECK(!boilerplate_object.is_null()); |
|
|
ElementsKind boilerplate_elements_kind = expr->constant_elements_kind(); |
|
|
DCHECK(site->SitePointsToLiteral()); |
|
|
if (!boilerplate_object.is_null()) { |
|
|
|
|
|
boilerplate_elements_kind = boilerplate_object->GetElementsKind(); |
|
|
ElementsKind boilerplate_elements_kind = |
|
|
} |
|
|
boilerplate_object->GetElementsKind(); |
|
|
|
|
|
|
|
|
|
|
|
// Check whether to use fast or slow deep-copying for boilerplate.
|
|
|
// Check whether to use fast or slow deep-copying for boilerplate.
|
|
|
int max_properties = kMaxFastLiteralProperties; |
|
|
int max_properties = kMaxFastLiteralProperties; |
|
|
if (IsFastLiteral(boilerplate_object, |
|
|
if (!boilerplate_object.is_null() && |
|
|
kMaxFastLiteralDepth, |
|
|
IsFastLiteral(boilerplate_object, kMaxFastLiteralDepth, |
|
|
&max_properties)) { |
|
|
&max_properties)) { |
|
|
|
|
|
DCHECK(site->SitePointsToLiteral()); |
|
|
AllocationSiteUsageContext site_context(isolate(), site, false); |
|
|
AllocationSiteUsageContext site_context(isolate(), site, false); |
|
|
site_context.EnterNewScope(); |
|
|
site_context.EnterNewScope(); |
|
|
literal = BuildFastLiteral(boilerplate_object, &site_context); |
|
|
literal = BuildFastLiteral(boilerplate_object, &site_context); |
|
|
site_context.ExitScope(site, boilerplate_object); |
|
|
site_context.ExitScope(site, boilerplate_object); |
|
|
} else { |
|
|
} else { |
|
|
NoObservableSideEffectsScope no_effects(this); |
|
|
NoObservableSideEffectsScope no_effects(this); |
|
|
// Boilerplate already exists and constant elements are never accessed,
|
|
|
Handle<FixedArray> constants = expr->constant_elements(); |
|
|
// pass an empty fixed array to the runtime function instead.
|
|
|
|
|
|
Handle<FixedArray> constants = isolate()->factory()->empty_fixed_array(); |
|
|
|
|
|
int literal_index = expr->literal_index(); |
|
|
int literal_index = expr->literal_index(); |
|
|
int flags = expr->ComputeFlags(true); |
|
|
int flags = expr->ComputeFlags(true); |
|
|
|
|
|
|
|
@ -6058,8 +6033,10 @@ void HOptimizedGraphBuilder::VisitArrayLiteral(ArrayLiteral* expr) { |
|
|
literal = Add<HCallRuntime>(Runtime::FunctionForId(function_id), 4); |
|
|
literal = Add<HCallRuntime>(Runtime::FunctionForId(function_id), 4); |
|
|
|
|
|
|
|
|
// Register to deopt if the boilerplate ElementsKind changes.
|
|
|
// Register to deopt if the boilerplate ElementsKind changes.
|
|
|
|
|
|
if (!site.is_null()) { |
|
|
top_info()->dependencies()->AssumeTransitionStable(site); |
|
|
top_info()->dependencies()->AssumeTransitionStable(site); |
|
|
} |
|
|
} |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
// The array is expected in the bailout environment during computation
|
|
|
// The array is expected in the bailout environment during computation
|
|
|
// of the property values and is the value of the entire expression.
|
|
|
// of the property values and is the value of the entire expression.
|
|
@ -6090,9 +6067,8 @@ void HOptimizedGraphBuilder::VisitArrayLiteral(ArrayLiteral* expr) { |
|
|
case FAST_HOLEY_ELEMENTS: |
|
|
case FAST_HOLEY_ELEMENTS: |
|
|
case FAST_DOUBLE_ELEMENTS: |
|
|
case FAST_DOUBLE_ELEMENTS: |
|
|
case FAST_HOLEY_DOUBLE_ELEMENTS: { |
|
|
case FAST_HOLEY_DOUBLE_ELEMENTS: { |
|
|
HStoreKeyed* instr = Add<HStoreKeyed>(elements, key, value, nullptr, |
|
|
Add<HStoreKeyed>(elements, key, value, nullptr, |
|
|
boilerplate_elements_kind); |
|
|
boilerplate_elements_kind); |
|
|
instr->SetUninitialized(uninitialized); |
|
|
|
|
|
break; |
|
|
break; |
|
|
} |
|
|
} |
|
|
default: |
|
|
default: |
|
|