Home | History | Annotate | Download | only in Analysis
      1 //===- DomPrinter.cpp - DOT printer for the dominance trees    ------------===//
      2 //
      3 //                     The LLVM Compiler Infrastructure
      4 //
      5 // This file is distributed under the University of Illinois Open Source
      6 // License. See LICENSE.TXT for details.
      7 //
      8 //===----------------------------------------------------------------------===//
      9 //
     10 // This file defines '-dot-dom' and '-dot-postdom' analysis passes, which emit
     11 // a dom.<fnname>.dot or postdom.<fnname>.dot file for each function in the
     12 // program, with a graph of the dominance/postdominance tree of that
     13 // function.
     14 //
     15 // There are also passes available to directly call dotty ('-view-dom' or
     16 // '-view-postdom'). By appending '-only' like '-dot-dom-only' only the
     17 // names of the bbs are printed, but the content is hidden.
     18 //
     19 //===----------------------------------------------------------------------===//
     20 
     21 #include "llvm/Analysis/DomPrinter.h"
     22 #include "llvm/Analysis/DOTGraphTraitsPass.h"
     23 #include "llvm/Analysis/PostDominators.h"
     24 
     25 using namespace llvm;
     26 
     27 namespace llvm {
     28 template<>
     29 struct DOTGraphTraits<DomTreeNode*> : public DefaultDOTGraphTraits {
     30 
     31   DOTGraphTraits (bool isSimple=false)
     32     : DefaultDOTGraphTraits(isSimple) {}
     33 
     34   std::string getNodeLabel(DomTreeNode *Node, DomTreeNode *Graph) {
     35 
     36     BasicBlock *BB = Node->getBlock();
     37 
     38     if (!BB)
     39       return "Post dominance root node";
     40 
     41 
     42     if (isSimple())
     43       return DOTGraphTraits<const Function*>
     44         ::getSimpleNodeLabel(BB, BB->getParent());
     45     else
     46       return DOTGraphTraits<const Function*>
     47         ::getCompleteNodeLabel(BB, BB->getParent());
     48   }
     49 };
     50 
     51 template<>
     52 struct DOTGraphTraits<DominatorTree*> : public DOTGraphTraits<DomTreeNode*> {
     53 
     54   DOTGraphTraits (bool isSimple=false)
     55     : DOTGraphTraits<DomTreeNode*>(isSimple) {}
     56 
     57   static std::string getGraphName(DominatorTree *DT) {
     58     return "Dominator tree";
     59   }
     60 
     61   std::string getNodeLabel(DomTreeNode *Node, DominatorTree *G) {
     62     return DOTGraphTraits<DomTreeNode*>::getNodeLabel(Node, G->getRootNode());
     63   }
     64 };
     65 
     66 template<>
     67 struct DOTGraphTraits<PostDominatorTree*>
     68   : public DOTGraphTraits<DomTreeNode*> {
     69 
     70   DOTGraphTraits (bool isSimple=false)
     71     : DOTGraphTraits<DomTreeNode*>(isSimple) {}
     72 
     73   static std::string getGraphName(PostDominatorTree *DT) {
     74     return "Post dominator tree";
     75   }
     76 
     77   std::string getNodeLabel(DomTreeNode *Node, PostDominatorTree *G ) {
     78     return DOTGraphTraits<DomTreeNode*>::getNodeLabel(Node, G->getRootNode());
     79   }
     80 };
     81 }
     82 
     83 namespace {
     84 struct DominatorTreeWrapperPassAnalysisGraphTraits {
     85   static DominatorTree *getGraph(DominatorTreeWrapperPass *DTWP) {
     86     return &DTWP->getDomTree();
     87   }
     88 };
     89 
     90 struct DomViewer : public DOTGraphTraitsViewer<
     91                        DominatorTreeWrapperPass, false, DominatorTree *,
     92                        DominatorTreeWrapperPassAnalysisGraphTraits> {
     93   static char ID;
     94   DomViewer()
     95       : DOTGraphTraitsViewer<DominatorTreeWrapperPass, false, DominatorTree *,
     96                              DominatorTreeWrapperPassAnalysisGraphTraits>(
     97             "dom", ID) {
     98     initializeDomViewerPass(*PassRegistry::getPassRegistry());
     99   }
    100 };
    101 
    102 struct DomOnlyViewer : public DOTGraphTraitsViewer<
    103                            DominatorTreeWrapperPass, true, DominatorTree *,
    104                            DominatorTreeWrapperPassAnalysisGraphTraits> {
    105   static char ID;
    106   DomOnlyViewer()
    107       : DOTGraphTraitsViewer<DominatorTreeWrapperPass, true, DominatorTree *,
    108                              DominatorTreeWrapperPassAnalysisGraphTraits>(
    109             "domonly", ID) {
    110     initializeDomOnlyViewerPass(*PassRegistry::getPassRegistry());
    111   }
    112 };
    113 
    114 struct PostDominatorTreeWrapperPassAnalysisGraphTraits {
    115   static PostDominatorTree *getGraph(PostDominatorTreeWrapperPass *PDTWP) {
    116     return &PDTWP->getPostDomTree();
    117   }
    118 };
    119 
    120 struct PostDomViewer : public DOTGraphTraitsViewer<
    121                           PostDominatorTreeWrapperPass, false,
    122                           PostDominatorTree *,
    123                           PostDominatorTreeWrapperPassAnalysisGraphTraits> {
    124   static char ID;
    125   PostDomViewer() :
    126     DOTGraphTraitsViewer<PostDominatorTreeWrapperPass, false,
    127                          PostDominatorTree *,
    128                          PostDominatorTreeWrapperPassAnalysisGraphTraits>(
    129         "postdom", ID){
    130       initializePostDomViewerPass(*PassRegistry::getPassRegistry());
    131     }
    132 };
    133 
    134 struct PostDomOnlyViewer : public DOTGraphTraitsViewer<
    135                             PostDominatorTreeWrapperPass, true,
    136                             PostDominatorTree *,
    137                             PostDominatorTreeWrapperPassAnalysisGraphTraits> {
    138   static char ID;
    139   PostDomOnlyViewer() :
    140     DOTGraphTraitsViewer<PostDominatorTreeWrapperPass, true,
    141                          PostDominatorTree *,
    142                          PostDominatorTreeWrapperPassAnalysisGraphTraits>(
    143         "postdomonly", ID){
    144       initializePostDomOnlyViewerPass(*PassRegistry::getPassRegistry());
    145     }
    146 };
    147 } // end anonymous namespace
    148 
    149 char DomViewer::ID = 0;
    150 INITIALIZE_PASS(DomViewer, "view-dom",
    151                 "View dominance tree of function", false, false)
    152 
    153 char DomOnlyViewer::ID = 0;
    154 INITIALIZE_PASS(DomOnlyViewer, "view-dom-only",
    155                 "View dominance tree of function (with no function bodies)",
    156                 false, false)
    157 
    158 char PostDomViewer::ID = 0;
    159 INITIALIZE_PASS(PostDomViewer, "view-postdom",
    160                 "View postdominance tree of function", false, false)
    161 
    162 char PostDomOnlyViewer::ID = 0;
    163 INITIALIZE_PASS(PostDomOnlyViewer, "view-postdom-only",
    164                 "View postdominance tree of function "
    165                 "(with no function bodies)",
    166                 false, false)
    167 
    168 namespace {
    169 struct DomPrinter : public DOTGraphTraitsPrinter<
    170                         DominatorTreeWrapperPass, false, DominatorTree *,
    171                         DominatorTreeWrapperPassAnalysisGraphTraits> {
    172   static char ID;
    173   DomPrinter()
    174       : DOTGraphTraitsPrinter<DominatorTreeWrapperPass, false, DominatorTree *,
    175                               DominatorTreeWrapperPassAnalysisGraphTraits>(
    176             "dom", ID) {
    177     initializeDomPrinterPass(*PassRegistry::getPassRegistry());
    178   }
    179 };
    180 
    181 struct DomOnlyPrinter : public DOTGraphTraitsPrinter<
    182                             DominatorTreeWrapperPass, true, DominatorTree *,
    183                             DominatorTreeWrapperPassAnalysisGraphTraits> {
    184   static char ID;
    185   DomOnlyPrinter()
    186       : DOTGraphTraitsPrinter<DominatorTreeWrapperPass, true, DominatorTree *,
    187                               DominatorTreeWrapperPassAnalysisGraphTraits>(
    188             "domonly", ID) {
    189     initializeDomOnlyPrinterPass(*PassRegistry::getPassRegistry());
    190   }
    191 };
    192 
    193 struct PostDomPrinter
    194   : public DOTGraphTraitsPrinter<
    195                             PostDominatorTreeWrapperPass, false,
    196                             PostDominatorTree *,
    197                             PostDominatorTreeWrapperPassAnalysisGraphTraits> {
    198   static char ID;
    199   PostDomPrinter() :
    200     DOTGraphTraitsPrinter<PostDominatorTreeWrapperPass, false,
    201                           PostDominatorTree *,
    202                           PostDominatorTreeWrapperPassAnalysisGraphTraits>(
    203         "postdom", ID) {
    204       initializePostDomPrinterPass(*PassRegistry::getPassRegistry());
    205     }
    206 };
    207 
    208 struct PostDomOnlyPrinter
    209   : public DOTGraphTraitsPrinter<
    210                             PostDominatorTreeWrapperPass, true,
    211                             PostDominatorTree *,
    212                             PostDominatorTreeWrapperPassAnalysisGraphTraits> {
    213   static char ID;
    214   PostDomOnlyPrinter() :
    215     DOTGraphTraitsPrinter<PostDominatorTreeWrapperPass, true,
    216                           PostDominatorTree *,
    217                           PostDominatorTreeWrapperPassAnalysisGraphTraits>(
    218         "postdomonly", ID) {
    219       initializePostDomOnlyPrinterPass(*PassRegistry::getPassRegistry());
    220     }
    221 };
    222 } // end anonymous namespace
    223 
    224 
    225 
    226 char DomPrinter::ID = 0;
    227 INITIALIZE_PASS(DomPrinter, "dot-dom",
    228                 "Print dominance tree of function to 'dot' file",
    229                 false, false)
    230 
    231 char DomOnlyPrinter::ID = 0;
    232 INITIALIZE_PASS(DomOnlyPrinter, "dot-dom-only",
    233                 "Print dominance tree of function to 'dot' file "
    234                 "(with no function bodies)",
    235                 false, false)
    236 
    237 char PostDomPrinter::ID = 0;
    238 INITIALIZE_PASS(PostDomPrinter, "dot-postdom",
    239                 "Print postdominance tree of function to 'dot' file",
    240                 false, false)
    241 
    242 char PostDomOnlyPrinter::ID = 0;
    243 INITIALIZE_PASS(PostDomOnlyPrinter, "dot-postdom-only",
    244                 "Print postdominance tree of function to 'dot' file "
    245                 "(with no function bodies)",
    246                 false, false)
    247 
    248 // Create methods available outside of this file, to use them
    249 // "include/llvm/LinkAllPasses.h". Otherwise the pass would be deleted by
    250 // the link time optimization.
    251 
    252 FunctionPass *llvm::createDomPrinterPass() {
    253   return new DomPrinter();
    254 }
    255 
    256 FunctionPass *llvm::createDomOnlyPrinterPass() {
    257   return new DomOnlyPrinter();
    258 }
    259 
    260 FunctionPass *llvm::createDomViewerPass() {
    261   return new DomViewer();
    262 }
    263 
    264 FunctionPass *llvm::createDomOnlyViewerPass() {
    265   return new DomOnlyViewer();
    266 }
    267 
    268 FunctionPass *llvm::createPostDomPrinterPass() {
    269   return new PostDomPrinter();
    270 }
    271 
    272 FunctionPass *llvm::createPostDomOnlyPrinterPass() {
    273   return new PostDomOnlyPrinter();
    274 }
    275 
    276 FunctionPass *llvm::createPostDomViewerPass() {
    277   return new PostDomViewer();
    278 }
    279 
    280 FunctionPass *llvm::createPostDomOnlyViewerPass() {
    281   return new PostDomOnlyViewer();
    282 }
    283