summaryrefslogtreecommitdiff
path: root/lib/Transforms
diff options
context:
space:
mode:
authorDuncan Sands <baldrick@free.fr>2008-10-04 13:24:24 +0000
committerDuncan Sands <baldrick@free.fr>2008-10-04 13:24:24 +0000
commitc82b6a1ed2496ce0d61d2e51ed3a4cac4db786da (patch)
tree64fec2d7d3157af9f6595b2bc6e554a543e201b9 /lib/Transforms
parent26e4b216c0c065dece616c1c9931252e2a0e40c6 (diff)
downloadllvm-c82b6a1ed2496ce0d61d2e51ed3a4cac4db786da.tar.gz
llvm-c82b6a1ed2496ce0d61d2e51ed3a4cac4db786da.tar.bz2
llvm-c82b6a1ed2496ce0d61d2e51ed3a4cac4db786da.tar.xz
Ignore loads from and stores to local memory (i.e. allocas)
when deciding whether to mark a function readnone/readonly. Since the pass is currently run before SROA, this may be quite helpful. Requested by Chris on IRC. git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@57050 91177308-0d34-0410-b5e6-96231b3b80d8
Diffstat (limited to 'lib/Transforms')
-rw-r--r--lib/Transforms/IPO/AddReadAttrs.cpp35
1 files changed, 26 insertions, 9 deletions
diff --git a/lib/Transforms/IPO/AddReadAttrs.cpp b/lib/Transforms/IPO/AddReadAttrs.cpp
index 7af071c40b..5460a5bde7 100644
--- a/lib/Transforms/IPO/AddReadAttrs.cpp
+++ b/lib/Transforms/IPO/AddReadAttrs.cpp
@@ -86,17 +86,34 @@ bool AddReadAttrs::runOnSCC(const std::vector<CallGraphNode *> &SCC) {
// Scan the function body for instructions that may read or write memory.
for (inst_iterator II = inst_begin(F), E = inst_end(F); II != E; ++II) {
- CallSite CS = CallSite::get(&*II);
-
- // Ignore calls to functions in the same SCC.
- if (CS.getInstruction() && SCCNodes.count(CG[CS.getCalledFunction()]))
- continue;
-
- if (II->mayWriteToMemory())
+ Instruction *I = &*II;
+
+ // Some instructions can be ignored even if they read or write memory.
+ // Detect these now, skipping to the next instruction if one is found.
+ CallSite CS = CallSite::get(I);
+ if (CS.getInstruction()) {
+ // Ignore calls to functions in the same SCC.
+ if (SCCNodes.count(CG[CS.getCalledFunction()]))
+ continue;
+ } else if (LoadInst *LI = dyn_cast<LoadInst>(I)) {
+ Value *Target = LI->getPointerOperand()->getUnderlyingObject();
+ // Ignore loads from local memory.
+ if (isa<AllocaInst>(Target))
+ continue;
+ } else if (StoreInst *SI = dyn_cast<StoreInst>(I)) {
+ Value *Target = SI->getPointerOperand()->getUnderlyingObject();
+ // Ignore stores to local memory.
+ if (isa<AllocaInst>(Target))
+ continue;
+ }
+
+ // Any remaining instructions need to be taken seriously! Check if they
+ // read or write memory.
+ if (I->mayWriteToMemory())
// Writes memory. Just give up.
return false;
-
- ReadsMemory |= II->mayReadFromMemory();
+ // If this instruction may read memory, remember that.
+ ReadsMemory |= I->mayReadFromMemory();
}
}