Chromium Code Reviews| Index: src/x64/lithium-codegen-x64.cc |
| =================================================================== |
| --- src/x64/lithium-codegen-x64.cc (revision 6931) |
| +++ src/x64/lithium-codegen-x64.cc (working copy) |
| @@ -2271,12 +2271,106 @@ |
| void LCodeGen::DoDeferredMathAbsTaggedHeapNumber(LUnaryMathOperation* instr) { |
| - Abort("Unimplemented: %s", "DoDeferredMathAbsTaggedHeapNumber"); |
| + Register input_reg = ToRegister(instr->InputAt(0)); |
| + __ CompareRoot(FieldOperand(input_reg, HeapObject::kMapOffset), |
| + Heap::kHeapNumberMapRootIndex); |
| + DeoptimizeIf(not_equal, instr->environment()); |
| + |
| + Label done; |
| + Register tmp = input_reg.is(rax) ? rcx : rax; |
| + Register tmp2 = tmp.is(rcx) ? rdx : input_reg.is(rcx) ? rdx : rcx; |
| + |
| + // Preserve the value of all registers. |
| + __ PushSafepointRegisters(); |
| + |
| + Label negative; |
| + __ movl(tmp, FieldOperand(input_reg, HeapNumber::kExponentOffset)); |
| + // Check the sign of the argument. If the argument is positive, just |
| + // return it. We do not need to patch the stack since |input| and |
| + // |result| are the same register and |input| will be restored |
| + // unchanged by popping safepoint registers. |
| + __ testl(tmp, Immediate(HeapNumber::kSignMask)); |
| + __ j(not_zero, &negative); |
| + __ jmp(&done); |
| + |
| + __ bind(&negative); |
| + |
| + Label allocated, slow; |
| + __ AllocateHeapNumber(tmp, tmp2, &slow); |
| + __ jmp(&allocated); |
| + |
| + // Slow case: Call the runtime system to do the number allocation. |
| + __ bind(&slow); |
| + |
| + __ CallRuntimeSaveDoubles(Runtime::kAllocateHeapNumber); |
| + RecordSafepointWithRegisters( |
| + instr->pointer_map(), 0, Safepoint::kNoDeoptimizationIndex); |
| + // Set the pointer to the new heap number in tmp. |
| + if (!tmp.is(rax)) { |
| + __ movq(tmp, rax); |
| + } |
| + |
| + // Restore input_reg after call to runtime. |
| + __ LoadFromSafepointRegisterSlot(input_reg, input_reg); |
| + |
| + __ bind(&allocated); |
| + __ movq(tmp2, FieldOperand(input_reg, HeapNumber::kValueOffset)); |
| + __ shl(tmp2, Immediate(1)); |
| + __ shr(tmp2, Immediate(1)); |
| + __ movq(FieldOperand(tmp, HeapNumber::kValueOffset), tmp2); |
| + __ StoreToSafepointRegisterSlot(input_reg, tmp); |
| + |
| + __ bind(&done); |
| + __ PopSafepointRegisters(); |
| } |
| +void LCodeGen::EmitIntegerMathAbs(LUnaryMathOperation* instr) { |
| + Register input_reg = ToRegister(instr->InputAt(0)); |
| + __ testl(input_reg, input_reg); |
| + Label is_positive; |
| + __ j(not_sign, &is_positive); |
| + __ negl(input_reg); |
|
Lasse Reichstein
2011/02/24 13:57:02
If you want to avoid the jumps, you can do:
movl(
|
| + __ testl(input_reg, input_reg); |
|
Lasse Reichstein
2011/02/24 13:57:02
The negl instruction sets the sign flag, so you ca
|
| + DeoptimizeIf(negative, instr->environment()); |
| + __ bind(&is_positive); |
| +} |
| + |
| + |
| void LCodeGen::DoMathAbs(LUnaryMathOperation* instr) { |
| - Abort("Unimplemented: %s", "DoMathAbs"); |
| + // Class for deferred case. |
| + class DeferredMathAbsTaggedHeapNumber: public LDeferredCode { |
| + public: |
| + DeferredMathAbsTaggedHeapNumber(LCodeGen* codegen, |
| + LUnaryMathOperation* instr) |
| + : LDeferredCode(codegen), instr_(instr) { } |
| + virtual void Generate() { |
| + codegen()->DoDeferredMathAbsTaggedHeapNumber(instr_); |
| + } |
| + private: |
| + LUnaryMathOperation* instr_; |
| + }; |
| + |
| + ASSERT(instr->InputAt(0)->Equals(instr->result())); |
| + Representation r = instr->hydrogen()->value()->representation(); |
| + |
| + if (r.IsDouble()) { |
| + XMMRegister scratch = xmm0; |
|
Lasse Reichstein
2011/02/24 13:57:02
Extra space before "scratch".
|
| + XMMRegister input_reg = ToDoubleRegister(instr->InputAt(0)); |
| + __ xorpd(scratch, scratch); |
| + __ subsd(scratch, input_reg); |
| + __ andpd(input_reg, scratch); |
|
Lasse Reichstein
2011/02/24 13:57:02
Neat.
|
| + } else if (r.IsInteger32()) { |
| + EmitIntegerMathAbs(instr); |
| + } else { // Tagged case. |
| + DeferredMathAbsTaggedHeapNumber* deferred = |
| + new DeferredMathAbsTaggedHeapNumber(this, instr); |
| + Register input_reg = ToRegister(instr->InputAt(0)); |
| + // Smi check. |
| + __ JumpIfNotSmi(input_reg, deferred->entry()); |
| + EmitIntegerMathAbs(instr); |
| + __ bind(deferred->exit()); |
| + } |
| } |