1 //===- RegionPrinter.cpp - Print regions tree pass ------------------------===//
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
7 //===----------------------------------------------------------------------===//
8 // Print out the region tree of a function using dotty/graphviz.
9 //===----------------------------------------------------------------------===//
11 #include "llvm/Analysis/RegionPrinter.h"
12 #include "llvm/ADT/DepthFirstIterator.h"
13 #include "llvm/ADT/PostOrderIterator.h"
14 #include "llvm/ADT/Statistic.h"
15 #include "llvm/Analysis/DOTGraphTraitsPass.h"
16 #include "llvm/Analysis/Passes.h"
17 #include "llvm/Analysis/RegionInfo.h"
18 #include "llvm/Analysis/RegionIterator.h"
19 #include "llvm/Support/CommandLine.h"
20 #include "llvm/Support/Debug.h"
21 #include "llvm/Support/raw_ostream.h"
23 #include "llvm/IR/LegacyPassManager.h"
28 //===----------------------------------------------------------------------===//
29 /// onlySimpleRegion - Show only the simple regions in the RegionViewer.
31 onlySimpleRegions("only-simple-regions",
32 cl::desc("Show only simple regions in the graphviz viewer"),
38 struct DOTGraphTraits
<RegionNode
*> : public DefaultDOTGraphTraits
{
40 DOTGraphTraits (bool isSimple
=false)
41 : DefaultDOTGraphTraits(isSimple
) {}
43 std::string
getNodeLabel(RegionNode
*Node
, RegionNode
*Graph
) {
45 if (!Node
->isSubRegion()) {
46 BasicBlock
*BB
= Node
->getNodeAs
<BasicBlock
>();
49 return DOTGraphTraits
<const Function
*>
50 ::getSimpleNodeLabel(BB
, BB
->getParent());
52 return DOTGraphTraits
<const Function
*>
53 ::getCompleteNodeLabel(BB
, BB
->getParent());
56 return "Not implemented";
61 struct DOTGraphTraits
<RegionInfo
*> : public DOTGraphTraits
<RegionNode
*> {
63 DOTGraphTraits (bool isSimple
= false)
64 : DOTGraphTraits
<RegionNode
*>(isSimple
) {}
66 static std::string
getGraphName(const RegionInfo
*) { return "Region Graph"; }
68 std::string
getNodeLabel(RegionNode
*Node
, RegionInfo
*G
) {
69 return DOTGraphTraits
<RegionNode
*>::getNodeLabel(
70 Node
, reinterpret_cast<RegionNode
*>(G
->getTopLevelRegion()));
73 std::string
getEdgeAttributes(RegionNode
*srcNode
,
74 GraphTraits
<RegionInfo
*>::ChildIteratorType CI
,
76 RegionNode
*destNode
= *CI
;
78 if (srcNode
->isSubRegion() || destNode
->isSubRegion())
81 // In case of a backedge, do not use it to define the layout of the nodes.
82 BasicBlock
*srcBB
= srcNode
->getNodeAs
<BasicBlock
>();
83 BasicBlock
*destBB
= destNode
->getNodeAs
<BasicBlock
>();
85 Region
*R
= G
->getRegionFor(destBB
);
87 while (R
&& R
->getParent())
88 if (R
->getParent()->getEntry() == destBB
)
93 if (R
&& R
->getEntry() == destBB
&& R
->contains(srcBB
))
94 return "constraint=false";
99 // Print the cluster of the subregions. This groups the single basic blocks
100 // and adds a different background color for each group.
101 static void printRegionCluster(const Region
&R
, GraphWriter
<RegionInfo
*> &GW
,
102 unsigned depth
= 0) {
103 raw_ostream
&O
= GW
.getOStream();
104 O
.indent(2 * depth
) << "subgraph cluster_" << static_cast<const void*>(&R
)
106 O
.indent(2 * (depth
+ 1)) << "label = \"\";\n";
108 if (!onlySimpleRegions
|| R
.isSimple()) {
109 O
.indent(2 * (depth
+ 1)) << "style = filled;\n";
110 O
.indent(2 * (depth
+ 1)) << "color = "
111 << ((R
.getDepth() * 2 % 12) + 1) << "\n";
114 O
.indent(2 * (depth
+ 1)) << "style = solid;\n";
115 O
.indent(2 * (depth
+ 1)) << "color = "
116 << ((R
.getDepth() * 2 % 12) + 2) << "\n";
119 for (const auto &RI
: R
)
120 printRegionCluster(*RI
, GW
, depth
+ 1);
122 const RegionInfo
&RI
= *static_cast<const RegionInfo
*>(R
.getRegionInfo());
124 for (auto *BB
: R
.blocks())
125 if (RI
.getRegionFor(BB
) == &R
)
126 O
.indent(2 * (depth
+ 1)) << "Node"
127 << static_cast<const void*>(RI
.getTopLevelRegion()->getBBNode(BB
))
130 O
.indent(2 * depth
) << "}\n";
133 static void addCustomGraphFeatures(const RegionInfo
*G
,
134 GraphWriter
<RegionInfo
*> &GW
) {
135 raw_ostream
&O
= GW
.getOStream();
136 O
<< "\tcolorscheme = \"paired12\"\n";
137 printRegionCluster(*G
->getTopLevelRegion(), GW
, 4);
140 } //end namespace llvm
144 struct RegionInfoPassGraphTraits
{
145 static RegionInfo
*getGraph(RegionInfoPass
*RIP
) {
146 return &RIP
->getRegionInfo();
151 : public DOTGraphTraitsPrinter
<RegionInfoPass
, false, RegionInfo
*,
152 RegionInfoPassGraphTraits
> {
155 : DOTGraphTraitsPrinter
<RegionInfoPass
, false, RegionInfo
*,
156 RegionInfoPassGraphTraits
>("reg", ID
) {
157 initializeRegionPrinterPass(*PassRegistry::getPassRegistry());
160 char RegionPrinter::ID
= 0;
162 struct RegionOnlyPrinter
163 : public DOTGraphTraitsPrinter
<RegionInfoPass
, true, RegionInfo
*,
164 RegionInfoPassGraphTraits
> {
167 : DOTGraphTraitsPrinter
<RegionInfoPass
, true, RegionInfo
*,
168 RegionInfoPassGraphTraits
>("reg", ID
) {
169 initializeRegionOnlyPrinterPass(*PassRegistry::getPassRegistry());
172 char RegionOnlyPrinter::ID
= 0;
175 : public DOTGraphTraitsViewer
<RegionInfoPass
, false, RegionInfo
*,
176 RegionInfoPassGraphTraits
> {
179 : DOTGraphTraitsViewer
<RegionInfoPass
, false, RegionInfo
*,
180 RegionInfoPassGraphTraits
>("reg", ID
) {
181 initializeRegionViewerPass(*PassRegistry::getPassRegistry());
184 char RegionViewer::ID
= 0;
186 struct RegionOnlyViewer
187 : public DOTGraphTraitsViewer
<RegionInfoPass
, true, RegionInfo
*,
188 RegionInfoPassGraphTraits
> {
191 : DOTGraphTraitsViewer
<RegionInfoPass
, true, RegionInfo
*,
192 RegionInfoPassGraphTraits
>("regonly", ID
) {
193 initializeRegionOnlyViewerPass(*PassRegistry::getPassRegistry());
196 char RegionOnlyViewer::ID
= 0;
198 } //end anonymous namespace
200 INITIALIZE_PASS(RegionPrinter
, "dot-regions",
201 "Print regions of function to 'dot' file", true, true)
204 RegionOnlyPrinter
, "dot-regions-only",
205 "Print regions of function to 'dot' file (with no function bodies)", true,
208 INITIALIZE_PASS(RegionViewer
, "view-regions", "View regions of function",
211 INITIALIZE_PASS(RegionOnlyViewer
, "view-regions-only",
212 "View regions of function (with no function bodies)",
215 FunctionPass
*llvm::createRegionPrinterPass() { return new RegionPrinter(); }
217 FunctionPass
*llvm::createRegionOnlyPrinterPass() {
218 return new RegionOnlyPrinter();
221 FunctionPass
* llvm::createRegionViewerPass() {
222 return new RegionViewer();
225 FunctionPass
* llvm::createRegionOnlyViewerPass() {
226 return new RegionOnlyViewer();
230 static void viewRegionInfo(RegionInfo
*RI
, bool ShortNames
) {
231 assert(RI
&& "Argument must be non-null");
233 llvm::Function
*F
= RI
->getTopLevelRegion()->getEntry()->getParent();
234 std::string GraphName
= DOTGraphTraits
<RegionInfo
*>::getGraphName(RI
);
236 llvm::ViewGraph(RI
, "reg", ShortNames
,
237 Twine(GraphName
) + " for '" + F
->getName() + "' function");
240 static void invokeFunctionPass(const Function
*F
, FunctionPass
*ViewerPass
) {
241 assert(F
&& "Argument must be non-null");
242 assert(!F
->isDeclaration() && "Function must have an implementation");
244 // The viewer and analysis passes do not modify anything, so we can safely
245 // remove the const qualifier
246 auto NonConstF
= const_cast<Function
*>(F
);
248 llvm::legacy::FunctionPassManager
FPM(NonConstF
->getParent());
250 FPM
.doInitialization();
252 FPM
.doFinalization();
255 void llvm::viewRegion(RegionInfo
*RI
) { viewRegionInfo(RI
, false); }
257 void llvm::viewRegion(const Function
*F
) {
258 invokeFunctionPass(F
, createRegionViewerPass());
261 void llvm::viewRegionOnly(RegionInfo
*RI
) { viewRegionInfo(RI
, true); }
263 void llvm::viewRegionOnly(const Function
*F
) {
264 invokeFunctionPass(F
, createRegionOnlyViewerPass());