107 lines
3.6 KiB
LLVM
107 lines
3.6 KiB
LLVM
; RUN: opt -gvn-hoist -newgvn -gvn-hoist -S < %s | FileCheck %s
|
|
|
|
; Check that recursive GEPs are hoisted. Since hoisting creates
|
|
; fully redundant instructions, newgvn is run to remove them which then
|
|
; creates more opportunites for hoisting.
|
|
|
|
; CHECK-LABEL: @fun
|
|
; CHECK: load
|
|
; CHECK: fdiv
|
|
; CHECK: load
|
|
; CHECK: load
|
|
; CHECK: load
|
|
; CHECK: fsub
|
|
; CHECK: fmul
|
|
; CHECK: fsub
|
|
; CHECK: fmul
|
|
; CHECK-NOT: fsub
|
|
; CHECK-NOT: fmul
|
|
|
|
%0 = type { double, double, double }
|
|
%1 = type { double, double, double }
|
|
%2 = type { %3, %1, %1 }
|
|
%3 = type { i32 (...)**, %4, %10*, %11, %11, %11, %11, %11, %11, %11, %11, %11 }
|
|
%4 = type { %5 }
|
|
%5 = type { %6 }
|
|
%6 = type { %7 }
|
|
%7 = type { %8 }
|
|
%8 = type { %9 }
|
|
%9 = type { i64, i64, i8* }
|
|
%10 = type <{ i32 (...)**, i32, [4 x i8] }>
|
|
%11 = type { [4 x [4 x double]] }
|
|
%12 = type <{ %1, %0, i32, [4 x i8] }>
|
|
%13 = type { %1, %0, %12, %3*, %14* }
|
|
%14 = type opaque
|
|
|
|
@d = external global %0, align 8
|
|
@p = external global %1, align 8
|
|
|
|
define zeroext i1 @fun(%2*, %12* dereferenceable(56), double*, %13*) {
|
|
%5 = alloca %2*, align 8
|
|
%6 = alloca %12*, align 8
|
|
%7 = alloca double*, align 8
|
|
%8 = alloca %13*, align 8
|
|
%9 = alloca double, align 8
|
|
%10 = alloca double, align 8
|
|
%11 = alloca double, align 8
|
|
%12 = alloca double, align 8
|
|
%13 = alloca double, align 8
|
|
%14 = alloca double, align 8
|
|
%15 = alloca double, align 8
|
|
store %2* %0, %2** %5, align 8
|
|
store %12* %1, %12** %6, align 8
|
|
store double* %2, double** %7, align 8
|
|
store %13* %3, %13** %8, align 8
|
|
%16 = load %2*, %2** %5, align 8
|
|
%17 = load double, double* getelementptr inbounds (%0, %0* @d, i32 0, i32 0), align 8
|
|
%18 = fdiv double 1.000000e+00, %17
|
|
store double %18, double* %15, align 8
|
|
%19 = load double, double* %15, align 8
|
|
%20 = fcmp oge double %19, 0.000000e+00
|
|
br i1 %20, label %21, label %36
|
|
|
|
; <label>:21: ; preds = %4
|
|
%22 = getelementptr inbounds %2, %2* %16, i32 0, i32 1
|
|
%23 = getelementptr inbounds %1, %1* %22, i32 0, i32 0
|
|
%24 = load double, double* %23, align 8
|
|
%25 = load double, double* getelementptr inbounds (%1, %1* @p, i32 0, i32 0), align 8
|
|
%26 = fsub double %24, %25
|
|
%27 = load double, double* %15, align 8
|
|
%28 = fmul double %26, %27
|
|
store double %28, double* %9, align 8
|
|
%29 = getelementptr inbounds %2, %2* %16, i32 0, i32 2
|
|
%30 = getelementptr inbounds %1, %1* %29, i32 0, i32 0
|
|
%31 = load double, double* %30, align 8
|
|
%32 = load double, double* getelementptr inbounds (%1, %1* @p, i32 0, i32 0), align 8
|
|
%33 = fsub double %31, %32
|
|
%34 = load double, double* %15, align 8
|
|
%35 = fmul double %33, %34
|
|
store double %35, double* %12, align 8
|
|
br label %51
|
|
|
|
; <label>:36: ; preds = %4
|
|
%37 = getelementptr inbounds %2, %2* %16, i32 0, i32 2
|
|
%38 = getelementptr inbounds %1, %1* %37, i32 0, i32 0
|
|
%39 = load double, double* %38, align 8
|
|
%40 = load double, double* getelementptr inbounds (%1, %1* @p, i32 0, i32 0), align 8
|
|
%41 = fsub double %39, %40
|
|
%42 = load double, double* %15, align 8
|
|
%43 = fmul double %41, %42
|
|
store double %43, double* %9, align 8
|
|
%44 = getelementptr inbounds %2, %2* %16, i32 0, i32 1
|
|
%45 = getelementptr inbounds %1, %1* %44, i32 0, i32 0
|
|
%46 = load double, double* %45, align 8
|
|
%47 = load double, double* getelementptr inbounds (%1, %1* @p, i32 0, i32 0), align 8
|
|
%48 = fsub double %46, %47
|
|
%49 = load double, double* %15, align 8
|
|
%50 = fmul double %48, %49
|
|
store double %50, double* %12, align 8
|
|
br label %51
|
|
|
|
; <label>:51: ; preds = %36, %21
|
|
%52 = load double, double* %12, align 8
|
|
%53 = load double, double* %9, align 8
|
|
%54 = fcmp olt double %52, %53
|
|
ret i1 %54
|
|
}
|