Index: src/ia32/codegen-ia32.cc |
diff --git a/src/ia32/codegen-ia32.cc b/src/ia32/codegen-ia32.cc |
index 25ad52de6b2f2b803def52ff43268fd8725c15d2..ba09b23fae3d01857836c3569090bd107a34ba77 100644 |
--- a/src/ia32/codegen-ia32.cc |
+++ b/src/ia32/codegen-ia32.cc |
@@ -5710,12 +5710,17 @@ void CodeGenerator::VisitArrayLiteral(ArrayLiteral* node) { |
frame_->Push(node->constant_elements()); |
int length = node->values()->length(); |
Result clone; |
- if (node->depth() > 1) { |
+ if (node->constant_elements()->map() == Heap::fixed_cow_array_map()) { |
+ FastCloneShallowArrayStub stub( |
+ FastCloneShallowArrayStub::COPY_ON_WRITE_ELEMENTS, length); |
+ clone = frame_->CallStub(&stub, 3); |
+ } else if (node->depth() > 1) { |
clone = frame_->CallRuntime(Runtime::kCreateArrayLiteral, 3); |
- } else if (length > FastCloneShallowArrayStub::kMaximumLength) { |
+ } else if (length > FastCloneShallowArrayStub::kMaximumClonedLength) { |
clone = frame_->CallRuntime(Runtime::kCreateArrayLiteralShallow, 3); |
} else { |
- FastCloneShallowArrayStub stub(length); |
+ FastCloneShallowArrayStub stub( |
+ FastCloneShallowArrayStub::CLONE_ELEMENTS, length); |
clone = frame_->CallStub(&stub, 3); |
} |
frame_->Push(&clone); |
@@ -5725,12 +5730,9 @@ void CodeGenerator::VisitArrayLiteral(ArrayLiteral* node) { |
for (int i = 0; i < length; i++) { |
Expression* value = node->values()->at(i); |
- // If value is a literal the property value is already set in the |
- // boilerplate object. |
- if (value->AsLiteral() != NULL) continue; |
- // If value is a materialized literal the property value is already set |
- // in the boilerplate object if it is simple. |
- if (CompileTimeValue::IsCompileTimeValue(value)) continue; |
+ if (!CompileTimeValue::ArrayLiteralElementNeedsInitialization(value)) { |
+ continue; |
+ } |
// The property must be set by generated code. |
Load(value); |
@@ -7599,7 +7601,7 @@ void CodeGenerator::GenerateSwapElements(ZoneList<Expression*>* args) { |
KeyedLoadIC::kSlowCaseBitFieldMask); |
deferred->Branch(not_zero); |
- // Check the object's elements are in fast case. |
+ // Check the object's elements are in fast case and writable. |
__ mov(tmp1.reg(), FieldOperand(object.reg(), JSObject::kElementsOffset)); |
__ cmp(FieldOperand(tmp1.reg(), HeapObject::kMapOffset), |
Immediate(Factory::fixed_array_map())); |
@@ -9494,15 +9496,10 @@ Result CodeGenerator::EmitKeyedLoad() { |
if (FLAG_debug_code) __ AbortIfNotSmi(key.reg()); |
} |
- // Get the elements array from the receiver and check that it |
- // is not a dictionary. |
+ // Get the elements array from the receiver. |
__ mov(elements.reg(), |
FieldOperand(receiver.reg(), JSObject::kElementsOffset)); |
- if (FLAG_debug_code) { |
- __ cmp(FieldOperand(elements.reg(), HeapObject::kMapOffset), |
- Immediate(Factory::fixed_array_map())); |
- __ Assert(equal, "JSObject with fast elements map has slow elements"); |
- } |
+ __ AssertFastElements(elements.reg()); |
// Check that the key is within bounds. |
__ cmp(key.reg(), |
@@ -9899,6 +9896,24 @@ void FastCloneShallowArrayStub::Generate(MacroAssembler* masm) { |
__ cmp(ecx, Factory::undefined_value()); |
__ j(equal, &slow_case); |
+ if (FLAG_debug_code) { |
+ const char* message; |
+ Handle<Map> expected_map; |
+ if (mode_ == CLONE_ELEMENTS) { |
+ message = "Expected (writable) fixed array"; |
+ expected_map = Factory::fixed_array_map(); |
+ } else { |
+ ASSERT(mode_ == COPY_ON_WRITE_ELEMENTS); |
+ message = "Expected copy-on-write fixed array"; |
+ expected_map = Factory::fixed_cow_array_map(); |
+ } |
+ __ push(ecx); |
+ __ mov(ecx, FieldOperand(ecx, JSArray::kElementsOffset)); |
+ __ cmp(FieldOperand(ecx, HeapObject::kMapOffset), expected_map); |
+ __ Assert(equal, message); |
+ __ pop(ecx); |
+ } |
+ |
// Allocate both the JS array and the elements array in one big |
// allocation. This avoids multiple limit checks. |
__ AllocateInNewSpace(size, eax, ebx, edx, &slow_case, TAG_OBJECT); |
@@ -9925,6 +9940,10 @@ void FastCloneShallowArrayStub::Generate(MacroAssembler* masm) { |
} |
} |
+ if (mode_ == COPY_ON_WRITE_ELEMENTS) { |
+ __ IncrementCounter(&Counters::cow_arrays_created_stub, 1); |
+ } |
+ |
// Return and remove the on-stack parameters. |
__ ret(3 * kPointerSize); |