mirror of
https://github.com/c64scene-ar/llvm-6502.git
synced 2024-12-16 11:30:51 +00:00
Fix PR7328: when turning a tail recursion into a loop, need to preserve
the returned value after the tail call if it differs from other return values. The optimal thing to do would be to introduce a phi node for the return value, but for the moment just fix the miscompile. git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@106947 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
parent
7a1d92a2cb
commit
d50e9e2566
@ -270,16 +270,16 @@ static bool isDynamicConstant(Value *V, CallInst *CI, ReturnInst *RI) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// getCommonReturnValue - Check to see if the function containing the specified
|
// getCommonReturnValue - Check to see if the function containing the specified
|
||||||
// return instruction and tail call consistently returns the same
|
// tail call consistently returns the same runtime-constant value at all exit
|
||||||
// runtime-constant value at all exit points. If so, return the returned value.
|
// points except for IgnoreRI. If so, return the returned value.
|
||||||
//
|
//
|
||||||
static Value *getCommonReturnValue(ReturnInst *TheRI, CallInst *CI) {
|
static Value *getCommonReturnValue(ReturnInst *IgnoreRI, CallInst *CI) {
|
||||||
Function *F = TheRI->getParent()->getParent();
|
Function *F = CI->getParent()->getParent();
|
||||||
Value *ReturnedValue = 0;
|
Value *ReturnedValue = 0;
|
||||||
|
|
||||||
for (Function::iterator BBI = F->begin(), E = F->end(); BBI != E; ++BBI)
|
for (Function::iterator BBI = F->begin(), E = F->end(); BBI != E; ++BBI)
|
||||||
if (ReturnInst *RI = dyn_cast<ReturnInst>(BBI->getTerminator()))
|
if (ReturnInst *RI = dyn_cast<ReturnInst>(BBI->getTerminator()))
|
||||||
if (RI != TheRI) {
|
if (RI != IgnoreRI) {
|
||||||
Value *RetOp = RI->getOperand(0);
|
Value *RetOp = RI->getOperand(0);
|
||||||
|
|
||||||
// We can only perform this transformation if the value returned is
|
// We can only perform this transformation if the value returned is
|
||||||
@ -404,7 +404,7 @@ bool TailCallElim::ProcessReturningBlock(ReturnInst *Ret, BasicBlock *&OldEntry,
|
|||||||
if (Ret->getNumOperands() == 1 && Ret->getReturnValue() != CI &&
|
if (Ret->getNumOperands() == 1 && Ret->getReturnValue() != CI &&
|
||||||
!isa<UndefValue>(Ret->getReturnValue()) &&
|
!isa<UndefValue>(Ret->getReturnValue()) &&
|
||||||
AccumulatorRecursionEliminationInitVal == 0 &&
|
AccumulatorRecursionEliminationInitVal == 0 &&
|
||||||
!getCommonReturnValue(Ret, CI))
|
!getCommonReturnValue(0, CI))
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
// OK! We can transform this tail call. If this is the first one found,
|
// OK! We can transform this tail call. If this is the first one found,
|
||||||
|
@ -0,0 +1,17 @@
|
|||||||
|
; RUN: opt < %s -tailcallelim -S | FileCheck %s
|
||||||
|
; PR7328
|
||||||
|
define i32 @foo(i32 %x) {
|
||||||
|
; CHECK: define i32 @foo
|
||||||
|
entry:
|
||||||
|
%cond = icmp ugt i32 %x, 0 ; <i1> [#uses=1]
|
||||||
|
br i1 %cond, label %return, label %body
|
||||||
|
|
||||||
|
body: ; preds = %entry
|
||||||
|
%y = add i32 %x, 1 ; <i32> [#uses=1]
|
||||||
|
%tmp = call i32 @foo(i32 %y) ; <i32> [#uses=0]
|
||||||
|
ret i32 0
|
||||||
|
; CHECK: ret i32 0
|
||||||
|
|
||||||
|
return: ; preds = %entry
|
||||||
|
ret i32 1
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user