From: Davide Italiano Date: Thu, 19 Jan 2017 23:07:51 +0000 (+0000) Subject: [SCCP] Teach the pass how to handle `div` with overdefined operands. X-Git-Url: https://granicus.if.org/sourcecode?a=commitdiff_plain;h=2037eb7f158f6a7b745b1c8d7be2f7015d8bca38;p=llvm [SCCP] Teach the pass how to handle `div` with overdefined operands. This can prove that: extern int f; int g() { int x = 0; for (int i = 0; i < 365; ++i) { x /= f; } return x; } always returns zero. Thanks to Sanjoy for confirming this transformation actually made sense (bugs are mine). git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@292531 91177308-0d34-0410-b5e6-96231b3b80d8 --- diff --git a/lib/Transforms/Scalar/SCCP.cpp b/lib/Transforms/Scalar/SCCP.cpp index 9c5fa5c1426..334642fe35d 100644 --- a/lib/Transforms/Scalar/SCCP.cpp +++ b/lib/Transforms/Scalar/SCCP.cpp @@ -910,6 +910,12 @@ void SCCPSolver::visitBinaryOperator(Instruction &I) { // Otherwise, one of our operands is overdefined. Try to produce something // better than overdefined with some tricks. + // If this is 0 / Y, it doesn't matter that the second operand is + // overdefined, and we can replace it with zero. + if (I.getOpcode() == Instruction::UDiv || I.getOpcode() == Instruction::SDiv) + if (V1State.isConstant() && V1State.getConstant()->isNullValue()) + return markConstant(IV, &I, V1State.getConstant()); + // If this is: // -> AND/MUL with 0 // -> OR with -1 diff --git a/test/Transforms/SCCP/overdefined-div.ll b/test/Transforms/SCCP/overdefined-div.ll new file mode 100644 index 00000000000..f0b16155c17 --- /dev/null +++ b/test/Transforms/SCCP/overdefined-div.ll @@ -0,0 +1,32 @@ +; RUN: opt < %s -sccp -S | FileCheck %s + +; Test that SCCP has basic knowledge of when div can nuke overdefined values. + +; 0 / X = 0 even if X is overdefined. +; CHECK-LABEL: test1 +; CHECK-NEXT: ret i32 0 +define i32 @test1(i32 %foo) { + %tinkywinky = udiv i32 0, %foo + ret i32 %tinkywinky +} + +; CHECK-LABEL: test2 +; CHECK-NEXT: ret i32 0 +define i32 @test2(i32 %foo) { + %tinkywinky = sdiv i32 0, %foo + ret i32 %tinkywinky +} + +; CHECK-LABEL: test3 +; CHECK: ret i32 %tinkywinky +define i32 @test3(i32 %foo) { + %tinkywinky = udiv i32 %foo, 0 + ret i32 %tinkywinky +} + +; CHECK-LABEL: test4 +; CHECK: ret i32 %tinkywinky +define i32 @test4(i32 %foo) { + %tinkywinky = sdiv i32 %foo, 0 + ret i32 %tinkywinky +}