diff options
author | Duncan Sands <baldrick@free.fr> | 2008-10-04 13:24:24 +0000 |
---|---|---|
committer | Duncan Sands <baldrick@free.fr> | 2008-10-04 13:24:24 +0000 |
commit | c82b6a1ed2496ce0d61d2e51ed3a4cac4db786da (patch) | |
tree | 64fec2d7d3157af9f6595b2bc6e554a543e201b9 /lib/Transforms | |
parent | 26e4b216c0c065dece616c1c9931252e2a0e40c6 (diff) | |
download | external_llvm-c82b6a1ed2496ce0d61d2e51ed3a4cac4db786da.zip external_llvm-c82b6a1ed2496ce0d61d2e51ed3a4cac4db786da.tar.gz external_llvm-c82b6a1ed2496ce0d61d2e51ed3a4cac4db786da.tar.bz2 |
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.cpp | 35 |
1 files changed, 26 insertions, 9 deletions
diff --git a/lib/Transforms/IPO/AddReadAttrs.cpp b/lib/Transforms/IPO/AddReadAttrs.cpp index 7af071c..5460a5b 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(); } } |