summaryrefslogtreecommitdiffstats
path: root/llvm
diff options
context:
space:
mode:
authorEvan Cheng <evan.cheng@apple.com>2010-01-31 06:44:49 +0000
committerEvan Cheng <evan.cheng@apple.com>2010-01-31 06:44:49 +0000
commit7f62def0f9d019493cdcaf014dc4c4774f9fa01a (patch)
tree4c5987d68cf3956b3d4f639f0b85243c850d9b32 /llvm
parent0de0b3677ac91f0f864181220c8da7333e113d5a (diff)
downloadbcm5719-llvm-7f62def0f9d019493cdcaf014dc4c4774f9fa01a.tar.gz
bcm5719-llvm-7f62def0f9d019493cdcaf014dc4c4774f9fa01a.zip
Avoid recursive sibcall's.
llvm-svn: 94946
Diffstat (limited to 'llvm')
-rw-r--r--llvm/lib/Target/X86/X86ISelLowering.cpp26
-rw-r--r--llvm/test/CodeGen/X86/tailcall2.ll24
2 files changed, 44 insertions, 6 deletions
diff --git a/llvm/lib/Target/X86/X86ISelLowering.cpp b/llvm/lib/Target/X86/X86ISelLowering.cpp
index c0ebade2dcd..e27d93cf7c8 100644
--- a/llvm/lib/Target/X86/X86ISelLowering.cpp
+++ b/llvm/lib/Target/X86/X86ISelLowering.cpp
@@ -2252,10 +2252,26 @@ X86TargetLowering::IsEligibleForTailCallOptimization(SDValue Callee,
// If -tailcallopt is specified, make fastcc functions tail-callable.
const Function *CallerF = DAG.getMachineFunction().getFunction();
- if (PerformTailCallOpt &&
- CalleeCC == CallingConv::Fast &&
- CallerF->getCallingConv() == CalleeCC)
- return true;
+ if (PerformTailCallOpt) {
+ if (CalleeCC == CallingConv::Fast &&
+ CallerF->getCallingConv() == CalleeCC)
+ return true;
+ return false;
+ }
+
+ // Do not tail call optimize vararg calls for now.
+ if (isVarArg)
+ return false;
+
+ // Don't tail call optimize recursive call.
+ GlobalAddressSDNode *G = dyn_cast<GlobalAddressSDNode>(Callee);
+ const Function *CalleeF = G ? cast<Function>(G->getGlobal()) : 0;
+ if (CallerF == CalleeF)
+ return false;
+ // If it's an indirect call, conversatively return false if the caller's
+ // address is taken.
+ if (!isa<ExternalSymbolSDNode>(Callee) && CallerF->hasAddressTaken())
+ return false;
// Look for obvious safe cases to perform tail call optimization.
// If the callee takes no arguments then go on to check the results of the
@@ -2279,9 +2295,7 @@ X86TargetLowering::IsEligibleForTailCallOptimization(SDValue Callee,
return true;
// If the return types match, then it's safe.
- GlobalAddressSDNode *G = dyn_cast<GlobalAddressSDNode>(Callee);
if (!G) return false; // FIXME: common external symbols?
- Function *CalleeF = cast<Function>(G->getGlobal());
const Type *CalleeRetTy = CalleeF->getReturnType();
return CallerRetTy == CalleeRetTy;
}
diff --git a/llvm/test/CodeGen/X86/tailcall2.ll b/llvm/test/CodeGen/X86/tailcall2.ll
index 6b0916fc675..e78e213d646 100644
--- a/llvm/test/CodeGen/X86/tailcall2.ll
+++ b/llvm/test/CodeGen/X86/tailcall2.ll
@@ -65,3 +65,27 @@ entry:
tail call void %x() nounwind
ret void
}
+
+define i32 @t6(i32 %x) nounwind ssp {
+entry:
+; 32: t6:
+; 32: call {{_?}}t6
+; 32: call {{_?}}bar
+
+; 64: t6:
+; 64: callq {{_?}}t6
+; 64: jmp {{_?}}bar
+ %0 = icmp slt i32 %x, 10
+ br i1 %0, label %bb, label %bb1
+
+bb:
+ %1 = add nsw i32 %x, -1
+ %2 = tail call i32 @t6(i32 %1) nounwind ssp
+ ret i32 %2
+
+bb1:
+ %3 = tail call i32 @bar(i32 %x) nounwind
+ ret i32 %3
+}
+
+declare i32 @bar(i32)
OpenPOWER on IntegriCloud