forked from OSchip/llvm-project
647 lines
18 KiB
LLVM
647 lines
18 KiB
LLVM
|
; RUN: opt -gvn-hoist -S < %s | FileCheck %s
|
||
|
target datalayout = "e-m:e-i64:64-f80:128-n8:16:32:64-S128"
|
||
|
target triple = "x86_64-unknown-linux-gnu"
|
||
|
|
||
|
@GlobalVar = internal global float 1.000000e+00
|
||
|
|
||
|
; Check that all scalar expressions are hoisted.
|
||
|
;
|
||
|
; CHECK-LABEL: @scalarsHoisting
|
||
|
; CHECK: fsub
|
||
|
; CHECK: fmul
|
||
|
; CHECK: fsub
|
||
|
; CHECK: fmul
|
||
|
; CHECK-NOT: fmul
|
||
|
; CHECK-NOT: fsub
|
||
|
define float @scalarsHoisting(float %d, float %min, float %max, float %a) {
|
||
|
entry:
|
||
|
%div = fdiv float 1.000000e+00, %d
|
||
|
%cmp = fcmp oge float %div, 0.000000e+00
|
||
|
br i1 %cmp, label %if.then, label %if.else
|
||
|
|
||
|
if.then: ; preds = %entry
|
||
|
%sub = fsub float %min, %a
|
||
|
%mul = fmul float %sub, %div
|
||
|
%sub1 = fsub float %max, %a
|
||
|
%mul2 = fmul float %sub1, %div
|
||
|
br label %if.end
|
||
|
|
||
|
if.else: ; preds = %entry
|
||
|
%sub3 = fsub float %max, %a
|
||
|
%mul4 = fmul float %sub3, %div
|
||
|
%sub5 = fsub float %min, %a
|
||
|
%mul6 = fmul float %sub5, %div
|
||
|
br label %if.end
|
||
|
|
||
|
if.end: ; preds = %if.else, %if.then
|
||
|
%tmax.0 = phi float [ %mul2, %if.then ], [ %mul6, %if.else ]
|
||
|
%tmin.0 = phi float [ %mul, %if.then ], [ %mul4, %if.else ]
|
||
|
%add = fadd float %tmax.0, %tmin.0
|
||
|
ret float %add
|
||
|
}
|
||
|
|
||
|
; Check that all loads and scalars depending on the loads are hoisted.
|
||
|
; Check that getelementptr computation gets hoisted before the load.
|
||
|
;
|
||
|
; CHECK-LABEL: @readsAndScalarsHoisting
|
||
|
; CHECK: load
|
||
|
; CHECK: load
|
||
|
; CHECK: load
|
||
|
; CHECK: fsub
|
||
|
; CHECK: fmul
|
||
|
; CHECK: fsub
|
||
|
; CHECK: fmul
|
||
|
; CHECK-NOT: load
|
||
|
; CHECK-NOT: fmul
|
||
|
; CHECK-NOT: fsub
|
||
|
define float @readsAndScalarsHoisting(float %d, float* %min, float* %max, float* %a) {
|
||
|
entry:
|
||
|
%div = fdiv float 1.000000e+00, %d
|
||
|
%cmp = fcmp oge float %div, 0.000000e+00
|
||
|
br i1 %cmp, label %if.then, label %if.else
|
||
|
|
||
|
if.then: ; preds = %entry
|
||
|
%A = getelementptr float, float* %min, i32 1
|
||
|
%0 = load float, float* %A, align 4
|
||
|
%1 = load float, float* %a, align 4
|
||
|
%sub = fsub float %0, %1
|
||
|
%mul = fmul float %sub, %div
|
||
|
%2 = load float, float* %max, align 4
|
||
|
%sub1 = fsub float %2, %1
|
||
|
%mul2 = fmul float %sub1, %div
|
||
|
br label %if.end
|
||
|
|
||
|
if.else: ; preds = %entry
|
||
|
%3 = load float, float* %max, align 4
|
||
|
%4 = load float, float* %a, align 4
|
||
|
%sub3 = fsub float %3, %4
|
||
|
%mul4 = fmul float %sub3, %div
|
||
|
%B = getelementptr float, float* %min, i32 1
|
||
|
%5 = load float, float* %B, align 4
|
||
|
%sub5 = fsub float %5, %4
|
||
|
%mul6 = fmul float %sub5, %div
|
||
|
br label %if.end
|
||
|
|
||
|
if.end: ; preds = %if.else, %if.then
|
||
|
%tmax.0 = phi float [ %mul2, %if.then ], [ %mul6, %if.else ]
|
||
|
%tmin.0 = phi float [ %mul, %if.then ], [ %mul4, %if.else ]
|
||
|
%add = fadd float %tmax.0, %tmin.0
|
||
|
ret float %add
|
||
|
}
|
||
|
|
||
|
; Check that we do not hoist loads after a store: the first two loads will be
|
||
|
; hoisted, and then the third load will not be hoisted.
|
||
|
;
|
||
|
; CHECK-LABEL: @readsAndWrites
|
||
|
; CHECK: load
|
||
|
; CHECK: load
|
||
|
; CHECK: fsub
|
||
|
; CHECK: fmul
|
||
|
; CHECK: store
|
||
|
; CHECK: load
|
||
|
; CHECK: fsub
|
||
|
; CHECK: fmul
|
||
|
; CHECK: load
|
||
|
; CHECK: fsub
|
||
|
; CHECK: fmul
|
||
|
; CHECK-NOT: load
|
||
|
; CHECK-NOT: fmul
|
||
|
; CHECK-NOT: fsub
|
||
|
define float @readsAndWrites(float %d, float* %min, float* %max, float* %a) {
|
||
|
entry:
|
||
|
%div = fdiv float 1.000000e+00, %d
|
||
|
%cmp = fcmp oge float %div, 0.000000e+00
|
||
|
br i1 %cmp, label %if.then, label %if.else
|
||
|
|
||
|
if.then: ; preds = %entry
|
||
|
%0 = load float, float* %min, align 4
|
||
|
%1 = load float, float* %a, align 4
|
||
|
store float %0, float* @GlobalVar
|
||
|
%sub = fsub float %0, %1
|
||
|
%mul = fmul float %sub, %div
|
||
|
%2 = load float, float* %max, align 4
|
||
|
%sub1 = fsub float %2, %1
|
||
|
%mul2 = fmul float %sub1, %div
|
||
|
br label %if.end
|
||
|
|
||
|
if.else: ; preds = %entry
|
||
|
%3 = load float, float* %max, align 4
|
||
|
%4 = load float, float* %a, align 4
|
||
|
%sub3 = fsub float %3, %4
|
||
|
%mul4 = fmul float %sub3, %div
|
||
|
%5 = load float, float* %min, align 4
|
||
|
%sub5 = fsub float %5, %4
|
||
|
%mul6 = fmul float %sub5, %div
|
||
|
br label %if.end
|
||
|
|
||
|
if.end: ; preds = %if.else, %if.then
|
||
|
%tmax.0 = phi float [ %mul2, %if.then ], [ %mul6, %if.else ]
|
||
|
%tmin.0 = phi float [ %mul, %if.then ], [ %mul4, %if.else ]
|
||
|
%add = fadd float %tmax.0, %tmin.0
|
||
|
ret float %add
|
||
|
}
|
||
|
|
||
|
; Check that we do hoist loads when the store is above the insertion point.
|
||
|
;
|
||
|
; CHECK-LABEL: @readsAndWriteAboveInsertPt
|
||
|
; CHECK: load
|
||
|
; CHECK: load
|
||
|
; CHECK: load
|
||
|
; CHECK: fsub
|
||
|
; CHECK: fmul
|
||
|
; CHECK: fsub
|
||
|
; CHECK: fmul
|
||
|
; CHECK-NOT: load
|
||
|
; CHECK-NOT: fmul
|
||
|
; CHECK-NOT: fsub
|
||
|
define float @readsAndWriteAboveInsertPt(float %d, float* %min, float* %max, float* %a) {
|
||
|
entry:
|
||
|
%div = fdiv float 1.000000e+00, %d
|
||
|
store float 0.000000e+00, float* @GlobalVar
|
||
|
%cmp = fcmp oge float %div, 0.000000e+00
|
||
|
br i1 %cmp, label %if.then, label %if.else
|
||
|
|
||
|
if.then: ; preds = %entry
|
||
|
%0 = load float, float* %min, align 4
|
||
|
%1 = load float, float* %a, align 4
|
||
|
%sub = fsub float %0, %1
|
||
|
%mul = fmul float %sub, %div
|
||
|
%2 = load float, float* %max, align 4
|
||
|
%sub1 = fsub float %2, %1
|
||
|
%mul2 = fmul float %sub1, %div
|
||
|
br label %if.end
|
||
|
|
||
|
if.else: ; preds = %entry
|
||
|
%3 = load float, float* %max, align 4
|
||
|
%4 = load float, float* %a, align 4
|
||
|
%sub3 = fsub float %3, %4
|
||
|
%mul4 = fmul float %sub3, %div
|
||
|
%5 = load float, float* %min, align 4
|
||
|
%sub5 = fsub float %5, %4
|
||
|
%mul6 = fmul float %sub5, %div
|
||
|
br label %if.end
|
||
|
|
||
|
if.end: ; preds = %if.else, %if.then
|
||
|
%tmax.0 = phi float [ %mul2, %if.then ], [ %mul6, %if.else ]
|
||
|
%tmin.0 = phi float [ %mul, %if.then ], [ %mul4, %if.else ]
|
||
|
%add = fadd float %tmax.0, %tmin.0
|
||
|
ret float %add
|
||
|
}
|
||
|
|
||
|
; Check that dependent expressions are hoisted.
|
||
|
; CHECK-LABEL: @dependentScalarsHoisting
|
||
|
; CHECK: fsub
|
||
|
; CHECK: fadd
|
||
|
; CHECK: fdiv
|
||
|
; CHECK: fmul
|
||
|
; CHECK-NOT: fsub
|
||
|
; CHECK-NOT: fadd
|
||
|
; CHECK-NOT: fdiv
|
||
|
; CHECK-NOT: fmul
|
||
|
define float @dependentScalarsHoisting(float %a, float %b, i1 %c) {
|
||
|
entry:
|
||
|
br i1 %c, label %if.then, label %if.else
|
||
|
|
||
|
if.then:
|
||
|
%d = fsub float %b, %a
|
||
|
%e = fadd float %d, %a
|
||
|
%f = fdiv float %e, %a
|
||
|
%g = fmul float %f, %a
|
||
|
br label %if.end
|
||
|
|
||
|
if.else:
|
||
|
%h = fsub float %b, %a
|
||
|
%i = fadd float %h, %a
|
||
|
%j = fdiv float %i, %a
|
||
|
%k = fmul float %j, %a
|
||
|
br label %if.end
|
||
|
|
||
|
if.end:
|
||
|
%r = phi float [ %g, %if.then ], [ %k, %if.else ]
|
||
|
ret float %r
|
||
|
}
|
||
|
|
||
|
; Check that all independent expressions are hoisted.
|
||
|
; CHECK-LABEL: @independentScalarsHoisting
|
||
|
; CHECK: fsub
|
||
|
; CHECK: fdiv
|
||
|
; CHECK: fmul
|
||
|
; CHECK: fadd
|
||
|
; CHECK-NOT: fsub
|
||
|
; CHECK-NOT: fdiv
|
||
|
; CHECK-NOT: fmul
|
||
|
define float @independentScalarsHoisting(float %a, float %b, i1 %c) {
|
||
|
entry:
|
||
|
br i1 %c, label %if.then, label %if.else
|
||
|
|
||
|
if.then:
|
||
|
%d = fadd float %b, %a
|
||
|
%e = fsub float %b, %a
|
||
|
%f = fdiv float %b, %a
|
||
|
%g = fmul float %b, %a
|
||
|
br label %if.end
|
||
|
|
||
|
if.else:
|
||
|
%i = fadd float %b, %a
|
||
|
%h = fsub float %b, %a
|
||
|
%j = fdiv float %b, %a
|
||
|
%k = fmul float %b, %a
|
||
|
br label %if.end
|
||
|
|
||
|
if.end:
|
||
|
%p = phi float [ %d, %if.then ], [ %i, %if.else ]
|
||
|
%q = phi float [ %e, %if.then ], [ %h, %if.else ]
|
||
|
%r = phi float [ %f, %if.then ], [ %j, %if.else ]
|
||
|
%s = phi float [ %g, %if.then ], [ %k, %if.else ]
|
||
|
%t = fadd float %p, %q
|
||
|
%u = fadd float %r, %s
|
||
|
%v = fadd float %t, %u
|
||
|
ret float %v
|
||
|
}
|
||
|
|
||
|
; Check that we hoist load and scalar expressions in triangles.
|
||
|
; CHECK-LABEL: @triangleHoisting
|
||
|
; CHECK: load
|
||
|
; CHECK: load
|
||
|
; CHECK: load
|
||
|
; CHECK: fsub
|
||
|
; CHECK: fmul
|
||
|
; CHECK: fsub
|
||
|
; CHECK: fmul
|
||
|
; CHECK-NOT: load
|
||
|
; CHECK-NOT: fmul
|
||
|
; CHECK-NOT: fsub
|
||
|
define float @triangleHoisting(float %d, float* %min, float* %max, float* %a) {
|
||
|
entry:
|
||
|
%div = fdiv float 1.000000e+00, %d
|
||
|
%cmp = fcmp oge float %div, 0.000000e+00
|
||
|
br i1 %cmp, label %if.then, label %if.end
|
||
|
|
||
|
if.then: ; preds = %entry
|
||
|
%0 = load float, float* %min, align 4
|
||
|
%1 = load float, float* %a, align 4
|
||
|
%sub = fsub float %0, %1
|
||
|
%mul = fmul float %sub, %div
|
||
|
%2 = load float, float* %max, align 4
|
||
|
%sub1 = fsub float %2, %1
|
||
|
%mul2 = fmul float %sub1, %div
|
||
|
br label %if.end
|
||
|
|
||
|
if.end: ; preds = %entry
|
||
|
%p1 = phi float [ %mul2, %if.then ], [ 0.000000e+00, %entry ]
|
||
|
%p2 = phi float [ %mul, %if.then ], [ 0.000000e+00, %entry ]
|
||
|
%3 = load float, float* %max, align 4
|
||
|
%4 = load float, float* %a, align 4
|
||
|
%sub3 = fsub float %3, %4
|
||
|
%mul4 = fmul float %sub3, %div
|
||
|
%5 = load float, float* %min, align 4
|
||
|
%sub5 = fsub float %5, %4
|
||
|
%mul6 = fmul float %sub5, %div
|
||
|
|
||
|
%x = fadd float %p1, %mul6
|
||
|
%y = fadd float %p2, %mul4
|
||
|
%z = fadd float %x, %y
|
||
|
ret float %z
|
||
|
}
|
||
|
|
||
|
; Check that we do not hoist loads past stores within a same basic block.
|
||
|
; CHECK-LABEL: @noHoistInSingleBBWithStore
|
||
|
; CHECK: load
|
||
|
; CHECK: store
|
||
|
; CHECK: load
|
||
|
; CHECK: store
|
||
|
define i32 @noHoistInSingleBBWithStore() {
|
||
|
entry:
|
||
|
%D = alloca i32, align 4
|
||
|
%0 = bitcast i32* %D to i8*
|
||
|
%bf = load i8, i8* %0, align 4
|
||
|
%bf.clear = and i8 %bf, -3
|
||
|
store i8 %bf.clear, i8* %0, align 4
|
||
|
%bf1 = load i8, i8* %0, align 4
|
||
|
%bf.clear1 = and i8 %bf1, 1
|
||
|
store i8 %bf.clear1, i8* %0, align 4
|
||
|
ret i32 0
|
||
|
}
|
||
|
|
||
|
; Check that we do not hoist loads past calls within a same basic block.
|
||
|
; CHECK-LABEL: @noHoistInSingleBBWithCall
|
||
|
; CHECK: load
|
||
|
; CHECK: call
|
||
|
; CHECK: load
|
||
|
declare void @foo()
|
||
|
define i32 @noHoistInSingleBBWithCall() {
|
||
|
entry:
|
||
|
%D = alloca i32, align 4
|
||
|
%0 = bitcast i32* %D to i8*
|
||
|
%bf = load i8, i8* %0, align 4
|
||
|
%bf.clear = and i8 %bf, -3
|
||
|
call void @foo()
|
||
|
%bf1 = load i8, i8* %0, align 4
|
||
|
%bf.clear1 = and i8 %bf1, 1
|
||
|
ret i32 0
|
||
|
}
|
||
|
|
||
|
; Check that we do not hoist loads past stores in any branch of a diamond.
|
||
|
; CHECK-LABEL: @noHoistInDiamondWithOneStore1
|
||
|
; CHECK: fdiv
|
||
|
; CHECK: fcmp
|
||
|
; CHECK: br
|
||
|
define float @noHoistInDiamondWithOneStore1(float %d, float* %min, float* %max, float* %a) {
|
||
|
entry:
|
||
|
%div = fdiv float 1.000000e+00, %d
|
||
|
%cmp = fcmp oge float %div, 0.000000e+00
|
||
|
br i1 %cmp, label %if.then, label %if.else
|
||
|
|
||
|
if.then: ; preds = %entry
|
||
|
store float 0.000000e+00, float* @GlobalVar
|
||
|
%0 = load float, float* %min, align 4
|
||
|
%1 = load float, float* %a, align 4
|
||
|
%sub = fsub float %0, %1
|
||
|
%mul = fmul float %sub, %div
|
||
|
%2 = load float, float* %max, align 4
|
||
|
%sub1 = fsub float %2, %1
|
||
|
%mul2 = fmul float %sub1, %div
|
||
|
br label %if.end
|
||
|
|
||
|
if.else: ; preds = %entry
|
||
|
; There are no side effects on the if.else branch.
|
||
|
%3 = load float, float* %max, align 4
|
||
|
%4 = load float, float* %a, align 4
|
||
|
%sub3 = fsub float %3, %4
|
||
|
%mul4 = fmul float %sub3, %div
|
||
|
%5 = load float, float* %min, align 4
|
||
|
%sub5 = fsub float %5, %4
|
||
|
%mul6 = fmul float %sub5, %div
|
||
|
br label %if.end
|
||
|
|
||
|
if.end: ; preds = %if.else, %if.then
|
||
|
%tmax.0 = phi float [ %mul2, %if.then ], [ %mul6, %if.else ]
|
||
|
%tmin.0 = phi float [ %mul, %if.then ], [ %mul4, %if.else ]
|
||
|
|
||
|
%6 = load float, float* %max, align 4
|
||
|
%7 = load float, float* %a, align 4
|
||
|
%sub6 = fsub float %6, %7
|
||
|
%mul7 = fmul float %sub6, %div
|
||
|
%8 = load float, float* %min, align 4
|
||
|
%sub8 = fsub float %8, %7
|
||
|
%mul9 = fmul float %sub8, %div
|
||
|
|
||
|
%add = fadd float %tmax.0, %tmin.0
|
||
|
ret float %add
|
||
|
}
|
||
|
|
||
|
; Check that we do not hoist loads past stores from half diamond.
|
||
|
; CHECK-LABEL: @noHoistInHalfDiamondPastStore
|
||
|
; CHECK: load
|
||
|
; CHECK-NEXT: load
|
||
|
; CHECK-NEXT: store
|
||
|
; CHECK-NEXT: br
|
||
|
; CHECK: load
|
||
|
; CHECK: load
|
||
|
; CHECK: load
|
||
|
; CHECK: br
|
||
|
define float @noHoistInHalfDiamondPastStore(float %d, float* %min, float* %max, float* %a) {
|
||
|
entry:
|
||
|
%div = fdiv float 1.000000e+00, %d
|
||
|
%cmp = fcmp oge float %div, 0.000000e+00
|
||
|
%0 = load float, float* %min, align 4
|
||
|
%1 = load float, float* %a, align 4
|
||
|
|
||
|
; Loads should not be hoisted above this store.
|
||
|
store float 0.000000e+00, float* @GlobalVar
|
||
|
|
||
|
br i1 %cmp, label %if.then, label %if.end
|
||
|
|
||
|
if.then:
|
||
|
; There are no side effects on the if.then branch.
|
||
|
%2 = load float, float* %max, align 4
|
||
|
%3 = load float, float* %a, align 4
|
||
|
%sub3 = fsub float %2, %3
|
||
|
%mul4 = fmul float %sub3, %div
|
||
|
%4 = load float, float* %min, align 4
|
||
|
%sub5 = fsub float %4, %3
|
||
|
%mul6 = fmul float %sub5, %div
|
||
|
br label %if.end
|
||
|
|
||
|
if.end:
|
||
|
%tmax.0 = phi float [ %mul4, %if.then ], [ %0, %entry ]
|
||
|
%tmin.0 = phi float [ %mul6, %if.then ], [ %1, %entry ]
|
||
|
|
||
|
%add = fadd float %tmax.0, %tmin.0
|
||
|
ret float %add
|
||
|
}
|
||
|
|
||
|
; Check that we do not hoist loads past a store in any branch of a diamond.
|
||
|
; CHECK-LABEL: @noHoistInDiamondWithOneStore2
|
||
|
; CHECK: fdiv
|
||
|
; CHECK: fcmp
|
||
|
; CHECK: br
|
||
|
define float @noHoistInDiamondWithOneStore2(float %d, float* %min, float* %max, float* %a) {
|
||
|
entry:
|
||
|
%div = fdiv float 1.000000e+00, %d
|
||
|
%cmp = fcmp oge float %div, 0.000000e+00
|
||
|
br i1 %cmp, label %if.then, label %if.else
|
||
|
|
||
|
if.then: ; preds = %entry
|
||
|
; There are no side effects on the if.then branch.
|
||
|
%0 = load float, float* %min, align 4
|
||
|
%1 = load float, float* %a, align 4
|
||
|
%sub = fsub float %0, %1
|
||
|
%mul = fmul float %sub, %div
|
||
|
%2 = load float, float* %max, align 4
|
||
|
%sub1 = fsub float %2, %1
|
||
|
%mul2 = fmul float %sub1, %div
|
||
|
br label %if.end
|
||
|
|
||
|
if.else: ; preds = %entry
|
||
|
store float 0.000000e+00, float* @GlobalVar
|
||
|
%3 = load float, float* %max, align 4
|
||
|
%4 = load float, float* %a, align 4
|
||
|
%sub3 = fsub float %3, %4
|
||
|
%mul4 = fmul float %sub3, %div
|
||
|
%5 = load float, float* %min, align 4
|
||
|
%sub5 = fsub float %5, %4
|
||
|
%mul6 = fmul float %sub5, %div
|
||
|
br label %if.end
|
||
|
|
||
|
if.end: ; preds = %if.else, %if.then
|
||
|
%tmax.0 = phi float [ %mul2, %if.then ], [ %mul6, %if.else ]
|
||
|
%tmin.0 = phi float [ %mul, %if.then ], [ %mul4, %if.else ]
|
||
|
|
||
|
%6 = load float, float* %max, align 4
|
||
|
%7 = load float, float* %a, align 4
|
||
|
%sub6 = fsub float %6, %7
|
||
|
%mul7 = fmul float %sub6, %div
|
||
|
%8 = load float, float* %min, align 4
|
||
|
%sub8 = fsub float %8, %7
|
||
|
%mul9 = fmul float %sub8, %div
|
||
|
|
||
|
%add = fadd float %tmax.0, %tmin.0
|
||
|
ret float %add
|
||
|
}
|
||
|
|
||
|
; Check that we do not hoist loads outside a loop containing stores.
|
||
|
; CHECK-LABEL: @noHoistInLoopsWithStores
|
||
|
; CHECK: fdiv
|
||
|
; CHECK: fcmp
|
||
|
; CHECK: br
|
||
|
define float @noHoistInLoopsWithStores(float %d, float* %min, float* %max, float* %a) {
|
||
|
entry:
|
||
|
%div = fdiv float 1.000000e+00, %d
|
||
|
%cmp = fcmp oge float %div, 0.000000e+00
|
||
|
br i1 %cmp, label %do.body, label %if.else
|
||
|
|
||
|
do.body:
|
||
|
%0 = load float, float* %min, align 4
|
||
|
%1 = load float, float* %a, align 4
|
||
|
|
||
|
; It is unsafe to hoist the loads outside the loop because of the store.
|
||
|
store float 0.000000e+00, float* @GlobalVar
|
||
|
|
||
|
%sub = fsub float %0, %1
|
||
|
%mul = fmul float %sub, %div
|
||
|
%2 = load float, float* %max, align 4
|
||
|
%sub1 = fsub float %2, %1
|
||
|
%mul2 = fmul float %sub1, %div
|
||
|
br label %while.cond
|
||
|
|
||
|
while.cond:
|
||
|
%cmp1 = fcmp oge float %mul2, 0.000000e+00
|
||
|
br i1 %cmp1, label %if.end, label %do.body
|
||
|
|
||
|
if.else:
|
||
|
%3 = load float, float* %max, align 4
|
||
|
%4 = load float, float* %a, align 4
|
||
|
%sub3 = fsub float %3, %4
|
||
|
%mul4 = fmul float %sub3, %div
|
||
|
%5 = load float, float* %min, align 4
|
||
|
%sub5 = fsub float %5, %4
|
||
|
%mul6 = fmul float %sub5, %div
|
||
|
br label %if.end
|
||
|
|
||
|
if.end:
|
||
|
%tmax.0 = phi float [ %mul2, %while.cond ], [ %mul6, %if.else ]
|
||
|
%tmin.0 = phi float [ %mul, %while.cond ], [ %mul4, %if.else ]
|
||
|
|
||
|
%add = fadd float %tmax.0, %tmin.0
|
||
|
ret float %add
|
||
|
}
|
||
|
|
||
|
; Check that we hoist stores: all the instructions from the then branch
|
||
|
; should be hoisted.
|
||
|
; CHECK-LABEL: @hoistStores
|
||
|
; CHECK: zext
|
||
|
; CHECK-NEXT: trunc
|
||
|
; CHECK-NEXT: getelementptr
|
||
|
; CHECK-NEXT: load
|
||
|
; CHECK-NEXT: getelementptr
|
||
|
; CHECK-NEXT: getelementptr
|
||
|
; CHECK-NEXT: store
|
||
|
; CHECK-NEXT: load
|
||
|
; CHECK-NEXT: load
|
||
|
; CHECK-NEXT: zext
|
||
|
; CHECK-NEXT: add
|
||
|
; CHECK-NEXT: store
|
||
|
; CHECK-NEXT: br
|
||
|
; CHECK: if.then
|
||
|
; CHECK: br
|
||
|
|
||
|
%struct.foo = type { i16* }
|
||
|
|
||
|
define void @hoistStores(%struct.foo* %s, i32* %coord, i1 zeroext %delta) {
|
||
|
entry:
|
||
|
%frombool = zext i1 %delta to i8
|
||
|
%tobool = trunc i8 %frombool to i1
|
||
|
br i1 %tobool, label %if.then, label %if.else
|
||
|
|
||
|
if.then: ; preds = %entry
|
||
|
%p = getelementptr inbounds %struct.foo, %struct.foo* %s, i32 0, i32 0
|
||
|
%0 = load i16*, i16** %p, align 8
|
||
|
%incdec.ptr = getelementptr inbounds i16, i16* %0, i32 1
|
||
|
store i16* %incdec.ptr, i16** %p, align 8
|
||
|
%1 = load i16, i16* %0, align 2
|
||
|
%conv = zext i16 %1 to i32
|
||
|
%2 = load i32, i32* %coord, align 4
|
||
|
%add = add i32 %2, %conv
|
||
|
store i32 %add, i32* %coord, align 4
|
||
|
br label %if.end
|
||
|
|
||
|
if.else: ; preds = %entry
|
||
|
%p1 = getelementptr inbounds %struct.foo, %struct.foo* %s, i32 0, i32 0
|
||
|
%3 = load i16*, i16** %p1, align 8
|
||
|
%incdec.ptr2 = getelementptr inbounds i16, i16* %3, i32 1
|
||
|
store i16* %incdec.ptr2, i16** %p1, align 8
|
||
|
%4 = load i16, i16* %3, align 2
|
||
|
%conv3 = zext i16 %4 to i32
|
||
|
%5 = load i32, i32* %coord, align 4
|
||
|
%add4 = add i32 %5, %conv3
|
||
|
store i32 %add4, i32* %coord, align 4
|
||
|
%6 = load i16*, i16** %p1, align 8
|
||
|
%incdec.ptr6 = getelementptr inbounds i16, i16* %6, i32 1
|
||
|
store i16* %incdec.ptr6, i16** %p1, align 8
|
||
|
%7 = load i16, i16* %6, align 2
|
||
|
%conv7 = zext i16 %7 to i32
|
||
|
%shl = shl i32 %conv7, 8
|
||
|
%8 = load i32, i32* %coord, align 4
|
||
|
%add8 = add i32 %8, %shl
|
||
|
store i32 %add8, i32* %coord, align 4
|
||
|
br label %if.end
|
||
|
|
||
|
if.end: ; preds = %if.else, %if.then
|
||
|
ret void
|
||
|
}
|
||
|
|
||
|
define i32 @mergeAlignments(i1 %b, i32* %y) {
|
||
|
entry:
|
||
|
br i1 %b, label %if.then, label %if.end
|
||
|
|
||
|
if.then: ; preds = %entry
|
||
|
%l1 = load i32, i32* %y, align 4
|
||
|
br label %return
|
||
|
|
||
|
if.end: ; preds = %entry
|
||
|
%l2 = load i32, i32* %y, align 1
|
||
|
br label %return
|
||
|
|
||
|
return: ; preds = %if.end, %if.then
|
||
|
%retval.0 = phi i32 [ %l1, %if.then ], [ %l2, %if.end ]
|
||
|
ret i32 %retval.0
|
||
|
}
|
||
|
; CHECK-LABEL: define i32 @mergeAlignments(
|
||
|
; CHECK: %[[load:.*]] = load i32, i32* %y, align 1
|
||
|
; CHECK: %[[phi:.*]] = phi i32 [ %[[load]], %{{.*}} ], [ %[[load]], %{{.*}} ]
|
||
|
; CHECK: i32 %[[phi]]
|
||
|
|
||
|
|
||
|
declare i8 @pr30991_f() nounwind readonly
|
||
|
declare void @pr30991_f1(i8)
|
||
|
define i8 @pr30991(i8* %sp, i8* %word, i1 %b1, i1 %b2) {
|
||
|
entry:
|
||
|
br i1 %b1, label %a, label %b
|
||
|
|
||
|
a:
|
||
|
%r0 = load i8, i8* %word, align 1
|
||
|
%incdec.ptr = getelementptr i8, i8* %sp, i32 1
|
||
|
%rr0 = call i8 @pr30991_f() nounwind readonly
|
||
|
call void @pr30991_f1(i8 %r0)
|
||
|
ret i8 %rr0
|
||
|
|
||
|
b:
|
||
|
br i1 %b2, label %c, label %x
|
||
|
|
||
|
c:
|
||
|
%r1 = load i8, i8* %word, align 1
|
||
|
%incdec.ptr115 = getelementptr i8, i8* %sp, i32 1
|
||
|
%rr1 = call i8 @pr30991_f() nounwind readonly
|
||
|
call void @pr30991_f1(i8 %r1)
|
||
|
ret i8 %rr1
|
||
|
|
||
|
x:
|
||
|
%r2 = load i8, i8* %word, align 1
|
||
|
ret i8 %r2
|
||
|
}
|
||
|
|
||
|
; CHECK-LABEL: define i8 @pr30991
|
||
|
; CHECK: %r0 = load i8, i8* %word, align 1
|
||
|
; CHECK-NEXT: br i1 %b1, label %a, label %b
|