
Steal impliesEquivalanceIf{True,False} (sic) from GVN, and extend it for floating-point constant vectors, and accounting for denormal values. Since InstCombine also performs GVN-like replacements, introduce CmpInst::isEquivalence, and remove the corresponding code in GVN, with the intent of using it in more places. The code in GVN also has a bad FIXME saying that the optimization may be valid in the nsz case, but this is not the case. Alive2 proof: https://alive2.llvm.org/ce/z/vEaK8M
392 lines
12 KiB
LLVM
392 lines
12 KiB
LLVM
; NOTE: Assertions have been autogenerated by utils/update_test_checks.py UTC_ARGS: --version 5
|
|
; RUN: opt -passes=gvn -S < %s | FileCheck %s
|
|
|
|
define i32 @f1(i32 %x) {
|
|
; CHECK-LABEL: define i32 @f1(
|
|
; CHECK-SAME: i32 [[X:%.*]]) {
|
|
; CHECK-NEXT: [[BB0:.*:]]
|
|
; CHECK-NEXT: [[CMP:%.*]] = icmp eq i32 [[X]], 0
|
|
; CHECK-NEXT: br i1 [[CMP]], label %[[BB2:.*]], label %[[BB1:.*]]
|
|
; CHECK: [[BB1]]:
|
|
; CHECK-NEXT: br label %[[BB2]]
|
|
; CHECK: [[BB2]]:
|
|
; CHECK-NEXT: ret i32 [[X]]
|
|
;
|
|
bb0:
|
|
%cmp = icmp eq i32 %x, 0
|
|
br i1 %cmp, label %bb2, label %bb1
|
|
bb1:
|
|
br label %bb2
|
|
bb2:
|
|
%cond = phi i32 [ %x, %bb0 ], [ 0, %bb1 ]
|
|
%foo = add i32 %cond, %x
|
|
ret i32 %foo
|
|
}
|
|
|
|
define i32 @f2(i32 %x) {
|
|
; CHECK-LABEL: define i32 @f2(
|
|
; CHECK-SAME: i32 [[X:%.*]]) {
|
|
; CHECK-NEXT: [[BB0:.*:]]
|
|
; CHECK-NEXT: [[CMP:%.*]] = icmp ne i32 [[X]], 0
|
|
; CHECK-NEXT: br i1 [[CMP]], label %[[BB1:.*]], label %[[BB2:.*]]
|
|
; CHECK: [[BB1]]:
|
|
; CHECK-NEXT: br label %[[BB2]]
|
|
; CHECK: [[BB2]]:
|
|
; CHECK-NEXT: ret i32 [[X]]
|
|
;
|
|
bb0:
|
|
%cmp = icmp ne i32 %x, 0
|
|
br i1 %cmp, label %bb1, label %bb2
|
|
bb1:
|
|
br label %bb2
|
|
bb2:
|
|
%cond = phi i32 [ %x, %bb0 ], [ 0, %bb1 ]
|
|
%foo = add i32 %cond, %x
|
|
ret i32 %foo
|
|
}
|
|
|
|
define i32 @f3(i32 %x) {
|
|
; CHECK-LABEL: define i32 @f3(
|
|
; CHECK-SAME: i32 [[X:%.*]]) {
|
|
; CHECK-NEXT: [[BB0:.*:]]
|
|
; CHECK-NEXT: switch i32 [[X]], label %[[BB1:.*]] [
|
|
; CHECK-NEXT: i32 0, label %[[BB2:.*]]
|
|
; CHECK-NEXT: ]
|
|
; CHECK: [[BB1]]:
|
|
; CHECK-NEXT: br label %[[BB2]]
|
|
; CHECK: [[BB2]]:
|
|
; CHECK-NEXT: ret i32 [[X]]
|
|
;
|
|
bb0:
|
|
switch i32 %x, label %bb1 [ i32 0, label %bb2]
|
|
bb1:
|
|
br label %bb2
|
|
bb2:
|
|
%cond = phi i32 [ %x, %bb0 ], [ 0, %bb1 ]
|
|
%foo = add i32 %cond, %x
|
|
ret i32 %foo
|
|
}
|
|
|
|
declare void @g(i1)
|
|
define void @f4(ptr %x) {
|
|
; CHECK-LABEL: define void @f4(
|
|
; CHECK-SAME: ptr [[X:%.*]]) {
|
|
; CHECK-NEXT: [[BB0:.*:]]
|
|
; CHECK-NEXT: [[Y:%.*]] = icmp eq ptr null, [[X]]
|
|
; CHECK-NEXT: br i1 [[Y]], label %[[BB2:.*]], label %[[BB1:.*]]
|
|
; CHECK: [[BB1]]:
|
|
; CHECK-NEXT: br label %[[BB2]]
|
|
; CHECK: [[BB2]]:
|
|
; CHECK-NEXT: call void @g(i1 [[Y]])
|
|
; CHECK-NEXT: ret void
|
|
;
|
|
bb0:
|
|
%y = icmp eq ptr null, %x
|
|
br i1 %y, label %bb2, label %bb1
|
|
bb1:
|
|
br label %bb2
|
|
bb2:
|
|
%zed = icmp eq ptr null, %x
|
|
call void @g(i1 %zed)
|
|
ret void
|
|
}
|
|
|
|
define double @fcmp_oeq_not_zero(double %x, double %y) {
|
|
; CHECK-LABEL: define double @fcmp_oeq_not_zero(
|
|
; CHECK-SAME: double [[X:%.*]], double [[Y:%.*]]) {
|
|
; CHECK-NEXT: [[ENTRY:.*]]:
|
|
; CHECK-NEXT: [[CMP:%.*]] = fcmp oeq double [[Y]], 2.000000e+00
|
|
; CHECK-NEXT: br i1 [[CMP]], label %[[IF:.*]], label %[[RETURN:.*]]
|
|
; CHECK: [[IF]]:
|
|
; CHECK-NEXT: [[DIV:%.*]] = fdiv double [[X]], 2.000000e+00
|
|
; CHECK-NEXT: br label %[[RETURN]]
|
|
; CHECK: [[RETURN]]:
|
|
; CHECK-NEXT: [[RETVAL:%.*]] = phi double [ [[DIV]], %[[IF]] ], [ [[X]], %[[ENTRY]] ]
|
|
; CHECK-NEXT: ret double [[RETVAL]]
|
|
;
|
|
entry:
|
|
%cmp = fcmp oeq double %y, 2.0
|
|
br i1 %cmp, label %if, label %return
|
|
|
|
if:
|
|
%div = fdiv double %x, %y
|
|
br label %return
|
|
|
|
return:
|
|
%retval = phi double [ %div, %if ], [ %x, %entry ]
|
|
ret double %retval
|
|
}
|
|
|
|
define double @fcmp_une_not_zero(double %x, double %y) {
|
|
; CHECK-LABEL: define double @fcmp_une_not_zero(
|
|
; CHECK-SAME: double [[X:%.*]], double [[Y:%.*]]) {
|
|
; CHECK-NEXT: [[ENTRY:.*]]:
|
|
; CHECK-NEXT: [[CMP:%.*]] = fcmp une double [[Y]], 2.000000e+00
|
|
; CHECK-NEXT: br i1 [[CMP]], label %[[RETURN:.*]], label %[[ELSE:.*]]
|
|
; CHECK: [[ELSE]]:
|
|
; CHECK-NEXT: [[DIV:%.*]] = fdiv double [[X]], 2.000000e+00
|
|
; CHECK-NEXT: br label %[[RETURN]]
|
|
; CHECK: [[RETURN]]:
|
|
; CHECK-NEXT: [[RETVAL:%.*]] = phi double [ [[DIV]], %[[ELSE]] ], [ [[X]], %[[ENTRY]] ]
|
|
; CHECK-NEXT: ret double [[RETVAL]]
|
|
;
|
|
entry:
|
|
%cmp = fcmp une double %y, 2.0
|
|
br i1 %cmp, label %return, label %else
|
|
|
|
else:
|
|
%div = fdiv double %x, %y
|
|
br label %return
|
|
|
|
return:
|
|
%retval = phi double [ %div, %else ], [ %x, %entry ]
|
|
ret double %retval
|
|
}
|
|
|
|
define double @fcmp_one_possibly_nan(double %x, double %y) {
|
|
; CHECK-LABEL: define double @fcmp_one_possibly_nan(
|
|
; CHECK-SAME: double [[X:%.*]], double [[Y:%.*]]) {
|
|
; CHECK-NEXT: [[ENTRY:.*]]:
|
|
; CHECK-NEXT: [[CMP:%.*]] = fcmp one double [[Y]], 2.000000e+00
|
|
; CHECK-NEXT: br i1 [[CMP]], label %[[RETURN:.*]], label %[[ELSE:.*]]
|
|
; CHECK: [[ELSE]]:
|
|
; CHECK-NEXT: [[DIV:%.*]] = fdiv double [[X]], [[Y]]
|
|
; CHECK-NEXT: br label %[[RETURN]]
|
|
; CHECK: [[RETURN]]:
|
|
; CHECK-NEXT: [[RETVAL:%.*]] = phi double [ [[DIV]], %[[ELSE]] ], [ [[X]], %[[ENTRY]] ]
|
|
; CHECK-NEXT: ret double [[RETVAL]]
|
|
;
|
|
entry:
|
|
%cmp = fcmp one double %y, 2.0
|
|
br i1 %cmp, label %return, label %else
|
|
|
|
else:
|
|
%div = fdiv double %x, %y
|
|
br label %return
|
|
|
|
return:
|
|
%retval = phi double [ %div, %else ], [ %x, %entry ]
|
|
ret double %retval
|
|
}
|
|
|
|
define double @fcmp_one_not_zero_or_nan(double %x, double %y) {
|
|
; CHECK-LABEL: define double @fcmp_one_not_zero_or_nan(
|
|
; CHECK-SAME: double [[X:%.*]], double [[Y:%.*]]) {
|
|
; CHECK-NEXT: [[ENTRY:.*]]:
|
|
; CHECK-NEXT: [[CMP:%.*]] = fcmp nnan one double [[Y]], 2.000000e+00
|
|
; CHECK-NEXT: br i1 [[CMP]], label %[[RETURN:.*]], label %[[ELSE:.*]]
|
|
; CHECK: [[ELSE]]:
|
|
; CHECK-NEXT: [[DIV:%.*]] = fdiv double [[X]], 2.000000e+00
|
|
; CHECK-NEXT: br label %[[RETURN]]
|
|
; CHECK: [[RETURN]]:
|
|
; CHECK-NEXT: [[RETVAL:%.*]] = phi double [ [[DIV]], %[[ELSE]] ], [ [[X]], %[[ENTRY]] ]
|
|
; CHECK-NEXT: ret double [[RETVAL]]
|
|
;
|
|
entry:
|
|
%cmp = fcmp nnan one double %y, 2.0
|
|
br i1 %cmp, label %return, label %else
|
|
|
|
else:
|
|
%div = fdiv double %x, %y
|
|
br label %return
|
|
|
|
return:
|
|
%retval = phi double [ %div, %else ], [ %x, %entry ]
|
|
ret double %retval
|
|
}
|
|
|
|
; PR22376 - We can't propagate zero constants because -0.0
|
|
; compares equal to 0.0. If %y is -0.0 in this test case,
|
|
; we would produce the wrong sign on the infinity return value.
|
|
define double @fcmp_oeq_zero(double %x, double %y) {
|
|
; CHECK-LABEL: define double @fcmp_oeq_zero(
|
|
; CHECK-SAME: double [[X:%.*]], double [[Y:%.*]]) {
|
|
; CHECK-NEXT: [[ENTRY:.*]]:
|
|
; CHECK-NEXT: [[CMP:%.*]] = fcmp oeq double [[Y]], 0.000000e+00
|
|
; CHECK-NEXT: br i1 [[CMP]], label %[[IF:.*]], label %[[RETURN:.*]]
|
|
; CHECK: [[IF]]:
|
|
; CHECK-NEXT: [[DIV:%.*]] = fdiv double [[X]], [[Y]]
|
|
; CHECK-NEXT: br label %[[RETURN]]
|
|
; CHECK: [[RETURN]]:
|
|
; CHECK-NEXT: [[RETVAL:%.*]] = phi double [ [[DIV]], %[[IF]] ], [ [[X]], %[[ENTRY]] ]
|
|
; CHECK-NEXT: ret double [[RETVAL]]
|
|
;
|
|
entry:
|
|
%cmp = fcmp oeq double %y, 0.0
|
|
br i1 %cmp, label %if, label %return
|
|
|
|
if:
|
|
%div = fdiv double %x, %y
|
|
br label %return
|
|
|
|
return:
|
|
%retval = phi double [ %div, %if ], [ %x, %entry ]
|
|
ret double %retval
|
|
}
|
|
|
|
; Denormals may be flushed to zero in some cases by the backend.
|
|
; Hence, treat denormals as 0.
|
|
define float @fcmp_oeq_denormal(float %x, float %y) {
|
|
; CHECK-LABEL: define float @fcmp_oeq_denormal(
|
|
; CHECK-SAME: float [[X:%.*]], float [[Y:%.*]]) {
|
|
; CHECK-NEXT: [[ENTRY:.*]]:
|
|
; CHECK-NEXT: [[CMP:%.*]] = fcmp oeq float [[Y]], 0x3800000000000000
|
|
; CHECK-NEXT: br i1 [[CMP]], label %[[IF:.*]], label %[[RETURN:.*]]
|
|
; CHECK: [[IF]]:
|
|
; CHECK-NEXT: [[DIV:%.*]] = fdiv float [[X]], [[Y]]
|
|
; CHECK-NEXT: br label %[[RETURN]]
|
|
; CHECK: [[RETURN]]:
|
|
; CHECK-NEXT: [[RETVAL:%.*]] = phi float [ [[DIV]], %[[IF]] ], [ [[X]], %[[ENTRY]] ]
|
|
; CHECK-NEXT: ret float [[RETVAL]]
|
|
;
|
|
entry:
|
|
%cmp = fcmp oeq float %y, 0x3800000000000000
|
|
br i1 %cmp, label %if, label %return
|
|
|
|
if:
|
|
%div = fdiv float %x, %y
|
|
br label %return
|
|
|
|
return:
|
|
%retval = phi float [ %div, %if ], [ %x, %entry ]
|
|
ret float %retval
|
|
}
|
|
|
|
define double @fcmp_une_zero(double %x, double %y) {
|
|
; CHECK-LABEL: define double @fcmp_une_zero(
|
|
; CHECK-SAME: double [[X:%.*]], double [[Y:%.*]]) {
|
|
; CHECK-NEXT: [[ENTRY:.*]]:
|
|
; CHECK-NEXT: [[CMP:%.*]] = fcmp une double [[Y]], -0.000000e+00
|
|
; CHECK-NEXT: br i1 [[CMP]], label %[[RETURN:.*]], label %[[ELSE:.*]]
|
|
; CHECK: [[ELSE]]:
|
|
; CHECK-NEXT: [[DIV:%.*]] = fdiv double [[X]], [[Y]]
|
|
; CHECK-NEXT: br label %[[RETURN]]
|
|
; CHECK: [[RETURN]]:
|
|
; CHECK-NEXT: [[RETVAL:%.*]] = phi double [ [[DIV]], %[[ELSE]] ], [ [[X]], %[[ENTRY]] ]
|
|
; CHECK-NEXT: ret double [[RETVAL]]
|
|
;
|
|
entry:
|
|
%cmp = fcmp une double %y, -0.0
|
|
br i1 %cmp, label %return, label %else
|
|
|
|
else:
|
|
%div = fdiv double %x, %y
|
|
br label %return
|
|
|
|
return:
|
|
%retval = phi double [ %div, %else ], [ %x, %entry ]
|
|
ret double %retval
|
|
}
|
|
|
|
; We also cannot propagate a value if it's not a constant.
|
|
; This is because the value could be 0.0, -0.0, or a denormal.
|
|
|
|
define double @fcmp_oeq_maybe_zero(double %x, double %y, double %z1, double %z2) {
|
|
; CHECK-LABEL: define double @fcmp_oeq_maybe_zero(
|
|
; CHECK-SAME: double [[X:%.*]], double [[Y:%.*]], double [[Z1:%.*]], double [[Z2:%.*]]) {
|
|
; CHECK-NEXT: [[ENTRY:.*]]:
|
|
; CHECK-NEXT: [[Z:%.*]] = fadd double [[Z1]], [[Z2]]
|
|
; CHECK-NEXT: [[CMP:%.*]] = fcmp oeq double [[Y]], [[Z]]
|
|
; CHECK-NEXT: br i1 [[CMP]], label %[[IF:.*]], label %[[RETURN:.*]]
|
|
; CHECK: [[IF]]:
|
|
; CHECK-NEXT: [[DIV:%.*]] = fdiv double [[X]], [[Z]]
|
|
; CHECK-NEXT: br label %[[RETURN]]
|
|
; CHECK: [[RETURN]]:
|
|
; CHECK-NEXT: [[RETVAL:%.*]] = phi double [ [[DIV]], %[[IF]] ], [ [[X]], %[[ENTRY]] ]
|
|
; CHECK-NEXT: ret double [[RETVAL]]
|
|
;
|
|
entry:
|
|
%z = fadd double %z1, %z2
|
|
%cmp = fcmp oeq double %y, %z
|
|
br i1 %cmp, label %if, label %return
|
|
|
|
if:
|
|
%div = fdiv double %x, %z
|
|
br label %return
|
|
|
|
return:
|
|
%retval = phi double [ %div, %if ], [ %x, %entry ]
|
|
ret double %retval
|
|
}
|
|
|
|
define double @fcmp_une_maybe_zero(double %x, double %y, double %z1, double %z2) {
|
|
; CHECK-LABEL: define double @fcmp_une_maybe_zero(
|
|
; CHECK-SAME: double [[X:%.*]], double [[Y:%.*]], double [[Z1:%.*]], double [[Z2:%.*]]) {
|
|
; CHECK-NEXT: [[ENTRY:.*]]:
|
|
; CHECK-NEXT: [[Z:%.*]] = fadd double [[Z1]], [[Z2]]
|
|
; CHECK-NEXT: [[CMP:%.*]] = fcmp une double [[Y]], [[Z]]
|
|
; CHECK-NEXT: br i1 [[CMP]], label %[[RETURN:.*]], label %[[ELSE:.*]]
|
|
; CHECK: [[ELSE]]:
|
|
; CHECK-NEXT: [[DIV:%.*]] = fdiv double [[X]], [[Z]]
|
|
; CHECK-NEXT: br label %[[RETURN]]
|
|
; CHECK: [[RETURN]]:
|
|
; CHECK-NEXT: [[RETVAL:%.*]] = phi double [ [[DIV]], %[[ELSE]] ], [ [[X]], %[[ENTRY]] ]
|
|
; CHECK-NEXT: ret double [[RETVAL]]
|
|
;
|
|
entry:
|
|
%z = fadd double %z1, %z2
|
|
%cmp = fcmp une double %y, %z
|
|
br i1 %cmp, label %return, label %else
|
|
|
|
else:
|
|
%div = fdiv double %x, %z
|
|
br label %return
|
|
|
|
return:
|
|
%retval = phi double [ %div, %else ], [ %x, %entry ]
|
|
ret double %retval
|
|
}
|
|
|
|
|
|
define double @fcmp_ueq_possibly_nan(double %x, double %y) {
|
|
; CHECK-LABEL: define double @fcmp_ueq_possibly_nan(
|
|
; CHECK-SAME: double [[X:%.*]], double [[Y:%.*]]) {
|
|
; CHECK-NEXT: [[ENTRY:.*]]:
|
|
; CHECK-NEXT: [[CMP:%.*]] = fcmp ueq double [[Y]], 2.000000e+00
|
|
; CHECK-NEXT: br i1 [[CMP]], label %[[DO_DIV:.*]], label %[[RETURN:.*]]
|
|
; CHECK: [[DO_DIV]]:
|
|
; CHECK-NEXT: [[DIV:%.*]] = fdiv double [[X]], [[Y]]
|
|
; CHECK-NEXT: br label %[[RETURN]]
|
|
; CHECK: [[RETURN]]:
|
|
; CHECK-NEXT: [[RETVAL:%.*]] = phi double [ [[DIV]], %[[DO_DIV]] ], [ [[X]], %[[ENTRY]] ]
|
|
; CHECK-NEXT: ret double [[RETVAL]]
|
|
;
|
|
entry:
|
|
%cmp = fcmp ueq double %y, 2.0
|
|
br i1 %cmp, label %do_div, label %return
|
|
|
|
do_div:
|
|
%div = fdiv double %x, %y
|
|
br label %return
|
|
|
|
return:
|
|
%retval = phi double [ %div, %do_div ], [ %x, %entry ]
|
|
ret double %retval
|
|
}
|
|
|
|
define double @fcmp_ueq_not_zero_or_nan(double %x, double %y) {
|
|
; CHECK-LABEL: define double @fcmp_ueq_not_zero_or_nan(
|
|
; CHECK-SAME: double [[X:%.*]], double [[Y:%.*]]) {
|
|
; CHECK-NEXT: [[ENTRY:.*]]:
|
|
; CHECK-NEXT: [[CMP:%.*]] = fcmp nnan ueq double [[Y]], 2.000000e+00
|
|
; CHECK-NEXT: br i1 [[CMP]], label %[[DO_DIV:.*]], label %[[RETURN:.*]]
|
|
; CHECK: [[DO_DIV]]:
|
|
; CHECK-NEXT: [[DIV:%.*]] = fdiv double [[X]], 2.000000e+00
|
|
; CHECK-NEXT: br label %[[RETURN]]
|
|
; CHECK: [[RETURN]]:
|
|
; CHECK-NEXT: [[RETVAL:%.*]] = phi double [ [[DIV]], %[[DO_DIV]] ], [ [[X]], %[[ENTRY]] ]
|
|
; CHECK-NEXT: ret double [[RETVAL]]
|
|
;
|
|
entry:
|
|
%cmp = fcmp nnan ueq double %y, 2.0
|
|
br i1 %cmp, label %do_div, label %return
|
|
|
|
do_div:
|
|
%div = fdiv double %x, %y
|
|
br label %return
|
|
|
|
return:
|
|
%retval = phi double [ %div, %do_div ], [ %x, %entry ]
|
|
ret double %retval
|
|
}
|