1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
|
/*
* Copyright 2017 WebAssembly Community Group participants
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#include <iterator>
#include <wasm-builder.h>
#include <wasm-printing.h>
#include <ir/find_all.h>
#include <ir/local-graph.h>
#include <cfg/cfg-traversal.h>
namespace wasm {
namespace LocalGraphInternal {
// Information about a basic block.
struct Info {
std::vector<Expression*> actions; // actions occurring in this block: local.gets and local.sets
std::unordered_map<Index, SetLocal*> lastSets; // for each index, the last local.set for it
};
// flow helper class. flows the gets to their sets
struct Flower : public CFGWalker<Flower, Visitor<Flower>, Info> {
LocalGraph::GetSetses& getSetses;
LocalGraph::Locations& locations;
Flower(LocalGraph::GetSetses& getSetses, LocalGraph::Locations& locations, Function* func) : getSetses(getSetses), locations(locations) {
setFunction(func);
// create the CFG by walking the IR
CFGWalker<Flower, Visitor<Flower>, Info>::doWalkFunction(func);
// flow gets across blocks
flow(func);
}
BasicBlock* makeBasicBlock() {
return new BasicBlock();
}
// cfg traversal work
static void doVisitGetLocal(Flower* self, Expression** currp) {
auto* curr = (*currp)->cast<GetLocal>();
// if in unreachable code, skip
if (!self->currBasicBlock) return;
self->currBasicBlock->contents.actions.emplace_back(curr);
self->locations[curr] = currp;
}
static void doVisitSetLocal(Flower* self, Expression** currp) {
auto* curr = (*currp)->cast<SetLocal>();
// if in unreachable code, skip
if (!self->currBasicBlock) return;
self->currBasicBlock->contents.actions.emplace_back(curr);
self->currBasicBlock->contents.lastSets[curr->index] = curr;
self->locations[curr] = currp;
}
void flow(Function* func) {
// This block struct is optimized for this flow process (Minimal information, iteration index).
struct FlowBlock {
// Last Traversed Iteration: This value helps us to find if this block has been seen while traversing blocks.
// We compare this value to the current iteration index in order to determine if we already process this block in the current iteration.
// This speeds up the processing compared to unordered_set or other struct usage. (No need to reset internal values, lookup into container, ...)
size_t lastTraversedIteration;
std::vector<Expression*> actions;
std::vector<FlowBlock*> in;
// Sor each index, the last local.set for it
// The unordered_map from BasicBlock.Info is converted into a vector
// This speeds up search as there are usually few sets in a block, so just scanning
// them linearly is efficient, avoiding hash computations (while in Info,
// it's convenient to have a map so we can assign them easily, where
// the last one seen overwrites the previous; and, we do that O(1)).
std::vector<std::pair<Index, SetLocal*>> lastSets;
};
auto numLocals = func->getNumLocals();
std::vector<std::vector<GetLocal*>> allGets;
allGets.resize(numLocals);
std::vector<FlowBlock*> work;
// Convert input blocks (basicBlocks) into more efficient flow blocks to improve memory access.
std::vector<FlowBlock> flowBlocks;
flowBlocks.resize(basicBlocks.size());
// Init mapping between basicblocks and flowBlocks
std::unordered_map<BasicBlock*, FlowBlock*> basicToFlowMap;
for (Index i = 0; i < basicBlocks.size(); ++i) {
basicToFlowMap[basicBlocks[i].get()] = &flowBlocks[i];
}
const size_t NULL_ITERATION = -1;
FlowBlock* entryFlowBlock = nullptr;
for (Index i = 0; i < flowBlocks.size(); ++i) {
auto& block = basicBlocks[i];
auto& flowBlock = flowBlocks[i];
// Get the equivalent block to entry in the flow list
if (block.get() == entry) entryFlowBlock = &flowBlock;
flowBlock.lastTraversedIteration = NULL_ITERATION;
flowBlock.actions.swap(block->contents.actions);
// Map in block to flow blocks
auto& in = block->in;
flowBlock.in.resize(in.size());
std::transform(in.begin(), in.end(), flowBlock.in.begin(), [&](BasicBlock* block) {
return basicToFlowMap[block];
});
// Convert unordered_map to vector.
flowBlock.lastSets.reserve(block->contents.lastSets.size());
for (auto set : block->contents.lastSets) {
flowBlock.lastSets.emplace_back(std::make_pair(set.first, set.second));
}
}
assert(entryFlowBlock != nullptr);
size_t currentIteration = 0;
for (auto& block : flowBlocks) {
#ifdef LOCAL_GRAPH_DEBUG
std::cout << "basic block " << block.get() << " :\n";
for (auto& action : block->contents.actions) {
std::cout << " action: " << *action << '\n';
}
for (auto* lastSet : block->contents.lastSets) {
std::cout << " last set " << lastSet << '\n';
}
#endif
// go through the block, finding each get and adding it to its index,
// and seeing how sets affect that
auto& actions = block.actions;
// move towards the front, handling things as we go
for (int i = int(actions.size()) - 1; i >= 0; i--) {
auto* action = actions[i];
if (auto* get = action->dynCast<GetLocal>()) {
allGets[get->index].push_back(get);
} else {
// This set is the only set for all those gets.
auto* set = action->cast<SetLocal>();
auto& gets = allGets[set->index];
for (auto* get : gets) {
getSetses[get].insert(set);
}
gets.clear();
}
}
// If anything is left, we must flow it back through other blocks. we
// can do that for all gets as a whole, they will get the same results.
for (Index index = 0; index < numLocals; index++) {
auto& gets = allGets[index];
if (gets.empty()) continue;
work.push_back(&block);
// Note that we may need to revisit the later parts of this initial
// block, if we are in a loop, so don't mark it as seen.
while (!work.empty()) {
auto* curr = work.back();
work.pop_back();
// We have gone through this block; now we must handle flowing to
// the inputs.
if (curr->in.empty()) {
if (curr == entryFlowBlock) {
// These receive a param or zero init value.
for (auto* get : gets) {
getSetses[get].insert(nullptr);
}
}
} else {
for (auto* pred : curr->in) {
if (pred->lastTraversedIteration == currentIteration) {
// We've already seen pred in this iteration.
continue;
}
pred->lastTraversedIteration = currentIteration;
auto lastSet = std::find_if(pred->lastSets.begin(), pred->lastSets.end(), [&](std::pair<Index, SetLocal*>& value) {
return value.first == index;
});
if (lastSet != pred->lastSets.end()) {
// There is a set here, apply it, and stop the flow.
for (auto* get : gets) {
getSetses[get].insert(lastSet->second);
}
} else {
// Keep on flowing.
work.push_back(pred);
}
}
}
}
gets.clear();
currentIteration++;
}
}
}
};
} // namespace LocalGraphInternal
// LocalGraph implementation
LocalGraph::LocalGraph(Function* func) {
LocalGraphInternal::Flower flower(getSetses, locations, func);
#ifdef LOCAL_GRAPH_DEBUG
std::cout << "LocalGraph::dump\n";
for (auto& pair : getSetses) {
auto* get = pair.first;
auto& sets = pair.second;
std::cout << "GET\n" << get << " is influenced by\n";
for (auto* set : sets) {
std::cout << set << '\n';
}
}
std::cout << "total locations: " << locations.size() << '\n';
#endif
}
void LocalGraph::computeInfluences() {
for (auto& pair : locations) {
auto* curr = pair.first;
if (auto* set = curr->dynCast<SetLocal>()) {
FindAll<GetLocal> findAll(set->value);
for (auto* get : findAll.list) {
getInfluences[get].insert(set);
}
} else {
auto* get = curr->cast<GetLocal>();
for (auto* set : getSetses[get]) {
setInfluences[set].insert(get);
}
}
}
}
void LocalGraph::computeSSAIndexes() {
std::unordered_map<Index, std::set<SetLocal*>> indexSets;
for (auto& pair : getSetses) {
auto* get = pair.first;
auto& sets = pair.second;
for (auto* set : sets) {
indexSets[get->index].insert(set);
}
}
for (auto& pair : locations) {
auto* curr = pair.first;
if (auto* set = curr->dynCast<SetLocal>()) {
auto& sets = indexSets[set->index];
if (sets.size() == 1 && *sets.begin() != curr) {
// While it has just one set, it is not the right one (us),
// so mark it invalid.
sets.clear();
}
}
}
for (auto& pair : indexSets) {
auto index = pair.first;
auto& sets = pair.second;
if (sets.size() == 1) {
SSAIndexes.insert(index);
}
}
}
bool LocalGraph::isSSA(Index x) {
return SSAIndexes.count(x);
}
} // namespace wasm
|