20
20
// ===-------------------------------------------------------------------===//
21
21
22
22
#include " Relooper.h"
23
+ #include " WebAssembly.h"
23
24
24
- #include < llvm/ADT/STLExtras.h>
25
- #include < llvm/Support/raw_ostream.h>
26
- #include < llvm/Support/CommandLine.h>
25
+ #include " llvm/ADT/STLExtras.h"
26
+ #include " llvm/IR/CFG.h"
27
+ #include " llvm/IR/Function.h"
28
+ #include " llvm/Pass.h"
29
+ #include " llvm/Support/CommandLine.h"
30
+ #include " llvm/Support/Debug.h"
31
+ #include " llvm/Support/raw_ostream.h"
27
32
28
33
#include < cstring>
29
34
#include < cstdlib>
32
37
#include < stack>
33
38
#include < string>
34
39
40
+ #define DEBUG_TYPE " relooper"
41
+
35
42
using namespace llvm ;
43
+ using namespace Relooper ;
36
44
37
45
static cl::opt<int > RelooperSplittingFactor (
38
46
" relooper-splitting-factor" ,
@@ -52,15 +60,89 @@ static cl::opt<unsigned> RelooperNestingLimit(
52
60
" How much nesting is acceptable" ),
53
61
cl::init(20 ));
54
62
55
- namespace llvm {
56
63
57
- namespace Relooper {
64
+ namespace {
65
+ // /
66
+ // / Implements the relooper algorithm for a function's blocks.
67
+ // /
68
+ // / Implementation details: The Relooper instance has
69
+ // / ownership of the blocks and shapes, and frees them when done.
70
+ // /
71
+ struct RelooperAlgorithm {
72
+ std::deque<Block *> Blocks;
73
+ std::deque<Shape *> Shapes;
74
+ Shape *Root;
75
+ bool MinSize;
76
+ int BlockIdCounter;
77
+ int ShapeIdCounter;
78
+
79
+ RelooperAlgorithm ();
80
+ ~RelooperAlgorithm ();
81
+
82
+ void AddBlock (Block *New, int Id = -1 );
83
+
84
+ // Calculates the shapes
85
+ void Calculate (Block *Entry);
86
+
87
+ // Sets us to try to minimize size
88
+ void SetMinSize (bool MinSize_) { MinSize = MinSize_; }
89
+ };
90
+
91
+ struct RelooperAnalysis final : public FunctionPass {
92
+ static char ID;
93
+ RelooperAnalysis () : FunctionPass(ID) {}
94
+ const char *getPassName () const override { return " relooper" ; }
95
+ void getAnalysisUsage (AnalysisUsage &AU) const override {
96
+ AU.setPreservesAll ();
97
+ }
98
+ bool runOnFunction (Function &F) override ;
99
+ };
100
+ }
101
+
102
+ // RelooperAnalysis
103
+
104
+ char RelooperAnalysis::ID = 0 ;
105
+ FunctionPass *llvm::createWebAssemblyRelooper () {
106
+ return new RelooperAnalysis ();
107
+ }
108
+
109
+ bool RelooperAnalysis::runOnFunction (Function &F) {
110
+ DEBUG (dbgs () << " Relooping function '" << F.getName () << " '\n " );
111
+ RelooperAlgorithm R;
112
+ // FIXME: remove duplication between relooper's and LLVM's BBs.
113
+ std::map<const BasicBlock *, Block *> BB2B;
114
+ std::map<const Block *, const BasicBlock *> B2BB;
115
+ for (const BasicBlock &BB : F) {
116
+ // FIXME: getName is wrong here, Code is meant to represent amount of code.
117
+ // FIXME: use BranchVarInit for switch.
118
+ Block *B = new Block (BB.getName ().str ().data (), /* BranchVarInit=*/ nullptr );
119
+ R.AddBlock (B);
120
+ assert (BB2B.find (&BB) == BB2B.end () && " Inserting the same block twice" );
121
+ assert (B2BB.find (B) == B2BB.end () && " Inserting the same block twice" );
122
+ BB2B[&BB] = B;
123
+ B2BB[B] = &BB;
124
+ }
125
+ for (Block *B : R.Blocks ) {
126
+ const BasicBlock *BB = B2BB[B];
127
+ for (const BasicBlock *Successor : successors (BB))
128
+ // FIXME: add branch's Condition and Code below.
129
+ B->AddBranchTo (BB2B[Successor], /* Condition=*/ nullptr , /* Code=*/ nullptr );
130
+ }
131
+ R.Calculate (BB2B[&F.getEntryBlock ()]);
132
+ return false ; // Analysis passes don't modify anything.
133
+ }
134
+
135
+ // Helpers
136
+
137
+ typedef MapVector<Block *, BlockSet> BlockBlockSetMap;
138
+ typedef std::list<Block *> BlockList;
58
139
59
140
template <class T , class U >
60
141
static bool contains (const T &container, const U &contained) {
61
142
return container.count (contained);
62
143
}
63
144
145
+
64
146
// Branch
65
147
66
148
Branch::Branch (const char *ConditionInit, const char *CodeInit)
@@ -93,42 +175,39 @@ Block::~Block() {
93
175
94
176
void Block::AddBranchTo (Block *Target, const char *Condition,
95
177
const char *Code) {
96
- assert (
97
- !contains (BranchesOut,
98
- Target)); // cannot add more than one branch to the same target
178
+ assert (!contains (BranchesOut, Target) &&
179
+ " cannot add more than one branch to the same target" );
99
180
BranchesOut[Target] = make_unique<Branch>(Condition, Code);
100
181
}
101
182
102
183
// Relooper
103
184
104
- Relooper::Relooper ()
185
+ RelooperAlgorithm::RelooperAlgorithm ()
105
186
: Root(nullptr ), MinSize(false ), BlockIdCounter(1 ),
106
187
ShapeIdCounter(0 ) { // block ID 0 is reserved for clearings
107
188
}
108
189
109
- Relooper ::~Relooper () {
190
+ RelooperAlgorithm ::~RelooperAlgorithm () {
110
191
for (auto Curr : Blocks)
111
192
delete Curr;
112
193
for (auto Curr : Shapes)
113
194
delete Curr;
114
195
}
115
196
116
- void Relooper ::AddBlock (Block *New, int Id) {
197
+ void RelooperAlgorithm ::AddBlock (Block *New, int Id) {
117
198
New->Id = Id == -1 ? BlockIdCounter++ : Id;
118
199
Blocks.push_back (New);
119
200
}
120
201
121
202
struct RelooperRecursor {
122
- Relooper *Parent;
123
- RelooperRecursor (Relooper *ParentInit) : Parent(ParentInit) {}
203
+ RelooperAlgorithm *Parent;
204
+ RelooperRecursor (RelooperAlgorithm *ParentInit) : Parent(ParentInit) {}
124
205
};
125
206
126
- typedef std::list<Block *> BlockList;
127
-
128
- void Relooper::Calculate (Block *Entry) {
207
+ void RelooperAlgorithm::Calculate (Block *Entry) {
129
208
// Scan and optimize the input
130
209
struct PreOptimizer : public RelooperRecursor {
131
- PreOptimizer (Relooper *Parent) : RelooperRecursor(Parent) {}
210
+ PreOptimizer (RelooperAlgorithm *Parent) : RelooperRecursor(Parent) {}
132
211
BlockSet Live;
133
212
134
213
void FindLive (Block *Root) {
@@ -169,6 +248,7 @@ void Relooper::Calculate(Block *Entry) {
169
248
// amount, abort
170
249
// Split the node (for simplicity, we replace all the blocks, even
171
250
// though we could have reused the original)
251
+ DEBUG (dbgs () << " Splitting '" << Original->Code << " '\n " );
172
252
for (const auto &Prior : Original->BranchesIn ) {
173
253
Block *Split = new Block (Original->Code , Original->BranchVar );
174
254
Parent->AddBlock (Split, Original->Id );
@@ -216,7 +296,7 @@ void Relooper::Calculate(Block *Entry) {
216
296
// Recursively process the graph
217
297
218
298
struct Analyzer : public RelooperRecursor {
219
- Analyzer (Relooper *Parent) : RelooperRecursor(Parent) {}
299
+ Analyzer (RelooperAlgorithm *Parent) : RelooperRecursor(Parent) {}
220
300
221
301
// Add a shape to the list of shapes in this Relooper calculation
222
302
void Notice (Shape *New) {
@@ -236,6 +316,8 @@ void Relooper::Calculate(Block *Entry) {
236
316
// Converts/processes all branchings to a specific target
237
317
void Solipsize (Block *Target, Branch::FlowType Type, Shape *Ancestor,
238
318
BlockSet &From) {
319
+ DEBUG (dbgs () << " Solipsize '" << Target->Code << " ' type " << Type
320
+ << " \n " );
239
321
for (auto iter = Target->BranchesIn .begin ();
240
322
iter != Target->BranchesIn .end ();) {
241
323
Block *Prior = *iter;
@@ -258,6 +340,7 @@ void Relooper::Calculate(Block *Entry) {
258
340
}
259
341
260
342
Shape *MakeSimple (BlockSet &Blocks, Block *Inner, BlockSet &NextEntries) {
343
+ DEBUG (dbgs () << " MakeSimple inner block '" << Inner->Code << " '\n " );
261
344
SimpleShape *Simple = new SimpleShape;
262
345
Notice (Simple);
263
346
Simple->Inner = Inner;
@@ -649,10 +732,10 @@ void Relooper::Calculate(Block *Entry) {
649
732
// / Relooper post-optimizer
650
733
// /
651
734
struct PostOptimizer {
652
- Relooper *Parent;
735
+ RelooperAlgorithm *Parent;
653
736
std::stack<Shape *> LoopStack;
654
737
655
- PostOptimizer (Relooper *ParentInit) : Parent(ParentInit) {}
738
+ PostOptimizer (RelooperAlgorithm *ParentInit) : Parent(ParentInit) {}
656
739
657
740
void ShapeSwitch (Shape* var,
658
741
std::function<void (SimpleShape*)> simple,
@@ -900,7 +983,3 @@ void Relooper::Calculate(Block *Entry) {
900
983
901
984
PostOptimizer (this ).Process (Root);
902
985
}
903
-
904
- } // namespace Relooper
905
-
906
- } // namespace llvm
0 commit comments