Add support for viewing graphviz graphs with xdot.py.
[oota-llvm.git] / lib / Support / GraphWriter.cpp
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/System/Path.h"
16 #include "llvm/System/Program.h"
17 #include "llvm/Config/config.h"
18 using namespace llvm;
19
20 std::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
54 void 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 viewing graph " << Filename.str() << ": " << ErrMsg
68            << "\n";
69   else
70     Filename.eraseFromDisk();
71
72 #elif HAVE_XDOT_PY
73   sys::Path XDotPy();
74
75   std::vector<const char*> args;
76   args.push_back(LLVM_PATH_XDOT_PY);
77   args.push_back(Filename.c_str());
78
79   switch (program) {
80   case GraphProgram::DOT:   args.push_back("-f"); args.push_back("dot"); break;
81   case GraphProgram::FDP:   args.push_back("-f"); args.push_back("fdp"); break;
82   case GraphProgram::NEATO: args.push_back("-f"); args.push_back("neato");break;
83   case GraphProgram::TWOPI: args.push_back("-f"); args.push_back("twopi");break;
84   case GraphProgram::CIRCO: args.push_back("-f"); args.push_back("circo");break;
85   default: errs() << "Unknown graph layout name; using default.\n";
86   }
87   
88   args.push_back(0);
89
90   errs() << "Running 'xdot.py' program... ";
91   if (sys::Program::ExecuteAndWait(sys::Path(LLVM_PATH_XDOT_PY),
92                                    &args[0],0,0,0,0,&ErrMsg))
93     errs() << "Error viewing graph " << Filename.str() << ": " << ErrMsg
94            << "\n";
95   else
96     Filename.eraseFromDisk();
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 viewing graph " << Filename.str() << ": '"
158             << ErrMsg << "\n";
159     return;
160   }
161   errs() << " done. \n";
162
163   sys::Path gv(LLVM_PATH_GV);
164   args.clear();
165   args.push_back(gv.c_str());
166   args.push_back(PSFilename.c_str());
167   args.push_back("--spartan");
168   args.push_back(0);
169   
170   ErrMsg.clear();
171   if (wait) {
172      if (sys::Program::ExecuteAndWait(gv, &args[0],0,0,0,0,&ErrMsg))
173         errs() << "Error viewing graph: " << ErrMsg << "\n";
174      Filename.eraseFromDisk();
175      PSFilename.eraseFromDisk();
176   }
177   else {
178      sys::Program::ExecuteNoWait(gv, &args[0],0,0,0,&ErrMsg);
179      errs() << "Remember to erase graph files: " << Filename.str() << " "
180             << PSFilename.str() << "\n";
181   }
182 #elif HAVE_DOTTY
183   sys::Path dotty(LLVM_PATH_DOTTY);
184
185   std::vector<const char*> args;
186   args.push_back(dotty.c_str());
187   args.push_back(Filename.c_str());
188   args.push_back(0);
189   
190   errs() << "Running 'dotty' program... ";
191   if (sys::Program::ExecuteAndWait(dotty, &args[0],0,0,0,0,&ErrMsg)) {
192      errs() << "Error viewing graph " << Filename.str() << ": "
193             << ErrMsg << "\n";
194   } else {
195 // Dotty spawns another app and doesn't wait until it returns
196 #if defined (__MINGW32__) || defined (_WINDOWS)
197     return;
198 #endif
199     Filename.eraseFromDisk();
200   }
201 #endif
202 }