| Chris Lattner | 7f65075 | 2005-01-10 23:08:40 +0000 | [diff] [blame] | 1 | //===-- SelectionDAGPrinter.cpp - Implement SelectionDAG::viewGraph() -----===// | 
| Misha Brukman | 835702a | 2005-04-21 22:36:52 +0000 | [diff] [blame] | 2 | // | 
| Chris Lattner | 7f65075 | 2005-01-10 23:08:40 +0000 | [diff] [blame] | 3 | //                     The LLVM Compiler Infrastructure | 
|  | 4 | // | 
| Chris Lattner | f3ebc3f | 2007-12-29 20:36:04 +0000 | [diff] [blame] | 5 | // This file is distributed under the University of Illinois Open Source | 
|  | 6 | // License. See LICENSE.TXT for details. | 
| Misha Brukman | 835702a | 2005-04-21 22:36:52 +0000 | [diff] [blame] | 7 | // | 
| Chris Lattner | 7f65075 | 2005-01-10 23:08:40 +0000 | [diff] [blame] | 8 | //===----------------------------------------------------------------------===// | 
|  | 9 | // | 
|  | 10 | // This implements the SelectionDAG::viewGraph method. | 
|  | 11 | // | 
|  | 12 | //===----------------------------------------------------------------------===// | 
|  | 13 |  | 
| Dan Gohman | 483377c | 2009-02-06 17:22:58 +0000 | [diff] [blame] | 14 | #include "ScheduleDAGSDNodes.h" | 
| Chris Lattner | c30405e | 2005-08-26 17:15:30 +0000 | [diff] [blame] | 15 | #include "llvm/Constants.h" | 
|  | 16 | #include "llvm/Function.h" | 
| Chris Lattner | c610e62 | 2006-03-05 09:38:03 +0000 | [diff] [blame] | 17 | #include "llvm/Assembly/Writer.h" | 
| Chris Lattner | 7f65075 | 2005-01-10 23:08:40 +0000 | [diff] [blame] | 18 | #include "llvm/CodeGen/SelectionDAG.h" | 
| Evan Cheng | 45fe3bc | 2006-09-12 21:00:35 +0000 | [diff] [blame] | 19 | #include "llvm/CodeGen/MachineConstantPool.h" | 
| Chris Lattner | 7f65075 | 2005-01-10 23:08:40 +0000 | [diff] [blame] | 20 | #include "llvm/CodeGen/MachineFunction.h" | 
| Bill Wendling | 2e506894 | 2008-07-03 22:53:42 +0000 | [diff] [blame] | 21 | #include "llvm/CodeGen/MachineModuleInfo.h" | 
| Dan Gohman | 7168de7 | 2008-07-17 21:12:16 +0000 | [diff] [blame] | 22 | #include "llvm/CodeGen/PseudoSourceValue.h" | 
| Devang Patel | 5c6e1e3 | 2009-01-13 00:35:13 +0000 | [diff] [blame] | 23 | #include "llvm/Analysis/DebugInfo.h" | 
| Dan Gohman | 3a4be0f | 2008-02-10 18:45:23 +0000 | [diff] [blame] | 24 | #include "llvm/Target/TargetRegisterInfo.h" | 
| Chris Lattner | 4990335 | 2005-08-19 21:21:16 +0000 | [diff] [blame] | 25 | #include "llvm/Target/TargetMachine.h" | 
| David Greene | b04e7c3 | 2008-10-27 18:17:03 +0000 | [diff] [blame] | 26 | #include "llvm/Support/Debug.h" | 
| Chris Lattner | 7f65075 | 2005-01-10 23:08:40 +0000 | [diff] [blame] | 27 | #include "llvm/Support/GraphWriter.h" | 
| Chris Lattner | 838aff3 | 2008-08-23 22:53:13 +0000 | [diff] [blame] | 28 | #include "llvm/Support/raw_ostream.h" | 
| David Greene | b04e7c3 | 2008-10-27 18:17:03 +0000 | [diff] [blame] | 29 | #include "llvm/ADT/DenseSet.h" | 
| Chris Lattner | 1308b48 | 2005-01-11 00:34:33 +0000 | [diff] [blame] | 30 | #include "llvm/ADT/StringExtras.h" | 
| Chris Lattner | b47f5e6 | 2005-07-15 22:48:31 +0000 | [diff] [blame] | 31 | #include "llvm/Config/config.h" | 
| Chris Lattner | 7f65075 | 2005-01-10 23:08:40 +0000 | [diff] [blame] | 32 | using namespace llvm; | 
|  | 33 |  | 
| Chris Lattner | 12be027 | 2005-01-10 23:26:00 +0000 | [diff] [blame] | 34 | namespace llvm { | 
|  | 35 | template<> | 
|  | 36 | struct DOTGraphTraits<SelectionDAG*> : public DefaultDOTGraphTraits { | 
| Tobias Grosser | 90d3340 | 2009-11-30 12:38:13 +0000 | [diff] [blame] | 37 |  | 
| Dan Gohman | 8def6e3 | 2009-12-01 19:16:15 +0000 | [diff] [blame] | 38 | explicit DOTGraphTraits(bool isSimple=false) : | 
|  | 39 | DefaultDOTGraphTraits(isSimple) {} | 
| Tobias Grosser | 90d3340 | 2009-11-30 12:38:13 +0000 | [diff] [blame] | 40 |  | 
| Dan Gohman | f1dc362 | 2008-07-21 21:06:55 +0000 | [diff] [blame] | 41 | static bool hasEdgeDestLabels() { | 
|  | 42 | return true; | 
|  | 43 | } | 
|  | 44 |  | 
|  | 45 | static unsigned numEdgeDestLabels(const void *Node) { | 
|  | 46 | return ((const SDNode *) Node)->getNumValues(); | 
|  | 47 | } | 
|  | 48 |  | 
|  | 49 | static std::string getEdgeDestLabel(const void *Node, unsigned i) { | 
| Owen Anderson | 53aa7a9 | 2009-08-10 22:56:29 +0000 | [diff] [blame] | 50 | return ((const SDNode *) Node)->getValueType(i).getEVTString(); | 
| Dan Gohman | f1dc362 | 2008-07-21 21:06:55 +0000 | [diff] [blame] | 51 | } | 
|  | 52 |  | 
| Dan Gohman | b2ae029 | 2009-12-01 19:20:00 +0000 | [diff] [blame] | 53 | template<typename EdgeIter> | 
|  | 54 | static std::string getEdgeSourceLabel(const void *Node, EdgeIter I) { | 
|  | 55 | return itostr(I - SDNodeIterator::begin((SDNode *) Node)); | 
|  | 56 | } | 
|  | 57 |  | 
| Dan Gohman | f1dc362 | 2008-07-21 21:06:55 +0000 | [diff] [blame] | 58 | /// edgeTargetsEdgeSource - This method returns true if this outgoing edge | 
| Tobias Grosser | 90d3340 | 2009-11-30 12:38:13 +0000 | [diff] [blame] | 59 | /// should actually target another edge source, not a node.  If this method | 
|  | 60 | /// is implemented, getEdgeTarget should be implemented. | 
| Dan Gohman | f1dc362 | 2008-07-21 21:06:55 +0000 | [diff] [blame] | 61 | template<typename EdgeIter> | 
|  | 62 | static bool edgeTargetsEdgeSource(const void *Node, EdgeIter I) { | 
|  | 63 | return true; | 
|  | 64 | } | 
|  | 65 |  | 
|  | 66 | /// getEdgeTarget - If edgeTargetsEdgeSource returns true, this method is | 
|  | 67 | /// called to determine which outgoing edge of Node is the target of this | 
|  | 68 | /// edge. | 
|  | 69 | template<typename EdgeIter> | 
|  | 70 | static EdgeIter getEdgeTarget(const void *Node, EdgeIter I) { | 
|  | 71 | SDNode *TargetNode = *I; | 
|  | 72 | SDNodeIterator NI = SDNodeIterator::begin(TargetNode); | 
| Gabor Greif | abfdf92 | 2008-08-26 22:36:50 +0000 | [diff] [blame] | 73 | std::advance(NI, I.getNode()->getOperand(I.getOperand()).getResNo()); | 
| Dan Gohman | f1dc362 | 2008-07-21 21:06:55 +0000 | [diff] [blame] | 74 | return NI; | 
|  | 75 | } | 
|  | 76 |  | 
| Chris Lattner | 12be027 | 2005-01-10 23:26:00 +0000 | [diff] [blame] | 77 | static std::string getGraphName(const SelectionDAG *G) { | 
|  | 78 | return G->getMachineFunction().getFunction()->getName(); | 
|  | 79 | } | 
| Chris Lattner | 1308b48 | 2005-01-11 00:34:33 +0000 | [diff] [blame] | 80 |  | 
|  | 81 | static bool renderGraphFromBottomUp() { | 
|  | 82 | return true; | 
| Chris Lattner | 12be027 | 2005-01-10 23:26:00 +0000 | [diff] [blame] | 83 | } | 
| Dan Gohman | 8def6e3 | 2009-12-01 19:16:15 +0000 | [diff] [blame] | 84 |  | 
| Chris Lattner | fda6944 | 2005-10-01 00:17:07 +0000 | [diff] [blame] | 85 | static bool hasNodeAddressLabel(const SDNode *Node, | 
|  | 86 | const SelectionDAG *Graph) { | 
|  | 87 | return true; | 
|  | 88 | } | 
| Dan Gohman | 8def6e3 | 2009-12-01 19:16:15 +0000 | [diff] [blame] | 89 |  | 
| Chris Lattner | c5ab6ce | 2006-10-20 18:06:09 +0000 | [diff] [blame] | 90 | /// If you want to override the dot attributes printed for a particular | 
|  | 91 | /// edge, override this method. | 
|  | 92 | template<typename EdgeIter> | 
| Tobias Grosser | 3ac8689 | 2011-02-27 04:11:03 +0000 | [diff] [blame] | 93 | static std::string getEdgeAttributes(const void *Node, EdgeIter EI, | 
|  | 94 | const SelectionDAG *Graph) { | 
| Dan Gohman | 2ce6f2a | 2008-07-27 21:46:04 +0000 | [diff] [blame] | 95 | SDValue Op = EI.getNode()->getOperand(EI.getOperand()); | 
| Owen Anderson | 53aa7a9 | 2009-08-10 22:56:29 +0000 | [diff] [blame] | 96 | EVT VT = Op.getValueType(); | 
| Chris Lattner | 3e5fbd7 | 2010-12-21 02:38:05 +0000 | [diff] [blame] | 97 | if (VT == MVT::Glue) | 
| Chris Lattner | c5ab6ce | 2006-10-20 18:06:09 +0000 | [diff] [blame] | 98 | return "color=red,style=bold"; | 
| Owen Anderson | 9f94459 | 2009-08-11 20:47:22 +0000 | [diff] [blame] | 99 | else if (VT == MVT::Other) | 
| Dan Gohman | 8c73332 | 2007-06-18 15:30:16 +0000 | [diff] [blame] | 100 | return "color=blue,style=dashed"; | 
| Chris Lattner | c5ab6ce | 2006-10-20 18:06:09 +0000 | [diff] [blame] | 101 | return ""; | 
|  | 102 | } | 
| Dan Gohman | 8def6e3 | 2009-12-01 19:16:15 +0000 | [diff] [blame] | 103 |  | 
| Chris Lattner | 12be027 | 2005-01-10 23:26:00 +0000 | [diff] [blame] | 104 |  | 
| Tobias Grosser | dd7f2e7 | 2009-11-30 12:38:47 +0000 | [diff] [blame] | 105 | static std::string getSimpleNodeLabel(const SDNode *Node, | 
|  | 106 | const SelectionDAG *G) { | 
|  | 107 | std::string Result = Node->getOperationName(G); | 
|  | 108 | { | 
|  | 109 | raw_string_ostream OS(Result); | 
|  | 110 | Node->print_details(OS, G); | 
|  | 111 | } | 
|  | 112 | return Result; | 
|  | 113 | } | 
|  | 114 | std::string getNodeLabel(const SDNode *Node, const SelectionDAG *Graph); | 
| Jim Laskey | 1368c26 | 2006-10-02 12:26:53 +0000 | [diff] [blame] | 115 | static std::string getNodeAttributes(const SDNode *N, | 
|  | 116 | const SelectionDAG *Graph) { | 
|  | 117 | #ifndef NDEBUG | 
|  | 118 | const std::string &Attrs = Graph->getGraphAttrs(N); | 
|  | 119 | if (!Attrs.empty()) { | 
|  | 120 | if (Attrs.find("shape=") == std::string::npos) | 
|  | 121 | return std::string("shape=Mrecord,") + Attrs; | 
|  | 122 | else | 
|  | 123 | return Attrs; | 
|  | 124 | } | 
|  | 125 | #endif | 
| Chris Lattner | 12be027 | 2005-01-10 23:26:00 +0000 | [diff] [blame] | 126 | return "shape=Mrecord"; | 
|  | 127 | } | 
| Chris Lattner | b241b44 | 2005-01-10 23:52:04 +0000 | [diff] [blame] | 128 |  | 
|  | 129 | static void addCustomGraphFeatures(SelectionDAG *G, | 
|  | 130 | GraphWriter<SelectionDAG*> &GW) { | 
|  | 131 | GW.emitSimpleNode(0, "plaintext=circle", "GraphRoot"); | 
| Gabor Greif | f304a7a | 2008-08-28 21:40:38 +0000 | [diff] [blame] | 132 | if (G->getRoot().getNode()) | 
|  | 133 | GW.emitEdge(0, -1, G->getRoot().getNode(), G->getRoot().getResNo(), | 
| Dan Gohman | 57c7492 | 2008-07-22 17:52:59 +0000 | [diff] [blame] | 134 | "color=blue,style=dashed"); | 
| Chris Lattner | b241b44 | 2005-01-10 23:52:04 +0000 | [diff] [blame] | 135 | } | 
| Chris Lattner | 12be027 | 2005-01-10 23:26:00 +0000 | [diff] [blame] | 136 | }; | 
|  | 137 | } | 
|  | 138 |  | 
| Chris Lattner | 1308b48 | 2005-01-11 00:34:33 +0000 | [diff] [blame] | 139 | std::string DOTGraphTraits<SelectionDAG*>::getNodeLabel(const SDNode *Node, | 
| Tobias Grosser | dd7f2e7 | 2009-11-30 12:38:47 +0000 | [diff] [blame] | 140 | const SelectionDAG *G) { | 
| Dan Gohman | 8def6e3 | 2009-12-01 19:16:15 +0000 | [diff] [blame] | 141 | return DOTGraphTraits<SelectionDAG*>::getSimpleNodeLabel(Node, G); | 
| Chris Lattner | 1308b48 | 2005-01-11 00:34:33 +0000 | [diff] [blame] | 142 | } | 
| Misha Brukman | 835702a | 2005-04-21 22:36:52 +0000 | [diff] [blame] | 143 |  | 
| Chris Lattner | 1308b48 | 2005-01-11 00:34:33 +0000 | [diff] [blame] | 144 |  | 
| Chris Lattner | 7f65075 | 2005-01-10 23:08:40 +0000 | [diff] [blame] | 145 | /// viewGraph - Pop up a ghostview window with the reachable parts of the DAG | 
|  | 146 | /// rendered using 'dot'. | 
|  | 147 | /// | 
| Dan Gohman | 581cc87 | 2008-07-21 20:00:07 +0000 | [diff] [blame] | 148 | void SelectionDAG::viewGraph(const std::string &Title) { | 
| Chris Lattner | fcc53ad | 2005-07-14 05:17:43 +0000 | [diff] [blame] | 149 | // This code is only for debugging! | 
| Chris Lattner | 46524e2 | 2005-07-14 05:33:13 +0000 | [diff] [blame] | 150 | #ifndef NDEBUG | 
| Dan Gohman | 8def6e3 | 2009-12-01 19:16:15 +0000 | [diff] [blame] | 151 | ViewGraph(this, "dag." + getMachineFunction().getFunction()->getNameStr(), | 
| Daniel Dunbar | 12368685 | 2009-07-24 08:24:36 +0000 | [diff] [blame] | 152 | false, Title); | 
| Reid Spencer | ee7eaa2 | 2006-06-27 16:49:46 +0000 | [diff] [blame] | 153 | #else | 
| Daniel Dunbar | 34ee203 | 2009-08-23 08:50:52 +0000 | [diff] [blame] | 154 | errs() << "SelectionDAG::viewGraph is only available in debug builds on " | 
|  | 155 | << "systems with Graphviz or gv!\n"; | 
| Reid Spencer | ee7eaa2 | 2006-06-27 16:49:46 +0000 | [diff] [blame] | 156 | #endif  // NDEBUG | 
| Chris Lattner | 7f65075 | 2005-01-10 23:08:40 +0000 | [diff] [blame] | 157 | } | 
| Jim Laskey | 1368c26 | 2006-10-02 12:26:53 +0000 | [diff] [blame] | 158 |  | 
| Dan Gohman | 88e0df0 | 2008-07-30 18:48:53 +0000 | [diff] [blame] | 159 | // This overload is defined out-of-line here instead of just using a | 
|  | 160 | // default parameter because this is easiest for gdb to call. | 
|  | 161 | void SelectionDAG::viewGraph() { | 
|  | 162 | viewGraph(""); | 
|  | 163 | } | 
| Jim Laskey | 1368c26 | 2006-10-02 12:26:53 +0000 | [diff] [blame] | 164 |  | 
|  | 165 | /// clearGraphAttrs - Clear all previously defined node graph attributes. | 
|  | 166 | /// Intended to be used from a debugging tool (eg. gdb). | 
|  | 167 | void SelectionDAG::clearGraphAttrs() { | 
|  | 168 | #ifndef NDEBUG | 
|  | 169 | NodeGraphAttrs.clear(); | 
|  | 170 | #else | 
| Daniel Dunbar | 34ee203 | 2009-08-23 08:50:52 +0000 | [diff] [blame] | 171 | errs() << "SelectionDAG::clearGraphAttrs is only available in debug builds" | 
|  | 172 | << " on systems with Graphviz or gv!\n"; | 
| Jim Laskey | 1368c26 | 2006-10-02 12:26:53 +0000 | [diff] [blame] | 173 | #endif | 
|  | 174 | } | 
|  | 175 |  | 
|  | 176 |  | 
|  | 177 | /// setGraphAttrs - Set graph attributes for a node. (eg. "color=red".) | 
|  | 178 | /// | 
|  | 179 | void SelectionDAG::setGraphAttrs(const SDNode *N, const char *Attrs) { | 
|  | 180 | #ifndef NDEBUG | 
|  | 181 | NodeGraphAttrs[N] = Attrs; | 
|  | 182 | #else | 
| Daniel Dunbar | 34ee203 | 2009-08-23 08:50:52 +0000 | [diff] [blame] | 183 | errs() << "SelectionDAG::setGraphAttrs is only available in debug builds" | 
|  | 184 | << " on systems with Graphviz or gv!\n"; | 
| Jim Laskey | 1368c26 | 2006-10-02 12:26:53 +0000 | [diff] [blame] | 185 | #endif | 
|  | 186 | } | 
|  | 187 |  | 
|  | 188 |  | 
|  | 189 | /// getGraphAttrs - Get graph attributes for a node. (eg. "color=red".) | 
|  | 190 | /// Used from getNodeAttributes. | 
|  | 191 | const std::string SelectionDAG::getGraphAttrs(const SDNode *N) const { | 
|  | 192 | #ifndef NDEBUG | 
|  | 193 | std::map<const SDNode *, std::string>::const_iterator I = | 
|  | 194 | NodeGraphAttrs.find(N); | 
| Dan Gohman | 8def6e3 | 2009-12-01 19:16:15 +0000 | [diff] [blame] | 195 |  | 
| Jim Laskey | 1368c26 | 2006-10-02 12:26:53 +0000 | [diff] [blame] | 196 | if (I != NodeGraphAttrs.end()) | 
|  | 197 | return I->second; | 
|  | 198 | else | 
|  | 199 | return ""; | 
|  | 200 | #else | 
| Daniel Dunbar | 34ee203 | 2009-08-23 08:50:52 +0000 | [diff] [blame] | 201 | errs() << "SelectionDAG::getGraphAttrs is only available in debug builds" | 
|  | 202 | << " on systems with Graphviz or gv!\n"; | 
| Dan Gohman | 5cae103 | 2010-08-04 01:39:08 +0000 | [diff] [blame] | 203 | return std::string(); | 
| Jim Laskey | 1368c26 | 2006-10-02 12:26:53 +0000 | [diff] [blame] | 204 | #endif | 
|  | 205 | } | 
|  | 206 |  | 
|  | 207 | /// setGraphColor - Convenience for setting node color attribute. | 
|  | 208 | /// | 
|  | 209 | void SelectionDAG::setGraphColor(const SDNode *N, const char *Color) { | 
|  | 210 | #ifndef NDEBUG | 
|  | 211 | NodeGraphAttrs[N] = std::string("color=") + Color; | 
|  | 212 | #else | 
| Daniel Dunbar | 34ee203 | 2009-08-23 08:50:52 +0000 | [diff] [blame] | 213 | errs() << "SelectionDAG::setGraphColor is only available in debug builds" | 
|  | 214 | << " on systems with Graphviz or gv!\n"; | 
| Jim Laskey | 1368c26 | 2006-10-02 12:26:53 +0000 | [diff] [blame] | 215 | #endif | 
|  | 216 | } | 
|  | 217 |  | 
| David Greene | b04e7c3 | 2008-10-27 18:17:03 +0000 | [diff] [blame] | 218 | /// setSubgraphColorHelper - Implement setSubgraphColor.  Return | 
|  | 219 | /// whether we truncated the search. | 
|  | 220 | /// | 
|  | 221 | bool SelectionDAG::setSubgraphColorHelper(SDNode *N, const char *Color, DenseSet<SDNode *> &visited, | 
|  | 222 | int level, bool &printed) { | 
|  | 223 | bool hit_limit = false; | 
|  | 224 |  | 
|  | 225 | #ifndef NDEBUG | 
|  | 226 | if (level >= 20) { | 
|  | 227 | if (!printed) { | 
|  | 228 | printed = true; | 
| David Greene | 40deefd | 2010-01-05 01:24:45 +0000 | [diff] [blame] | 229 | DEBUG(dbgs() << "setSubgraphColor hit max level\n"); | 
| David Greene | b04e7c3 | 2008-10-27 18:17:03 +0000 | [diff] [blame] | 230 | } | 
|  | 231 | return true; | 
|  | 232 | } | 
|  | 233 |  | 
|  | 234 | unsigned oldSize = visited.size(); | 
|  | 235 | visited.insert(N); | 
|  | 236 | if (visited.size() != oldSize) { | 
|  | 237 | setGraphColor(N, Color); | 
|  | 238 | for(SDNodeIterator i = SDNodeIterator::begin(N), iend = SDNodeIterator::end(N); | 
|  | 239 | i != iend; | 
|  | 240 | ++i) { | 
|  | 241 | hit_limit = setSubgraphColorHelper(*i, Color, visited, level+1, printed) || hit_limit; | 
|  | 242 | } | 
|  | 243 | } | 
|  | 244 | #else | 
| Daniel Dunbar | 34ee203 | 2009-08-23 08:50:52 +0000 | [diff] [blame] | 245 | errs() << "SelectionDAG::setSubgraphColor is only available in debug builds" | 
|  | 246 | << " on systems with Graphviz or gv!\n"; | 
| David Greene | b04e7c3 | 2008-10-27 18:17:03 +0000 | [diff] [blame] | 247 | #endif | 
|  | 248 | return hit_limit; | 
|  | 249 | } | 
|  | 250 |  | 
|  | 251 | /// setSubgraphColor - Convenience for setting subgraph color attribute. | 
|  | 252 | /// | 
|  | 253 | void SelectionDAG::setSubgraphColor(SDNode *N, const char *Color) { | 
|  | 254 | #ifndef NDEBUG | 
|  | 255 | DenseSet<SDNode *> visited; | 
|  | 256 | bool printed = false; | 
|  | 257 | if (setSubgraphColorHelper(N, Color, visited, 0, printed)) { | 
|  | 258 | // Visually mark that we hit the limit | 
| Ted Kremenek | 8fcff4d | 2008-10-27 22:43:07 +0000 | [diff] [blame] | 259 | if (strcmp(Color, "red") == 0) { | 
| David Greene | b04e7c3 | 2008-10-27 18:17:03 +0000 | [diff] [blame] | 260 | setSubgraphColorHelper(N, "blue", visited, 0, printed); | 
| Dan Gohman | 8def6e3 | 2009-12-01 19:16:15 +0000 | [diff] [blame] | 261 | } else if (strcmp(Color, "yellow") == 0) { | 
| David Greene | b04e7c3 | 2008-10-27 18:17:03 +0000 | [diff] [blame] | 262 | setSubgraphColorHelper(N, "green", visited, 0, printed); | 
|  | 263 | } | 
|  | 264 | } | 
|  | 265 |  | 
|  | 266 | #else | 
| Daniel Dunbar | 34ee203 | 2009-08-23 08:50:52 +0000 | [diff] [blame] | 267 | errs() << "SelectionDAG::setSubgraphColor is only available in debug builds" | 
|  | 268 | << " on systems with Graphviz or gv!\n"; | 
| David Greene | b04e7c3 | 2008-10-27 18:17:03 +0000 | [diff] [blame] | 269 | #endif | 
|  | 270 | } | 
|  | 271 |  | 
| Dan Gohman | 60cb69e | 2008-11-19 23:18:57 +0000 | [diff] [blame] | 272 | std::string ScheduleDAGSDNodes::getGraphNodeLabel(const SUnit *SU) const { | 
| Dan Gohman | f4d95fd | 2008-11-19 22:09:45 +0000 | [diff] [blame] | 273 | std::string s; | 
|  | 274 | raw_string_ostream O(s); | 
|  | 275 | O << "SU(" << SU->NodeNum << "): "; | 
|  | 276 | if (SU->getNode()) { | 
| Chris Lattner | 11a3381 | 2010-12-23 17:24:32 +0000 | [diff] [blame] | 277 | SmallVector<SDNode *, 4> GluedNodes; | 
|  | 278 | for (SDNode *N = SU->getNode(); N; N = N->getGluedNode()) | 
|  | 279 | GluedNodes.push_back(N); | 
|  | 280 | while (!GluedNodes.empty()) { | 
| Tobias Grosser | dd7f2e7 | 2009-11-30 12:38:47 +0000 | [diff] [blame] | 281 | O << DOTGraphTraits<SelectionDAG*> | 
| Chris Lattner | 11a3381 | 2010-12-23 17:24:32 +0000 | [diff] [blame] | 282 | ::getSimpleNodeLabel(GluedNodes.back(), DAG); | 
|  | 283 | GluedNodes.pop_back(); | 
|  | 284 | if (!GluedNodes.empty()) | 
| Dan Gohman | f4d95fd | 2008-11-19 22:09:45 +0000 | [diff] [blame] | 285 | O << "\n    "; | 
|  | 286 | } | 
|  | 287 | } else { | 
|  | 288 | O << "CROSS RC COPY"; | 
|  | 289 | } | 
|  | 290 | return O.str(); | 
|  | 291 | } | 
| Dan Gohman | 81b62e1 | 2007-08-28 20:32:58 +0000 | [diff] [blame] | 292 |  | 
| Dan Gohman | 60cb69e | 2008-11-19 23:18:57 +0000 | [diff] [blame] | 293 | void ScheduleDAGSDNodes::getCustomGraphFeatures(GraphWriter<ScheduleDAG*> &GW) const { | 
|  | 294 | if (DAG) { | 
|  | 295 | // Draw a special "GraphRoot" node to indicate the root of the graph. | 
|  | 296 | GW.emitSimpleNode(0, "plaintext=circle", "GraphRoot"); | 
|  | 297 | const SDNode *N = DAG->getRoot().getNode(); | 
|  | 298 | if (N && N->getNodeId() != -1) | 
|  | 299 | GW.emitEdge(0, -1, &SUnits[N->getNodeId()], -1, | 
|  | 300 | "color=blue,style=dashed"); | 
|  | 301 | } | 
| Dan Gohman | 81b62e1 | 2007-08-28 20:32:58 +0000 | [diff] [blame] | 302 | } |