The absence of noreturn doesn't ensure mayReturn

There are two separate issues:
- LLVM doesn't consider infinite loops to be side effects: we happily
  hoist/sink above/below loops whose bounds are unknown.
- The absence of the noreturn attribute is insufficient for us to know
  if a function will definitely return.  Relying on noreturn in the
  middle-end for any property is an accident waiting to happen.

llvm-svn: 273762
This commit is contained in:
David Majnemer 2016-06-25 00:55:12 +00:00
parent f4c56e97df
commit 0f45572761
3 changed files with 1 additions and 33 deletions

View File

@ -394,21 +394,13 @@ public:
/// Return true if this instruction may throw an exception.
bool mayThrow() const;
/// Return true if this is a function that may return.
/// This is true for all normal instructions. The only exception
/// is functions that are marked with the 'noreturn' attribute.
///
bool mayReturn() const;
/// Return true if the instruction may have side effects.
///
/// Note that this does not consider malloc and alloca to have side
/// effects because the newly allocated memory is completely invisible to
/// instructions which don't use the returned value. For cases where this
/// matters, isSafeToSpeculativelyExecute may be more appropriate.
bool mayHaveSideEffects() const {
return mayWriteToMemory() || mayThrow() || !mayReturn();
}
bool mayHaveSideEffects() const { return mayWriteToMemory() || mayThrow(); }
/// Return true if the instruction is a variety of EH-block.
bool isEHPad() const {

View File

@ -545,12 +545,6 @@ bool Instruction::mayThrow() const {
return isa<ResumeInst>(this);
}
bool Instruction::mayReturn() const {
if (const CallInst *CI = dyn_cast<CallInst>(this))
return !CI->doesNotReturn();
return true;
}
/// isAssociative - Return true if the instruction is associative:
///
/// Associative operators satisfy: x op (y op z) === (x op y) op z

View File

@ -1,18 +0,0 @@
; RUN: opt < %s -functionattrs -instcombine -S | FileCheck %s
define void @endless_loop() noreturn nounwind readnone ssp uwtable {
entry:
br label %while.body
while.body:
br label %while.body
}
;CHECK-LABEL: @main(
;CHECK: endless_loop
;CHECK: ret
define i32 @main() noreturn nounwind ssp uwtable {
entry:
tail call void @endless_loop()
unreachable
}