[llvm] r351475 - [InstCombine] Don't sink dynamic allocas
Reid Kleckner via llvm-commits
llvm-commits at lists.llvm.org
Thu Jan 17 12:46:53 PST 2019
Author: rnk
Date: Thu Jan 17 12:46:53 2019
New Revision: 351475
URL: http://llvm.org/viewvc/llvm-project?rev=351475&view=rev
Log:
[InstCombine] Don't sink dynamic allocas
Summary:
InstCombine's sinking algorithm only thinks about memory. It doesn't
think about non-memory constraints like stack object lifetime. It can
sink dynamic allocas across a stacksave call, which may be used with
stackrestore, which can incorrectly reduce the lifetime of the dynamic
alloca.
Fixes PR40365
Reviewers: hfinkel, efriedma
Subscribers: hiraditya, llvm-commits
Differential Revision: https://reviews.llvm.org/D56872
Added:
llvm/trunk/test/Transforms/InstCombine/sink-alloca.ll
Modified:
llvm/trunk/lib/Transforms/InstCombine/InstructionCombining.cpp
Modified: llvm/trunk/lib/Transforms/InstCombine/InstructionCombining.cpp
URL: http://llvm.org/viewvc/llvm-project/llvm/trunk/lib/Transforms/InstCombine/InstructionCombining.cpp?rev=351475&r1=351474&r2=351475&view=diff
==============================================================================
--- llvm/trunk/lib/Transforms/InstCombine/InstructionCombining.cpp (original)
+++ llvm/trunk/lib/Transforms/InstCombine/InstructionCombining.cpp Thu Jan 17 12:46:53 2019
@@ -3065,9 +3065,11 @@ static bool TryToSinkInstruction(Instruc
I->isTerminator())
return false;
- // Do not sink alloca instructions out of the entry block.
- if (isa<AllocaInst>(I) && I->getParent() ==
- &DestBlock->getParent()->getEntryBlock())
+ // Do not sink static or dynamic alloca instructions. Static allocas must
+ // remain in the entry block, and dynamic allocas must not be sunk in between
+ // a stacksave / stackrestore pair, which would incorrectly shorten its
+ // lifetime.
+ if (isa<AllocaInst>(I))
return false;
// Do not sink into catchswitch blocks.
Added: llvm/trunk/test/Transforms/InstCombine/sink-alloca.ll
URL: http://llvm.org/viewvc/llvm-project/llvm/trunk/test/Transforms/InstCombine/sink-alloca.ll?rev=351475&view=auto
==============================================================================
--- llvm/trunk/test/Transforms/InstCombine/sink-alloca.ll (added)
+++ llvm/trunk/test/Transforms/InstCombine/sink-alloca.ll Thu Jan 17 12:46:53 2019
@@ -0,0 +1,52 @@
+; RUN: opt -instcombine -S < %s | FileCheck %s
+
+target datalayout = "e-m:e-p:32:32-f64:32:64-f80:32-n8:16:32-S128"
+target triple = "i686-unknown-linux-gnu"
+
+; Check that instcombine doesn't sink dynamic allocas across llvm.stacksave.
+
+; Helper to generate branch conditions.
+declare i1 @cond()
+
+declare i32* @use_and_return(i32*)
+
+declare i8* @llvm.stacksave() #0
+
+declare void @llvm.stackrestore(i8*) #0
+
+define void @foo(i32 %x) {
+entry:
+ %c1 = call i1 @cond()
+ br i1 %c1, label %ret, label %nonentry
+
+nonentry: ; preds = %entry
+ %argmem = alloca i32, i32 %x, align 4
+ %sp = call i8* @llvm.stacksave()
+ %c2 = call i1 @cond()
+ br i1 %c2, label %ret, label %sinktarget
+
+sinktarget: ; preds = %nonentry
+ ; Arrange for there to be a single use of %argmem by returning it.
+ %p = call i32* @use_and_return(i32* nonnull %argmem)
+ store i32 13, i32* %p, align 4
+ call void @llvm.stackrestore(i8* %sp)
+ %0 = call i32* @use_and_return(i32* %p)
+ br label %ret
+
+ret: ; preds = %sinktarget, %nonentry, %entry
+ ret void
+}
+
+; CHECK-LABEL: define void @foo(i32 %x)
+; CHECK: nonentry:
+; CHECK: %argmem = alloca i32, i32 %x
+; CHECK: %sp = call i8* @llvm.stacksave()
+; CHECK: %c2 = call i1 @cond()
+; CHECK: br i1 %c2, label %ret, label %sinktarget
+; CHECK: sinktarget:
+; CHECK: %p = call i32* @use_and_return(i32* nonnull %argmem)
+; CHECK: store i32 13, i32* %p
+; CHECK: call void @llvm.stackrestore(i8* %sp)
+; CHECK: %0 = call i32* @use_and_return(i32* %p)
+
+attributes #0 = { nounwind }
More information about the llvm-commits
mailing list