Commit 9e14155d authored by epertoso's avatar epertoso Committed by Commit bot

[turbofan] Fix CheckedInt32Mod lowering.

We now deopt when the lhs of a mod is negative and the rhs is 1 too (previously, we erroneusly returned 0 instead of -0).

BUG=v8:5278
R=bmeurer@chromium.org

Review-Url: https://codereview.chromium.org/2233713002
Cr-Commit-Position: refs/heads/master@{#38525}
parent 2b15dd52
...@@ -1307,13 +1307,13 @@ EffectControlLinearizer::ValueEffectControl ...@@ -1307,13 +1307,13 @@ EffectControlLinearizer::ValueEffectControl
EffectControlLinearizer::LowerCheckedInt32Mod(Node* node, Node* frame_state, EffectControlLinearizer::LowerCheckedInt32Mod(Node* node, Node* frame_state,
Node* effect, Node* control) { Node* effect, Node* control) {
Node* zero = jsgraph()->Int32Constant(0); Node* zero = jsgraph()->Int32Constant(0);
Node* one = jsgraph()->Int32Constant(1);
Node* minusone = jsgraph()->Int32Constant(-1); Node* minusone = jsgraph()->Int32Constant(-1);
Node* minint = jsgraph()->Int32Constant(std::numeric_limits<int32_t>::min());
// General case for signed integer modulus, with optimization for (unknown) // General case for signed integer modulus, with optimization for (unknown)
// power of 2 right hand side. // power of 2 right hand side.
// //
// if 0 < rhs then // if 1 < rhs then
// msk = rhs - 1 // msk = rhs - 1
// if rhs & msk == 0 then // if rhs & msk == 0 then
// if lhs < 0 then // if lhs < 0 then
...@@ -1327,14 +1327,14 @@ EffectControlLinearizer::LowerCheckedInt32Mod(Node* node, Node* frame_state, ...@@ -1327,14 +1327,14 @@ EffectControlLinearizer::LowerCheckedInt32Mod(Node* node, Node* frame_state,
// lhs % rhs // lhs % rhs
// else // else
// deopt if rhs == 0 // deopt if rhs == 0
// deopt if lhs == minint // deopt if lhs < 0
// zero // zero
// //
Node* lhs = node->InputAt(0); Node* lhs = node->InputAt(0);
Node* rhs = node->InputAt(1); Node* rhs = node->InputAt(1);
// Check if {rhs} is strictly positive. // Check if {rhs} is strictly greater than one.
Node* check0 = graph()->NewNode(machine()->Int32LessThan(), zero, rhs); Node* check0 = graph()->NewNode(machine()->Int32LessThan(), one, rhs);
Node* branch0 = Node* branch0 =
graph()->NewNode(common()->Branch(BranchHint::kTrue), check0, control); graph()->NewNode(common()->Branch(BranchHint::kTrue), check0, control);
...@@ -1410,9 +1410,9 @@ EffectControlLinearizer::LowerCheckedInt32Mod(Node* node, Node* frame_state, ...@@ -1410,9 +1410,9 @@ EffectControlLinearizer::LowerCheckedInt32Mod(Node* node, Node* frame_state,
common()->DeoptimizeIf(DeoptimizeReason::kDivisionByZero), check2, common()->DeoptimizeIf(DeoptimizeReason::kDivisionByZero), check2,
frame_state, efalse1, if_false1); frame_state, efalse1, if_false1);
// Now we know that {rhs} is -1, so make sure {lhs} is not kMinInt, as // Now we know that {rhs} is -1, so make sure {lhs} is >= 0, as we would
// we would otherwise have to return -0. // otherwise have to return -0.
Node* check3 = graph()->NewNode(machine()->Word32Equal(), lhs, minint); Node* check3 = graph()->NewNode(machine()->Int32LessThan(), lhs, zero);
if_false1 = efalse1 = if_false1 = efalse1 =
graph()->NewNode(common()->DeoptimizeIf(DeoptimizeReason::kMinusZero), graph()->NewNode(common()->DeoptimizeIf(DeoptimizeReason::kMinusZero),
check3, frame_state, efalse1, if_false1); check3, frame_state, efalse1, if_false1);
......
// Copyright 2016 the V8 project authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
// Flags: --allow-natives-syntax
function foo(a, b) {
return a % b;
}
foo(2, 1);
foo(2, 1);
%OptimizeFunctionOnNextCall(foo);
assertEquals(-0, foo(-2, 1));
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment