forked from OSchip/llvm-project
[ArgumentPromotion] don't break musttail invariant PR36543
Summary: Do not break musttail invariant by promoting arguments of musttail callee or caller. Reviewers: sanjoy, dberlin, hfinkel, george.burgess.iv, fhahn, rnk Reviewed By: rnk Subscribers: rnk, llvm-commits Differential Revision: https://reviews.llvm.org/D43926 llvm-svn: 326521
This commit is contained in:
parent
0ffcaf7437
commit
1571b1271e
|
@ -853,10 +853,20 @@ promoteArguments(Function *F, function_ref<AAResults &(Function &F)> AARGetter,
|
|||
if (CS.getInstruction() == nullptr || !CS.isCallee(&U))
|
||||
return nullptr;
|
||||
|
||||
// Can't change signature of musttail callee
|
||||
if (CS.isMustTailCall())
|
||||
return nullptr;
|
||||
|
||||
if (CS.getInstruction()->getParent()->getParent() == F)
|
||||
isSelfRecursive = true;
|
||||
}
|
||||
|
||||
// Can't change signature of musttail caller
|
||||
// FIXME: Support promoting whole chain of musttail functions
|
||||
for (BasicBlock &BB : *F)
|
||||
if (BB.getTerminatingMustTailCall())
|
||||
return nullptr;
|
||||
|
||||
const DataLayout &DL = F->getParent()->getDataLayout();
|
||||
|
||||
AAResults &AAR = AARGetter(*F);
|
||||
|
|
|
@ -0,0 +1,45 @@
|
|||
; RUN: opt < %s -argpromotion -S | FileCheck %s
|
||||
; PR36543
|
||||
|
||||
; Don't promote arguments of musttail callee
|
||||
|
||||
%T = type { i32, i32, i32, i32 }
|
||||
|
||||
; CHECK-LABEL: define internal i32 @test(%T* %p)
|
||||
define internal i32 @test(%T* %p) {
|
||||
%a.gep = getelementptr %T, %T* %p, i64 0, i32 3
|
||||
%b.gep = getelementptr %T, %T* %p, i64 0, i32 2
|
||||
%a = load i32, i32* %a.gep
|
||||
%b = load i32, i32* %b.gep
|
||||
%v = add i32 %a, %b
|
||||
ret i32 %v
|
||||
}
|
||||
|
||||
; CHECK-LABEL: define i32 @caller(%T* %p)
|
||||
define i32 @caller(%T* %p) {
|
||||
%v = musttail call i32 @test(%T* %p)
|
||||
ret i32 %v
|
||||
}
|
||||
|
||||
; Don't promote arguments of musttail caller
|
||||
|
||||
define i32 @foo(%T* %p, i32 %v) {
|
||||
ret i32 0
|
||||
}
|
||||
|
||||
; CHECK-LABEL: define internal i32 @test2(%T* %p, i32 %p2)
|
||||
define internal i32 @test2(%T* %p, i32 %p2) {
|
||||
%a.gep = getelementptr %T, %T* %p, i64 0, i32 3
|
||||
%b.gep = getelementptr %T, %T* %p, i64 0, i32 2
|
||||
%a = load i32, i32* %a.gep
|
||||
%b = load i32, i32* %b.gep
|
||||
%v = add i32 %a, %b
|
||||
%ca = musttail call i32 @foo(%T* undef, i32 %v)
|
||||
ret i32 %ca
|
||||
}
|
||||
|
||||
; CHECK-LABEL: define i32 @caller2(%T* %g)
|
||||
define i32 @caller2(%T* %g) {
|
||||
%v = call i32 @test2(%T* %g, i32 0)
|
||||
ret i32 %v
|
||||
}
|
Loading…
Reference in New Issue