[llvm] InstCombine: Baseline tests for SimplifyDemandedFPClass phi support (PR #184123)

Matt Arsenault via llvm-commits llvm-commits at lists.llvm.org
Mon Mar 2 05:54:38 PST 2026


https://github.com/arsenm created https://github.com/llvm/llvm-project/pull/184123

None

>From 1adb6813a026e7349ff1fd527aeb652203e7896c Mon Sep 17 00:00:00 2001
From: Matt Arsenault <Matthew.Arsenault at amd.com>
Date: Sat, 28 Feb 2026 08:18:18 +0100
Subject: [PATCH] InstCombine: Baseline tests for SimplifyDemandedFPClass phi
 support

---
 .../simplify-demanded-fpclass-phi.ll          | 661 ++++++++++++++++++
 1 file changed, 661 insertions(+)
 create mode 100644 llvm/test/Transforms/InstCombine/simplify-demanded-fpclass-phi.ll

diff --git a/llvm/test/Transforms/InstCombine/simplify-demanded-fpclass-phi.ll b/llvm/test/Transforms/InstCombine/simplify-demanded-fpclass-phi.ll
new file mode 100644
index 0000000000000..a064d8b65a457
--- /dev/null
+++ b/llvm/test/Transforms/InstCombine/simplify-demanded-fpclass-phi.ll
@@ -0,0 +1,661 @@
+; NOTE: Assertions have been autogenerated by utils/update_test_checks.py UTC_ARGS: --version 6
+; RUN: opt -S -passes=instcombine < %s | FileCheck %s
+
+declare nofpclass(inf norm sub zero) float @nan_only()
+
+define nofpclass(nan) half @ret_phi_if_ret_0(i1 %cond1, i1 %cond2, half %unknown) {
+; CHECK-LABEL: define nofpclass(nan) half @ret_phi_if_ret_0(
+; CHECK-SAME: i1 [[COND1:%.*]], i1 [[COND2:%.*]], half [[UNKNOWN:%.*]]) {
+; CHECK-NEXT:  [[ENTRY:.*]]:
+; CHECK-NEXT:    br i1 [[COND1]], label %[[IF:.*]], label %[[RET:.*]]
+; CHECK:       [[IF]]:
+; CHECK-NEXT:    [[SELECT:%.*]] = select i1 [[COND2]], half 0xH7E00, half [[UNKNOWN]]
+; CHECK-NEXT:    br label %[[RET]]
+; CHECK:       [[RET]]:
+; CHECK-NEXT:    [[PHI:%.*]] = phi half [ [[SELECT]], %[[IF]] ], [ 0xH0000, %[[ENTRY]] ]
+; CHECK-NEXT:    ret half [[PHI]]
+;
+entry:
+  br i1 %cond1, label %if, label %ret
+
+if:
+  %select = select i1 %cond2, half 0xH7E00, half %unknown
+  br label %ret
+
+ret:
+  %phi = phi half [ %select, %if ], [ 0.0, %entry ]
+  ret half %phi
+}
+
+define nofpclass(nan inf norm sub nzero) half @ret_phi_only_pzero(i1 %cond1, i1 %cond2, half %unknown) {
+; CHECK-LABEL: define nofpclass(nan inf nzero sub norm) half @ret_phi_only_pzero(
+; CHECK-SAME: i1 [[COND1:%.*]], i1 [[COND2:%.*]], half [[UNKNOWN:%.*]]) {
+; CHECK-NEXT:  [[ENTRY:.*:]]
+; CHECK-NEXT:    br i1 [[COND1]], label %[[IF:.*]], label %[[RET:.*]]
+; CHECK:       [[IF]]:
+; CHECK-NEXT:    br label %[[RET]]
+; CHECK:       [[RET]]:
+; CHECK-NEXT:    ret half 0xH0000
+;
+entry:
+  br i1 %cond1, label %if, label %ret
+
+if:
+  %select = select i1 %cond2, half 0xH7E00, half %unknown
+  br label %ret
+
+ret:
+  %phi = phi half [ %select, %if ], [ 0.0, %entry ]
+  ret half %phi
+}
+
+define nofpclass(inf norm sub zero) half @ret_phi_only_nan(i1 %cond1, i1 %cond2, half %unknown) {
+; CHECK-LABEL: define nofpclass(inf zero sub norm) half @ret_phi_only_nan(
+; CHECK-SAME: i1 [[COND1:%.*]], i1 [[COND2:%.*]], half [[UNKNOWN:%.*]]) {
+; CHECK-NEXT:  [[ENTRY:.*]]:
+; CHECK-NEXT:    br i1 [[COND1]], label %[[IF:.*]], label %[[RET:.*]]
+; CHECK:       [[IF]]:
+; CHECK-NEXT:    [[SELECT:%.*]] = select i1 [[COND2]], half 0xH7E00, half [[UNKNOWN]]
+; CHECK-NEXT:    br label %[[RET]]
+; CHECK:       [[RET]]:
+; CHECK-NEXT:    [[PHI:%.*]] = phi half [ [[SELECT]], %[[IF]] ], [ 0xH0000, %[[ENTRY]] ]
+; CHECK-NEXT:    ret half [[PHI]]
+;
+entry:
+  br i1 %cond1, label %if, label %ret
+
+if:
+  %select = select i1 %cond2, half 0xH7E00, half %unknown
+  br label %ret
+
+ret:
+  %phi = phi half [ %select, %if ], [ 0.0, %entry ]
+  ret half %phi
+}
+
+define nofpclass(nan) half @ret_phi_if_ret_1(i1 %cond1, i1 %cond2, half %unknown) {
+; CHECK-LABEL: define nofpclass(nan) half @ret_phi_if_ret_1(
+; CHECK-SAME: i1 [[COND1:%.*]], i1 [[COND2:%.*]], half [[UNKNOWN:%.*]]) {
+; CHECK-NEXT:  [[ENTRY:.*]]:
+; CHECK-NEXT:    br i1 [[COND1]], label %[[IF:.*]], label %[[RET:.*]]
+; CHECK:       [[IF]]:
+; CHECK-NEXT:    [[SELECT:%.*]] = select i1 [[COND2]], half 0xH7E00, half [[UNKNOWN]]
+; CHECK-NEXT:    br label %[[RET]]
+; CHECK:       [[RET]]:
+; CHECK-NEXT:    [[PHI:%.*]] = phi half [ 0xH0000, %[[ENTRY]] ], [ [[SELECT]], %[[IF]] ]
+; CHECK-NEXT:    ret half [[PHI]]
+;
+entry:
+  br i1 %cond1, label %if, label %ret
+
+if:
+  %select = select i1 %cond2, half 0xH7E00, half %unknown
+  br label %ret
+
+ret:
+  %phi = phi half [ 0.0, %entry ], [ %select, %if ]
+  ret half %phi
+}
+
+; Make sure all block references use the same simplified value.
+define nofpclass(nan) half @ret_repeated_switch_pred(i8 %switch.cond, i1 %cond2, half %unknown) {
+; CHECK-LABEL: define nofpclass(nan) half @ret_repeated_switch_pred(
+; CHECK-SAME: i8 [[SWITCH_COND:%.*]], i1 [[COND2:%.*]], half [[UNKNOWN:%.*]]) {
+; CHECK-NEXT:  [[ENTRY:.*]]:
+; CHECK-NEXT:    [[SELECT:%.*]] = select i1 [[COND2]], half 0xH7E00, half [[UNKNOWN]]
+; CHECK-NEXT:    switch i8 [[SWITCH_COND]], label %[[DEFAULT:.*]] [
+; CHECK-NEXT:      i8 0, label %[[RET:.*]]
+; CHECK-NEXT:      i8 1, label %[[RET]]
+; CHECK-NEXT:      i8 2, label %[[IF:.*]]
+; CHECK-NEXT:    ]
+; CHECK:       [[IF]]:
+; CHECK-NEXT:    br label %[[RET]]
+; CHECK:       [[DEFAULT]]:
+; CHECK-NEXT:    unreachable
+; CHECK:       [[RET]]:
+; CHECK-NEXT:    [[PHI:%.*]] = phi half [ 0xH0000, %[[IF]] ], [ [[SELECT]], %[[ENTRY]] ], [ [[SELECT]], %[[ENTRY]] ]
+; CHECK-NEXT:    ret half [[PHI]]
+;
+entry:
+  %select = select i1 %cond2, half 0xH7E00, half %unknown
+  switch i8 %switch.cond, label %default
+  [ i8 0, label %ret
+  i8 1, label %ret
+  i8 2, label %if
+  ]
+
+if:
+  br label %ret
+
+default:
+  unreachable
+
+ret:
+  %phi = phi half [ 0.0, %if ], [ %select, %entry ], [ %select, %entry ]
+  ret half %phi
+}
+
+
+define nofpclass(nan) half @ret_phi_chain(i1 %cond1, i1 %cond2, i1 %cond3, half %unknown) {
+; CHECK-LABEL: define nofpclass(nan) half @ret_phi_chain(
+; CHECK-SAME: i1 [[COND1:%.*]], i1 [[COND2:%.*]], i1 [[COND3:%.*]], half [[UNKNOWN:%.*]]) {
+; CHECK-NEXT:  [[ENTRY:.*]]:
+; CHECK-NEXT:    br i1 [[COND1]], label %[[IF0:.*]], label %[[IF1:.*]]
+; CHECK:       [[IF0]]:
+; CHECK-NEXT:    [[SELECT0:%.*]] = select i1 [[COND2]], half 0xH7E00, half [[UNKNOWN]]
+; CHECK-NEXT:    br i1 [[COND2]], label %[[IF1]], label %[[RET:.*]]
+; CHECK:       [[IF1]]:
+; CHECK-NEXT:    [[PHI0:%.*]] = phi half [ [[SELECT0]], %[[IF0]] ], [ 0xH0000, %[[ENTRY]] ]
+; CHECK-NEXT:    br label %[[RET]]
+; CHECK:       [[RET]]:
+; CHECK-NEXT:    [[PHI1:%.*]] = phi half [ [[PHI0]], %[[IF1]] ], [ 0xH0000, %[[IF0]] ]
+; CHECK-NEXT:    ret half [[PHI1]]
+;
+entry:
+  br i1 %cond1, label %if0, label %if1
+
+if0:
+  %select0 = select i1 %cond2, half 0xH7E00, half %unknown
+  br i1 %cond2, label %if1, label %ret
+
+if1:
+  %phi0 = phi half [ %select0, %if0 ], [ 0.0, %entry ]
+  br label %ret
+
+ret:
+  %phi1 = phi half [ %phi0, %if1 ], [ 0.0, %if0 ]
+  ret half %phi1
+}
+
+
+define nofpclass(nan) half @basic_loop_break_entry_block(i1 %cond1, i1 %cond2, half %unknown, ptr %p0, ptr %p1) {
+; CHECK-LABEL: define nofpclass(nan) half @basic_loop_break_entry_block(
+; CHECK-SAME: i1 [[COND1:%.*]], i1 [[COND2:%.*]], half [[UNKNOWN:%.*]], ptr [[P0:%.*]], ptr [[P1:%.*]]) {
+; CHECK-NEXT:  [[ENTRY:.*]]:
+; CHECK-NEXT:    [[SELECT:%.*]] = select i1 [[COND2]], half 0xH7E00, half [[UNKNOWN]]
+; CHECK-NEXT:    br label %[[LOOP:.*]]
+; CHECK:       [[LOOP]]:
+; CHECK-NEXT:    [[PHI:%.*]] = phi half [ [[LOAD:%.*]], %[[LOOP]] ], [ [[SELECT]], %[[ENTRY]] ]
+; CHECK-NEXT:    [[LOAD]] = load half, ptr [[P1]], align 2
+; CHECK-NEXT:    [[BREAK_COND:%.*]] = load i1, ptr [[P0]], align 1
+; CHECK-NEXT:    br i1 [[BREAK_COND]], label %[[RET:.*]], label %[[LOOP]]
+; CHECK:       [[RET]]:
+; CHECK-NEXT:    ret half [[PHI]]
+;
+entry:
+  %select = select i1 %cond2, half 0xH7E00, half %unknown
+  br label %loop
+
+loop:
+  %phi = phi half [ %load, %loop ], [ %select, %entry ]
+  %load = load half, ptr %p1
+  %break.cond = load i1, ptr %p0
+  br i1 %break.cond, label %ret, label %loop
+
+ret:
+  ret half %phi
+}
+
+define nofpclass(nan) half @basic_loop_break_mid_loop(i1 %cond1, i1 %cond2, half %unknown, ptr %p0, ptr %p1) {
+; CHECK-LABEL: define nofpclass(nan) half @basic_loop_break_mid_loop(
+; CHECK-SAME: i1 [[COND1:%.*]], i1 [[COND2:%.*]], half [[UNKNOWN:%.*]], ptr [[P0:%.*]], ptr [[P1:%.*]]) {
+; CHECK-NEXT:  [[ENTRY:.*]]:
+; CHECK-NEXT:    br label %[[LOOP:.*]]
+; CHECK:       [[LOOP]]:
+; CHECK-NEXT:    [[PHI:%.*]] = phi half [ [[SELECT:%.*]], %[[LOOP]] ], [ 0xH0000, %[[ENTRY]] ]
+; CHECK-NEXT:    [[LOAD:%.*]] = load half, ptr [[P1]], align 2
+; CHECK-NEXT:    [[SELECT]] = select i1 [[COND2]], half 0xH7E00, half [[LOAD]]
+; CHECK-NEXT:    [[BREAK_COND:%.*]] = load i1, ptr [[P0]], align 1
+; CHECK-NEXT:    br i1 [[BREAK_COND]], label %[[RET:.*]], label %[[LOOP]]
+; CHECK:       [[RET]]:
+; CHECK-NEXT:    ret half [[PHI]]
+;
+entry:
+  br label %loop
+
+loop:
+  %phi = phi half [ %select, %loop ], [ 0.0, %entry ]
+  %load = load half, ptr %p1
+  %select = select i1 %cond2, half 0xH7E00, half %load
+  %break.cond = load i1, ptr %p0
+  br i1 %break.cond, label %ret, label %loop
+
+ret:
+  ret half %phi
+}
+
+define nofpclass(inf) half @recurrence(i1 %select.cond, half %unknown, ptr %p0, ptr %p1) {
+; CHECK-LABEL: define nofpclass(inf) half @recurrence(
+; CHECK-SAME: i1 [[SELECT_COND:%.*]], half [[UNKNOWN:%.*]], ptr [[P0:%.*]], ptr [[P1:%.*]]) {
+; CHECK-NEXT:  [[ENTRY:.*]]:
+; CHECK-NEXT:    br label %[[LOOP:.*]]
+; CHECK:       [[LOOP]]:
+; CHECK-NEXT:    [[PHI:%.*]] = phi half [ [[SELECT:%.*]], %[[LOOP]] ], [ 0xH0000, %[[ENTRY]] ]
+; CHECK-NEXT:    [[LOAD:%.*]] = load half, ptr [[P1]], align 2
+; CHECK-NEXT:    [[SELECT]] = select i1 [[SELECT_COND]], half 0xH7C00, half [[LOAD]]
+; CHECK-NEXT:    [[BREAK_COND:%.*]] = load i1, ptr [[P0]], align 1
+; CHECK-NEXT:    br i1 [[BREAK_COND]], label %[[RET:.*]], label %[[LOOP]]
+; CHECK:       [[RET]]:
+; CHECK-NEXT:    ret half [[PHI]]
+;
+entry:
+  br label %loop
+
+loop:
+  %phi = phi half [ %select, %loop ], [ 0.0, %entry ]
+  %load = load half, ptr %p1
+  %select = select i1 %select.cond, half 0xH7C00, half %load
+  %break.cond = load i1, ptr %p0
+  br i1 %break.cond, label %ret, label %loop
+
+ret:
+  ret half %phi
+}
+
+define nofpclass(nan) half @path_dependent(i1 %select.cond, half %unknown, half %x) {
+; CHECK-LABEL: define nofpclass(nan) half @path_dependent(
+; CHECK-SAME: i1 [[SELECT_COND:%.*]], half [[UNKNOWN:%.*]], half [[X:%.*]]) {
+; CHECK-NEXT:  [[ENTRY:.*]]:
+; CHECK-NEXT:    [[IS_NAN:%.*]] = fcmp uno half [[X]], 0xH0000
+; CHECK-NEXT:    br i1 [[IS_NAN]], label %[[IF:.*]], label %[[RET:.*]]
+; CHECK:       [[IF]]:
+; CHECK-NEXT:    [[SELECT:%.*]] = select i1 [[SELECT_COND]], half [[X]], half [[UNKNOWN]]
+; CHECK-NEXT:    br label %[[RET]]
+; CHECK:       [[RET]]:
+; CHECK-NEXT:    [[PHI:%.*]] = phi half [ [[SELECT]], %[[IF]] ], [ 0xH0000, %[[ENTRY]] ]
+; CHECK-NEXT:    ret half [[PHI]]
+;
+entry:
+  %is.nan = fcmp uno half %x, 0.0
+  br i1 %is.nan, label %if, label %ret
+
+if:
+  %select = select i1 %select.cond, half %x, half %unknown
+  br label %ret
+
+ret:
+  %phi = phi half [ %select, %if ], [ 0.0, %entry ]
+  ret half %phi
+}
+
+define nofpclass(nan) half @path_dependent_wrong(i1 %select.cond, half %unknown, half %x) {
+; CHECK-LABEL: define nofpclass(nan) half @path_dependent_wrong(
+; CHECK-SAME: i1 [[SELECT_COND:%.*]], half [[UNKNOWN:%.*]], half [[X:%.*]]) {
+; CHECK-NEXT:  [[ENTRY:.*]]:
+; CHECK-NEXT:    [[IS_NOT_NAN:%.*]] = fcmp ord half [[X]], 0xH0000
+; CHECK-NEXT:    br i1 [[IS_NOT_NAN]], label %[[IF:.*]], label %[[RET:.*]]
+; CHECK:       [[IF]]:
+; CHECK-NEXT:    [[SELECT:%.*]] = select i1 [[SELECT_COND]], half [[X]], half [[UNKNOWN]]
+; CHECK-NEXT:    br label %[[RET]]
+; CHECK:       [[RET]]:
+; CHECK-NEXT:    [[PHI:%.*]] = phi half [ [[SELECT]], %[[IF]] ], [ 0xH0000, %[[ENTRY]] ]
+; CHECK-NEXT:    ret half [[PHI]]
+;
+entry:
+  %is.not.nan = fcmp ord half %x, 0.0
+  br i1 %is.not.nan, label %if, label %ret
+
+if:
+  %select = select i1 %select.cond, half %x, half %unknown
+  br label %ret
+
+ret:
+  %phi = phi half [ %select, %if ], [ 0.0, %entry ]
+  ret half %phi
+}
+
+
+define nofpclass(nan) half @diamond_same_select(i1 %select.cond, half %unknown, half %x) {
+; CHECK-LABEL: define nofpclass(nan) half @diamond_same_select(
+; CHECK-SAME: i1 [[SELECT_COND:%.*]], half [[UNKNOWN:%.*]], half [[X:%.*]]) {
+; CHECK-NEXT:  [[ENTRY:.*:]]
+; CHECK-NEXT:    [[IS_NAN:%.*]] = fcmp uno half [[X]], 0xH0000
+; CHECK-NEXT:    br i1 [[IS_NAN]], label %[[IF:.*]], label %[[ELSE:.*]]
+; CHECK:       [[IF]]:
+; CHECK-NEXT:    [[SELECT0:%.*]] = select i1 [[SELECT_COND]], half [[X]], half [[UNKNOWN]]
+; CHECK-NEXT:    br label %[[ENDIF:.*]]
+; CHECK:       [[ELSE]]:
+; CHECK-NEXT:    [[SELECT1:%.*]] = select i1 [[SELECT_COND]], half [[X]], half [[UNKNOWN]]
+; CHECK-NEXT:    br label %[[ENDIF]]
+; CHECK:       [[ENDIF]]:
+; CHECK-NEXT:    [[PHI:%.*]] = phi half [ [[SELECT0]], %[[IF]] ], [ [[SELECT1]], %[[ELSE]] ]
+; CHECK-NEXT:    ret half [[PHI]]
+;
+entry:
+  %is.nan = fcmp uno half %x, 0.0
+  br i1 %is.nan, label %if, label %else
+
+if:
+  %select0 = select i1 %select.cond, half %x, half %unknown
+  br label %endif
+
+else:
+  %select1 = select i1 %select.cond, half %x, half %unknown
+  br label %endif
+
+endif:
+  %phi = phi half [ %select0, %if ], [ %select1, %else ]
+  ret half %phi
+}
+
+define nofpclass(nan) half @remove_nan_start_value(ptr %p0, ptr %p1) {
+; CHECK-LABEL: define nofpclass(nan) half @remove_nan_start_value(
+; CHECK-SAME: ptr [[P0:%.*]], ptr [[P1:%.*]]) {
+; CHECK-NEXT:  [[ENTRY:.*]]:
+; CHECK-NEXT:    br label %[[LOOP:.*]]
+; CHECK:       [[LOOP]]:
+; CHECK-NEXT:    [[PHI:%.*]] = phi half [ [[LOAD:%.*]], %[[LOOP]] ], [ 0xH7E00, %[[ENTRY]] ]
+; CHECK-NEXT:    [[LOAD]] = load half, ptr [[P1]], align 2
+; CHECK-NEXT:    [[BREAK_COND:%.*]] = load i1, ptr [[P0]], align 1
+; CHECK-NEXT:    br i1 [[BREAK_COND]], label %[[RET:.*]], label %[[LOOP]]
+; CHECK:       [[RET]]:
+; CHECK-NEXT:    ret half [[PHI]]
+;
+entry:
+  br label %loop
+
+loop:
+  %phi = phi half [ %load, %loop ], [ 0xH7E00, %entry ]
+  %load = load half, ptr %p1
+  %break.cond = load i1, ptr %p0
+  br i1 %break.cond, label %ret, label %loop
+
+ret:
+  ret half %phi
+}
+
+define nofpclass(nan) half @keep_nan_start_value_multi_use(i1 %cond1, i1 %cond2, ptr %p0, ptr %p1) {
+; CHECK-LABEL: define nofpclass(nan) half @keep_nan_start_value_multi_use(
+; CHECK-SAME: i1 [[COND1:%.*]], i1 [[COND2:%.*]], ptr [[P0:%.*]], ptr [[P1:%.*]]) {
+; CHECK-NEXT:  [[ENTRY:.*]]:
+; CHECK-NEXT:    br label %[[LOOP:.*]]
+; CHECK:       [[LOOP]]:
+; CHECK-NEXT:    [[PHI:%.*]] = phi half [ [[LOAD:%.*]], %[[LOOP]] ], [ 0xH7E00, %[[ENTRY]] ]
+; CHECK-NEXT:    [[LOAD]] = load half, ptr [[P1]], align 2
+; CHECK-NEXT:    [[BREAK_COND:%.*]] = load i1, ptr [[P0]], align 1
+; CHECK-NEXT:    br i1 [[BREAK_COND]], label %[[RET:.*]], label %[[LOOP]]
+; CHECK:       [[RET]]:
+; CHECK-NEXT:    store half [[PHI]], ptr [[P0]], align 2
+; CHECK-NEXT:    ret half [[PHI]]
+;
+entry:
+  br label %loop
+
+loop:
+  %phi = phi half [ %load, %loop ], [ 0xH7E00, %entry ]
+  %load = load half, ptr %p1
+  %break.cond = load i1, ptr %p0
+  br i1 %break.cond, label %ret, label %loop
+
+ret:
+  store half %phi, ptr %p0
+  ret half %phi
+}
+
+; TODO: Why doesn't this fold to ret poison?
+define nofpclass(nan) half @loop_break_if_nan(ptr %p0) {
+; CHECK-LABEL: define nofpclass(nan) half @loop_break_if_nan(
+; CHECK-SAME: ptr [[P0:%.*]]) {
+; CHECK-NEXT:  [[ENTRY:.*]]:
+; CHECK-NEXT:    br label %[[LOOP:.*]]
+; CHECK:       [[LOOP]]:
+; CHECK-NEXT:    [[PHI:%.*]] = phi half [ [[LOAD:%.*]], %[[LOOP]] ], [ 0xH7E00, %[[ENTRY]] ]
+; CHECK-NEXT:    [[LOAD]] = load half, ptr [[P0]], align 2
+; CHECK-NEXT:    [[IS_NAN:%.*]] = fcmp uno half [[LOAD]], 0xH0000
+; CHECK-NEXT:    br i1 [[IS_NAN]], label %[[RET:.*]], label %[[LOOP]]
+; CHECK:       [[RET]]:
+; CHECK-NEXT:    ret half [[PHI]]
+;
+entry:
+  br label %loop
+
+loop:
+  %phi = phi half [ %load, %loop ], [ 0xH7E00, %entry ]
+  %load = load half, ptr %p0
+  %is.nan = fcmp uno half %load, 0.0
+  br i1 %is.nan, label %ret, label %loop
+
+ret:
+  ret half %phi
+}
+
+define nofpclass(nan) half @loop_break_if_nan_abs(ptr %p0) {
+; CHECK-LABEL: define nofpclass(nan) half @loop_break_if_nan_abs(
+; CHECK-SAME: ptr [[P0:%.*]]) {
+; CHECK-NEXT:  [[ENTRY:.*:]]
+; CHECK-NEXT:    br label %[[LOOP:.*]]
+; CHECK:       [[LOOP]]:
+; CHECK-NEXT:    [[LOAD:%.*]] = load half, ptr [[P0]], align 2
+; CHECK-NEXT:    [[IS_NAN:%.*]] = fcmp uno half [[LOAD]], 0xH0000
+; CHECK-NEXT:    br i1 [[IS_NAN]], label %[[RET:.*]], label %[[LOOP]]
+; CHECK:       [[RET]]:
+; CHECK-NEXT:    ret half poison
+;
+entry:
+  br label %loop
+
+loop:
+  %phi = phi half [ %load, %loop ], [ 0xH7E00, %entry ]
+  %load = load half, ptr %p0
+  %abs = call half @llvm.fabs.f16(half %load)
+  %is.nan = fcmp uno half %load, 0.0
+  br i1 %is.nan, label %ret, label %loop
+
+ret:
+  ret half %abs
+}
+
+define nofpclass(nan) half @ret_loop_under_if_phi(half %unknown, i1 %cond, i1 %cond2, ptr %p0, ptr %p1) {
+; CHECK-LABEL: define nofpclass(nan) half @ret_loop_under_if_phi(
+; CHECK-SAME: half [[UNKNOWN:%.*]], i1 [[COND:%.*]], i1 [[COND2:%.*]], ptr [[P0:%.*]], ptr [[P1:%.*]]) {
+; CHECK-NEXT:  [[ENTRY:.*]]:
+; CHECK-NEXT:    br i1 [[COND]], label %[[LOOP:.*]], label %[[RET:.*]]
+; CHECK:       [[LOOP]]:
+; CHECK-NEXT:    [[LOOP_PHI:%.*]] = phi half [ [[LOOP_SELECT:%.*]], %[[LOOP]] ], [ 0xH7E00, %[[ENTRY]] ]
+; CHECK-NEXT:    [[LOOP_SELECT]] = select i1 [[COND2]], half 0xH7E00, half [[UNKNOWN]]
+; CHECK-NEXT:    [[LOOP_COND:%.*]] = load i1, ptr [[P1]], align 1
+; CHECK-NEXT:    br i1 [[LOOP_COND]], label %[[RET]], label %[[LOOP]]
+; CHECK:       [[RET]]:
+; CHECK-NEXT:    [[OUTER_PHI:%.*]] = phi half [ 0xH7E00, %[[ENTRY]] ], [ [[LOOP_PHI]], %[[LOOP]] ]
+; CHECK-NEXT:    ret half [[OUTER_PHI]]
+;
+entry:
+  %entry.select = select i1 %cond2, half 0xH7E00, half %unknown
+  br i1 %cond, label %loop, label %ret
+
+loop:
+  %loop.phi = phi half [ %loop.select, %loop ], [ 0xH7E00, %entry ]
+  %loop.select = select i1 %cond2, half 0xH7E00, half %unknown
+  %loop.cond = load i1, ptr %p1
+  br i1 %loop.cond, label %ret, label %loop
+
+ret:
+  %outer.phi = phi half [ 0xH7E00, %entry ], [ %loop.phi, %loop ]
+  ret half %outer.phi
+}
+
+
+define nofpclass(nan) half @assume_in_loop(half %assumed.nan.in.loop, half %unknown, i1 %cond, i1 %cond2, ptr %p0, ptr %p1) {
+; CHECK-LABEL: define nofpclass(nan) half @assume_in_loop(
+; CHECK-SAME: half [[ASSUMED_NAN_IN_LOOP:%.*]], half [[UNKNOWN:%.*]], i1 [[COND:%.*]], i1 [[COND2:%.*]], ptr [[P0:%.*]], ptr [[P1:%.*]]) {
+; CHECK-NEXT:  [[ENTRY:.*]]:
+; CHECK-NEXT:    br i1 [[COND]], label %[[LOOP:.*]], label %[[RET:.*]]
+; CHECK:       [[LOOP]]:
+; CHECK-NEXT:    [[LOOP_PHI:%.*]] = phi half [ [[LOOP_SELECT:%.*]], %[[LOOP]] ], [ [[ASSUMED_NAN_IN_LOOP]], %[[ENTRY]] ]
+; CHECK-NEXT:    [[IS_NAN:%.*]] = fcmp uno half [[ASSUMED_NAN_IN_LOOP]], 0xH0000
+; CHECK-NEXT:    call void @llvm.assume(i1 [[IS_NAN]])
+; CHECK-NEXT:    [[LOOP_SELECT]] = select i1 [[COND2]], half [[ASSUMED_NAN_IN_LOOP]], half [[UNKNOWN]]
+; CHECK-NEXT:    [[LOOP_COND:%.*]] = load i1, ptr [[P1]], align 1
+; CHECK-NEXT:    br i1 [[LOOP_COND]], label %[[RET]], label %[[LOOP]]
+; CHECK:       [[RET]]:
+; CHECK-NEXT:    [[OUTER_PHI:%.*]] = phi half [ [[ASSUMED_NAN_IN_LOOP]], %[[ENTRY]] ], [ [[LOOP_PHI]], %[[LOOP]] ]
+; CHECK-NEXT:    ret half [[OUTER_PHI]]
+;
+entry:
+  br i1 %cond, label %loop, label %ret
+
+loop:
+  %loop.phi = phi half [ %loop.select, %loop ], [ %assumed.nan.in.loop, %entry ]
+  %is.nan = fcmp uno half %assumed.nan.in.loop, 0.0
+  call void @llvm.assume(i1 %is.nan)
+  %loop.select = select i1 %cond2, half %assumed.nan.in.loop, half %unknown
+  %loop.cond = load i1, ptr %p1
+  br i1 %loop.cond, label %ret, label %loop
+
+ret:
+  %outer.phi = phi half [ %assumed.nan.in.loop, %entry ], [ %loop.phi, %loop ]
+  ret half %outer.phi
+}
+
+
+define nofpclass(nan) half @ret_phi_nan_check(half %checked.if.nan, half %unknown, i1 %cond) {
+; CHECK-LABEL: define nofpclass(nan) half @ret_phi_nan_check(
+; CHECK-SAME: half [[CHECKED_IF_NAN:%.*]], half [[UNKNOWN:%.*]], i1 [[COND:%.*]]) {
+; CHECK-NEXT:  [[ENTRY:.*]]:
+; CHECK-NEXT:    [[IS_NAN:%.*]] = fcmp uno half [[CHECKED_IF_NAN]], 0xH0000
+; CHECK-NEXT:    br i1 [[IS_NAN]], label %[[IF_NAN:.*]], label %[[RET:.*]]
+; CHECK:       [[IF_NAN]]:
+; CHECK-NEXT:    [[SELECT:%.*]] = select i1 [[COND]], half [[CHECKED_IF_NAN]], half [[UNKNOWN]]
+; CHECK-NEXT:    br label %[[RET]]
+; CHECK:       [[RET]]:
+; CHECK-NEXT:    [[PHI:%.*]] = phi half [ [[CHECKED_IF_NAN]], %[[ENTRY]] ], [ [[SELECT]], %[[IF_NAN]] ]
+; CHECK-NEXT:    ret half [[PHI]]
+;
+entry:
+  %is.nan = fcmp uno half %checked.if.nan, 0.0
+  br i1 %is.nan, label %if_nan, label %ret
+
+if_nan:
+  %select = select i1 %cond, half %checked.if.nan, half %unknown
+  br label %ret
+
+ret:
+  %phi = phi half [ %checked.if.nan, %entry ], [ %select, %if_nan ]
+  ret half %phi
+}
+
+define nofpclass(nan inf) half @edge_case_if_chain(half %x, i1 %cond, ptr %p0, ptr %p1, ptr %p2) {
+; CHECK-LABEL: define nofpclass(nan inf) half @edge_case_if_chain(
+; CHECK-SAME: half [[X:%.*]], i1 [[COND:%.*]], ptr [[P0:%.*]], ptr [[P1:%.*]], ptr [[P2:%.*]]) {
+; CHECK-NEXT:  [[ENTRY:.*:]]
+; CHECK-NEXT:    [[IS_INF:%.*]] = fcmp oeq half [[X]], 0xH7C00
+; CHECK-NEXT:    br i1 [[IS_INF]], label %[[IS_INF:.*]], label %[[ELSE:.*]]
+; CHECK:       [[IS_INF]]:
+; CHECK-NEXT:    [[LOAD0:%.*]] = load half, ptr [[P0]], align 2
+; CHECK-NEXT:    [[VAL0:%.*]] = select i1 [[COND]], half [[X]], half [[LOAD0]]
+; CHECK-NEXT:    br label %[[RET:.*]]
+; CHECK:       [[ELSE]]:
+; CHECK-NEXT:    [[IS_NAN:%.*]] = fcmp uno half [[X]], 0xH0000
+; CHECK-NEXT:    [[LOAD1:%.*]] = load half, ptr [[P0]], align 2
+; CHECK-NEXT:    [[VAL1:%.*]] = select i1 [[COND]], half [[LOAD1]], half [[X]]
+; CHECK-NEXT:    br i1 [[IS_NAN]], label %[[IS_NAN:.*]], label %[[RET]]
+; CHECK:       [[IS_NAN]]:
+; CHECK-NEXT:    [[LOAD2:%.*]] = load half, ptr [[P2]], align 2
+; CHECK-NEXT:    [[VAL2:%.*]] = select i1 [[COND]], half [[X]], half [[LOAD2]]
+; CHECK-NEXT:    br label %[[RET]]
+; CHECK:       [[RET]]:
+; CHECK-NEXT:    [[PHI:%.*]] = phi half [ [[VAL0]], %[[IS_INF]] ], [ [[VAL1]], %[[ELSE]] ], [ [[VAL2]], %[[IS_NAN]] ]
+; CHECK-NEXT:    ret half [[PHI]]
+;
+entry:
+  %is.inf = fcmp oeq half %x, 0xH7C00
+  br i1 %is.inf, label %is_inf, label %else
+
+is_inf:
+  %load0 = load half, ptr %p0
+  %val0 = select i1 %cond, half %x, half %load0
+  br label %ret
+
+else:
+  %is.nan = fcmp uno half %x, 0.0
+  %load1 = load half, ptr %p0
+  %val1 = select i1 %cond, half %load1, half %x
+  br i1 %is.nan, label %is_nan, label %ret
+
+is_nan:
+  %load2 = load half, ptr %p2
+  %val2 = select i1 %cond, half %x, half %load2
+  br label %ret
+
+ret:
+  %phi = phi half [ %val0, %is_inf ], [ %val1, %else ], [ %val2, %is_nan ]
+  ret half %phi
+}
+
+; Fold the %is_nan input value to %unknown, though the select is
+; defined in a different block.
+define nofpclass(nan) half @evaluate_phi_input_at_incoming_edge(half %known.nan.in.branch, half %unknown, i1 %cond) {
+; CHECK-LABEL: define nofpclass(nan) half @evaluate_phi_input_at_incoming_edge(
+; CHECK-SAME: half [[KNOWN_NAN_IN_BRANCH:%.*]], half [[UNKNOWN:%.*]], i1 [[COND:%.*]]) {
+; CHECK-NEXT:  [[ENTRY:.*]]:
+; CHECK-NEXT:    [[IS_NAN:%.*]] = fcmp uno half [[KNOWN_NAN_IN_BRANCH]], 0xH0000
+; CHECK-NEXT:    br i1 [[IS_NAN]], label %[[IS_NAN:.*]], label %[[RET:.*]]
+; CHECK:       [[IS_NAN]]:
+; CHECK-NEXT:    [[ENTRY_SELECT:%.*]] = select i1 [[COND]], half [[KNOWN_NAN_IN_BRANCH]], half [[UNKNOWN]]
+; CHECK-NEXT:    br label %[[RET]]
+; CHECK:       [[RET]]:
+; CHECK-NEXT:    [[PHI:%.*]] = phi half [ [[KNOWN_NAN_IN_BRANCH]], %[[ENTRY]] ], [ [[ENTRY_SELECT]], %[[IS_NAN]] ]
+; CHECK-NEXT:    ret half [[PHI]]
+;
+entry:
+  %entry.select = select i1 %cond, half %known.nan.in.branch, half %unknown
+  %is.nan = fcmp uno half %known.nan.in.branch, 0.0
+  br i1 %is.nan, label %is_nan, label %ret
+
+is_nan:
+  br label %ret
+
+ret:
+  %phi = phi half [ %known.nan.in.branch, %entry ], [ %entry.select, %is_nan ]
+  ret half %phi
+}
+
+; This phi should fold out into a ret 0. This depends on trying to
+; simplify both phi inputs at once, with the predecessor terminator as
+; the context.
+define nofpclass(nan) half @multi_simplify_lost_phi_context_(half %x, half %y, i1 %cond0, i1 %cond1, i1 %cond2) {
+; CHECK-LABEL: define nofpclass(nan) half @multi_simplify_lost_phi_context_(
+; CHECK-SAME: half [[X:%.*]], half [[Y:%.*]], i1 [[COND0:%.*]], i1 [[COND1:%.*]], i1 [[COND2:%.*]]) {
+; CHECK-NEXT:  [[ENTRY:.*:]]
+; CHECK-NEXT:    br i1 [[COND0]], label %[[ASSUME_A:.*]], label %[[ASSUME_B:.*]]
+; CHECK:       [[ASSUME_A]]:
+; CHECK-NEXT:    [[X_IS_NAN:%.*]] = fcmp uno half [[X]], 0xH0000
+; CHECK-NEXT:    call void @llvm.assume(i1 [[X_IS_NAN]])
+; CHECK-NEXT:    br label %[[A:.*]]
+; CHECK:       [[A]]:
+; CHECK-NEXT:    br label %[[RET:.*]]
+; CHECK:       [[ASSUME_B]]:
+; CHECK-NEXT:    [[Y_IS_NAN:%.*]] = fcmp uno half [[Y]], 0xH0000
+; CHECK-NEXT:    call void @llvm.assume(i1 [[Y_IS_NAN]])
+; CHECK-NEXT:    br label %[[B:.*]]
+; CHECK:       [[B]]:
+; CHECK-NEXT:    br label %[[RET]]
+; CHECK:       [[RET]]:
+; CHECK-NEXT:    ret half 0xH0000
+;
+entry:
+  br i1 %cond0, label %assume_a, label %assume_b
+
+assume_a:
+  %x.is.nan = fcmp uno half %x, 0.0
+  call void @llvm.assume(i1 %x.is.nan)
+  br label %a
+
+a:
+  %select_a = select i1 %cond2, half %x, half 0xH7E00
+  br label %ret
+
+assume_b:
+  %y.is.nan = fcmp uno half %y, 0.0
+  call void @llvm.assume(i1 %y.is.nan)
+  br label %b
+
+b:
+  %select_b = select i1 %cond2, half %y, half 0xH7E00
+  br label %ret
+
+ret:
+  %phi.must.be.nan = phi half [ %select_a, %a ], [ %select_b, %b ]
+  %select = select i1 %cond2, half %phi.must.be.nan, half 0.0
+  ret half %select
+}



More information about the llvm-commits mailing list