1//===-- GraphWriter.cpp - Implements GraphWriter support routines ---------===// 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 implements misc. GraphWriter support routines. 11// 12//===----------------------------------------------------------------------===// 13 14#include "llvm/Support/GraphWriter.h" 15#include "llvm/Support/Path.h" 16#include "llvm/Support/Program.h" 17#include "llvm/Config/config.h" 18using namespace llvm; 19 20std::string llvm::DOT::EscapeString(const std::string &Label) { 21 std::string Str(Label); 22 for (unsigned i = 0; i != Str.length(); ++i) 23 switch (Str[i]) { 24 case '\n': 25 Str.insert(Str.begin()+i, '\\'); // Escape character... 26 ++i; 27 Str[i] = 'n'; 28 break; 29 case '\t': 30 Str.insert(Str.begin()+i, ' '); // Convert to two spaces 31 ++i; 32 Str[i] = ' '; 33 break; 34 case '\\': 35 if (i+1 != Str.length()) 36 switch (Str[i+1]) { 37 case 'l': continue; // don't disturb \l 38 case '|': case '{': case '}': 39 Str.erase(Str.begin()+i); continue; 40 default: break; 41 } 42 case '{': case '}': 43 case '<': case '>': 44 case '|': case '"': 45 Str.insert(Str.begin()+i, '\\'); // Escape character... 46 ++i; // don't infinite loop 47 break; 48 } 49 return Str; 50} 51 52 53 54void llvm::DisplayGraph(const sys::Path &Filename, bool wait, 55 GraphProgram::Name program) { 56 std::string ErrMsg; 57#if HAVE_GRAPHVIZ 58 sys::Path Graphviz(LLVM_PATH_GRAPHVIZ); 59 60 std::vector<const char*> args; 61 args.push_back(Graphviz.c_str()); 62 args.push_back(Filename.c_str()); 63 args.push_back(0); 64 65 errs() << "Running 'Graphviz' program... "; 66 if (sys::Program::ExecuteAndWait(Graphviz, &args[0],0,0,0,0,&ErrMsg)) { 67 errs() << "Error: " << ErrMsg << "\n"; 68 return; 69 } 70 Filename.eraseFromDisk(); 71 errs() << " done. \n"; 72 73#elif HAVE_XDOT_PY 74 std::vector<const char*> args; 75 args.push_back(LLVM_PATH_XDOT_PY); 76 args.push_back(Filename.c_str()); 77 78 switch (program) { 79 case GraphProgram::DOT: args.push_back("-f"); args.push_back("dot"); break; 80 case GraphProgram::FDP: args.push_back("-f"); args.push_back("fdp"); break; 81 case GraphProgram::NEATO: args.push_back("-f"); args.push_back("neato");break; 82 case GraphProgram::TWOPI: args.push_back("-f"); args.push_back("twopi");break; 83 case GraphProgram::CIRCO: args.push_back("-f"); args.push_back("circo");break; 84 default: errs() << "Unknown graph layout name; using default.\n"; 85 } 86 87 args.push_back(0); 88 89 errs() << "Running 'xdot.py' program... "; 90 if (sys::Program::ExecuteAndWait(sys::Path(LLVM_PATH_XDOT_PY), 91 &args[0],0,0,0,0,&ErrMsg)) { 92 errs() << "Error: " << ErrMsg << "\n"; 93 return; 94 } 95 Filename.eraseFromDisk(); 96 errs() << " done. \n"; 97 98#elif (HAVE_GV && (HAVE_DOT || HAVE_FDP || HAVE_NEATO || \ 99 HAVE_TWOPI || HAVE_CIRCO)) 100 sys::Path PSFilename = Filename; 101 PSFilename.appendSuffix("ps"); 102 103 sys::Path prog; 104 105 // Set default grapher 106#if HAVE_CIRCO 107 prog = sys::Path(LLVM_PATH_CIRCO); 108#endif 109#if HAVE_TWOPI 110 prog = sys::Path(LLVM_PATH_TWOPI); 111#endif 112#if HAVE_NEATO 113 prog = sys::Path(LLVM_PATH_NEATO); 114#endif 115#if HAVE_FDP 116 prog = sys::Path(LLVM_PATH_FDP); 117#endif 118#if HAVE_DOT 119 prog = sys::Path(LLVM_PATH_DOT); 120#endif 121 122 // Find which program the user wants 123#if HAVE_DOT 124 if (program == GraphProgram::DOT) 125 prog = sys::Path(LLVM_PATH_DOT); 126#endif 127#if (HAVE_FDP) 128 if (program == GraphProgram::FDP) 129 prog = sys::Path(LLVM_PATH_FDP); 130#endif 131#if (HAVE_NEATO) 132 if (program == GraphProgram::NEATO) 133 prog = sys::Path(LLVM_PATH_NEATO); 134#endif 135#if (HAVE_TWOPI) 136 if (program == GraphProgram::TWOPI) 137 prog = sys::Path(LLVM_PATH_TWOPI); 138#endif 139#if (HAVE_CIRCO) 140 if (program == GraphProgram::CIRCO) 141 prog = sys::Path(LLVM_PATH_CIRCO); 142#endif 143 144 std::vector<const char*> args; 145 args.push_back(prog.c_str()); 146 args.push_back("-Tps"); 147 args.push_back("-Nfontname=Courier"); 148 args.push_back("-Gsize=7.5,10"); 149 args.push_back(Filename.c_str()); 150 args.push_back("-o"); 151 args.push_back(PSFilename.c_str()); 152 args.push_back(0); 153 154 errs() << "Running '" << prog.str() << "' program... "; 155 156 if (sys::Program::ExecuteAndWait(prog, &args[0], 0, 0, 0, 0, &ErrMsg)) { 157 errs() << "Error: " << ErrMsg << "\n"; 158 return; 159 } 160 errs() << " done. \n"; 161 162 sys::Path gv(LLVM_PATH_GV); 163 args.clear(); 164 args.push_back(gv.c_str()); 165 args.push_back(PSFilename.c_str()); 166 args.push_back("--spartan"); 167 args.push_back(0); 168 169 ErrMsg.clear(); 170 if (wait) { 171 if (sys::Program::ExecuteAndWait(gv, &args[0],0,0,0,0,&ErrMsg)) 172 errs() << "Error: " << ErrMsg << "\n"; 173 Filename.eraseFromDisk(); 174 PSFilename.eraseFromDisk(); 175 } 176 else { 177 sys::Program::ExecuteNoWait(gv, &args[0],0,0,0,&ErrMsg); 178 errs() << "Remember to erase graph files: " << Filename.str() << " " 179 << PSFilename.str() << "\n"; 180 } 181#elif HAVE_DOTTY 182 sys::Path dotty(LLVM_PATH_DOTTY); 183 184 std::vector<const char*> args; 185 args.push_back(dotty.c_str()); 186 args.push_back(Filename.c_str()); 187 args.push_back(0); 188 189 errs() << "Running 'dotty' program... "; 190 if (sys::Program::ExecuteAndWait(dotty, &args[0],0,0,0,0,&ErrMsg)) { 191 errs() << "Error: " << ErrMsg << "\n"; 192 } else { 193// Dotty spawns another app and doesn't wait until it returns 194#if defined (__MINGW32__) || defined (_WINDOWS) 195 return; 196#endif 197 Filename.eraseFromDisk(); 198 } 199#endif 200} 201