[PATCH] D53519: [CodeGen] skip lifetime end marker in isInTailCallPosition
Robert Lougher via Phabricator via llvm-commits
llvm-commits at lists.llvm.org
Mon Oct 22 11:57:10 PDT 2018
rob.lougher created this revision.
rob.lougher added reviewers: majnemer, t.p.northover, hfinkel.
Consider the following program:
------------- test.c ---------------
extern void bar(void);
__attribute__((__noinline__))
void foo(int *p) {
*p = 10;
}
void foobar() {
int i;
foo(&i);
bar();
}
------------------------------------
TailCallElim will correctly identify the call to bar as a tail call (as foo is nocapture). However, it will not be tail call optimized as it fails isInTailCallPosition() due to the lifetime end marker for the variable i:
define void @foobar() {
entry:
%i = alloca i32, align 4
%0 = bitcast i32* %i to i8*
call void @llvm.lifetime.start.p0i8(i64 4, i8* nonnull %0)
call void @foo(i32* nonnull %i)
tail call void @bar()
call void @llvm.lifetime.end.p0i8(i64 4, i8* nonnull %0)
ret void
}
On x86_64 this generates:
foobar:
pushq %rax
leaq 4(%rsp), %rdi
callq foo
callq bar
popq %rax
retq
After this patch we get:
foobar:
pushq %rax
leaq 4(%rsp), %rdi
callq foo
popq %rax
jmp bar # TAILCALL
Repository:
rL LLVM
https://reviews.llvm.org/D53519
Files:
lib/CodeGen/Analysis.cpp
test/CodeGen/X86/tailcall-lifetime-end.ll
Index: test/CodeGen/X86/tailcall-lifetime-end.ll
===================================================================
--- test/CodeGen/X86/tailcall-lifetime-end.ll
+++ test/CodeGen/X86/tailcall-lifetime-end.ll
@@ -0,0 +1,27 @@
+; RUN: llc -mtriple=x86_64-unknown-linux-gnu -o - %s | FileCheck %s
+
+; A lifetime end intrinsic should not prevent a call from being tail call
+; optimized.
+
+define void @foobar() {
+; CHECK-LABEL: foobar
+; CHECK: pushq %rax
+; CHECK: leaq 4(%rsp), %rdi
+; CHECK: callq foo
+; CHECK: popq %rax
+; CHECK: jmp bar
+entry:
+ %i = alloca i32
+ %0 = bitcast i32* %i to i8*
+ call void @llvm.lifetime.start.p0i8(i64 4, i8* nonnull %0)
+ call void @foo(i32* nonnull %i)
+ tail call void @bar()
+ call void @llvm.lifetime.end.p0i8(i64 4, i8* nonnull %0)
+ ret void
+}
+
+declare void @foo(i32* nocapture %p)
+declare void @bar()
+
+declare void @llvm.lifetime.start.p0i8(i64, i8* nocapture)
+declare void @llvm.lifetime.end.p0i8(i64, i8* nocapture)
Index: lib/CodeGen/Analysis.cpp
===================================================================
--- lib/CodeGen/Analysis.cpp
+++ lib/CodeGen/Analysis.cpp
@@ -496,6 +496,10 @@
// Debug info intrinsics do not get in the way of tail call optimization.
if (isa<DbgInfoIntrinsic>(BBI))
continue;
+ // A lifetime end intrinsic should not stop tail call optimization.
+ if (const IntrinsicInst *II = dyn_cast<IntrinsicInst>(BBI))
+ if (II->getIntrinsicID() == Intrinsic::lifetime_end)
+ continue;
if (BBI->mayHaveSideEffects() || BBI->mayReadFromMemory() ||
!isSafeToSpeculativelyExecute(&*BBI))
return false;
-------------- next part --------------
A non-text attachment was scrubbed...
Name: D53519.170458.patch
Type: text/x-patch
Size: 1663 bytes
Desc: not available
URL: <http://lists.llvm.org/pipermail/llvm-commits/attachments/20181022/02fe494a/attachment.bin>
More information about the llvm-commits
mailing list