[llvm] [SimplifyLibCalls] Constant fold scalbxx (PR #114417)
via llvm-commits
llvm-commits at lists.llvm.org
Wed Nov 6 08:15:32 PST 2024
https://github.com/fawdlstty updated https://github.com/llvm/llvm-project/pull/114417
>From 16ff8929b0dd53cda1f22d2a00ff36737a5d03d4 Mon Sep 17 00:00:00 2001
From: fawdlstty <f at fawdlstty.com>
Date: Thu, 31 Oct 2024 23:36:51 +0800
Subject: [PATCH 01/15] add optimize method & test
---
.../llvm/Transforms/Utils/SimplifyLibCalls.h | 1 +
.../lib/Transforms/Utils/SimplifyLibCalls.cpp | 43 ++++++++++
llvm/test/Transforms/InstCombine/scalbn.ll | 85 +++++++++++++++++++
3 files changed, 129 insertions(+)
create mode 100644 llvm/test/Transforms/InstCombine/scalbn.ll
diff --git a/llvm/include/llvm/Transforms/Utils/SimplifyLibCalls.h b/llvm/include/llvm/Transforms/Utils/SimplifyLibCalls.h
index 128502b99d9a37..93ab5f5974c189 100644
--- a/llvm/include/llvm/Transforms/Utils/SimplifyLibCalls.h
+++ b/llvm/include/llvm/Transforms/Utils/SimplifyLibCalls.h
@@ -213,6 +213,7 @@ class LibCallSimplifier {
Value *optimizeSymmetric(CallInst *CI, LibFunc Func, IRBuilderBase &B);
Value *optimizeRemquo(CallInst *CI, IRBuilderBase &B);
Value *optimizeFdim(CallInst *CI, IRBuilderBase &B);
+ Value *optimizeScalbn(CallInst *CI, IRBuilderBase &B);
// Wrapper for all floating point library call optimizations
Value *optimizeFloatingPointLibCall(CallInst *CI, LibFunc Func,
IRBuilderBase &B);
diff --git a/llvm/lib/Transforms/Utils/SimplifyLibCalls.cpp b/llvm/lib/Transforms/Utils/SimplifyLibCalls.cpp
index d85e0d99466022..afa2196243b538 100644
--- a/llvm/lib/Transforms/Utils/SimplifyLibCalls.cpp
+++ b/llvm/lib/Transforms/Utils/SimplifyLibCalls.cpp
@@ -3157,6 +3157,42 @@ Value *LibCallSimplifier::optimizeFdim(CallInst *CI, IRBuilderBase &B) {
return ConstantFP::get(CI->getType(), MaxVal);
}
+Value *LibCallSimplifier::optimizeScalbn(CallInst *CI, IRBuilderBase &B) {
+ Value *Base = CI->getArgOperand(0);
+ Value *Expo = CI->getArgOperand(1);
+ // Function *Callee = CI->getCalledFunction();
+ // StringRef Name = Callee->getName();
+ Type *Ty = CI->getType();
+ Module *M = CI->getModule();
+ // bool AllowApprox = CI->hasApproxFunc();
+ // bool Ignored;
+
+ // Propagate the math semantics from the call to any created instructions.
+ IRBuilderBase::FastMathFlagGuard Guard(B);
+ B.setFastMathFlags(CI->getFastMathFlags());
+ // Evaluate special cases related to the base.
+
+ // Scalbn(0.0, exp) -> 0.0
+ if (match(Base, m_SpecificFP(0.0)))
+ return Base;
+
+ // Scalbn(arg, 0) -> arg
+ if (match(Expo, m_Zero()))
+ return Base;
+
+ // Scalbn(arg, 1) -> arg * 2.0
+ if (match(Expo, m_SpecificInt(1)))
+ return B.CreateFMul(Base, ConstantFP::get(Ty, 2.0), "mul");
+
+ // Scalbn(1.0, exp) -> FLT_RADIX ^ exp
+ auto pow = createPowWithIntegerExponent(ConstantFP::get(Ty, 2.0), Expo, M, B);
+ if (match(Base, m_SpecificFP(1.0)))
+ return pow;
+
+ // otherwise
+ return B.CreateFMul(Base, pow, "mul");
+}
+
//===----------------------------------------------------------------------===//
// Integer Library Call Optimizations
//===----------------------------------------------------------------------===//
@@ -4105,6 +4141,13 @@ Value *LibCallSimplifier::optimizeFloatingPointLibCall(CallInst *CI,
case LibFunc_remquof:
case LibFunc_remquol:
return optimizeRemquo(CI, Builder);
+ case LibFunc_scalbln:
+ case LibFunc_scalblnf:
+ case LibFunc_scalblnl:
+ case LibFunc_scalbn:
+ case LibFunc_scalbnf:
+ case LibFunc_scalbnl:
+ return optimizeScalbn(CI, Builder);
case LibFunc_nan:
case LibFunc_nanf:
case LibFunc_nanl:
diff --git a/llvm/test/Transforms/InstCombine/scalbn.ll b/llvm/test/Transforms/InstCombine/scalbn.ll
new file mode 100644
index 00000000000000..4dccef967973cf
--- /dev/null
+++ b/llvm/test/Transforms/InstCombine/scalbn.ll
@@ -0,0 +1,85 @@
+; NOTE: Assertions have been autogenerated by utils/update_test_checks.py UTC_ARGS: --version 2
+; RUN: opt -S -passes=instcombine < %s | FileCheck %s
+
+declare double @scalbln(double, i64)
+declare float @scalblnf(float, i64)
+declare x86_fp80 @scalblnl(x86_fp80, i64)
+declare double @scalbn(double, i32)
+declare float @scalbnf(float, i32)
+declare x86_fp80 @scalbnl(x86_fp80, i32)
+
+;---------------------------------------------------------------------
+; scalbn(scalbn(x, a), b) -> scalbn(x, a + b)
+;---------------------------------------------------------------------
+
+define double @scalbln_scalbln(double %x, i64 %a, i64 %b) {
+; CHECK-LABEL: define double @scalbln
+; CHECK-SAME: (double [[X:%.*]], i64 [[A:%.*]], i64 [[B:%.*]]) {
+; CHECK-NEXT: [[SCALBXX0:%.*]] = call double @scalbln(double [[X]], i64 [[A]])
+; CHECK-NEXT: [[SCALBXX1:%.*]] = call double @scalbln(double [[SCALBXX0]], i64 [[B]])
+; CHECK-NEXT: ret double [[SCALBXX1]]
+;
+ %scalbxx0 = call double @scalbln(double %x, i64 %a)
+ %scalbxx1 = call double @scalbln(double %scalbxx0, i64 %b)
+ ret double %scalbxx1
+}
+
+define float @scalblnf_scalblnf(float %x, i64 %a, i64 %b) {
+; CHECK-LABEL: define float @scalblnf
+; CHECK-SAME: (float [[X:%.*]], i64 [[A:%.*]], i64 [[B:%.*]]) {
+; CHECK-NEXT: [[SCALBXX0:%.*]] = call float @scalblnf(float [[X]], i64 [[A]])
+; CHECK-NEXT: [[SCALBXX1:%.*]] = call float @scalblnf(float [[SCALBXX0]], i64 [[B]])
+; CHECK-NEXT: ret float [[SCALBXX1]]
+;
+ %scalbxx0 = call float @scalblnf(float %x, i64 %a)
+ %scalbxx1 = call float @scalblnf(float %scalbxx0, i64 %b)
+ ret float %scalbxx1
+}
+
+define x86_fp80 @scalblnl_scalblnl(x86_fp80 %x, i64 %a, i64 %b) {
+; CHECK-LABEL: define x86_fp80 @scalblnl
+; CHECK-SAME: (x86_fp80 [[X:%.*]], i64 [[A:%.*]], i64 [[B:%.*]]) {
+; CHECK-NEXT: [[SCALBXX0:%.*]] = call x86_fp80 @scalblnl(x86_fp80 [[X]], i64 [[A]])
+; CHECK-NEXT: [[SCALBXX1:%.*]] = call x86_fp80 @scalblnl(x86_fp80 [[SCALBXX0]], i64 [[B]])
+; CHECK-NEXT: ret x86_fp80 [[SCALBXX1]]
+;
+ %scalbxx0 = call x86_fp80 @scalblnl(x86_fp80 %x, i64 %a)
+ %scalbxx1 = call x86_fp80 @scalblnl(x86_fp80 %scalbxx0, i64 %b)
+ ret x86_fp80 %scalbxx1
+}
+
+define double @scalbn_scalbn(double %x, i32 %a, i32 %b) {
+; CHECK-LABEL: define double @scalbn
+; CHECK-SAME: (double [[X:%.*]], i32 [[A:%.*]], i32 [[B:%.*]]) {
+; CHECK-NEXT: [[SCALBXX0:%.*]] = call double @scalbn(double [[X]], i32 [[A]])
+; CHECK-NEXT: [[SCALBXX1:%.*]] = call double @scalbn(double [[SCALBXX0]], i32 [[B]])
+; CHECK-NEXT: ret double [[SCALBXX1]]
+;
+ %scalbxx0 = call double @scalbn(double %x, i32 %a)
+ %scalbxx1 = call double @scalbn(double %scalbxx0, i32 %b)
+ ret double %scalbxx1
+}
+
+define float @scalbnf_scalbnf(float %x, i32 %a, i32 %b) {
+; CHECK-LABEL: define float @scalbnf
+; CHECK-SAME: (float [[X:%.*]], i32 [[A:%.*]], i32 [[B:%.*]]) {
+; CHECK-NEXT: [[SCALBXX0:%.*]] = call float @scalbnf(float [[X]], i32 [[A]])
+; CHECK-NEXT: [[SCALBXX1:%.*]] = call float @scalbnf(float [[SCALBXX0]], i32 [[B]])
+; CHECK-NEXT: ret float [[SCALBXX1]]
+;
+ %scalbxx0 = call float @scalbnf(float %x, i32 %a)
+ %scalbxx1 = call float @scalbnf(float %scalbxx0, i32 %b)
+ ret float %scalbxx1
+}
+
+define x86_fp80 @scalbnl_scalbnl(x86_fp80 %x, i32 %a, i32 %b) {
+; CHECK-LABEL: define x86_fp80 @scalbnl
+; CHECK-SAME: (x86_fp80 [[X:%.*]], i32 [[A:%.*]], i32 [[B:%.*]]) {
+; CHECK-NEXT: [[SCALBXX0:%.*]] = call x86_fp80 @scalbnl(x86_fp80 [[X]], i32 [[A]])
+; CHECK-NEXT: [[SCALBXX1:%.*]] = call x86_fp80 @scalbnl(x86_fp80 [[SCALBXX0]], i32 [[B]])
+; CHECK-NEXT: ret x86_fp80 [[SCALBXX1]]
+;
+ %scalbxx0 = call x86_fp80 @scalbnl(x86_fp80 %x, i32 %a)
+ %scalbxx1 = call x86_fp80 @scalbnl(x86_fp80 %scalbxx0, i32 %b)
+ ret x86_fp80 %scalbxx1
+}
>From 11b33db7f51162a6caba02a1a9e98a5441bc810a Mon Sep 17 00:00:00 2001
From: fawdlstty <f at fawdlstty.com>
Date: Sat, 2 Nov 2024 23:53:03 +0800
Subject: [PATCH 02/15] impl in Analysis/ConstantFolding.cpp
---
llvm/include/llvm/IR/Intrinsics.td | 1 +
.../llvm/Transforms/Utils/SimplifyLibCalls.h | 1 -
llvm/lib/Analysis/ConstantFolding.cpp | 19 ++++++++--
.../lib/Transforms/Utils/SimplifyLibCalls.cpp | 36 -------------------
4 files changed, 18 insertions(+), 39 deletions(-)
diff --git a/llvm/include/llvm/IR/Intrinsics.td b/llvm/include/llvm/IR/Intrinsics.td
index 8ed57f818d6006..a9a015a51e2286 100644
--- a/llvm/include/llvm/IR/Intrinsics.td
+++ b/llvm/include/llvm/IR/Intrinsics.td
@@ -1034,6 +1034,7 @@ let IntrProperties = [IntrNoMem, IntrSpeculatable, IntrWillReturn] in {
def int_tanh : DefaultAttrsIntrinsic<[llvm_anyfloat_ty], [LLVMMatchType<0>]>;
def int_pow : DefaultAttrsIntrinsic<[llvm_anyfloat_ty],
[LLVMMatchType<0>, LLVMMatchType<0>]>;
+ def int_ldexp : DefaultAttrsIntrinsic<[llvm_anyfloat_ty], [LLVMMatchType<0>, LLVMMatchType<0>]>;
def int_log : DefaultAttrsIntrinsic<[llvm_anyfloat_ty], [LLVMMatchType<0>]>;
def int_log10: DefaultAttrsIntrinsic<[llvm_anyfloat_ty], [LLVMMatchType<0>]>;
def int_log2 : DefaultAttrsIntrinsic<[llvm_anyfloat_ty], [LLVMMatchType<0>]>;
diff --git a/llvm/include/llvm/Transforms/Utils/SimplifyLibCalls.h b/llvm/include/llvm/Transforms/Utils/SimplifyLibCalls.h
index 93ab5f5974c189..128502b99d9a37 100644
--- a/llvm/include/llvm/Transforms/Utils/SimplifyLibCalls.h
+++ b/llvm/include/llvm/Transforms/Utils/SimplifyLibCalls.h
@@ -213,7 +213,6 @@ class LibCallSimplifier {
Value *optimizeSymmetric(CallInst *CI, LibFunc Func, IRBuilderBase &B);
Value *optimizeRemquo(CallInst *CI, IRBuilderBase &B);
Value *optimizeFdim(CallInst *CI, IRBuilderBase &B);
- Value *optimizeScalbn(CallInst *CI, IRBuilderBase &B);
// Wrapper for all floating point library call optimizations
Value *optimizeFloatingPointLibCall(CallInst *CI, LibFunc Func,
IRBuilderBase &B);
diff --git a/llvm/lib/Analysis/ConstantFolding.cpp b/llvm/lib/Analysis/ConstantFolding.cpp
index c5a2c2f52f8dc2..dbbd3621d5e34e 100644
--- a/llvm/lib/Analysis/ConstantFolding.cpp
+++ b/llvm/lib/Analysis/ConstantFolding.cpp
@@ -1681,7 +1681,8 @@ bool llvm::canConstantFoldCallTo(const CallBase *Call, const Function *F) {
case 'i':
return Name == "ilogb" || Name == "ilogbf";
case 'l':
- return Name == "log" || Name == "logf" || Name == "logl" ||
+ return Name == "ldexp" || Name == "ldexpf" || Name == "ldexpl" ||
+ Name == "log" || Name == "logf" || Name == "logl" ||
Name == "log2" || Name == "log2f" || Name == "log10" ||
Name == "log10f" || Name == "logb" || Name == "logbf" ||
Name == "log1p" || Name == "log1pf";
@@ -1694,7 +1695,9 @@ bool llvm::canConstantFoldCallTo(const CallBase *Call, const Function *F) {
Name == "rint" || Name == "rintf" ||
Name == "round" || Name == "roundf";
case 's':
- return Name == "sin" || Name == "sinf" ||
+ return Name == "scalbn" || Name == "scalbnf" || Name == "scalbnl" ||
+ Name == "scalbln" || Name == "scalblnf" || Name == "scalblnl" ||
+ Name == "sin" || Name == "sinf" ||
Name == "sinh" || Name == "sinhf" ||
Name == "sqrt" || Name == "sqrtf";
case 't':
@@ -2372,6 +2375,18 @@ static Constant *ConstantFoldScalarCall1(StringRef Name,
return ConstantFP::get(Ty->getContext(), U);
}
break;
+ case LibFunc_ldexp:
+ case LibFunc_ldexpf:
+ case LibFunc_ldexpl:
+ case LibFunc_scalbn:
+ case LibFunc_scalbnf:
+ case LibFunc_scalbnl:
+ case LibFunc_scalbln:
+ case LibFunc_scalblnf:
+ case LibFunc_scalblnl:
+ if (TLI->has(Func))
+ return ConstantFoldFP(ldexp, APF, Ty);
+ break;
case LibFunc_log:
case LibFunc_logf:
case LibFunc_log_finite:
diff --git a/llvm/lib/Transforms/Utils/SimplifyLibCalls.cpp b/llvm/lib/Transforms/Utils/SimplifyLibCalls.cpp
index afa2196243b538..d333242762eb9e 100644
--- a/llvm/lib/Transforms/Utils/SimplifyLibCalls.cpp
+++ b/llvm/lib/Transforms/Utils/SimplifyLibCalls.cpp
@@ -3157,42 +3157,6 @@ Value *LibCallSimplifier::optimizeFdim(CallInst *CI, IRBuilderBase &B) {
return ConstantFP::get(CI->getType(), MaxVal);
}
-Value *LibCallSimplifier::optimizeScalbn(CallInst *CI, IRBuilderBase &B) {
- Value *Base = CI->getArgOperand(0);
- Value *Expo = CI->getArgOperand(1);
- // Function *Callee = CI->getCalledFunction();
- // StringRef Name = Callee->getName();
- Type *Ty = CI->getType();
- Module *M = CI->getModule();
- // bool AllowApprox = CI->hasApproxFunc();
- // bool Ignored;
-
- // Propagate the math semantics from the call to any created instructions.
- IRBuilderBase::FastMathFlagGuard Guard(B);
- B.setFastMathFlags(CI->getFastMathFlags());
- // Evaluate special cases related to the base.
-
- // Scalbn(0.0, exp) -> 0.0
- if (match(Base, m_SpecificFP(0.0)))
- return Base;
-
- // Scalbn(arg, 0) -> arg
- if (match(Expo, m_Zero()))
- return Base;
-
- // Scalbn(arg, 1) -> arg * 2.0
- if (match(Expo, m_SpecificInt(1)))
- return B.CreateFMul(Base, ConstantFP::get(Ty, 2.0), "mul");
-
- // Scalbn(1.0, exp) -> FLT_RADIX ^ exp
- auto pow = createPowWithIntegerExponent(ConstantFP::get(Ty, 2.0), Expo, M, B);
- if (match(Base, m_SpecificFP(1.0)))
- return pow;
-
- // otherwise
- return B.CreateFMul(Base, pow, "mul");
-}
-
//===----------------------------------------------------------------------===//
// Integer Library Call Optimizations
//===----------------------------------------------------------------------===//
>From 0d7c62cf833fec6d5e3205a4403c75623d03586e Mon Sep 17 00:00:00 2001
From: fawdlstty <f at fawdlstty.com>
Date: Sat, 2 Nov 2024 23:55:35 +0800
Subject: [PATCH 03/15] remove unused codes
---
llvm/lib/Transforms/Utils/SimplifyLibCalls.cpp | 7 -------
1 file changed, 7 deletions(-)
diff --git a/llvm/lib/Transforms/Utils/SimplifyLibCalls.cpp b/llvm/lib/Transforms/Utils/SimplifyLibCalls.cpp
index d333242762eb9e..d85e0d99466022 100644
--- a/llvm/lib/Transforms/Utils/SimplifyLibCalls.cpp
+++ b/llvm/lib/Transforms/Utils/SimplifyLibCalls.cpp
@@ -4105,13 +4105,6 @@ Value *LibCallSimplifier::optimizeFloatingPointLibCall(CallInst *CI,
case LibFunc_remquof:
case LibFunc_remquol:
return optimizeRemquo(CI, Builder);
- case LibFunc_scalbln:
- case LibFunc_scalblnf:
- case LibFunc_scalblnl:
- case LibFunc_scalbn:
- case LibFunc_scalbnf:
- case LibFunc_scalbnl:
- return optimizeScalbn(CI, Builder);
case LibFunc_nan:
case LibFunc_nanf:
case LibFunc_nanl:
>From 6ed6ba35846e7f614ce50c070756335bb5618eaf Mon Sep 17 00:00:00 2001
From: fawdlstty <f at fawdlstty.com>
Date: Sun, 3 Nov 2024 00:09:55 +0800
Subject: [PATCH 04/15] fix format error
---
llvm/lib/Analysis/ConstantFolding.cpp | 5 ++---
1 file changed, 2 insertions(+), 3 deletions(-)
diff --git a/llvm/lib/Analysis/ConstantFolding.cpp b/llvm/lib/Analysis/ConstantFolding.cpp
index dbbd3621d5e34e..2e07b300bb3c8f 100644
--- a/llvm/lib/Analysis/ConstantFolding.cpp
+++ b/llvm/lib/Analysis/ConstantFolding.cpp
@@ -1697,9 +1697,8 @@ bool llvm::canConstantFoldCallTo(const CallBase *Call, const Function *F) {
case 's':
return Name == "scalbn" || Name == "scalbnf" || Name == "scalbnl" ||
Name == "scalbln" || Name == "scalblnf" || Name == "scalblnl" ||
- Name == "sin" || Name == "sinf" ||
- Name == "sinh" || Name == "sinhf" ||
- Name == "sqrt" || Name == "sqrtf";
+ Name == "sin" || Name == "sinf" || Name == "sinh" ||
+ Name == "sinhf" || Name == "sqrt" || Name == "sqrtf";
case 't':
return Name == "tan" || Name == "tanf" ||
Name == "tanh" || Name == "tanhf" ||
>From 96955a6dd9c8d8194cd15f3563c223e6462dce63 Mon Sep 17 00:00:00 2001
From: fawdlstty <f at fawdlstty.com>
Date: Sun, 3 Nov 2024 00:17:02 +0800
Subject: [PATCH 05/15] a
---
llvm/lib/Analysis/ConstantFolding.cpp | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/llvm/lib/Analysis/ConstantFolding.cpp b/llvm/lib/Analysis/ConstantFolding.cpp
index 2e07b300bb3c8f..55d5b71bb28032 100644
--- a/llvm/lib/Analysis/ConstantFolding.cpp
+++ b/llvm/lib/Analysis/ConstantFolding.cpp
@@ -1698,7 +1698,7 @@ bool llvm::canConstantFoldCallTo(const CallBase *Call, const Function *F) {
return Name == "scalbn" || Name == "scalbnf" || Name == "scalbnl" ||
Name == "scalbln" || Name == "scalblnf" || Name == "scalblnl" ||
Name == "sin" || Name == "sinf" || Name == "sinh" ||
- Name == "sinhf" || Name == "sqrt" || Name == "sqrtf";
+ Name == "sinhf" || Name == "sqrt" || Name == "sqrtf";
case 't':
return Name == "tan" || Name == "tanf" ||
Name == "tanh" || Name == "tanhf" ||
>From bea62a32e89b1570b86fdfd79e1177035f84cb24 Mon Sep 17 00:00:00 2001
From: fawdlstty <f at fawdlstty.com>
Date: Sun, 3 Nov 2024 14:00:59 +0800
Subject: [PATCH 06/15] remove repeated item
---
llvm/include/llvm/IR/Intrinsics.td | 1 -
1 file changed, 1 deletion(-)
diff --git a/llvm/include/llvm/IR/Intrinsics.td b/llvm/include/llvm/IR/Intrinsics.td
index a9a015a51e2286..8ed57f818d6006 100644
--- a/llvm/include/llvm/IR/Intrinsics.td
+++ b/llvm/include/llvm/IR/Intrinsics.td
@@ -1034,7 +1034,6 @@ let IntrProperties = [IntrNoMem, IntrSpeculatable, IntrWillReturn] in {
def int_tanh : DefaultAttrsIntrinsic<[llvm_anyfloat_ty], [LLVMMatchType<0>]>;
def int_pow : DefaultAttrsIntrinsic<[llvm_anyfloat_ty],
[LLVMMatchType<0>, LLVMMatchType<0>]>;
- def int_ldexp : DefaultAttrsIntrinsic<[llvm_anyfloat_ty], [LLVMMatchType<0>, LLVMMatchType<0>]>;
def int_log : DefaultAttrsIntrinsic<[llvm_anyfloat_ty], [LLVMMatchType<0>]>;
def int_log10: DefaultAttrsIntrinsic<[llvm_anyfloat_ty], [LLVMMatchType<0>]>;
def int_log2 : DefaultAttrsIntrinsic<[llvm_anyfloat_ty], [LLVMMatchType<0>]>;
>From cdbc5408c4cc873c2423cf1a38c6a7b0fb99da69 Mon Sep 17 00:00:00 2001
From: fawdlstty <f at fawdlstty.com>
Date: Sun, 3 Nov 2024 14:53:56 +0800
Subject: [PATCH 07/15] 1
---
llvm/lib/Analysis/ConstantFolding.cpp | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/llvm/lib/Analysis/ConstantFolding.cpp b/llvm/lib/Analysis/ConstantFolding.cpp
index 55d5b71bb28032..32e160a3dfb06d 100644
--- a/llvm/lib/Analysis/ConstantFolding.cpp
+++ b/llvm/lib/Analysis/ConstantFolding.cpp
@@ -2384,7 +2384,7 @@ static Constant *ConstantFoldScalarCall1(StringRef Name,
case LibFunc_scalblnf:
case LibFunc_scalblnl:
if (TLI->has(Func))
- return ConstantFoldFP(ldexp, APF, Ty);
+ return ConstantFoldBinaryFP(ldexp, APF, Ty);
break;
case LibFunc_log:
case LibFunc_logf:
>From e6b21d0c5421722aa86f98b84f47c6621cdfa3f4 Mon Sep 17 00:00:00 2001
From: fawdlstty <f at fawdlstty.com>
Date: Sun, 3 Nov 2024 15:02:35 +0800
Subject: [PATCH 08/15] mov eval method
---
llvm/lib/Analysis/ConstantFolding.cpp | 24 ++++++++++++------------
1 file changed, 12 insertions(+), 12 deletions(-)
diff --git a/llvm/lib/Analysis/ConstantFolding.cpp b/llvm/lib/Analysis/ConstantFolding.cpp
index 32e160a3dfb06d..ba1a1c374e4fed 100644
--- a/llvm/lib/Analysis/ConstantFolding.cpp
+++ b/llvm/lib/Analysis/ConstantFolding.cpp
@@ -2374,18 +2374,6 @@ static Constant *ConstantFoldScalarCall1(StringRef Name,
return ConstantFP::get(Ty->getContext(), U);
}
break;
- case LibFunc_ldexp:
- case LibFunc_ldexpf:
- case LibFunc_ldexpl:
- case LibFunc_scalbn:
- case LibFunc_scalbnf:
- case LibFunc_scalbnl:
- case LibFunc_scalbln:
- case LibFunc_scalblnf:
- case LibFunc_scalblnl:
- if (TLI->has(Func))
- return ConstantFoldBinaryFP(ldexp, APF, Ty);
- break;
case LibFunc_log:
case LibFunc_logf:
case LibFunc_log_finite:
@@ -2635,6 +2623,18 @@ static Constant *ConstantFoldLibCall2(StringRef Name, Type *Ty,
switch (Func) {
default:
break;
+ case LibFunc_ldexp:
+ case LibFunc_ldexpf:
+ case LibFunc_ldexpl:
+ case LibFunc_scalbn:
+ case LibFunc_scalbnf:
+ case LibFunc_scalbnl:
+ case LibFunc_scalbln:
+ case LibFunc_scalblnf:
+ case LibFunc_scalblnl:
+ if (TLI->has(Func))
+ return ConstantFoldBinaryFP(ldexp, Op1V, Op2V, Ty);
+ break;
case LibFunc_pow:
case LibFunc_powf:
case LibFunc_pow_finite:
>From 810520f0f754db31a658b5cc58c297ee74880bee Mon Sep 17 00:00:00 2001
From: fawdlstty <f at fawdlstty.com>
Date: Sun, 3 Nov 2024 15:09:30 +0800
Subject: [PATCH 09/15] modify eval
---
llvm/lib/Analysis/ConstantFolding.cpp | 7 +++++--
1 file changed, 5 insertions(+), 2 deletions(-)
diff --git a/llvm/lib/Analysis/ConstantFolding.cpp b/llvm/lib/Analysis/ConstantFolding.cpp
index ba1a1c374e4fed..d0e2f0095a71d4 100644
--- a/llvm/lib/Analysis/ConstantFolding.cpp
+++ b/llvm/lib/Analysis/ConstantFolding.cpp
@@ -2632,8 +2632,11 @@ static Constant *ConstantFoldLibCall2(StringRef Name, Type *Ty,
case LibFunc_scalbln:
case LibFunc_scalblnf:
case LibFunc_scalblnl:
- if (TLI->has(Func))
- return ConstantFoldBinaryFP(ldexp, Op1V, Op2V, Ty);
+ if (TLI->has(Func)) {
+ APFloat V = Op1->getValueAPF();
+ if (APFloat::opStatus::opOK == V.ldexp(Op2->getValueAPF()))
+ return ConstantFP::get(Ty->getContext(), V);
+ }
break;
case LibFunc_pow:
case LibFunc_powf:
>From 0ec862d932c26f2de83d9dea8fbd8a861c191b05 Mon Sep 17 00:00:00 2001
From: fawdlstty <f at fawdlstty.com>
Date: Sun, 3 Nov 2024 23:07:27 +0800
Subject: [PATCH 10/15] update eval
---
llvm/lib/Analysis/ConstantFolding.cpp | 10 +++++++---
1 file changed, 7 insertions(+), 3 deletions(-)
diff --git a/llvm/lib/Analysis/ConstantFolding.cpp b/llvm/lib/Analysis/ConstantFolding.cpp
index d0e2f0095a71d4..2292643e20cbde 100644
--- a/llvm/lib/Analysis/ConstantFolding.cpp
+++ b/llvm/lib/Analysis/ConstantFolding.cpp
@@ -2633,9 +2633,13 @@ static Constant *ConstantFoldLibCall2(StringRef Name, Type *Ty,
case LibFunc_scalblnf:
case LibFunc_scalblnl:
if (TLI->has(Func)) {
- APFloat V = Op1->getValueAPF();
- if (APFloat::opStatus::opOK == V.ldexp(Op2->getValueAPF()))
- return ConstantFP::get(Ty->getContext(), V);
+ if (const auto *ConstrIntr =
+ dyn_cast_if_present<ConstrainedFPIntrinsic>(Call)) {
+ RoundingMode RM = getEvaluationRoundingMode(ConstrIntr);
+ auto tmp = ConstantFoldBinaryFP(pow, APFloat(2.0), Op2V, Ty);
+ Op1V.multiply(APFloat(tmp, RM));
+ return ConstantFP::get(Ty->getContext(), Op1V);
+ }
}
break;
case LibFunc_pow:
>From b5d5da5ba5d6b649fbb10bcdfb4eff7edd779e22 Mon Sep 17 00:00:00 2001
From: fawdlstty <f at fawdlstty.com>
Date: Mon, 4 Nov 2024 00:13:23 +0800
Subject: [PATCH 11/15] fix eval
---
llvm/lib/Analysis/ConstantFolding.cpp | 15 ++++++++-------
1 file changed, 8 insertions(+), 7 deletions(-)
diff --git a/llvm/lib/Analysis/ConstantFolding.cpp b/llvm/lib/Analysis/ConstantFolding.cpp
index 2292643e20cbde..b90d6d821ada32 100644
--- a/llvm/lib/Analysis/ConstantFolding.cpp
+++ b/llvm/lib/Analysis/ConstantFolding.cpp
@@ -2633,13 +2633,14 @@ static Constant *ConstantFoldLibCall2(StringRef Name, Type *Ty,
case LibFunc_scalblnf:
case LibFunc_scalblnl:
if (TLI->has(Func)) {
- if (const auto *ConstrIntr =
- dyn_cast_if_present<ConstrainedFPIntrinsic>(Call)) {
- RoundingMode RM = getEvaluationRoundingMode(ConstrIntr);
- auto tmp = ConstantFoldBinaryFP(pow, APFloat(2.0), Op2V, Ty);
- Op1V.multiply(APFloat(tmp, RM));
- return ConstantFP::get(Ty->getContext(), Op1V);
- }
+ Constant *tmp = ConstantFoldBinaryFP(pow, APFloat(2.0), Op2V, Ty);
+ const auto *tmp1 = dyn_cast<ConstantFP>(tmp);
+ if (!tmp1)
+ return nullptr;
+ APFloat Op1V2(Op1V);
+ auto eval_ret = Op1V2.multiply(tmp1->getValueAPF(), RoundingMode::TowardZero);
+ if (APFloat::opStatus::opOK == eval_ret)
+ return ConstantFP::get(Ty->getContext(), Op1V2);
}
break;
case LibFunc_pow:
>From 4e7db059732ca0690698d16313c6db9107d77e68 Mon Sep 17 00:00:00 2001
From: fawdlstty <f at fawdlstty.com>
Date: Tue, 5 Nov 2024 00:29:52 +0800
Subject: [PATCH 12/15] use apfloat scalbn impl
---
llvm/lib/Analysis/ConstantFolding.cpp | 10 ++--------
1 file changed, 2 insertions(+), 8 deletions(-)
diff --git a/llvm/lib/Analysis/ConstantFolding.cpp b/llvm/lib/Analysis/ConstantFolding.cpp
index b90d6d821ada32..c48cdc198706a4 100644
--- a/llvm/lib/Analysis/ConstantFolding.cpp
+++ b/llvm/lib/Analysis/ConstantFolding.cpp
@@ -2633,14 +2633,8 @@ static Constant *ConstantFoldLibCall2(StringRef Name, Type *Ty,
case LibFunc_scalblnf:
case LibFunc_scalblnl:
if (TLI->has(Func)) {
- Constant *tmp = ConstantFoldBinaryFP(pow, APFloat(2.0), Op2V, Ty);
- const auto *tmp1 = dyn_cast<ConstantFP>(tmp);
- if (!tmp1)
- return nullptr;
- APFloat Op1V2(Op1V);
- auto eval_ret = Op1V2.multiply(tmp1->getValueAPF(), RoundingMode::TowardZero);
- if (APFloat::opStatus::opOK == eval_ret)
- return ConstantFP::get(Ty->getContext(), Op1V2);
+ APFloat ret = llvm::scalbn(Op1V, (int)Op2V.convertToDouble(), RoundingMode::TowardZero);
+ return ConstantFP::get(Ty->getContext(), ret);
}
break;
case LibFunc_pow:
>From ab6d16161bbca330877c46f98a03791a1d3a1442 Mon Sep 17 00:00:00 2001
From: fawdlstty <f at fawdlstty.com>
Date: Tue, 5 Nov 2024 21:32:32 +0800
Subject: [PATCH 13/15] remove convert & fix test
---
llvm/lib/Analysis/ConstantFolding.cpp | 2 +-
llvm/test/Transforms/InstCombine/scalbn.ll | 2 +-
2 files changed, 2 insertions(+), 2 deletions(-)
diff --git a/llvm/lib/Analysis/ConstantFolding.cpp b/llvm/lib/Analysis/ConstantFolding.cpp
index c48cdc198706a4..be163666ae1e45 100644
--- a/llvm/lib/Analysis/ConstantFolding.cpp
+++ b/llvm/lib/Analysis/ConstantFolding.cpp
@@ -2633,7 +2633,7 @@ static Constant *ConstantFoldLibCall2(StringRef Name, Type *Ty,
case LibFunc_scalblnf:
case LibFunc_scalblnl:
if (TLI->has(Func)) {
- APFloat ret = llvm::scalbn(Op1V, (int)Op2V.convertToDouble(), RoundingMode::TowardZero);
+ APFloat ret = llvm::scalbn(Op1V, Op2V.convertToDouble(), RoundingMode::TowardZero);
return ConstantFP::get(Ty->getContext(), ret);
}
break;
diff --git a/llvm/test/Transforms/InstCombine/scalbn.ll b/llvm/test/Transforms/InstCombine/scalbn.ll
index 4dccef967973cf..59be867dfd2469 100644
--- a/llvm/test/Transforms/InstCombine/scalbn.ll
+++ b/llvm/test/Transforms/InstCombine/scalbn.ll
@@ -1,5 +1,5 @@
; NOTE: Assertions have been autogenerated by utils/update_test_checks.py UTC_ARGS: --version 2
-; RUN: opt -S -passes=instcombine < %s | FileCheck %s
+; RUN: opt -S -passes=instsimplify < %s | FileCheck %s
declare double @scalbln(double, i64)
declare float @scalblnf(float, i64)
>From 3c64e380c298bd36cd383c25c2d7fddcfa8aeb31 Mon Sep 17 00:00:00 2001
From: fawdlstty <f at fawdlstty.com>
Date: Tue, 5 Nov 2024 21:37:30 +0800
Subject: [PATCH 14/15] fix format check
---
llvm/lib/Analysis/ConstantFolding.cpp | 3 ++-
1 file changed, 2 insertions(+), 1 deletion(-)
diff --git a/llvm/lib/Analysis/ConstantFolding.cpp b/llvm/lib/Analysis/ConstantFolding.cpp
index be163666ae1e45..b6ccc9dc81194d 100644
--- a/llvm/lib/Analysis/ConstantFolding.cpp
+++ b/llvm/lib/Analysis/ConstantFolding.cpp
@@ -2633,7 +2633,8 @@ static Constant *ConstantFoldLibCall2(StringRef Name, Type *Ty,
case LibFunc_scalblnf:
case LibFunc_scalblnl:
if (TLI->has(Func)) {
- APFloat ret = llvm::scalbn(Op1V, Op2V.convertToDouble(), RoundingMode::TowardZero);
+ APFloat ret =
+ llvm::scalbn(Op1V, Op2V.convertToDouble(), RoundingMode::TowardZero);
return ConstantFP::get(Ty->getContext(), ret);
}
break;
>From 4dddafa85ed4d1e2f4e5853d50c85000f779f71a Mon Sep 17 00:00:00 2001
From: fawdlstty <f at fawdlstty.com>
Date: Thu, 7 Nov 2024 00:15:09 +0800
Subject: [PATCH 15/15] move test file
---
llvm/test/Transforms/InstCombine/scalbn.ll | 85 ---------------------
llvm/test/Transforms/InstSimplify/scalbn.ll | 69 +++++++++++++++++
2 files changed, 69 insertions(+), 85 deletions(-)
delete mode 100644 llvm/test/Transforms/InstCombine/scalbn.ll
create mode 100644 llvm/test/Transforms/InstSimplify/scalbn.ll
diff --git a/llvm/test/Transforms/InstCombine/scalbn.ll b/llvm/test/Transforms/InstCombine/scalbn.ll
deleted file mode 100644
index 59be867dfd2469..00000000000000
--- a/llvm/test/Transforms/InstCombine/scalbn.ll
+++ /dev/null
@@ -1,85 +0,0 @@
-; NOTE: Assertions have been autogenerated by utils/update_test_checks.py UTC_ARGS: --version 2
-; RUN: opt -S -passes=instsimplify < %s | FileCheck %s
-
-declare double @scalbln(double, i64)
-declare float @scalblnf(float, i64)
-declare x86_fp80 @scalblnl(x86_fp80, i64)
-declare double @scalbn(double, i32)
-declare float @scalbnf(float, i32)
-declare x86_fp80 @scalbnl(x86_fp80, i32)
-
-;---------------------------------------------------------------------
-; scalbn(scalbn(x, a), b) -> scalbn(x, a + b)
-;---------------------------------------------------------------------
-
-define double @scalbln_scalbln(double %x, i64 %a, i64 %b) {
-; CHECK-LABEL: define double @scalbln
-; CHECK-SAME: (double [[X:%.*]], i64 [[A:%.*]], i64 [[B:%.*]]) {
-; CHECK-NEXT: [[SCALBXX0:%.*]] = call double @scalbln(double [[X]], i64 [[A]])
-; CHECK-NEXT: [[SCALBXX1:%.*]] = call double @scalbln(double [[SCALBXX0]], i64 [[B]])
-; CHECK-NEXT: ret double [[SCALBXX1]]
-;
- %scalbxx0 = call double @scalbln(double %x, i64 %a)
- %scalbxx1 = call double @scalbln(double %scalbxx0, i64 %b)
- ret double %scalbxx1
-}
-
-define float @scalblnf_scalblnf(float %x, i64 %a, i64 %b) {
-; CHECK-LABEL: define float @scalblnf
-; CHECK-SAME: (float [[X:%.*]], i64 [[A:%.*]], i64 [[B:%.*]]) {
-; CHECK-NEXT: [[SCALBXX0:%.*]] = call float @scalblnf(float [[X]], i64 [[A]])
-; CHECK-NEXT: [[SCALBXX1:%.*]] = call float @scalblnf(float [[SCALBXX0]], i64 [[B]])
-; CHECK-NEXT: ret float [[SCALBXX1]]
-;
- %scalbxx0 = call float @scalblnf(float %x, i64 %a)
- %scalbxx1 = call float @scalblnf(float %scalbxx0, i64 %b)
- ret float %scalbxx1
-}
-
-define x86_fp80 @scalblnl_scalblnl(x86_fp80 %x, i64 %a, i64 %b) {
-; CHECK-LABEL: define x86_fp80 @scalblnl
-; CHECK-SAME: (x86_fp80 [[X:%.*]], i64 [[A:%.*]], i64 [[B:%.*]]) {
-; CHECK-NEXT: [[SCALBXX0:%.*]] = call x86_fp80 @scalblnl(x86_fp80 [[X]], i64 [[A]])
-; CHECK-NEXT: [[SCALBXX1:%.*]] = call x86_fp80 @scalblnl(x86_fp80 [[SCALBXX0]], i64 [[B]])
-; CHECK-NEXT: ret x86_fp80 [[SCALBXX1]]
-;
- %scalbxx0 = call x86_fp80 @scalblnl(x86_fp80 %x, i64 %a)
- %scalbxx1 = call x86_fp80 @scalblnl(x86_fp80 %scalbxx0, i64 %b)
- ret x86_fp80 %scalbxx1
-}
-
-define double @scalbn_scalbn(double %x, i32 %a, i32 %b) {
-; CHECK-LABEL: define double @scalbn
-; CHECK-SAME: (double [[X:%.*]], i32 [[A:%.*]], i32 [[B:%.*]]) {
-; CHECK-NEXT: [[SCALBXX0:%.*]] = call double @scalbn(double [[X]], i32 [[A]])
-; CHECK-NEXT: [[SCALBXX1:%.*]] = call double @scalbn(double [[SCALBXX0]], i32 [[B]])
-; CHECK-NEXT: ret double [[SCALBXX1]]
-;
- %scalbxx0 = call double @scalbn(double %x, i32 %a)
- %scalbxx1 = call double @scalbn(double %scalbxx0, i32 %b)
- ret double %scalbxx1
-}
-
-define float @scalbnf_scalbnf(float %x, i32 %a, i32 %b) {
-; CHECK-LABEL: define float @scalbnf
-; CHECK-SAME: (float [[X:%.*]], i32 [[A:%.*]], i32 [[B:%.*]]) {
-; CHECK-NEXT: [[SCALBXX0:%.*]] = call float @scalbnf(float [[X]], i32 [[A]])
-; CHECK-NEXT: [[SCALBXX1:%.*]] = call float @scalbnf(float [[SCALBXX0]], i32 [[B]])
-; CHECK-NEXT: ret float [[SCALBXX1]]
-;
- %scalbxx0 = call float @scalbnf(float %x, i32 %a)
- %scalbxx1 = call float @scalbnf(float %scalbxx0, i32 %b)
- ret float %scalbxx1
-}
-
-define x86_fp80 @scalbnl_scalbnl(x86_fp80 %x, i32 %a, i32 %b) {
-; CHECK-LABEL: define x86_fp80 @scalbnl
-; CHECK-SAME: (x86_fp80 [[X:%.*]], i32 [[A:%.*]], i32 [[B:%.*]]) {
-; CHECK-NEXT: [[SCALBXX0:%.*]] = call x86_fp80 @scalbnl(x86_fp80 [[X]], i32 [[A]])
-; CHECK-NEXT: [[SCALBXX1:%.*]] = call x86_fp80 @scalbnl(x86_fp80 [[SCALBXX0]], i32 [[B]])
-; CHECK-NEXT: ret x86_fp80 [[SCALBXX1]]
-;
- %scalbxx0 = call x86_fp80 @scalbnl(x86_fp80 %x, i32 %a)
- %scalbxx1 = call x86_fp80 @scalbnl(x86_fp80 %scalbxx0, i32 %b)
- ret x86_fp80 %scalbxx1
-}
diff --git a/llvm/test/Transforms/InstSimplify/scalbn.ll b/llvm/test/Transforms/InstSimplify/scalbn.ll
new file mode 100644
index 00000000000000..1c9925b0c6431d
--- /dev/null
+++ b/llvm/test/Transforms/InstSimplify/scalbn.ll
@@ -0,0 +1,69 @@
+; NOTE: Assertions have been autogenerated by utils/update_test_checks.py
+; RUN: opt < %s -passes=instsimplify -S | FileCheck %s
+
+declare double @scalbln(double, i64)
+declare float @scalblnf(float, i64)
+declare x86_fp80 @scalblnl(x86_fp80, i64)
+declare double @scalbn(double, i32)
+declare float @scalbnf(float, i32)
+declare x86_fp80 @scalbnl(x86_fp80, i32)
+
+;
+
+define double @scalbln_undef() {
+; CHECK-LABEL: @scalbln_undef(
+; CHECK-NEXT: ret double 0x7FF8000000000000
+;
+ %call = call double @scalbln(double undef, i64 undef)
+ ret double %call
+}
+
+define float @scalblnf_undef() {
+; CHECK-LABEL: @scalblnf_undef(
+; CHECK-NEXT: ret float 0x7FF8000000000000
+;
+ %call = call float @scalblnf(float undef, i64 undef)
+ ret float %call
+}
+
+define x86_fp80 @scalblnl_undef() {
+; CHECK-LABEL: @scalblnl_undef(
+; CHECK-NEXT: ret x86_fp80 0x7FF8000000000000
+;
+ %call = call x86_fp80 @scalblnl(x86_fp80 undef, i64 undef)
+ ret x86_fp80 %call
+}
+
+define double @scalbn_undef() {
+; CHECK-LABEL: @scalbn_undef(
+; CHECK-NEXT: ret double 0x7FF8000000000000
+;
+ %call = call double @scalbn(double undef, i32 undef)
+ ret double %call
+}
+
+define float @scalbnf_undef() {
+; CHECK-LABEL: @scalbnf_undef(
+; CHECK-NEXT: ret float 0x7FF8000000000000
+;
+ %call = call float @scalbnf(float undef, i32 undef)
+ ret float %call
+}
+
+define x86_fp80 @scalbnl_undef() {
+; CHECK-LABEL: @scalbnl_undef(
+; CHECK-NEXT: ret x86_fp80 0x7FF8000000000000
+;
+ %call = call x86_fp80 @scalbnl(x86_fp80 undef, i32 undef)
+ ret x86_fp80 %call
+}
+
+;
+
+define double @scalbn_check() {
+; CHECK-LABEL: @scalbn_check(
+; CHECK-NEXT: ret double 12
+;
+ %call = call double @scalbn(double 3.0, i32 2)
+ ret %call
+}
More information about the llvm-commits
mailing list