2009-10-18 04:10:40 +00:00
|
|
|
//===- DomPrinter.cpp - DOT printer for the dominance trees ------------===//
|
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// This file defines '-dot-dom' and '-dot-postdom' analysis passes, which emit
|
|
|
|
// a dom.<fnname>.dot or postdom.<fnname>.dot file for each function in the
|
|
|
|
// program, with a graph of the dominance/postdominance tree of that
|
|
|
|
// function.
|
|
|
|
//
|
|
|
|
// There are also passes available to directly call dotty ('-view-dom' or
|
|
|
|
// '-view-postdom'). By appending '-only' like '-dot-dom-only' only the
|
|
|
|
// names of the bbs are printed, but the content is hidden.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#include "llvm/Analysis/DomPrinter.h"
|
|
|
|
#include "llvm/Pass.h"
|
|
|
|
#include "llvm/Function.h"
|
|
|
|
#include "llvm/Analysis/CFGPrinter.h"
|
|
|
|
#include "llvm/Analysis/Dominators.h"
|
|
|
|
#include "llvm/Analysis/PostDominators.h"
|
|
|
|
|
|
|
|
using namespace llvm;
|
|
|
|
|
|
|
|
namespace llvm {
|
|
|
|
template<>
|
|
|
|
struct DOTGraphTraits<DomTreeNode*> : public DefaultDOTGraphTraits {
|
|
|
|
static std::string getNodeLabel(DomTreeNode *Node, DomTreeNode *Graph,
|
|
|
|
bool ShortNames) {
|
|
|
|
|
|
|
|
BasicBlock *BB = Node->getBlock();
|
|
|
|
|
|
|
|
if (!BB)
|
|
|
|
return "Post dominance root node";
|
|
|
|
|
|
|
|
return DOTGraphTraits<const Function*>::getNodeLabel(BB, BB->getParent(),
|
|
|
|
ShortNames);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
template<>
|
|
|
|
struct DOTGraphTraits<DominatorTree*> : public DOTGraphTraits<DomTreeNode*> {
|
|
|
|
|
|
|
|
static std::string getGraphName(DominatorTree *DT) {
|
|
|
|
return "Dominator tree";
|
|
|
|
}
|
|
|
|
|
|
|
|
static std::string getNodeLabel(DomTreeNode *Node,
|
|
|
|
DominatorTree *G,
|
|
|
|
bool ShortNames) {
|
|
|
|
return DOTGraphTraits<DomTreeNode*>::getNodeLabel(Node, G->getRootNode(),
|
|
|
|
ShortNames);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
template<>
|
|
|
|
struct DOTGraphTraits<PostDominatorTree*>
|
|
|
|
: public DOTGraphTraits<DomTreeNode*> {
|
|
|
|
static std::string getGraphName(PostDominatorTree *DT) {
|
|
|
|
return "Post dominator tree";
|
|
|
|
}
|
|
|
|
static std::string getNodeLabel(DomTreeNode *Node,
|
|
|
|
PostDominatorTree *G,
|
|
|
|
bool ShortNames) {
|
|
|
|
return DOTGraphTraits<DomTreeNode*>::getNodeLabel(Node,
|
|
|
|
G->getRootNode(),
|
|
|
|
ShortNames);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
template <class Analysis, bool OnlyBBS>
|
|
|
|
struct GenericGraphViewer : public FunctionPass {
|
|
|
|
std::string Name;
|
|
|
|
|
2009-10-18 04:27:14 +00:00
|
|
|
GenericGraphViewer(std::string GraphName, const void *ID) : FunctionPass(ID) {
|
2009-10-18 04:10:40 +00:00
|
|
|
Name = GraphName;
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual bool runOnFunction(Function &F) {
|
|
|
|
Analysis *Graph;
|
|
|
|
|
|
|
|
Graph = &getAnalysis<Analysis>();
|
|
|
|
ViewGraph(Graph, Name, OnlyBBS);
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual void getAnalysisUsage(AnalysisUsage &AU) const {
|
|
|
|
AU.setPreservesAll();
|
|
|
|
AU.addRequired<Analysis>();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
struct DomViewer
|
|
|
|
: public GenericGraphViewer<DominatorTree, false> {
|
|
|
|
static char ID;
|
2009-10-18 04:27:14 +00:00
|
|
|
DomViewer() : GenericGraphViewer<DominatorTree, false>("dom", &ID){}
|
2009-10-18 04:10:40 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
struct DomOnlyViewer
|
|
|
|
: public GenericGraphViewer<DominatorTree, true> {
|
|
|
|
static char ID;
|
2009-10-18 04:27:14 +00:00
|
|
|
DomOnlyViewer() : GenericGraphViewer<DominatorTree, true>("domonly", &ID){}
|
2009-10-18 04:10:40 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
struct PostDomViewer
|
|
|
|
: public GenericGraphViewer<PostDominatorTree, false> {
|
|
|
|
static char ID;
|
|
|
|
PostDomViewer() :
|
2009-10-18 04:27:14 +00:00
|
|
|
GenericGraphViewer<PostDominatorTree, false>("postdom", &ID){}
|
2009-10-18 04:10:40 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
struct PostDomOnlyViewer
|
|
|
|
: public GenericGraphViewer<PostDominatorTree, true> {
|
|
|
|
static char ID;
|
|
|
|
PostDomOnlyViewer() :
|
2009-10-18 04:27:14 +00:00
|
|
|
GenericGraphViewer<PostDominatorTree, true>("postdomonly", &ID){}
|
2009-10-18 04:10:40 +00:00
|
|
|
};
|
|
|
|
} // end anonymous namespace
|
|
|
|
|
|
|
|
char DomViewer::ID = 0;
|
|
|
|
RegisterPass<DomViewer> A("view-dom",
|
|
|
|
"View dominance tree of function");
|
|
|
|
|
|
|
|
char DomOnlyViewer::ID = 0;
|
|
|
|
RegisterPass<DomOnlyViewer> B("view-dom-only",
|
|
|
|
"View dominance tree of function "
|
|
|
|
"(with no function bodies)");
|
|
|
|
|
|
|
|
char PostDomViewer::ID = 0;
|
|
|
|
RegisterPass<PostDomViewer> C("view-postdom",
|
|
|
|
"View postdominance tree of function");
|
|
|
|
|
|
|
|
char PostDomOnlyViewer::ID = 0;
|
|
|
|
RegisterPass<PostDomOnlyViewer> D("view-postdom-only",
|
|
|
|
"View postdominance tree of function "
|
|
|
|
"(with no function bodies)");
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
template <class Analysis, bool OnlyBBS>
|
|
|
|
struct GenericGraphPrinter : public FunctionPass {
|
|
|
|
|
|
|
|
std::string Name;
|
|
|
|
|
2009-10-18 04:58:34 +00:00
|
|
|
GenericGraphPrinter(std::string GraphName, const void *ID)
|
|
|
|
: FunctionPass(ID) {
|
2009-10-18 04:10:40 +00:00
|
|
|
Name = GraphName;
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual bool runOnFunction(Function &F) {
|
|
|
|
Analysis *Graph;
|
|
|
|
std::string Filename = Name + "." + F.getNameStr() + ".dot";
|
|
|
|
errs() << "Writing '" << Filename << "'...";
|
|
|
|
|
|
|
|
std::string ErrorInfo;
|
|
|
|
raw_fd_ostream File(Filename.c_str(), ErrorInfo);
|
|
|
|
Graph = &getAnalysis<Analysis>();
|
|
|
|
|
|
|
|
if (ErrorInfo.empty())
|
|
|
|
WriteGraph(File, Graph, OnlyBBS);
|
|
|
|
else
|
|
|
|
errs() << " error opening file for writing!";
|
|
|
|
errs() << "\n";
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual void getAnalysisUsage(AnalysisUsage &AU) const {
|
|
|
|
AU.setPreservesAll();
|
|
|
|
AU.addRequired<Analysis>();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
struct DomPrinter
|
|
|
|
: public GenericGraphPrinter<DominatorTree, false> {
|
|
|
|
static char ID;
|
2009-10-18 04:58:34 +00:00
|
|
|
DomPrinter() : GenericGraphPrinter<DominatorTree, false>("dom", &ID) {}
|
2009-10-18 04:10:40 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
struct DomOnlyPrinter
|
|
|
|
: public GenericGraphPrinter<DominatorTree, true> {
|
|
|
|
static char ID;
|
2009-10-18 04:58:34 +00:00
|
|
|
DomOnlyPrinter() : GenericGraphPrinter<DominatorTree, true>("domonly", &ID) {}
|
2009-10-18 04:10:40 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
struct PostDomPrinter
|
|
|
|
: public GenericGraphPrinter<PostDominatorTree, false> {
|
|
|
|
static char ID;
|
|
|
|
PostDomPrinter() :
|
2009-10-18 04:58:34 +00:00
|
|
|
GenericGraphPrinter<PostDominatorTree, false>("postdom", &ID) {}
|
2009-10-18 04:10:40 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
struct PostDomOnlyPrinter
|
|
|
|
: public GenericGraphPrinter<PostDominatorTree, true> {
|
|
|
|
static char ID;
|
|
|
|
PostDomOnlyPrinter() :
|
2009-10-18 04:58:34 +00:00
|
|
|
GenericGraphPrinter<PostDominatorTree, true>("postdomonly", &ID) {}
|
2009-10-18 04:10:40 +00:00
|
|
|
};
|
|
|
|
} // end anonymous namespace
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
char DomPrinter::ID = 0;
|
|
|
|
RegisterPass<DomPrinter> E("dot-dom",
|
|
|
|
"Print dominance tree of function "
|
|
|
|
"to 'dot' file");
|
|
|
|
|
|
|
|
char DomOnlyPrinter::ID = 0;
|
|
|
|
RegisterPass<DomOnlyPrinter> F("dot-dom-only",
|
|
|
|
"Print dominance tree of function "
|
|
|
|
"to 'dot' file "
|
|
|
|
"(with no function bodies)");
|
|
|
|
|
|
|
|
char PostDomPrinter::ID = 0;
|
|
|
|
RegisterPass<PostDomPrinter> G("dot-postdom",
|
|
|
|
"Print postdominance tree of function "
|
|
|
|
"to 'dot' file");
|
|
|
|
|
|
|
|
char PostDomOnlyPrinter::ID = 0;
|
|
|
|
RegisterPass<PostDomOnlyPrinter> H("dot-postdom-only",
|
|
|
|
"Print postdominance tree of function "
|
|
|
|
"to 'dot' file "
|
|
|
|
"(with no function bodies)");
|
|
|
|
|
|
|
|
// Create methods available outside of this file, to use them
|
|
|
|
// "include/llvm/LinkAllPasses.h". Otherwise the pass would be deleted by
|
|
|
|
// the link time optimization.
|
|
|
|
|
|
|
|
FunctionPass *llvm::createDomPrinterPass() {
|
|
|
|
return new DomPrinter();
|
|
|
|
}
|
|
|
|
|
|
|
|
FunctionPass *llvm::createDomOnlyPrinterPass() {
|
|
|
|
return new DomOnlyPrinter();
|
|
|
|
}
|
|
|
|
|
|
|
|
FunctionPass *llvm::createDomViewerPass() {
|
|
|
|
return new DomViewer();
|
|
|
|
}
|
|
|
|
|
|
|
|
FunctionPass *llvm::createDomOnlyViewerPass() {
|
|
|
|
return new DomOnlyViewer();
|
|
|
|
}
|
|
|
|
|
|
|
|
FunctionPass *llvm::createPostDomPrinterPass() {
|
|
|
|
return new PostDomPrinter();
|
|
|
|
}
|
|
|
|
|
|
|
|
FunctionPass *llvm::createPostDomOnlyPrinterPass() {
|
|
|
|
return new PostDomOnlyPrinter();
|
|
|
|
}
|
|
|
|
|
|
|
|
FunctionPass *llvm::createPostDomViewerPass() {
|
|
|
|
return new PostDomViewer();
|
|
|
|
}
|
|
|
|
|
|
|
|
FunctionPass *llvm::createPostDomOnlyViewerPass() {
|
|
|
|
return new PostDomOnlyViewer();
|
|
|
|
}
|