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
|
/*
* Copyright 2016 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.
*/
#ifndef wasm_ir_localizer_h
#define wasm_ir_localizer_h
#include "ir/iteration.h"
#include "wasm-builder.h"
namespace wasm {
// Make an expression available in a local. If already in one, just
// use that local, otherwise use a new local.
//
// Note that if the local is reused, this assumes it is not modified in between
// the set and the get, which the caller must ensure.
struct Localizer {
Index index;
Expression* expr;
Localizer(Expression* input, Function* func, Module* wasm) {
expr = input;
if (auto* get = expr->dynCast<LocalGet>()) {
index = get->index;
} else if (auto* set = expr->dynCast<LocalSet>()) {
index = set->index;
} else {
index = Builder::addVar(func, expr->type);
expr = Builder(*wasm).makeLocalTee(index, expr, expr->type);
}
}
};
// Replaces all children with gets of locals, if they have any effects that
// interact with any of the others, or if they have side effects which cannot be
// removed. Also replace unreachable things with an unreachable, leaving in
// place only things without interacting effects. For example:
//
// (parent
// (call $foo)
// (br $out)
// (i32.const)
// )
//
// =>
//
// (local.set $temp.foo
// (call $foo) ;; moved out
// )
// (br $out) ;; moved out
// (parent
// (local.get $temp.foo) ;; value saved to a local
// (unreachable) ;; complex effect replaced by unreachable
// (i32.const)
// )
//
// After this it is safe to reorder and remove things from the parent: all
// interesting interactions happen before the parent.
//
// Typical usage is to call getReplacement() will produces the entire output
// just shown (i.e., possible initial local.sets and other stuff that was pulled
// out, followed by the parent, as relevant). Note that getReplacement() may
// omit the parent, if it had an unreachable child. That is useful behavior in
// that it removes unneeded code (& otherwise some users of this code would need
// to write their own removal logic). However, that does imply that it is valid
// to remove the parent in such cases, which is not so for e.g. br when it is
// the last thing keeping a block reachable. Calling this with something like a
// struct.new or a call (the current intended users) is valid; if we want to
// generalize this fully then we need to make changes here.
//
// TODO: use in more places
struct ChildLocalizer {
ChildLocalizer(Expression* parent,
Function* func,
Module& wasm,
const PassOptions& options)
: parent(parent), wasm(wasm) {
Builder builder(wasm);
ChildIterator iterator(parent);
auto& children = iterator.children;
auto num = children.size();
// Compute the effects of all children.
std::vector<EffectAnalyzer> effects;
for (Index i = 0; i < num; i++) {
// The children are in reverse order in ChildIterator, but we want to
// process them in the normal order.
auto* child = *children[num - 1 - i];
effects.emplace_back(options, wasm, child);
}
// Go through the children and move to locals those that we need to.
for (Index i = 0; i < num; i++) {
auto** childp = children[num - 1 - i];
auto* child = *childp;
if (child->type == Type::unreachable) {
// Move the child out, and put an unreachable in its place (note that we
// don't need an actual set here, as there is no value to set to a
// local).
sets.push_back(child);
*childp = builder.makeUnreachable();
hasUnreachableChild = true;
continue;
}
if (hasUnreachableChild) {
// Once we pass one unreachable, we only need to copy the children over.
// (The only reason we still need them is that they may be needed for
// validation, e.g. if one contains a break to a block that is the only
// reason the block has type none.)
sets.push_back(builder.makeDrop(child));
*childp = builder.makeUnreachable();
continue;
}
// Use a local if we need to. That is the case either if this has side
// effects we can't remove, or if it interacts with other children.
bool needLocal = effects[i].hasUnremovableSideEffects();
if (!needLocal) {
// TODO: Avoid quadratic time here by accumulating effects and checking
// vs the accumulation.
for (Index j = 0; j < num; j++) {
if (j != i && effects[i].invalidates(effects[j])) {
needLocal = true;
break;
}
}
}
if (needLocal) {
auto local = builder.addVar(func, child->type);
sets.push_back(builder.makeLocalSet(local, child));
*childp = builder.makeLocalGet(local, child->type);
}
}
}
// Helper that gets a replacement for the parent: a block containing the
// sets + the parent. This will not contain the parent if we don't need it
// (if it was never reached).
Expression* getReplacement() {
if (sets.empty()) {
// Nothing to add.
return parent;
}
auto* block = getChildrenReplacement();
if (!hasUnreachableChild) {
block->list.push_back(parent);
block->finalize();
}
return block;
}
// Like `getReplacement`, but the result never contains the parent.
Block* getChildrenReplacement() {
auto* block = Builder(wasm).makeBlock();
block->list.set(sets);
if (hasUnreachableChild) {
block->type = Type::unreachable;
}
return block;
}
private:
Expression* parent;
Module& wasm;
std::vector<Expression*> sets;
bool hasUnreachableChild = false;
};
} // namespace wasm
#endif // wasm_ir_localizer_h
|