|  | //===- llvm/Transforms/HoistPHIConstants.h - Normalize PHI nodes ------------=// | 
|  | // | 
|  | // HoistPHIConstants - Remove literal constants that are arguments of PHI nodes | 
|  | // by inserting cast instructions in the preceeding basic blocks, and changing | 
|  | // constant references into references of the casted value. | 
|  | // | 
|  | //===----------------------------------------------------------------------===// | 
|  |  | 
|  |  | 
|  |  | 
|  |  | 
|  | #include "llvm/Transforms/HoistPHIConstants.h" | 
|  | #include "llvm/iOther.h" | 
|  | #include "llvm/BasicBlock.h" | 
|  | #include "llvm/Method.h" | 
|  | #include <map> | 
|  | #include <vector> | 
|  |  | 
|  | typedef pair<BasicBlock *, Value*> BBConstTy; | 
|  | typedef map<BBConstTy, CastInst *> CachedCopyMap; | 
|  |  | 
|  | static Value *NormalizePhiOperand(PHINode *PN, Value *CPV, | 
|  | BasicBlock *Pred, CachedCopyMap &CopyCache) { | 
|  |  | 
|  | /* NOTE: CahedCopyMap was disabled to insert phi elimination code | 
|  | for all phi args -- Ruchira | 
|  | */ | 
|  |  | 
|  |  | 
|  | // Check if we've already inserted a copy for this constant in Pred | 
|  | // Note that `copyCache[Pred]' will create an empty vector the first time | 
|  | // | 
|  | //CachedCopyMap::iterator CCI = CopyCache.find(BBConstTy(Pred, CPV)); | 
|  | //if (CCI != CopyCache.end()) return CCI->second; | 
|  |  | 
|  | // Create a copy instruction and add it to the cache... | 
|  | CastInst *Inst = new CastInst(CPV, CPV->getType()); | 
|  | //CopyCache.insert(make_pair(BBConstTy(Pred, CPV), Inst)); | 
|  |  | 
|  | // Insert the copy just before the terminator inst of the predecessor BB | 
|  | assert(Pred->getTerminator() && "Degenerate BB encountered!"); | 
|  | Pred->getInstList().insert(Pred->getInstList().end()-1, Inst); | 
|  |  | 
|  | return Inst; | 
|  | } | 
|  |  | 
|  |  | 
|  | //--------------------------------------------------------------------------- | 
|  | // Entry point for normalizing constant args in PHIs | 
|  | //--------------------------------------------------------------------------- | 
|  |  | 
|  | bool HoistPHIConstants::doHoistPHIConstants(Method *M) { | 
|  | CachedCopyMap Cache; | 
|  | bool Changed = false; | 
|  |  | 
|  | for (Method::iterator BI = M->begin(), BE = M->end(); BI != BE; ++BI) | 
|  | { | 
|  | vector<PHINode*> phis;            // normalizing invalidates BB iterator | 
|  |  | 
|  | for (BasicBlock::iterator II = (*BI)->begin(); II != (*BI)->end(); ++II) | 
|  | { | 
|  | if (PHINode *PN = dyn_cast<PHINode>(*II)) | 
|  | phis.push_back(PN); | 
|  | else | 
|  | break;                      // All PHIs occur at top of BB! | 
|  | } | 
|  |  | 
|  | for (vector<PHINode*>::iterator PI=phis.begin(); PI != phis.end(); ++PI) | 
|  | for (unsigned i = 0; i < (*PI)->getNumIncomingValues(); ++i) | 
|  | { | 
|  | //if (isa<ConstPoolVal>(Op)) {--- Do for all phi args -- Ruchira | 
|  | (*PI)->setIncomingValue(i, | 
|  | NormalizePhiOperand((*PI), | 
|  | (*PI)->getIncomingValue(i), | 
|  | (*PI)->getIncomingBlock(i), Cache)); | 
|  | Changed = true; | 
|  | //} | 
|  | } | 
|  | } | 
|  |  | 
|  | return Changed; | 
|  | } |