xref: /freebsd/contrib/llvm-project/llvm/lib/CodeGen/SelectionDAG/SelectionDAGPrinter.cpp (revision fe6060f10f634930ff71b7c50291ddc610da2475)
10b57cec5SDimitry Andric //===-- SelectionDAGPrinter.cpp - Implement SelectionDAG::viewGraph() -----===//
20b57cec5SDimitry Andric //
30b57cec5SDimitry Andric // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
40b57cec5SDimitry Andric // See https://llvm.org/LICENSE.txt for license information.
50b57cec5SDimitry Andric // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
60b57cec5SDimitry Andric //
70b57cec5SDimitry Andric //===----------------------------------------------------------------------===//
80b57cec5SDimitry Andric //
90b57cec5SDimitry Andric // This implements the SelectionDAG::viewGraph method.
100b57cec5SDimitry Andric //
110b57cec5SDimitry Andric //===----------------------------------------------------------------------===//
120b57cec5SDimitry Andric 
130b57cec5SDimitry Andric #include "ScheduleDAGSDNodes.h"
140b57cec5SDimitry Andric #include "llvm/ADT/DenseSet.h"
150b57cec5SDimitry Andric #include "llvm/ADT/StringExtras.h"
160b57cec5SDimitry Andric #include "llvm/CodeGen/MachineConstantPool.h"
170b57cec5SDimitry Andric #include "llvm/CodeGen/MachineFunction.h"
180b57cec5SDimitry Andric #include "llvm/CodeGen/SelectionDAG.h"
190b57cec5SDimitry Andric #include "llvm/CodeGen/TargetRegisterInfo.h"
200b57cec5SDimitry Andric #include "llvm/IR/Constants.h"
210b57cec5SDimitry Andric #include "llvm/Support/Debug.h"
220b57cec5SDimitry Andric #include "llvm/Support/GraphWriter.h"
230b57cec5SDimitry Andric #include "llvm/Support/raw_ostream.h"
240b57cec5SDimitry Andric #include "llvm/Target/TargetMachine.h"
250b57cec5SDimitry Andric using namespace llvm;
260b57cec5SDimitry Andric 
270b57cec5SDimitry Andric #define DEBUG_TYPE "dag-printer"
280b57cec5SDimitry Andric 
290b57cec5SDimitry Andric namespace llvm {
300b57cec5SDimitry Andric   template<>
310b57cec5SDimitry Andric   struct DOTGraphTraits<SelectionDAG*> : public DefaultDOTGraphTraits {
320b57cec5SDimitry Andric 
330b57cec5SDimitry Andric     explicit DOTGraphTraits(bool isSimple=false) :
340b57cec5SDimitry Andric       DefaultDOTGraphTraits(isSimple) {}
350b57cec5SDimitry Andric 
360b57cec5SDimitry Andric     static bool hasEdgeDestLabels() {
370b57cec5SDimitry Andric       return true;
380b57cec5SDimitry Andric     }
390b57cec5SDimitry Andric 
400b57cec5SDimitry Andric     static unsigned numEdgeDestLabels(const void *Node) {
410b57cec5SDimitry Andric       return ((const SDNode *) Node)->getNumValues();
420b57cec5SDimitry Andric     }
430b57cec5SDimitry Andric 
440b57cec5SDimitry Andric     static std::string getEdgeDestLabel(const void *Node, unsigned i) {
450b57cec5SDimitry Andric       return ((const SDNode *) Node)->getValueType(i).getEVTString();
460b57cec5SDimitry Andric     }
470b57cec5SDimitry Andric 
480b57cec5SDimitry Andric     template<typename EdgeIter>
490b57cec5SDimitry Andric     static std::string getEdgeSourceLabel(const void *Node, EdgeIter I) {
500b57cec5SDimitry Andric       return itostr(I - SDNodeIterator::begin((const SDNode *) Node));
510b57cec5SDimitry Andric     }
520b57cec5SDimitry Andric 
530b57cec5SDimitry Andric     /// edgeTargetsEdgeSource - This method returns true if this outgoing edge
540b57cec5SDimitry Andric     /// should actually target another edge source, not a node.  If this method
550b57cec5SDimitry Andric     /// is implemented, getEdgeTarget should be implemented.
560b57cec5SDimitry Andric     template<typename EdgeIter>
570b57cec5SDimitry Andric     static bool edgeTargetsEdgeSource(const void *Node, EdgeIter I) {
580b57cec5SDimitry Andric       return true;
590b57cec5SDimitry Andric     }
600b57cec5SDimitry Andric 
610b57cec5SDimitry Andric     /// getEdgeTarget - If edgeTargetsEdgeSource returns true, this method is
620b57cec5SDimitry Andric     /// called to determine which outgoing edge of Node is the target of this
630b57cec5SDimitry Andric     /// edge.
640b57cec5SDimitry Andric     template<typename EdgeIter>
650b57cec5SDimitry Andric     static EdgeIter getEdgeTarget(const void *Node, EdgeIter I) {
660b57cec5SDimitry Andric       SDNode *TargetNode = *I;
670b57cec5SDimitry Andric       SDNodeIterator NI = SDNodeIterator::begin(TargetNode);
680b57cec5SDimitry Andric       std::advance(NI, I.getNode()->getOperand(I.getOperand()).getResNo());
690b57cec5SDimitry Andric       return NI;
700b57cec5SDimitry Andric     }
710b57cec5SDimitry Andric 
720b57cec5SDimitry Andric     static std::string getGraphName(const SelectionDAG *G) {
735ffd83dbSDimitry Andric       return std::string(G->getMachineFunction().getName());
740b57cec5SDimitry Andric     }
750b57cec5SDimitry Andric 
760b57cec5SDimitry Andric     static bool renderGraphFromBottomUp() {
770b57cec5SDimitry Andric       return true;
780b57cec5SDimitry Andric     }
790b57cec5SDimitry Andric 
800b57cec5SDimitry Andric     static std::string getNodeIdentifierLabel(const SDNode *Node,
810b57cec5SDimitry Andric                                               const SelectionDAG *Graph) {
820b57cec5SDimitry Andric       std::string R;
830b57cec5SDimitry Andric       raw_string_ostream OS(R);
840b57cec5SDimitry Andric #ifndef NDEBUG
850b57cec5SDimitry Andric       OS << 't' << Node->PersistentId;
860b57cec5SDimitry Andric #else
870b57cec5SDimitry Andric       OS << static_cast<const void *>(Node);
880b57cec5SDimitry Andric #endif
890b57cec5SDimitry Andric       return R;
900b57cec5SDimitry Andric     }
910b57cec5SDimitry Andric 
920b57cec5SDimitry Andric     /// If you want to override the dot attributes printed for a particular
930b57cec5SDimitry Andric     /// edge, override this method.
940b57cec5SDimitry Andric     template<typename EdgeIter>
950b57cec5SDimitry Andric     static std::string getEdgeAttributes(const void *Node, EdgeIter EI,
960b57cec5SDimitry Andric                                          const SelectionDAG *Graph) {
970b57cec5SDimitry Andric       SDValue Op = EI.getNode()->getOperand(EI.getOperand());
980b57cec5SDimitry Andric       EVT VT = Op.getValueType();
990b57cec5SDimitry Andric       if (VT == MVT::Glue)
1000b57cec5SDimitry Andric         return "color=red,style=bold";
1010b57cec5SDimitry Andric       else if (VT == MVT::Other)
1020b57cec5SDimitry Andric         return "color=blue,style=dashed";
1030b57cec5SDimitry Andric       return "";
1040b57cec5SDimitry Andric     }
1050b57cec5SDimitry Andric 
1060b57cec5SDimitry Andric 
1070b57cec5SDimitry Andric     static std::string getSimpleNodeLabel(const SDNode *Node,
1080b57cec5SDimitry Andric                                           const SelectionDAG *G) {
1090b57cec5SDimitry Andric       std::string Result = Node->getOperationName(G);
1100b57cec5SDimitry Andric       {
1110b57cec5SDimitry Andric         raw_string_ostream OS(Result);
1120b57cec5SDimitry Andric         Node->print_details(OS, G);
1130b57cec5SDimitry Andric       }
1140b57cec5SDimitry Andric       return Result;
1150b57cec5SDimitry Andric     }
1160b57cec5SDimitry Andric     std::string getNodeLabel(const SDNode *Node, const SelectionDAG *Graph);
1170b57cec5SDimitry Andric     static std::string getNodeAttributes(const SDNode *N,
1180b57cec5SDimitry Andric                                          const SelectionDAG *Graph) {
1190b57cec5SDimitry Andric #ifndef NDEBUG
1200b57cec5SDimitry Andric       const std::string &Attrs = Graph->getGraphAttrs(N);
1210b57cec5SDimitry Andric       if (!Attrs.empty()) {
1220b57cec5SDimitry Andric         if (Attrs.find("shape=") == std::string::npos)
1230b57cec5SDimitry Andric           return std::string("shape=Mrecord,") + Attrs;
1240b57cec5SDimitry Andric         else
1250b57cec5SDimitry Andric           return Attrs;
1260b57cec5SDimitry Andric       }
1270b57cec5SDimitry Andric #endif
1280b57cec5SDimitry Andric       return "shape=Mrecord";
1290b57cec5SDimitry Andric     }
1300b57cec5SDimitry Andric 
1310b57cec5SDimitry Andric     static void addCustomGraphFeatures(SelectionDAG *G,
1320b57cec5SDimitry Andric                                        GraphWriter<SelectionDAG*> &GW) {
1330b57cec5SDimitry Andric       GW.emitSimpleNode(nullptr, "plaintext=circle", "GraphRoot");
1340b57cec5SDimitry Andric       if (G->getRoot().getNode())
1350b57cec5SDimitry Andric         GW.emitEdge(nullptr, -1, G->getRoot().getNode(), G->getRoot().getResNo(),
1360b57cec5SDimitry Andric                     "color=blue,style=dashed");
1370b57cec5SDimitry Andric     }
1380b57cec5SDimitry Andric   };
1390b57cec5SDimitry Andric }
1400b57cec5SDimitry Andric 
1410b57cec5SDimitry Andric std::string DOTGraphTraits<SelectionDAG*>::getNodeLabel(const SDNode *Node,
1420b57cec5SDimitry Andric                                                         const SelectionDAG *G) {
1430b57cec5SDimitry Andric   return DOTGraphTraits<SelectionDAG*>::getSimpleNodeLabel(Node, G);
1440b57cec5SDimitry Andric }
1450b57cec5SDimitry Andric 
1460b57cec5SDimitry Andric 
1470b57cec5SDimitry Andric /// viewGraph - Pop up a ghostview window with the reachable parts of the DAG
1480b57cec5SDimitry Andric /// rendered using 'dot'.
1490b57cec5SDimitry Andric ///
1500b57cec5SDimitry Andric void SelectionDAG::viewGraph(const std::string &Title) {
1510b57cec5SDimitry Andric // This code is only for debugging!
1520b57cec5SDimitry Andric #ifndef NDEBUG
1530b57cec5SDimitry Andric   ViewGraph(this, "dag." + getMachineFunction().getName(),
1540b57cec5SDimitry Andric             false, Title);
1550b57cec5SDimitry Andric #else
1560b57cec5SDimitry Andric   errs() << "SelectionDAG::viewGraph is only available in debug builds on "
1570b57cec5SDimitry Andric          << "systems with Graphviz or gv!\n";
1580b57cec5SDimitry Andric #endif  // NDEBUG
1590b57cec5SDimitry Andric }
1600b57cec5SDimitry Andric 
1610b57cec5SDimitry Andric // This overload is defined out-of-line here instead of just using a
1620b57cec5SDimitry Andric // default parameter because this is easiest for gdb to call.
1630b57cec5SDimitry Andric void SelectionDAG::viewGraph() {
1640b57cec5SDimitry Andric   viewGraph("");
1650b57cec5SDimitry Andric }
1660b57cec5SDimitry Andric 
1675ffd83dbSDimitry Andric /// Just dump dot graph to a user-provided path and title.
1685ffd83dbSDimitry Andric /// This doesn't open the dot viewer program and
1695ffd83dbSDimitry Andric /// helps visualization when outside debugging session.
1705ffd83dbSDimitry Andric /// FileName expects absolute path. If provided
1715ffd83dbSDimitry Andric /// without any path separators then the file
1725ffd83dbSDimitry Andric /// will be created in the current directory.
1735ffd83dbSDimitry Andric /// Error will be emitted if the path is insane.
1745ffd83dbSDimitry Andric #if !defined(NDEBUG) || defined(LLVM_ENABLE_DUMP)
1755ffd83dbSDimitry Andric LLVM_DUMP_METHOD void SelectionDAG::dumpDotGraph(const Twine &FileName,
1765ffd83dbSDimitry Andric                                                  const Twine &Title) {
1775ffd83dbSDimitry Andric   dumpDotGraphToFile(this, FileName, Title);
1785ffd83dbSDimitry Andric }
1795ffd83dbSDimitry Andric #endif
1805ffd83dbSDimitry Andric 
1810b57cec5SDimitry Andric /// clearGraphAttrs - Clear all previously defined node graph attributes.
1820b57cec5SDimitry Andric /// Intended to be used from a debugging tool (eg. gdb).
1830b57cec5SDimitry Andric void SelectionDAG::clearGraphAttrs() {
1840b57cec5SDimitry Andric #ifndef NDEBUG
1850b57cec5SDimitry Andric   NodeGraphAttrs.clear();
1860b57cec5SDimitry Andric #else
1870b57cec5SDimitry Andric   errs() << "SelectionDAG::clearGraphAttrs is only available in debug builds"
1880b57cec5SDimitry Andric          << " on systems with Graphviz or gv!\n";
1890b57cec5SDimitry Andric #endif
1900b57cec5SDimitry Andric }
1910b57cec5SDimitry Andric 
1920b57cec5SDimitry Andric 
1930b57cec5SDimitry Andric /// setGraphAttrs - Set graph attributes for a node. (eg. "color=red".)
1940b57cec5SDimitry Andric ///
1950b57cec5SDimitry Andric void SelectionDAG::setGraphAttrs(const SDNode *N, const char *Attrs) {
1960b57cec5SDimitry Andric #ifndef NDEBUG
1970b57cec5SDimitry Andric   NodeGraphAttrs[N] = Attrs;
1980b57cec5SDimitry Andric #else
1990b57cec5SDimitry Andric   errs() << "SelectionDAG::setGraphAttrs is only available in debug builds"
2000b57cec5SDimitry Andric          << " on systems with Graphviz or gv!\n";
2010b57cec5SDimitry Andric #endif
2020b57cec5SDimitry Andric }
2030b57cec5SDimitry Andric 
2040b57cec5SDimitry Andric 
2050b57cec5SDimitry Andric /// getGraphAttrs - Get graph attributes for a node. (eg. "color=red".)
2060b57cec5SDimitry Andric /// Used from getNodeAttributes.
207*fe6060f1SDimitry Andric std::string SelectionDAG::getGraphAttrs(const SDNode *N) const {
2080b57cec5SDimitry Andric #ifndef NDEBUG
2090b57cec5SDimitry Andric   std::map<const SDNode *, std::string>::const_iterator I =
2100b57cec5SDimitry Andric     NodeGraphAttrs.find(N);
2110b57cec5SDimitry Andric 
2120b57cec5SDimitry Andric   if (I != NodeGraphAttrs.end())
2130b57cec5SDimitry Andric     return I->second;
2140b57cec5SDimitry Andric   else
2150b57cec5SDimitry Andric     return "";
2160b57cec5SDimitry Andric #else
2170b57cec5SDimitry Andric   errs() << "SelectionDAG::getGraphAttrs is only available in debug builds"
2180b57cec5SDimitry Andric          << " on systems with Graphviz or gv!\n";
2190b57cec5SDimitry Andric   return std::string();
2200b57cec5SDimitry Andric #endif
2210b57cec5SDimitry Andric }
2220b57cec5SDimitry Andric 
2230b57cec5SDimitry Andric /// setGraphColor - Convenience for setting node color attribute.
2240b57cec5SDimitry Andric ///
2250b57cec5SDimitry Andric void SelectionDAG::setGraphColor(const SDNode *N, const char *Color) {
2260b57cec5SDimitry Andric #ifndef NDEBUG
2270b57cec5SDimitry Andric   NodeGraphAttrs[N] = std::string("color=") + Color;
2280b57cec5SDimitry Andric #else
2290b57cec5SDimitry Andric   errs() << "SelectionDAG::setGraphColor is only available in debug builds"
2300b57cec5SDimitry Andric          << " on systems with Graphviz or gv!\n";
2310b57cec5SDimitry Andric #endif
2320b57cec5SDimitry Andric }
2330b57cec5SDimitry Andric 
2340b57cec5SDimitry Andric /// setSubgraphColorHelper - Implement setSubgraphColor.  Return
2350b57cec5SDimitry Andric /// whether we truncated the search.
2360b57cec5SDimitry Andric ///
2370b57cec5SDimitry Andric bool SelectionDAG::setSubgraphColorHelper(SDNode *N, const char *Color, DenseSet<SDNode *> &visited,
2380b57cec5SDimitry Andric                                           int level, bool &printed) {
2390b57cec5SDimitry Andric   bool hit_limit = false;
2400b57cec5SDimitry Andric 
2410b57cec5SDimitry Andric #ifndef NDEBUG
2420b57cec5SDimitry Andric   if (level >= 20) {
2430b57cec5SDimitry Andric     if (!printed) {
2440b57cec5SDimitry Andric       printed = true;
2450b57cec5SDimitry Andric       LLVM_DEBUG(dbgs() << "setSubgraphColor hit max level\n");
2460b57cec5SDimitry Andric     }
2470b57cec5SDimitry Andric     return true;
2480b57cec5SDimitry Andric   }
2490b57cec5SDimitry Andric 
2500b57cec5SDimitry Andric   unsigned oldSize = visited.size();
2510b57cec5SDimitry Andric   visited.insert(N);
2520b57cec5SDimitry Andric   if (visited.size() != oldSize) {
2530b57cec5SDimitry Andric     setGraphColor(N, Color);
2540b57cec5SDimitry Andric     for(SDNodeIterator i = SDNodeIterator::begin(N), iend = SDNodeIterator::end(N);
2550b57cec5SDimitry Andric         i != iend;
2560b57cec5SDimitry Andric         ++i) {
2570b57cec5SDimitry Andric       hit_limit = setSubgraphColorHelper(*i, Color, visited, level+1, printed) || hit_limit;
2580b57cec5SDimitry Andric     }
2590b57cec5SDimitry Andric   }
2600b57cec5SDimitry Andric #else
2610b57cec5SDimitry Andric   errs() << "SelectionDAG::setSubgraphColor is only available in debug builds"
2620b57cec5SDimitry Andric          << " on systems with Graphviz or gv!\n";
2630b57cec5SDimitry Andric #endif
2640b57cec5SDimitry Andric   return hit_limit;
2650b57cec5SDimitry Andric }
2660b57cec5SDimitry Andric 
2670b57cec5SDimitry Andric /// setSubgraphColor - Convenience for setting subgraph color attribute.
2680b57cec5SDimitry Andric ///
2690b57cec5SDimitry Andric void SelectionDAG::setSubgraphColor(SDNode *N, const char *Color) {
2700b57cec5SDimitry Andric #ifndef NDEBUG
2710b57cec5SDimitry Andric   DenseSet<SDNode *> visited;
2720b57cec5SDimitry Andric   bool printed = false;
2730b57cec5SDimitry Andric   if (setSubgraphColorHelper(N, Color, visited, 0, printed)) {
2740b57cec5SDimitry Andric     // Visually mark that we hit the limit
2750b57cec5SDimitry Andric     if (strcmp(Color, "red") == 0) {
2760b57cec5SDimitry Andric       setSubgraphColorHelper(N, "blue", visited, 0, printed);
2770b57cec5SDimitry Andric     } else if (strcmp(Color, "yellow") == 0) {
2780b57cec5SDimitry Andric       setSubgraphColorHelper(N, "green", visited, 0, printed);
2790b57cec5SDimitry Andric     }
2800b57cec5SDimitry Andric   }
2810b57cec5SDimitry Andric 
2820b57cec5SDimitry Andric #else
2830b57cec5SDimitry Andric   errs() << "SelectionDAG::setSubgraphColor is only available in debug builds"
2840b57cec5SDimitry Andric          << " on systems with Graphviz or gv!\n";
2850b57cec5SDimitry Andric #endif
2860b57cec5SDimitry Andric }
2870b57cec5SDimitry Andric 
2880b57cec5SDimitry Andric std::string ScheduleDAGSDNodes::getGraphNodeLabel(const SUnit *SU) const {
2890b57cec5SDimitry Andric   std::string s;
2900b57cec5SDimitry Andric   raw_string_ostream O(s);
2910b57cec5SDimitry Andric   O << "SU(" << SU->NodeNum << "): ";
2920b57cec5SDimitry Andric   if (SU->getNode()) {
2930b57cec5SDimitry Andric     SmallVector<SDNode *, 4> GluedNodes;
2940b57cec5SDimitry Andric     for (SDNode *N = SU->getNode(); N; N = N->getGluedNode())
2950b57cec5SDimitry Andric       GluedNodes.push_back(N);
2960b57cec5SDimitry Andric     while (!GluedNodes.empty()) {
2970b57cec5SDimitry Andric       O << DOTGraphTraits<SelectionDAG*>
2980b57cec5SDimitry Andric         ::getSimpleNodeLabel(GluedNodes.back(), DAG);
2990b57cec5SDimitry Andric       GluedNodes.pop_back();
3000b57cec5SDimitry Andric       if (!GluedNodes.empty())
3010b57cec5SDimitry Andric         O << "\n    ";
3020b57cec5SDimitry Andric     }
3030b57cec5SDimitry Andric   } else {
3040b57cec5SDimitry Andric     O << "CROSS RC COPY";
3050b57cec5SDimitry Andric   }
3060b57cec5SDimitry Andric   return O.str();
3070b57cec5SDimitry Andric }
3080b57cec5SDimitry Andric 
3090b57cec5SDimitry Andric void ScheduleDAGSDNodes::getCustomGraphFeatures(GraphWriter<ScheduleDAG*> &GW) const {
3100b57cec5SDimitry Andric   if (DAG) {
3110b57cec5SDimitry Andric     // Draw a special "GraphRoot" node to indicate the root of the graph.
3120b57cec5SDimitry Andric     GW.emitSimpleNode(nullptr, "plaintext=circle", "GraphRoot");
3130b57cec5SDimitry Andric     const SDNode *N = DAG->getRoot().getNode();
3140b57cec5SDimitry Andric     if (N && N->getNodeId() != -1)
3150b57cec5SDimitry Andric       GW.emitEdge(nullptr, -1, &SUnits[N->getNodeId()], -1,
3160b57cec5SDimitry Andric                   "color=blue,style=dashed");
3170b57cec5SDimitry Andric   }
3180b57cec5SDimitry Andric }
319