summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--llvm/lib/Transforms/Scalar/TailRecursionElimination.cpp3
-rw-r--r--llvm/test/Transforms/TailCallElim/deopt-bundle.ll57
2 files changed, 59 insertions, 1 deletions
diff --git a/llvm/lib/Transforms/Scalar/TailRecursionElimination.cpp b/llvm/lib/Transforms/Scalar/TailRecursionElimination.cpp
index 16c88270a84..2c0294ccc38 100644
--- a/llvm/lib/Transforms/Scalar/TailRecursionElimination.cpp
+++ b/llvm/lib/Transforms/Scalar/TailRecursionElimination.cpp
@@ -236,7 +236,7 @@ static bool markTails(Function &F, bool &AllCallsAreTailCalls) {
if (!CI || CI->isTailCall())
continue;
- bool IsNoTail = CI->isNoTailCall();
+ bool IsNoTail = CI->isNoTailCall() || CI->hasOperandBundles();
if (!IsNoTail && CI->doesNotAccessMemory()) {
// A call to a readnone function whose arguments are all things computed
@@ -256,6 +256,7 @@ static bool markTails(Function &F, bool &AllCallsAreTailCalls) {
SafeToTail = false;
break;
}
+ SafeToTail &= CI->hasOperandBundles();
if (SafeToTail) {
emitOptimizationRemark(
F.getContext(), "tailcallelim", F, CI->getDebugLoc(),
diff --git a/llvm/test/Transforms/TailCallElim/deopt-bundle.ll b/llvm/test/Transforms/TailCallElim/deopt-bundle.ll
new file mode 100644
index 00000000000..a98c47e9e6f
--- /dev/null
+++ b/llvm/test/Transforms/TailCallElim/deopt-bundle.ll
@@ -0,0 +1,57 @@
+; RUN: opt < %s -tailcallelim -S | FileCheck %s
+
+define i32 @f_1(i32 %x) {
+; CHECK-LABEL: @f_1(
+wentry:
+ %cond = icmp ugt i32 %x, 0
+ br i1 %cond, label %return, label %body
+
+body:
+; CHECK: body:
+; CHECK: call i32 @f_1(i32 %y) [ "deopt"() ]
+ %y = add i32 %x, 1
+ %tmp = call i32 @f_1(i32 %y) [ "deopt"() ]
+ ret i32 0
+
+return:
+ ret i32 1
+}
+
+define i32 @f_2(i32 %x) {
+; CHECK-LABEL: @f_2
+
+entry:
+ %cond = icmp ugt i32 %x, 0
+ br i1 %cond, label %return, label %body
+
+body:
+; CHECK: body:
+; CHECK: call i32 @f_2(i32 %y) [ "unknown"() ]
+ %y = add i32 %x, 1
+ %tmp = call i32 @f_2(i32 %y) [ "unknown"() ]
+ ret i32 0
+
+return:
+ ret i32 1
+}
+
+declare void @func()
+
+define void @f_3(i1 %B) personality i8 42 {
+; CHECK-LABEL: @f_3(
+entry:
+ invoke void @func()
+ to label %exit unwind label %merge
+merge:
+ %cs1 = catchswitch within none [label %catch] unwind to caller
+
+catch:
+; CHECK: catch:
+; CHECK: call void @f_3(i1 %B) [ "funclet"(token %cp) ]
+ %cp = catchpad within %cs1 []
+ call void @f_3(i1 %B) [ "funclet"(token %cp) ]
+ ret void
+
+exit:
+ ret void
+}
OpenPOWER on IntegriCloud