llvm/test/Transforms/InstCombine/pr27996.ll
Guozhi Wei 413fdf3b7f [InstCombine] Resubmit the combine of A->B->A BitCast and fix for pr27996
The original patch of the A->B->A BitCast optimization was reverted by r274094 because it may cause infinite loop inside compiler https://llvm.org/bugs/show_bug.cgi?id=27996.

The problem is with following code

xB = load (type B); 
xA = load (type A); 
+yA = (A)xB; B -> A
+zAn = PHI[yA, xA]; PHI 
+zBn = (B)zAn; // A -> B
store zAn;
store zBn;

optimizeBitCastFromPhi generates

+zBn = (B)zAn; // A -> B

and expects it will be combined with the following store instruction to another

store zAn 

Unfortunately before combineStoreToValueType is called on the store instruction, optimizeBitCastFromPhi is called on the new BitCast again, and this pattern repeats indefinitely.

optimizeBitCastFromPhi only generates BitCast for load/store instructions, only the BitCast before store can cause the reexecution of optimizeBitCastFromPhi, and BitCast before store can easily be handled by InstCombineLoadStoreAlloca.cpp. So the solution to the problem is if all users of a CI are store instructions, we should not do optimizeBitCastFromPhi on it. Then optimizeBitCastFromPhi will not be called on the new BitCast instructions.

Differential Revision: https://reviews.llvm.org/D23896



git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@285116 91177308-0d34-0410-b5e6-96231b3b80d8
2016-10-25 20:43:42 +00:00

42 lines
950 B
LLVM

; RUN: opt < %s -instcombine -S | FileCheck %s
@i = constant i32 1, align 4
@f = constant float 0x3FF19999A0000000, align 4
@cmp = common global i32 0, align 4
@resf = common global float* null, align 8
@resi = common global i32* null, align 8
define i32 @foo() {
entry:
br label %while.cond
while.cond:
%res.0 = phi i32* [ null, %entry ], [ @i, %if.then ], [ bitcast (float* @f to i32*), %if.else ]
%0 = load i32, i32* @cmp, align 4
%shr = ashr i32 %0, 1
store i32 %shr, i32* @cmp, align 4
%tobool = icmp ne i32 %shr, 0
br i1 %tobool, label %while.body, label %while.end
while.body:
%and = and i32 %shr, 1
%tobool1 = icmp ne i32 %and, 0
br i1 %tobool1, label %if.then, label %if.else
if.then:
br label %while.cond
if.else:
br label %while.cond
while.end:
%1 = bitcast i32* %res.0 to float*
store float* %1, float** @resf, align 8
store i32* %res.0, i32** @resi, align 8
ret i32 0
; CHECK-NOT: bitcast i32
}