Index: lib/Transforms/Scalar/TailRecursionElimination.cpp =================================================================== --- lib/Transforms/Scalar/TailRecursionElimination.cpp +++ lib/Transforms/Scalar/TailRecursionElimination.cpp @@ -236,7 +236,7 @@ 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 @@ SafeToTail = false; break; } + SafeToTail &= CI->hasOperandBundles(); if (SafeToTail) { emitOptimizationRemark( F.getContext(), "tailcallelim", F, CI->getDebugLoc(), Index: test/Transforms/TailCallElim/deopt-bundle.ll =================================================================== --- /dev/null +++ test/Transforms/TailCallElim/deopt-bundle.ll @@ -0,0 +1,36 @@ +; 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 +}