Index: src/x64/lithium-codegen-x64.cc |
diff --git a/src/x64/lithium-codegen-x64.cc b/src/x64/lithium-codegen-x64.cc |
index 5e11a36cb1c41c3b33dae3921828d9615eb20ee5..a94dcee2271d2efd696ea09cb3162b18cb87a44d 100644 |
--- a/src/x64/lithium-codegen-x64.cc |
+++ b/src/x64/lithium-codegen-x64.cc |
@@ -1152,55 +1152,38 @@ void LCodeGen::DoMathFloorOfDiv(LMathFloorOfDiv* instr) { |
void LCodeGen::DoDivI(LDivI* instr) { |
if (!instr->is_flooring() && instr->hydrogen()->RightIsPowerOf2()) { |
Register dividend = ToRegister(instr->left()); |
- int32_t divisor = |
- HConstant::cast(instr->hydrogen()->right())->Integer32Value(); |
- int32_t test_value = 0; |
- int32_t power = 0; |
- |
- if (divisor > 0) { |
- test_value = divisor - 1; |
- power = WhichPowerOf2(divisor); |
- } else { |
- // Check for (0 / -x) that will produce negative zero. |
- if (instr->hydrogen()->CheckFlag(HValue::kBailoutOnMinusZero)) { |
- __ testl(dividend, dividend); |
- DeoptimizeIf(zero, instr->environment()); |
- } |
- // Check for (kMinInt / -1). |
- if (divisor == -1 && instr->hydrogen()->CheckFlag(HValue::kCanOverflow)) { |
- __ cmpl(dividend, Immediate(kMinInt)); |
- DeoptimizeIf(zero, instr->environment()); |
- } |
- test_value = - divisor - 1; |
- power = WhichPowerOf2(-divisor); |
- } |
- |
- if (test_value != 0) { |
- if (instr->hydrogen()->CheckFlag( |
- HInstruction::kAllUsesTruncatingToInt32)) { |
- Label done, negative; |
- __ cmpl(dividend, Immediate(0)); |
- __ j(less, &negative, Label::kNear); |
- __ sarl(dividend, Immediate(power)); |
- if (divisor < 0) __ negl(dividend); |
- __ jmp(&done, Label::kNear); |
+ HDiv* hdiv = instr->hydrogen(); |
+ int32_t divisor = hdiv->right()->GetInteger32Constant(); |
+ Register result = ToRegister(instr->result()); |
+ ASSERT(!result.is(dividend)); |
- __ bind(&negative); |
- __ negl(dividend); |
- __ sarl(dividend, Immediate(power)); |
- if (divisor > 0) __ negl(dividend); |
- __ bind(&done); |
- return; // Don't fall through to "__ neg" below. |
- } else { |
- // Deoptimize if remainder is not 0. |
- __ testl(dividend, Immediate(test_value)); |
- DeoptimizeIf(not_zero, instr->environment()); |
- __ sarl(dividend, Immediate(power)); |
- } |
+ // Check for (0 / -x) that will produce negative zero. |
+ if (hdiv->left()->RangeCanInclude(0) && divisor < 0 && |
+ hdiv->CheckFlag(HValue::kBailoutOnMinusZero)) { |
+ __ testl(dividend, dividend); |
+ DeoptimizeIf(zero, instr->environment()); |
} |
- |
- if (divisor < 0) __ negl(dividend); |
- |
+ // Check for (kMinInt / -1). |
+ if (hdiv->left()->RangeCanInclude(kMinInt) && divisor == -1 && |
+ hdiv->CheckFlag(HValue::kCanOverflow)) { |
+ __ cmpl(dividend, Immediate(kMinInt)); |
+ DeoptimizeIf(zero, instr->environment()); |
+ } |
+ // Deoptimize if remainder will not be 0. |
+ if (!hdiv->CheckFlag(HInstruction::kAllUsesTruncatingToInt32)) { |
+ __ testl(dividend, Immediate(Abs(divisor) - 1)); |
+ DeoptimizeIf(not_zero, instr->environment()); |
+ } |
+ __ Move(result, dividend); |
+ int32_t shift = WhichPowerOf2(Abs(divisor)); |
+ if (shift > 0) { |
+ // The arithmetic shift is always OK, the 'if' is an optimization only. |
+ if (shift > 1) __ sarl(result, Immediate(31)); |
+ __ shrl(result, Immediate(32 - shift)); |
+ __ addl(result, dividend); |
+ __ sarl(result, Immediate(shift)); |
+ } |
+ if (divisor < 0) __ negl(result); |
return; |
} |