summaryrefslogtreecommitdiff
path: root/lib/Transforms/Utils/PromoteMemoryToRegister.cpp
diff options
context:
space:
mode:
authorChris Lattner <sabre@nondot.org>2007-08-04 01:41:18 +0000
committerChris Lattner <sabre@nondot.org>2007-08-04 01:41:18 +0000
commitbbe104002fe7398bbde5d5ec2c2e5e324cf72305 (patch)
tree317a661680b2d0b05f006e5d00ed7f9c0e2e8e65 /lib/Transforms/Utils/PromoteMemoryToRegister.cpp
parent483ce14bf4746d8078d65a14f61319528e1a00f1 (diff)
downloadllvm-bbe104002fe7398bbde5d5ec2c2e5e324cf72305.tar.gz
llvm-bbe104002fe7398bbde5d5ec2c2e5e324cf72305.tar.bz2
llvm-bbe104002fe7398bbde5d5ec2c2e5e324cf72305.tar.xz
refactor some code to shrink PromoteMem2Reg::run a bit
git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@40805 91177308-0d34-0410-b5e6-96231b3b80d8
Diffstat (limited to 'lib/Transforms/Utils/PromoteMemoryToRegister.cpp')
-rw-r--r--lib/Transforms/Utils/PromoteMemoryToRegister.cpp159
1 files changed, 96 insertions, 63 deletions
diff --git a/lib/Transforms/Utils/PromoteMemoryToRegister.cpp b/lib/Transforms/Utils/PromoteMemoryToRegister.cpp
index e2fd46e337..f895d1955c 100644
--- a/lib/Transforms/Utils/PromoteMemoryToRegister.cpp
+++ b/lib/Transforms/Utils/PromoteMemoryToRegister.cpp
@@ -16,6 +16,7 @@
//
//===----------------------------------------------------------------------===//
+#define DEBUG_TYPE "mem2reg"
#include "llvm/Transforms/Utils/PromoteMemToReg.h"
#include "llvm/Constants.h"
#include "llvm/DerivedTypes.h"
@@ -26,12 +27,17 @@
#include "llvm/ADT/DenseMap.h"
#include "llvm/ADT/SmallPtrSet.h"
#include "llvm/ADT/SmallVector.h"
+#include "llvm/ADT/Statistic.h"
#include "llvm/ADT/StringExtras.h"
#include "llvm/Support/CFG.h"
#include "llvm/Support/Compiler.h"
#include <algorithm>
using namespace llvm;
+STATISTIC(NumLocalPromoted, "Number of alloca's promoted within one block");
+STATISTIC(NumSingleStore, "Number of alloca's promoted with a single store");
+STATISTIC(NumDeadAlloca, "Number of dead alloca's removed");
+
// Provide DenseMapKeyInfo for all pointers.
namespace llvm {
template<>
@@ -153,6 +159,12 @@ namespace {
}
private:
+ void RemoveFromAllocasList(unsigned &AllocaIdx) {
+ Allocas[AllocaIdx] = Allocas.back();
+ Allocas.pop_back();
+ --AllocaIdx;
+ }
+
void MarkDominatingPHILive(BasicBlock *BB, unsigned AllocaNum,
SmallPtrSet<PHINode*, 16> &DeadPHINodes);
bool PromoteLocallyUsedAlloca(BasicBlock *BB, AllocaInst *AI);
@@ -165,6 +177,58 @@ namespace {
bool QueuePhiNode(BasicBlock *BB, unsigned AllocaIdx, unsigned &Version,
SmallPtrSet<PHINode*, 16> &InsertedPHINodes);
};
+
+ struct AllocaInfo {
+ std::vector<BasicBlock*> DefiningBlocks;
+ std::vector<BasicBlock*> UsingBlocks;
+
+ StoreInst *OnlyStore;
+ BasicBlock *OnlyBlock;
+ bool OnlyUsedInOneBlock;
+
+ Value *AllocaPointerVal;
+
+ void clear() {
+ DefiningBlocks.clear();
+ UsingBlocks.clear();
+ OnlyStore = 0;
+ OnlyBlock = 0;
+ OnlyUsedInOneBlock = true;
+ AllocaPointerVal = 0;
+ }
+
+ /// AnalyzeAlloca - Scan the uses of the specified alloca, filling in our
+ /// ivars.
+ void AnalyzeAlloca(AllocaInst *AI) {
+ clear();
+
+ // As we scan the uses of the alloca instruction, keep track of stores,
+ // and decide whether all of the loads and stores to the alloca are within
+ // the same basic block.
+ for (Value::use_iterator U = AI->use_begin(), E = AI->use_end();
+ U != E; ++U){
+ Instruction *User = cast<Instruction>(*U);
+ if (StoreInst *SI = dyn_cast<StoreInst>(User)) {
+ // Remember the basic blocks which define new values for the alloca
+ DefiningBlocks.push_back(SI->getParent());
+ AllocaPointerVal = SI->getOperand(0);
+ OnlyStore = SI;
+ } else {
+ LoadInst *LI = cast<LoadInst>(User);
+ // Otherwise it must be a load instruction, keep track of variable reads
+ UsingBlocks.push_back(LI->getParent());
+ AllocaPointerVal = LI;
+ }
+
+ if (OnlyUsedInOneBlock) {
+ if (OnlyBlock == 0)
+ OnlyBlock = User->getParent();
+ else if (OnlyBlock != User->getParent())
+ OnlyUsedInOneBlock = false;
+ }
+ }
+ }
+ };
} // end of anonymous namespace
@@ -180,6 +244,8 @@ void PromoteMem2Reg::run() {
if (AST) PointerAllocaValues.resize(Allocas.size());
+ AllocaInfo Info;
+
for (unsigned AllocaNum = 0; AllocaNum != Allocas.size(); ++AllocaNum) {
AllocaInst *AI = Allocas[AllocaNum];
@@ -194,80 +260,46 @@ void PromoteMem2Reg::run() {
AI->eraseFromParent();
// Remove the alloca from the Allocas list, since it has been processed
- Allocas[AllocaNum] = Allocas.back();
- Allocas.pop_back();
- --AllocaNum;
+ RemoveFromAllocasList(AllocaNum);
+ ++NumDeadAlloca;
continue;
}
-
+
// Calculate the set of read and write-locations for each alloca. This is
// analogous to finding the 'uses' and 'definitions' of each variable.
- std::vector<BasicBlock*> DefiningBlocks;
- std::vector<BasicBlock*> UsingBlocks;
-
- StoreInst *OnlyStore = 0;
- BasicBlock *OnlyBlock = 0;
- bool OnlyUsedInOneBlock = true;
-
- // As we scan the uses of the alloca instruction, keep track of stores, and
- // decide whether all of the loads and stores to the alloca are within the
- // same basic block.
- Value *AllocaPointerVal = 0;
- for (Value::use_iterator U =AI->use_begin(), E = AI->use_end(); U != E;++U){
- Instruction *User = cast<Instruction>(*U);
- if (StoreInst *SI = dyn_cast<StoreInst>(User)) {
- // Remember the basic blocks which define new values for the alloca
- DefiningBlocks.push_back(SI->getParent());
- AllocaPointerVal = SI->getOperand(0);
- OnlyStore = SI;
- } else {
- LoadInst *LI = cast<LoadInst>(User);
- // Otherwise it must be a load instruction, keep track of variable reads
- UsingBlocks.push_back(LI->getParent());
- AllocaPointerVal = LI;
- }
-
- if (OnlyUsedInOneBlock) {
- if (OnlyBlock == 0)
- OnlyBlock = User->getParent();
- else if (OnlyBlock != User->getParent())
- OnlyUsedInOneBlock = false;
- }
- }
+ Info.AnalyzeAlloca(AI);
// If the alloca is only read and written in one basic block, just perform a
// linear sweep over the block to eliminate it.
- if (OnlyUsedInOneBlock) {
- LocallyUsedAllocas[OnlyBlock].push_back(AI);
+ if (Info.OnlyUsedInOneBlock) {
+ LocallyUsedAllocas[Info.OnlyBlock].push_back(AI);
// Remove the alloca from the Allocas list, since it will be processed.
- Allocas[AllocaNum] = Allocas.back();
- Allocas.pop_back();
- --AllocaNum;
+ RemoveFromAllocasList(AllocaNum);
continue;
}
// If there is only a single store to this value, replace any loads of
// it that are directly dominated by the definition with the value stored.
- if (DefiningBlocks.size() == 1) {
+ if (Info.DefiningBlocks.size() == 1) {
// Be aware of loads before the store.
std::set<BasicBlock*> ProcessedBlocks;
- for (unsigned i = 0, e = UsingBlocks.size(); i != e; ++i)
+ for (unsigned i = 0, e = Info.UsingBlocks.size(); i != e; ++i)
// If the store dominates the block and if we haven't processed it yet,
// do so now.
- if (dominates(OnlyStore->getParent(), UsingBlocks[i]))
- if (ProcessedBlocks.insert(UsingBlocks[i]).second) {
- BasicBlock *UseBlock = UsingBlocks[i];
+ if (dominates(Info.OnlyStore->getParent(), Info.UsingBlocks[i]))
+ if (ProcessedBlocks.insert(Info.UsingBlocks[i]).second) {
+ BasicBlock *UseBlock = Info.UsingBlocks[i];
// If the use and store are in the same block, do a quick scan to
// verify that there are no uses before the store.
- if (UseBlock == OnlyStore->getParent()) {
+ if (UseBlock == Info.OnlyStore->getParent()) {
BasicBlock::iterator I = UseBlock->begin();
- for (; &*I != OnlyStore; ++I) { // scan block for store.
+ for (; &*I != Info.OnlyStore; ++I) { // scan block for store.
if (isa<LoadInst>(I) && I->getOperand(0) == AI)
break;
}
- if (&*I != OnlyStore) break; // Do not handle this case.
+ if (&*I != Info.OnlyStore) break; // Do not handle this case.
}
// Otherwise, if this is a different block or if all uses happen
@@ -277,7 +309,7 @@ void PromoteMem2Reg::run() {
I != E; ) {
if (LoadInst *LI = dyn_cast<LoadInst>(I++)) {
if (LI->getOperand(0) == AI) {
- LI->replaceAllUsesWith(OnlyStore->getOperand(0));
+ LI->replaceAllUsesWith(Info.OnlyStore->getOperand(0));
if (AST && isa<PointerType>(LI->getType()))
AST->deleteValue(LI);
LI->eraseFromParent();
@@ -286,23 +318,22 @@ void PromoteMem2Reg::run() {
}
// Finally, remove this block from the UsingBlock set.
- UsingBlocks[i] = UsingBlocks.back();
+ Info.UsingBlocks[i] = Info.UsingBlocks.back();
--i; --e;
}
// Finally, after the scan, check to see if the store is all that is left.
- if (UsingBlocks.empty()) {
+ if (Info.UsingBlocks.empty()) {
+ ++NumSingleStore;
// The alloca has been processed, move on.
- Allocas[AllocaNum] = Allocas.back();
- Allocas.pop_back();
- --AllocaNum;
+ RemoveFromAllocasList(AllocaNum);
continue;
}
}
if (AST)
- PointerAllocaValues[AllocaNum] = AllocaPointerVal;
+ PointerAllocaValues[AllocaNum] = Info.AllocaPointerVal;
// If we haven't computed a numbering for the BB's in the function, do so
// now.
@@ -318,9 +349,9 @@ void PromoteMem2Reg::run() {
unsigned CurrentVersion = 0;
SmallPtrSet<PHINode*, 16> InsertedPHINodes;
std::vector<std::pair<unsigned, BasicBlock*> > DFBlocks;
- while (!DefiningBlocks.empty()) {
- BasicBlock *BB = DefiningBlocks.back();
- DefiningBlocks.pop_back();
+ while (!Info.DefiningBlocks.empty()) {
+ BasicBlock *BB = Info.DefiningBlocks.back();
+ Info.DefiningBlocks.pop_back();
// Look up the DF for this write, add it to PhiNodes
DominanceFrontier::const_iterator it = DF.find(BB);
@@ -342,7 +373,7 @@ void PromoteMem2Reg::run() {
for (unsigned i = 0, e = DFBlocks.size(); i != e; ++i) {
BasicBlock *BB = DFBlocks[i].second;
if (QueuePhiNode(BB, AllocaNum, CurrentVersion, InsertedPHINodes))
- DefiningBlocks.push_back(BB);
+ Info.DefiningBlocks.push_back(BB);
}
DFBlocks.clear();
}
@@ -355,9 +386,9 @@ void PromoteMem2Reg::run() {
// marked alive because of loads which are dominated by stores, but there
// will be no unmarked PHI nodes which are actually used.
//
- for (unsigned i = 0, e = UsingBlocks.size(); i != e; ++i)
- MarkDominatingPHILive(UsingBlocks[i], AllocaNum, InsertedPHINodes);
- UsingBlocks.clear();
+ for (unsigned i = 0, e = Info.UsingBlocks.size(); i != e; ++i)
+ MarkDominatingPHILive(Info.UsingBlocks[i], AllocaNum, InsertedPHINodes);
+ Info.UsingBlocks.clear();
// If there are any PHI nodes which are now known to be dead, remove them!
for (SmallPtrSet<PHINode*, 16>::iterator I = InsertedPHINodes.begin(),
@@ -623,6 +654,8 @@ bool PromoteMem2Reg::PromoteLocallyUsedAlloca(BasicBlock *BB, AllocaInst *AI) {
assert(AI->use_empty() && "Uses of alloca from more than one BB??");
if (AST) AST->deleteValue(AI);
AI->getParent()->getInstList().erase(AI);
+
+ ++NumLocalPromoted;
return false;
}