mirror of
https://github.com/RPCS3/llvm.git
synced 2025-01-22 18:35:45 +00:00
Teach LazyValueInfo that allocas aren't NULL. Over all of llvm-test, this saves
half a million non-local queries, each of which would otherwise have triggered a linear scan over a basic block. Also fix a fixme for memory intrinsics which dereference pointers. With this, we prove that a pointer is non-null because it was dereferenced by an intrinsic 112 times in llvm-test. git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@123533 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
parent
a5eaa861e4
commit
786c7cd141
@ -139,6 +139,10 @@ namespace llvm {
|
|||||||
return !getVolatileCst()->isZero();
|
return !getVolatileCst()->isZero();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
unsigned getAddressSpace() const {
|
||||||
|
return cast<PointerType>(getRawDest()->getType())->getAddressSpace();
|
||||||
|
}
|
||||||
|
|
||||||
/// getDest - This is just like getRawDest, but it strips off any cast
|
/// getDest - This is just like getRawDest, but it strips off any cast
|
||||||
/// instructions that feed it, giving the original input. The returned
|
/// instructions that feed it, giving the original input. The returned
|
||||||
/// value is guaranteed to be a pointer.
|
/// value is guaranteed to be a pointer.
|
||||||
|
@ -17,6 +17,7 @@
|
|||||||
#include "llvm/Analysis/ValueTracking.h"
|
#include "llvm/Analysis/ValueTracking.h"
|
||||||
#include "llvm/Constants.h"
|
#include "llvm/Constants.h"
|
||||||
#include "llvm/Instructions.h"
|
#include "llvm/Instructions.h"
|
||||||
|
#include "llvm/IntrinsicInst.h"
|
||||||
#include "llvm/Analysis/ConstantFolding.h"
|
#include "llvm/Analysis/ConstantFolding.h"
|
||||||
#include "llvm/Target/TargetData.h"
|
#include "llvm/Target/TargetData.h"
|
||||||
#include "llvm/Support/CFG.h"
|
#include "llvm/Support/CFG.h"
|
||||||
@ -544,6 +545,11 @@ bool LazyValueInfoCache::solveBlockValue(Value *Val, BasicBlock *BB) {
|
|||||||
return ODCacheUpdater.markResult(solveBlockValuePHINode(BBLV, PN, BB));
|
return ODCacheUpdater.markResult(solveBlockValuePHINode(BBLV, PN, BB));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (AllocaInst *AI = dyn_cast<AllocaInst>(BBI)) {
|
||||||
|
BBLV = LVILatticeVal::getNot(ConstantPointerNull::get(AI->getType()));
|
||||||
|
return ODCacheUpdater.markResult(true);
|
||||||
|
}
|
||||||
|
|
||||||
// We can only analyze the definitions of certain classes of instructions
|
// We can only analyze the definitions of certain classes of instructions
|
||||||
// (integral binops and casts at the moment), so bail if this isn't one.
|
// (integral binops and casts at the moment), so bail if this isn't one.
|
||||||
LVILatticeVal Result;
|
LVILatticeVal Result;
|
||||||
@ -580,7 +586,19 @@ static bool InstructionDereferencesPointer(Instruction *I, Value *Ptr) {
|
|||||||
GetUnderlyingObject(S->getPointerOperand()) ==
|
GetUnderlyingObject(S->getPointerOperand()) ==
|
||||||
GetUnderlyingObject(Ptr);
|
GetUnderlyingObject(Ptr);
|
||||||
}
|
}
|
||||||
// FIXME: llvm.memset, etc.
|
if (MemIntrinsic *MI = dyn_cast<MemIntrinsic>(I)) {
|
||||||
|
if (MI->isVolatile()) return false;
|
||||||
|
if (MI->getAddressSpace() != 0) return false;
|
||||||
|
|
||||||
|
// FIXME: check whether it has a valuerange that excludes zero?
|
||||||
|
ConstantInt *Len = dyn_cast<ConstantInt>(MI->getLength());
|
||||||
|
if (!Len || Len->isZero()) return false;
|
||||||
|
|
||||||
|
if (MI->getRawDest() == Ptr || MI->getDest() == Ptr)
|
||||||
|
return true;
|
||||||
|
if (MemTransferInst *MTI = dyn_cast<MemTransferInst>(MI))
|
||||||
|
return MTI->getRawSource() == Ptr || MTI->getSource() == Ptr;
|
||||||
|
}
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -592,10 +610,14 @@ bool LazyValueInfoCache::solveBlockValueNonLocal(LVILatticeVal &BBLV,
|
|||||||
// then we know that the pointer can't be NULL.
|
// then we know that the pointer can't be NULL.
|
||||||
bool NotNull = false;
|
bool NotNull = false;
|
||||||
if (Val->getType()->isPointerTy()) {
|
if (Val->getType()->isPointerTy()) {
|
||||||
for (BasicBlock::iterator BI = BB->begin(), BE = BB->end();BI != BE;++BI){
|
if (isa<AllocaInst>(Val)) {
|
||||||
if (InstructionDereferencesPointer(BI, Val)) {
|
NotNull = true;
|
||||||
NotNull = true;
|
} else {
|
||||||
break;
|
for (BasicBlock::iterator BI = BB->begin(), BE = BB->end();BI != BE;++BI){
|
||||||
|
if (InstructionDereferencesPointer(BI, Val)) {
|
||||||
|
NotNull = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user