Index: llvm/test/Transforms/InstCombine/fneg-fabs.ll =================================================================== --- llvm/test/Transforms/InstCombine/fneg-fabs.ll +++ llvm/test/Transforms/InstCombine/fneg-fabs.ll @@ -1,238 +1,225 @@ ; NOTE: Assertions have been autogenerated by utils/update_test_checks.py ; RUN: opt < %s -instcombine -S | FileCheck %s -; (X < +/-0.0) ? X : -X --> -fabs(X) -; (X <= +/-0.0) ? X : -X --> -fabs(X) -; One negative test with no fmf -define double @select_noFMF_nfabs_lt(double %x) { -; CHECK-LABEL: @select_noFMF_nfabs_lt( -; CHECK-NEXT: [[CMP:%.*]] = fcmp olt double [[X:%.*]], 0.000000e+00 -; CHECK-NEXT: [[NEGX:%.*]] = fneg double [[X]] -; CHECK-NEXT: [[SEL:%.*]] = select i1 [[CMP]], double [[X]], double [[NEGX]] -; CHECK-NEXT: ret double [[SEL]] + +; (X <= +/-0.0) ? X : (0.0 - X) --> No FMF No transformation +; (X < +/-0.0) ? X : (0.0 - X) --> (0.0 - fabs(X)) +; (X < +/-0.0) ? X : -X --> No FMF No transformation +; (X < +/-0.0) ? X : -X --> (0.0 - fabs(X)) FMF on FNEG +; (X < +/-0.0) ? X : -X --> (0.0 - fabs(X)) FMF on Select + +; One negative test with <=. We need FMFs to transform <=. +; (X <= +/-0.0) ? X : (0.0 - X) --> No transformation +define double @negative_fsub_nofmf(double %x) { +; CHECK-LABEL: @negative_fsub_nofmf( +; CHECK-NEXT: [[CMP:%.*]] = fcmp ole double [[X:%.*]], 0.000000e+00 +; CHECK-NEXT: [[NEGX:%.*]] = fsub double 0.000000e+00, [[X]] +; CHECK-NEXT: [[RETVAL:%.*]] = select i1 [[CMP]], double [[X]], double [[NEGX]] +; CHECK-NEXT: ret double [[RETVAL]] ; - %cmp = fcmp olt double %x, 0.000000e+00 - %negX = fneg double %x - %sel = select i1 %cmp, double %x, double %negX - ret double %sel + %cmp = fcmp ole double %x, 0.000000e+00 + %negX = fsub double 0.000000e+00, %x + %retval = select i1 %cmp, double %x, double %negX + ret double %retval } -; One test where the neg has fmfs. -define double @select_nsz_nfabs_lt_fmfProp(double %x) { -; CHECK-LABEL: @select_nsz_nfabs_lt_fmfProp( +; (X < +/-0.0) ? X : (0.0 - X) --> (0.0 - fabs(X)) +define double @olt_fsub(double %x) { +; CHECK-LABEL: @olt_fsub( ; CHECK-NEXT: [[CMP:%.*]] = fcmp olt double [[X:%.*]], 0.000000e+00 -; CHECK-NEXT: [[NEGX:%.*]] = fneg fast double [[X]] -; CHECK-NEXT: [[SEL:%.*]] = select nsz i1 [[CMP]], double [[X]], double [[NEGX]] -; CHECK-NEXT: ret double [[SEL]] +; CHECK-NEXT: [[NEGX:%.*]] = fsub double 0.000000e+00, [[X]] +; CHECK-NEXT: [[RETVAL:%.*]] = select i1 [[CMP]], double [[X]], double [[NEGX]] +; CHECK-NEXT: ret double [[RETVAL]] ; %cmp = fcmp olt double %x, 0.000000e+00 - %negX = fneg fast double %x - %sel = select nsz i1 %cmp, double %x, double %negX - ret double %sel + %negX = fsub double 0.000000e+00, %x + %retval = select i1 %cmp, double %x, double %negX + ret double %retval } - -; Tests with various predicate types. -define double @select_nsz_nfabs_olt(double %x) { -; CHECK-LABEL: @select_nsz_nfabs_olt( -; CHECK-NEXT: [[CMP:%.*]] = fcmp olt double [[X:%.*]], 0.000000e+00 -; CHECK-NEXT: [[NEGX:%.*]] = fneg double [[X]] -; CHECK-NEXT: [[SEL:%.*]] = select nsz i1 [[CMP]], double [[X]], double [[NEGX]] -; CHECK-NEXT: ret double [[SEL]] +define double @ult_fsub(double %x) { +; CHECK-LABEL: @ult_fsub( +; CHECK-NEXT: [[CMP:%.*]] = fcmp ult double [[X:%.*]], 0.000000e+00 +; CHECK-NEXT: [[NEGX:%.*]] = fsub double 0.000000e+00, [[X]] +; CHECK-NEXT: [[RETVAL:%.*]] = select i1 [[CMP]], double [[X]], double [[NEGX]] +; CHECK-NEXT: ret double [[RETVAL]] ; - %cmp = fcmp olt double %x, 0.000000e+00 - %negX = fneg double %x - %sel = select nsz i1 %cmp, double %x, double %negX - ret double %sel + %cmp = fcmp ult double %x, 0.000000e+00 + %negX = fsub double 0.000000e+00, %x + %retval = select i1 %cmp, double %x, double %negX + ret double %retval } -define double @select_nsz_nfabs_ult(double %x) { -; CHECK-LABEL: @select_nsz_nfabs_ult( +; One negative test with no FMFs on FNeg or the select. +; (X < +/-0.0) ? X : -X --> No FMF No transformation +define double @negative_fneg_nofmf(double %x) { +; CHECK-LABEL: @negative_fneg_nofmf( ; CHECK-NEXT: [[CMP:%.*]] = fcmp ult double [[X:%.*]], 0.000000e+00 ; CHECK-NEXT: [[NEGX:%.*]] = fneg double [[X]] -; CHECK-NEXT: [[SEL:%.*]] = select nsz i1 [[CMP]], double [[X]], double [[NEGX]] -; CHECK-NEXT: ret double [[SEL]] +; CHECK-NEXT: [[RETVAL:%.*]] = select i1 [[CMP]], double [[X]], double [[NEGX]] +; CHECK-NEXT: ret double [[RETVAL]] ; %cmp = fcmp ult double %x, 0.000000e+00 %negX = fneg double %x - %sel = select nsz i1 %cmp, double %x, double %negX - ret double %sel + %retval = select i1 %cmp, double %x, double %negX + ret double %retval } - -define double @select_nsz_nfabs_ole(double %x) { -; CHECK-LABEL: @select_nsz_nfabs_ole( -; CHECK-NEXT: [[CMP:%.*]] = fcmp ole double [[X:%.*]], 0.000000e+00 -; CHECK-NEXT: [[NEGX:%.*]] = fneg double [[X]] -; CHECK-NEXT: [[SEL:%.*]] = select nsz i1 [[CMP]], double [[X]], double [[NEGX]] -; CHECK-NEXT: ret double [[SEL]] +; (X < +/-0.0) ? X : -X --> (0.0 - fabs(X)) FMF on FNEG +; (X < +/-0.0) ? X : -X --> (0.0 - fabs(X)) FMF on Select +define double @ult_fneg1(double %x) { +; CHECK-LABEL: @ult_fneg1( +; CHECK-NEXT: [[CMP:%.*]] = fcmp ult double [[X:%.*]], 0.000000e+00 +; CHECK-NEXT: [[NEGX:%.*]] = fneg nsz double [[X]] +; CHECK-NEXT: [[RETVAL:%.*]] = select i1 [[CMP]], double [[X]], double [[NEGX]] +; CHECK-NEXT: ret double [[RETVAL]] ; - %cmp = fcmp ole double %x, 0.000000e+00 - %negX = fneg double %x - %sel = select nsz i1 %cmp, double %x, double %negX - ret double %sel + %cmp = fcmp ult double %x, 0.000000e+00 + %negX = fneg nsz double %x + %retval = select i1 %cmp, double %x, double %negX + ret double %retval } - -define double @select_nsz_nfabs_ule(double %x) { -; CHECK-LABEL: @select_nsz_nfabs_ule( -; CHECK-NEXT: [[CMP:%.*]] = fcmp ule double [[X:%.*]], 0.000000e+00 +define double @ult_fneg2(double %x) { +; CHECK-LABEL: @ult_fneg2( +; CHECK-NEXT: [[CMP:%.*]] = fcmp ult double [[X:%.*]], 0.000000e+00 ; CHECK-NEXT: [[NEGX:%.*]] = fneg double [[X]] -; CHECK-NEXT: [[SEL:%.*]] = select nsz i1 [[CMP]], double [[X]], double [[NEGX]] -; CHECK-NEXT: ret double [[SEL]] +; CHECK-NEXT: [[RETVAL:%.*]] = select nsz i1 [[CMP]], double [[X]], double [[NEGX]] +; CHECK-NEXT: ret double [[RETVAL]] ; - %cmp = fcmp ule double %x, 0.000000e+00 + %cmp = fcmp ult double %x, 0.000000e+00 %negX = fneg double %x - %sel = select nsz i1 %cmp, double %x, double %negX - ret double %sel + %retval = select nsz i1 %cmp, double %x, double %negX + ret double %retval } - -; (X > +/-0.0) ? -X : X --> -fabs(X) -; (X >= +/-0.0) ? -X : X --> -fabs(X) -; One negative test with no fmf -define double @select_noFMF_nfabs_gt(double %x) { -; CHECK-LABEL: @select_noFMF_nfabs_gt( -; CHECK-NEXT: [[CMP:%.*]] = fcmp ogt double [[X:%.*]], 0.000000e+00 +define double @olt_fneg1(double %x) { +; CHECK-LABEL: @olt_fneg1( +; CHECK-NEXT: [[CMP:%.*]] = fcmp olt double [[X:%.*]], 0.000000e+00 +; CHECK-NEXT: [[NEGX:%.*]] = fneg nsz double [[X]] +; CHECK-NEXT: [[RETVAL:%.*]] = select i1 [[CMP]], double [[X]], double [[NEGX]] +; CHECK-NEXT: ret double [[RETVAL]] +; + %cmp = fcmp olt double %x, 0.000000e+00 + %negX = fneg nsz double %x + %retval = select i1 %cmp, double %x, double %negX + ret double %retval +} +define double @olt_fneg2(double %x) { +; CHECK-LABEL: @olt_fneg2( +; CHECK-NEXT: [[CMP:%.*]] = fcmp olt double [[X:%.*]], 0.000000e+00 ; CHECK-NEXT: [[NEGX:%.*]] = fneg double [[X]] -; CHECK-NEXT: [[SEL:%.*]] = select i1 [[CMP]], double [[NEGX]], double [[X]] -; CHECK-NEXT: ret double [[SEL]] +; CHECK-NEXT: [[RETVAL:%.*]] = select nsz i1 [[CMP]], double [[X]], double [[NEGX]] +; CHECK-NEXT: ret double [[RETVAL]] ; - %cmp = fcmp ogt double %x, 0.000000e+00 + %cmp = fcmp olt double %x, 0.000000e+00 %negX = fneg double %x - %sel = select i1 %cmp, double %negX, double %x - ret double %sel + %retval = select nsz i1 %cmp, double %x, double %negX + ret double %retval } -; One test where the neg has fmfs. -define double @select_nsz_nfabs_gt_fmfProp(double %x) { -; CHECK-LABEL: @select_nsz_nfabs_gt_fmfProp( -; CHECK-NEXT: [[CMP:%.*]] = fcmp ogt double [[X:%.*]], 0.000000e+00 -; CHECK-NEXT: [[NEGX:%.*]] = fneg fast double [[X]] -; CHECK-NEXT: [[SEL:%.*]] = select nsz i1 [[CMP]], double [[NEGX]], double [[X]] -; CHECK-NEXT: ret double [[SEL]] + +; ///////////////////////////////////// +; Similar variations of tests for ">". +; (X >= +/-0.0) ? X : (0.0 - X) --> No FMF No transformation +; (X > +/-0.0) ? (0.0 - X) : X --> (0.0 - fabs(X)) +; (X > +/-0.0) ? -X : X --> No FMF No transformation +; (X > +/-0.0) ? -X : X --> (0.0 - fabs(X)) FMF on FNEG +; (X > +/-0.0) ? -X : X --> (0.0 - fabs(X)) FMF on Select +; ///////////////////////////////////// + +define double @negative_fsub_nofmf_greater(double %x) { +; CHECK-LABEL: @negative_fsub_nofmf_greater( +; CHECK-NEXT: [[CMP:%.*]] = fcmp oge double [[X:%.*]], 0.000000e+00 +; CHECK-NEXT: [[NEGX:%.*]] = fsub double 0.000000e+00, [[X]] +; CHECK-NEXT: [[RETVAL:%.*]] = select i1 [[CMP]], double [[NEGX]], double [[X]] +; CHECK-NEXT: ret double [[RETVAL]] ; - %cmp = fcmp ogt double %x, 0.000000e+00 - %negX = fneg fast double %x - %sel = select nsz i1 %cmp, double %negX, double %x - ret double %sel + %cmp = fcmp oge double %x, 0.000000e+00 + %negX = fsub double 0.000000e+00, %x + %retval = select i1 %cmp, double %negX, double %x + ret double %retval } -; Tests with various predicate types. -define double @select_nsz_nfabs_ogt(double %x) { -; CHECK-LABEL: @select_nsz_nfabs_ogt( +define double @ogt_fsub(double %x) { +; CHECK-LABEL: @ogt_fsub( ; CHECK-NEXT: [[CMP:%.*]] = fcmp ogt double [[X:%.*]], 0.000000e+00 -; CHECK-NEXT: [[NEGX:%.*]] = fneg double [[X]] -; CHECK-NEXT: [[SEL:%.*]] = select nsz i1 [[CMP]], double [[NEGX]], double [[X]] -; CHECK-NEXT: ret double [[SEL]] +; CHECK-NEXT: [[NEGX:%.*]] = fsub double 0.000000e+00, [[X]] +; CHECK-NEXT: [[RETVAL:%.*]] = select i1 [[CMP]], double [[NEGX]], double [[X]] +; CHECK-NEXT: ret double [[RETVAL]] ; %cmp = fcmp ogt double %x, 0.000000e+00 - %negX = fneg double %x - %sel = select nsz i1 %cmp, double %negX, double %x - ret double %sel + %negX = fsub double 0.000000e+00, %x + %retval = select i1 %cmp, double %negX, double %x + ret double %retval } - -define double @select_nsz_nfabs_ugt(double %x) { -; CHECK-LABEL: @select_nsz_nfabs_ugt( +define double @ugt_fsub(double %x) { +; CHECK-LABEL: @ugt_fsub( ; CHECK-NEXT: [[CMP:%.*]] = fcmp ugt double [[X:%.*]], 0.000000e+00 -; CHECK-NEXT: [[NEGX:%.*]] = fneg double [[X]] -; CHECK-NEXT: [[SEL:%.*]] = select nsz i1 [[CMP]], double [[NEGX]], double [[X]] -; CHECK-NEXT: ret double [[SEL]] +; CHECK-NEXT: [[NEGX:%.*]] = fsub double 0.000000e+00, [[X]] +; CHECK-NEXT: [[RETVAL:%.*]] = select i1 [[CMP]], double [[NEGX]], double [[X]] +; CHECK-NEXT: ret double [[RETVAL]] ; %cmp = fcmp ugt double %x, 0.000000e+00 - %negX = fneg double %x - %sel = select nsz i1 %cmp, double %negX, double %x - ret double %sel -} - -define double @select_nsz_nfabs_oge(double %x) { -; CHECK-LABEL: @select_nsz_nfabs_oge( -; CHECK-NEXT: [[CMP:%.*]] = fcmp oge double [[X:%.*]], 0.000000e+00 -; CHECK-NEXT: [[NEGX:%.*]] = fneg double [[X]] -; CHECK-NEXT: [[SEL:%.*]] = select nsz i1 [[CMP]], double [[NEGX]], double [[X]] -; CHECK-NEXT: ret double [[SEL]] -; - %cmp = fcmp oge double %x, 0.000000e+00 - %negX = fneg double %x - %sel = select nsz i1 %cmp, double %negX, double %x - ret double %sel + %negX = fsub double 0.000000e+00, %x + %retval = select i1 %cmp, double %negX, double %x + ret double %retval } -define double @select_nsz_nfabs_uge(double %x) { -; CHECK-LABEL: @select_nsz_nfabs_uge( -; CHECK-NEXT: [[CMP:%.*]] = fcmp uge double [[X:%.*]], 0.000000e+00 +define double @negative_fneg_nofmf_greater(double %x) { +; CHECK-LABEL: @negative_fneg_nofmf_greater( +; CHECK-NEXT: [[CMP:%.*]] = fcmp ugt double [[X:%.*]], 0.000000e+00 ; CHECK-NEXT: [[NEGX:%.*]] = fneg double [[X]] -; CHECK-NEXT: [[SEL:%.*]] = select nsz i1 [[CMP]], double [[NEGX]], double [[X]] -; CHECK-NEXT: ret double [[SEL]] +; CHECK-NEXT: [[RETVAL:%.*]] = select i1 [[CMP]], double [[NEGX]], double [[X]] +; CHECK-NEXT: ret double [[RETVAL]] ; - %cmp = fcmp uge double %x, 0.000000e+00 + %cmp = fcmp ugt double %x, 0.000000e+00 %negX = fneg double %x - %sel = select nsz i1 %cmp, double %negX, double %x - ret double %sel + %retval = select i1 %cmp, double %negX, double %x + ret double %retval } -; (X < +/-0.0) ? X : (0.0 - X) --> (0.0 - fabs(X)) -; One negative test with <=. -define double @select_noFMF_fsubfabs_le(double %x) { -; CHECK-LABEL: @select_noFMF_fsubfabs_le( -; CHECK-NEXT: [[CMP:%.*]] = fcmp ole double [[X:%.*]], 0.000000e+00 -; CHECK-NEXT: [[SUB:%.*]] = fsub double 0.000000e+00, [[X]] -; CHECK-NEXT: [[RETVAL_0:%.*]] = select i1 [[CMP]], double [[X]], double [[SUB]] -; CHECK-NEXT: ret double [[RETVAL_0]] -; - %cmp = fcmp ole double %x, 0.000000e+00 - %sub = fsub double 0.000000e+00, %x - %retval.0 = select i1 %cmp, double %x, double %sub - ret double %retval.0 -} - -define double @select_noFMF_fsubfabs_olt(double %x) { -; CHECK-LABEL: @select_noFMF_fsubfabs_olt( -; CHECK-NEXT: [[CMP:%.*]] = fcmp olt double [[X:%.*]], 0.000000e+00 -; CHECK-NEXT: [[SUB:%.*]] = fsub double 0.000000e+00, [[X]] -; CHECK-NEXT: [[RETVAL_0:%.*]] = select i1 [[CMP]], double [[X]], double [[SUB]] -; CHECK-NEXT: ret double [[RETVAL_0]] +define double @ugt_fneg1(double %x) { +; CHECK-LABEL: @ugt_fneg1( +; CHECK-NEXT: [[CMP:%.*]] = fcmp ugt double [[X:%.*]], 0.000000e+00 +; CHECK-NEXT: [[NEGX:%.*]] = fneg nsz double [[X]] +; CHECK-NEXT: [[RETVAL:%.*]] = select i1 [[CMP]], double [[NEGX]], double [[X]] +; CHECK-NEXT: ret double [[RETVAL]] ; - %cmp = fcmp olt double %x, 0.000000e+00 - %sub = fsub double 0.000000e+00, %x - %retval.0 = select i1 %cmp, double %x, double %sub - ret double %retval.0 + %cmp = fcmp ugt double %x, 0.000000e+00 + %negX = fneg nsz double %x + %retval = select i1 %cmp, double %negX, double %x + ret double %retval } - -define double @select_noFMF_fsubfabs_ult(double %x) { -; CHECK-LABEL: @select_noFMF_fsubfabs_ult( -; CHECK-NEXT: [[CMP:%.*]] = fcmp ult double [[X:%.*]], 0.000000e+00 -; CHECK-NEXT: [[SUB:%.*]] = fsub double 0.000000e+00, [[X]] -; CHECK-NEXT: [[RETVAL_0:%.*]] = select i1 [[CMP]], double [[X]], double [[SUB]] -; CHECK-NEXT: ret double [[RETVAL_0]] +define double @ugt_fneg2(double %x) { +; CHECK-LABEL: @ugt_fneg2( +; CHECK-NEXT: [[CMP:%.*]] = fcmp ugt double [[X:%.*]], 0.000000e+00 +; CHECK-NEXT: [[NEGX:%.*]] = fneg double [[X]] +; CHECK-NEXT: [[RETVAL:%.*]] = select nsz i1 [[CMP]], double [[NEGX]], double [[X]] +; CHECK-NEXT: ret double [[RETVAL]] ; - %cmp = fcmp ult double %x, 0.000000e+00 - %sub = fsub double 0.000000e+00, %x - %retval.0 = select i1 %cmp, double %x, double %sub - ret double %retval.0 + %cmp = fcmp ugt double %x, 0.000000e+00 + %negX = fneg double %x + %retval = select nsz i1 %cmp, double %negX, double %x + ret double %retval } - - -; With nsz: -; (X < +/-0.0) ? X : -X --> -fabs(X) -define double @select_nsz_fnegfabs_olt(double %x) { -; CHECK-LABEL: @select_nsz_fnegfabs_olt( -; CHECK-NEXT: [[CMP:%.*]] = fcmp olt double [[X:%.*]], 0.000000e+00 +define double @ogt_fneg1(double %x) { +; CHECK-LABEL: @ogt_fneg1( +; CHECK-NEXT: [[CMP:%.*]] = fcmp ogt double [[X:%.*]], 0.000000e+00 ; CHECK-NEXT: [[NEGX:%.*]] = fneg nsz double [[X]] -; CHECK-NEXT: [[RETVAL_0:%.*]] = select i1 [[CMP]], double [[X]], double [[NEGX]] -; CHECK-NEXT: ret double [[RETVAL_0]] +; CHECK-NEXT: [[RETVAL:%.*]] = select i1 [[CMP]], double [[NEGX]], double [[X]] +; CHECK-NEXT: ret double [[RETVAL]] ; - %cmp = fcmp olt double %x, 0.000000e+00 + %cmp = fcmp ogt double %x, 0.000000e+00 %negX = fneg nsz double %x - %retval.0 = select i1 %cmp, double %x, double %negX - ret double %retval.0 + %retval = select i1 %cmp, double %negX, double %x + ret double %retval } - -define double @select_nsz_fnegfabs_ult(double %x) { -; CHECK-LABEL: @select_nsz_fnegfabs_ult( -; CHECK-NEXT: [[CMP:%.*]] = fcmp ult double [[X:%.*]], 0.000000e+00 -; CHECK-NEXT: [[NEGX:%.*]] = fneg nsz double [[X]] -; CHECK-NEXT: [[RETVAL_0:%.*]] = select i1 [[CMP]], double [[X]], double [[NEGX]] -; CHECK-NEXT: ret double [[RETVAL_0]] +define double @ogt_fneg2(double %x) { +; CHECK-LABEL: @ogt_fneg2( +; CHECK-NEXT: [[CMP:%.*]] = fcmp ogt double [[X:%.*]], 0.000000e+00 +; CHECK-NEXT: [[NEGX:%.*]] = fneg double [[X]] +; CHECK-NEXT: [[RETVAL:%.*]] = select nsz i1 [[CMP]], double [[NEGX]], double [[X]] +; CHECK-NEXT: ret double [[RETVAL]] ; - %cmp = fcmp ult double %x, 0.000000e+00 - %negX = fneg nsz double %x - %retval.0 = select i1 %cmp, double %x, double %negX - ret double %retval.0 + %cmp = fcmp ogt double %x, 0.000000e+00 + %negX = fneg double %x + %retval = select nsz i1 %cmp, double %negX, double %x + ret double %retval }