From: Davide Italiano Date: Fri, 9 Dec 2016 03:08:42 +0000 (+0000) Subject: [SCCP] Teach the pass about `mul %x 0` even if %x is overdefined. X-Git-Url: https://granicus.if.org/sourcecode?a=commitdiff_plain;h=45b7e0e48d51a4dc13bf19f08c421c9df7254038;p=llvm [SCCP] Teach the pass about `mul %x 0` even if %x is overdefined. The motivating example is: extern int patatino; int goo() { int x = 0; for (int i = 0; i < 1000000; ++i) { x *= patatino; } return x; } Currently SCCP will not realize that this function returns always zero, therefore will try to unroll and vectorize the loop at -O3 producing an awful lot of (useless) code. With this change, it will just produce: 0000000000000000 : xor %eax,%eax retq git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@289175 91177308-0d34-0410-b5e6-96231b3b80d8 --- diff --git a/lib/Transforms/Scalar/SCCP.cpp b/lib/Transforms/Scalar/SCCP.cpp index 2c1364d77cf..279c710f9e1 100644 --- a/lib/Transforms/Scalar/SCCP.cpp +++ b/lib/Transforms/Scalar/SCCP.cpp @@ -916,7 +916,8 @@ void SCCPSolver::visitBinaryOperator(Instruction &I) { // If this is an AND or OR with 0 or -1, it doesn't matter that the other // operand is overdefined. - if (I.getOpcode() == Instruction::And || I.getOpcode() == Instruction::Or) { + if (I.getOpcode() == Instruction::And || I.getOpcode() == Instruction::Mul || + I.getOpcode() == Instruction::Or) { LatticeVal *NonOverdefVal = nullptr; if (!V1State.isOverdefined()) NonOverdefVal = &V1State; @@ -927,8 +928,10 @@ void SCCPSolver::visitBinaryOperator(Instruction &I) { if (NonOverdefVal->isUnknown()) return; - if (I.getOpcode() == Instruction::And) { + if (I.getOpcode() == Instruction::And || + I.getOpcode() == Instruction::Mul) { // X and 0 = 0 + // X * 0 = 0 if (NonOverdefVal->getConstant()->isNullValue()) return markConstant(IV, &I, NonOverdefVal->getConstant()); } else { diff --git a/test/Transforms/SCCP/logical-nuke.ll b/test/Transforms/SCCP/logical-nuke.ll index 4ef52a25539..76884dbab4b 100644 --- a/test/Transforms/SCCP/logical-nuke.ll +++ b/test/Transforms/SCCP/logical-nuke.ll @@ -1,6 +1,6 @@ ; RUN: opt < %s -sccp -S | FileCheck %s -; Test that SCCP has basic knowledge of when and/or nuke overdefined values. +; Test that SCCP has basic knowledge of when and/or/mul nuke overdefined values. ; CHECK-LABEL: test ; CHECK: ret i32 0 @@ -29,3 +29,10 @@ define i32 @test4(i32 %X) { %Y = or i32 %X, undef ret i32 %Y } + +; CHECK-LABEL: test5 +; CHECK: ret i32 0 +define i32 @test5(i32 %foo) { + %patatino = mul i32 %foo, undef + ret i32 %patatino +}