blob: f20de1cfc2105ded48cf6874a880663cc8eef879 [file] [log] [blame]
Chris Lattner1bffea02001-10-15 17:31:51 +00001//===- llvm/Transforms/HoistPHIConstants.h - Normalize PHI nodes ------------=//
2//
3// HoistPHIConstants - Remove literal constants that are arguments of PHI nodes
4// by inserting cast instructions in the preceeding basic blocks, and changing
5// constant references into references of the casted value.
6//
7//===----------------------------------------------------------------------===//
8
9#include "llvm/Transforms/HoistPHIConstants.h"
Chris Lattner7061dc52001-12-03 18:02:31 +000010#include "llvm/iPHINode.h"
Chris Lattner1bffea02001-10-15 17:31:51 +000011#include "llvm/iOther.h"
12#include "llvm/BasicBlock.h"
13#include "llvm/Method.h"
Chris Lattnerbd0ef772002-02-26 21:46:54 +000014#include "llvm/Pass.h"
Chris Lattner1bffea02001-10-15 17:31:51 +000015#include <map>
Vikram S. Adveddf5ff62001-11-08 05:25:33 +000016#include <vector>
Chris Lattner1bffea02001-10-15 17:31:51 +000017
Chris Lattner697954c2002-01-20 22:54:45 +000018typedef std::pair<BasicBlock *, Value*> BBConstTy;
19typedef std::map<BBConstTy, CastInst *> CachedCopyMap;
Chris Lattner1bffea02001-10-15 17:31:51 +000020
21static Value *NormalizePhiOperand(PHINode *PN, Value *CPV,
22 BasicBlock *Pred, CachedCopyMap &CopyCache) {
Chris Lattner1bffea02001-10-15 17:31:51 +000023 // Check if we've already inserted a copy for this constant in Pred
24 // Note that `copyCache[Pred]' will create an empty vector the first time
25 //
Ruchira Sasanka061b8ed2001-11-12 19:32:04 +000026 CachedCopyMap::iterator CCI = CopyCache.find(BBConstTy(Pred, CPV));
27 if (CCI != CopyCache.end()) return CCI->second;
Chris Lattner1bffea02001-10-15 17:31:51 +000028
29 // Create a copy instruction and add it to the cache...
30 CastInst *Inst = new CastInst(CPV, CPV->getType());
Chris Lattner697954c2002-01-20 22:54:45 +000031 CopyCache.insert(std::make_pair(BBConstTy(Pred, CPV), Inst));
Chris Lattner1bffea02001-10-15 17:31:51 +000032
33 // Insert the copy just before the terminator inst of the predecessor BB
34 assert(Pred->getTerminator() && "Degenerate BB encountered!");
35 Pred->getInstList().insert(Pred->getInstList().end()-1, Inst);
36
37 return Inst;
38}
39
40
41//---------------------------------------------------------------------------
42// Entry point for normalizing constant args in PHIs
43//---------------------------------------------------------------------------
44
Chris Lattnerbd0ef772002-02-26 21:46:54 +000045static bool doHoistPHIConstants(Method *M) {
Chris Lattner1bffea02001-10-15 17:31:51 +000046 CachedCopyMap Cache;
Chris Lattner42c9c2c2001-10-18 05:27:33 +000047 bool Changed = false;
Chris Lattner1bffea02001-10-15 17:31:51 +000048
Chris Lattnere9bb2df2001-12-03 22:26:30 +000049 for (Method::iterator BI = M->begin(), BE = M->end(); BI != BE; ++BI) {
Chris Lattner697954c2002-01-20 22:54:45 +000050 std::vector<PHINode*> phis; // normalizing invalidates BB iterator
Chris Lattner1bffea02001-10-15 17:31:51 +000051
Chris Lattnere9bb2df2001-12-03 22:26:30 +000052 for (BasicBlock::iterator II = (*BI)->begin(); II != (*BI)->end(); ++II) {
53 if (PHINode *PN = dyn_cast<PHINode>(*II))
54 phis.push_back(PN);
55 else
56 break; // All PHIs occur at top of BB!
57 }
Vikram S. Adveddf5ff62001-11-08 05:25:33 +000058
Chris Lattner697954c2002-01-20 22:54:45 +000059 for (std::vector<PHINode*>::iterator PI=phis.begin(); PI != phis.end();++PI)
Chris Lattnere9bb2df2001-12-03 22:26:30 +000060 for (unsigned i = 0; i < (*PI)->getNumIncomingValues(); ++i) {
61 Value *Op = (*PI)->getIncomingValue(i);
62
63 if (isa<Constant>(Op)) {
64 (*PI)->setIncomingValue(i,
Vikram S. Adveddf5ff62001-11-08 05:25:33 +000065 NormalizePhiOperand((*PI),
66 (*PI)->getIncomingValue(i),
67 (*PI)->getIncomingBlock(i), Cache));
Chris Lattnere9bb2df2001-12-03 22:26:30 +000068 Changed = true;
69 }
70 }
71 }
Vikram S. Adveddf5ff62001-11-08 05:25:33 +000072
Chris Lattner42c9c2c2001-10-18 05:27:33 +000073 return Changed;
Chris Lattner1bffea02001-10-15 17:31:51 +000074}
Chris Lattnerbd0ef772002-02-26 21:46:54 +000075
76namespace {
77 struct HoistPHIConstants : public MethodPass {
78 virtual bool runOnMethod(Method *M) { return doHoistPHIConstants(M); }
79 };
80}
81
82Pass *createHoistPHIConstantsPass() { return new HoistPHIConstants(); }