1 //===-- CFGPrinter.h - CFG printer external interface -----------*- C++ -*-===// 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 external functions that can be called to explicitly 11 // instantiate the CFG printer. 12 // 13 //===----------------------------------------------------------------------===// 14 15 #ifndef LLVM_ANALYSIS_CFGPRINTER_H 16 #define LLVM_ANALYSIS_CFGPRINTER_H 17 18 #include "llvm/Assembly/Writer.h" 19 #include "llvm/IR/Constants.h" 20 #include "llvm/IR/Function.h" 21 #include "llvm/IR/Instructions.h" 22 #include "llvm/Support/CFG.h" 23 #include "llvm/Support/GraphWriter.h" 24 25 namespace llvm { 26 template<> 27 struct DOTGraphTraits<const Function*> : public DefaultDOTGraphTraits { 28 29 DOTGraphTraits (bool isSimple=false) : DefaultDOTGraphTraits(isSimple) {} 30 31 static std::string getGraphName(const Function *F) { 32 return "CFG for '" + F->getName().str() + "' function"; 33 } 34 35 static std::string getSimpleNodeLabel(const BasicBlock *Node, 36 const Function *) { 37 if (!Node->getName().empty()) 38 return Node->getName().str(); 39 40 std::string Str; 41 raw_string_ostream OS(Str); 42 43 WriteAsOperand(OS, Node, false); 44 return OS.str(); 45 } 46 47 static std::string getCompleteNodeLabel(const BasicBlock *Node, 48 const Function *) { 49 std::string Str; 50 raw_string_ostream OS(Str); 51 52 if (Node->getName().empty()) { 53 WriteAsOperand(OS, Node, false); 54 OS << ":"; 55 } 56 57 OS << *Node; 58 std::string OutStr = OS.str(); 59 if (OutStr[0] == '\n') OutStr.erase(OutStr.begin()); 60 61 // Process string output to make it nicer... 62 for (unsigned i = 0; i != OutStr.length(); ++i) 63 if (OutStr[i] == '\n') { // Left justify 64 OutStr[i] = '\\'; 65 OutStr.insert(OutStr.begin()+i+1, 'l'); 66 } else if (OutStr[i] == ';') { // Delete comments! 67 unsigned Idx = OutStr.find('\n', i+1); // Find end of line 68 OutStr.erase(OutStr.begin()+i, OutStr.begin()+Idx); 69 --i; 70 } 71 72 return OutStr; 73 } 74 75 std::string getNodeLabel(const BasicBlock *Node, 76 const Function *Graph) { 77 if (isSimple()) 78 return getSimpleNodeLabel(Node, Graph); 79 else 80 return getCompleteNodeLabel(Node, Graph); 81 } 82 83 static std::string getEdgeSourceLabel(const BasicBlock *Node, 84 succ_const_iterator I) { 85 // Label source of conditional branches with "T" or "F" 86 if (const BranchInst *BI = dyn_cast<BranchInst>(Node->getTerminator())) 87 if (BI->isConditional()) 88 return (I == succ_begin(Node)) ? "T" : "F"; 89 90 // Label source of switch edges with the associated value. 91 if (const SwitchInst *SI = dyn_cast<SwitchInst>(Node->getTerminator())) { 92 unsigned SuccNo = I.getSuccessorIndex(); 93 94 if (SuccNo == 0) return "def"; 95 96 std::string Str; 97 raw_string_ostream OS(Str); 98 SwitchInst::ConstCaseIt Case = 99 SwitchInst::ConstCaseIt::fromSuccessorIndex(SI, SuccNo); 100 OS << Case.getCaseValue()->getValue(); 101 return OS.str(); 102 } 103 return ""; 104 } 105 }; 106 } // End llvm namespace 107 108 namespace llvm { 109 class FunctionPass; 110 FunctionPass *createCFGPrinterPass (); 111 FunctionPass *createCFGOnlyPrinterPass (); 112 } // End llvm namespace 113 114 #endif 115