Chromium Code Reviews| Index: src/x64/code-stubs-x64.cc |
| diff --git a/src/x64/code-stubs-x64.cc b/src/x64/code-stubs-x64.cc |
| index 7d41ffe53932610814debaf8e91b47bb7430e365..bc0a34e6b74cc5da71818b93287ad1d4da893a47 100644 |
| --- a/src/x64/code-stubs-x64.cc |
| +++ b/src/x64/code-stubs-x64.cc |
| @@ -5665,6 +5665,12 @@ struct AheadOfTimeWriteBarrierStubList kAheadOfTime[] = { |
| { rbx, rdx, rcx, EMIT_REMEMBERED_SET}, |
| // KeyedStoreStubCompiler::GenerateStoreFastElement. |
| { rdi, rdx, rcx, EMIT_REMEMBERED_SET}, |
| + // FastElementsConversionStub::GenerateSmiOnlyToObject |
| + // and FastElementsConversionStub::GenerateDoubleToObject |
| + { rdx, rbx, rdi, EMIT_REMEMBERED_SET}, |
| + // FastElementsConversionStub::GenerateDoubleToObject |
| + { r11, r12, r13, EMIT_REMEMBERED_SET}, |
| + { rdx, r11, rdi, EMIT_REMEMBERED_SET}, |
| // Null termination. |
| { no_reg, no_reg, no_reg, EMIT_REMEMBERED_SET} |
| }; |
| @@ -5913,6 +5919,162 @@ void RecordWriteStub::CheckNeedsToInformIncrementalMarker( |
| } |
| +void FastElementsConversionStub::GenerateSmiOnlyToObject(MacroAssembler* masm) { |
| + // ----------- S t a t e ------------- |
| + // -- rax : value |
| + // -- rbx : target map |
| + // -- rcx : key |
| + // -- rdx : receiver |
| + // -- rsp[0] : return address |
| + // ----------------------------------- |
| + // Set transitioned map. |
| + __ movq(FieldOperand(rdx, HeapObject::kMapOffset), rbx); |
| + __ RecordWriteField(rdx, |
| + HeapObject::kMapOffset, |
| + rbx, |
| + rdi, |
| + kDontSaveFPRegs, |
| + EMIT_REMEMBERED_SET, |
| + OMIT_SMI_CHECK); |
| +} |
| + |
| + |
| +void FastElementsConversionStub::GenerateSmiOnlyToDouble( |
| + MacroAssembler* masm, StrictModeFlag strict_mode) { |
| + // ----------- S t a t e ------------- |
| + // -- rax : value |
| + // -- rbx : target map |
| + // -- rcx : key |
| + // -- rdx : receiver |
| + // -- rsp[0] : return address |
| + // ----------------------------------- |
| + // Set transitioned map. |
| + __ movq(FieldOperand(rdx, HeapObject::kMapOffset), rbx); |
| + __ RecordWriteField(rdx, |
| + HeapObject::kMapOffset, |
| + rbx, |
| + rdi, |
| + kDontSaveFPRegs, |
| + EMIT_REMEMBERED_SET, |
| + OMIT_SMI_CHECK); |
| + // Set backing store's map |
| + __ movq(r8, FieldOperand(rdx, JSObject::kElementsOffset)); |
| + __ LoadRoot(rdi, Heap::kFixedDoubleArrayMapRootIndex); |
| + __ movq(FieldOperand(r8, HeapObject::kMapOffset), rdi); |
| + |
| + // Convert smis to doubles and holes to hole NaNs. Since FixedArray and |
| + // FixedDoubleArray do not differ in size, we do not allocate a new array. |
| + STATIC_ASSERT(FixedDoubleArray::kLengthOffset == FixedArray::kLengthOffset); |
| + STATIC_ASSERT(FixedDoubleArray::kHeaderSize == FixedArray::kHeaderSize); |
| + __ SmiToInteger32(r9, FieldOperand(r8, FixedDoubleArray::kLengthOffset)); |
| + // r8 : elements array |
| + // r9 : elements array length |
| + Label loop, entry, convert_hole; |
| + __ jmp(&entry); |
| + __ bind(&loop); |
| + __ decq(r9); |
| + __ movq(rbx, |
| + FieldOperand(r8, r9, times_8, FixedArray::kHeaderSize)); |
| + // r9 : current element's index |
| + // rbx: current element (smi-tagged) |
| + __ JumpIfNotSmi(rbx, &convert_hole); |
| + __ SmiToInteger32(rbx, rbx); |
| + __ cvtlsi2sd(xmm0, rbx); |
| + __ movsd(FieldOperand(r8, r9, times_8, FixedDoubleArray::kHeaderSize), |
| + xmm0); |
| + __ jmp(&entry); |
| + __ bind(&convert_hole); |
| + __ movq(FieldOperand(r8, r9, times_8, FixedDoubleArray::kHeaderSize), |
| + Immediate(BitCast<int64_t>(kHoleNanInt64))); |
| + __ bind(&entry); |
| + __ testq(r9, r9); |
| + __ j(not_zero, &loop); |
| +} |
| + |
| + |
| +void FastElementsConversionStub::GenerateDoubleToObject( |
| + MacroAssembler* masm, StrictModeFlag strict_mode) { |
| + // ----------- S t a t e ------------- |
| + // -- rax : value |
| + // -- rbx : target map |
| + // -- rcx : key |
| + // -- rdx : receiver |
| + // -- rsp[0] : return address |
| + // ----------------------------------- |
| + Label loop, entry, convert_hole, gc_required; |
| + |
| + __ jmp(&gc_required); |
|
Jakob Kummerow
2011/10/14 09:25:38
lolwut?
Yang
2011/10/14 16:04:28
Forgot to take this out after some debugging.
|
| + __ movq(r8, FieldOperand(rdx, JSObject::kElementsOffset)); |
| + __ SmiToInteger32(r9, FieldOperand(r8, FixedDoubleArray::kLengthOffset)); |
| + // r8 : source FixedDoubleArray |
| + // r9 : number of elements |
| + __ lea(rdi, Operand(r9, times_8, FixedArray::kHeaderSize)); |
| + __ AllocateInNewSpace(rdi, r11, r12, r13, &gc_required, TAG_OBJECT); |
| + __ LoadRoot(rdi, Heap::kFixedArrayMapRootIndex); |
| + __ movq(FieldOperand(r11, HeapObject::kMapOffset), rdi); |
| + __ Integer32ToSmiField(FieldOperand(r11, FixedArray::kLengthOffset), r9); |
| + // r11: destination FixedArray |
|
Jakob Kummerow
2011/10/14 09:25:38
Please move this up to after the AllocateInNewSpac
Yang
2011/10/14 16:04:28
Done.
|
| + |
| + // Prepare for conversion loop. |
| + __ Move(rdi, masm->isolate()->factory()->the_hole_value()); |
| + __ jmp(&entry); |
| + |
| + // Call into runtime if GC is required. |
| + __ bind(&gc_required); |
| + KeyedStoreIC::GenerateRuntimeSetProperty(masm, strict_mode); |
| + |
| + // Box doubles into heap numbers. |
| + __ bind(&loop); |
| + __ decq(r9); |
| + __ movq(r14, FieldOperand(r8, r9, times_8, FixedDoubleArray::kHeaderSize)); |
| + // r9 : current element's index |
| + // r14: current element |
| + // rdi: the hole NaN |
| + __ cmpq(r14, Immediate(BitCast<int64_t>(kHoleNanInt64))); |
| + __ j(equal, &convert_hole); |
| + |
| + // Non-hole double, copy value into a heap number. |
| + __ AllocateHeapNumber(r12, r13, &gc_required); |
| + // r12: new heap number |
| + __ movq(FieldOperand(r12, HeapNumber::kValueOffset), r14); |
| + __ movq(FieldOperand(r11, r9, times_8, FixedDoubleArray::kHeaderSize), r12); |
|
Jakob Kummerow
2011/10/14 09:25:38
FixedArray::kHeaderSize (I know they're the same,
Yang
2011/10/14 16:04:28
Done.
|
| + __ movq(r13, r9); |
| + __ RecordWriteArray(r11, |
| + r12, |
| + r13, |
| + kDontSaveFPRegs, |
| + EMIT_REMEMBERED_SET, |
| + OMIT_SMI_CHECK); |
| + __ jmp(&entry, Label::kNear); |
| + |
| + // Replace the-hole NaN with the-hole pointer. |
| + __ bind(&convert_hole); |
| + __ movq(FieldOperand(r11, r9, times_8, FixedDoubleArray::kHeaderSize), rdi); |
|
Jakob Kummerow
2011/10/14 09:25:38
times_pointer_size, FixedArray::kHeaderSize
Yang
2011/10/14 16:04:28
Done.
|
| + |
| + __ bind(&entry); |
| + __ testq(r9, r9); |
| + __ j(not_zero, &loop); |
| + |
| + // Set transitioned map. |
| + __ movq(FieldOperand(rdx, HeapObject::kMapOffset), rbx); |
| + __ RecordWriteField(rdx, |
| + HeapObject::kMapOffset, |
| + rbx, |
| + rdi, |
| + kDontSaveFPRegs, |
| + EMIT_REMEMBERED_SET, |
| + OMIT_SMI_CHECK); |
| + // Replace receiver's backing store with newly created and filled FixedArray. |
| + __ movq(FieldOperand(rdx, JSObject::kElementsOffset), r11); |
| + __ RecordWriteField(rdx, |
| + HeapObject::kMapOffset, |
|
Jakob Kummerow
2011/10/14 09:25:38
JSObject::kElementsOffset!
Yang
2011/10/14 16:04:28
Done.
|
| + r11, |
| + rdi, |
| + kDontSaveFPRegs, |
| + EMIT_REMEMBERED_SET, |
| + OMIT_SMI_CHECK); |
| +} |
| + |
| #undef __ |
| } } // namespace v8::internal |