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
|
/*
* Copyright 2019 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.
*/
//
// Flattens code, removing nesting.e.g. an if return value would be
// converted to a local
//
// (i32.add
// (if (..condition..)
// (..if true..)
// (..if false..)
// )
// (i32.const 1)
// )
// =>
// (if (..condition..)
// (local.set $temp
// (..if true..)
// )
// (local.set $temp
// (..if false..)
// )
// )
// (i32.add
// (local.get $temp)
// (i32.const 1)
// )
//
// Formally, this pass flattens in the precise sense of
// making the AST have these properties:
//
// 1. Aside from a local.set, the operands of an instruction must be a
// local.get, a const, or an unreachable. Anything else is written
// to a local earlier.
// 2. Disallow block, loop, and if return values, and do not allow the
// function body to have a concrete type, i.e., do not use
// control flow to pass around values.
// 3. Disallow local.tee, setting a local is always done in a local.set
// on a non-nested-expression location.
//
#ifndef wasm_ir_flat_h
#define wasm_ir_flat_h
#include "ir/iteration.h"
#include "ir/properties.h"
#include "pass.h"
#include "wasm-traversal.h"
namespace wasm {
namespace Flat {
inline void verifyFlatness(Function* func) {
struct VerifyFlatness
: public PostWalker<VerifyFlatness,
UnifiedExpressionVisitor<VerifyFlatness>> {
void visitExpression(Expression* curr) {
if (Properties::isControlFlowStructure(curr)) {
verify(!curr->type.isConcrete(),
"control flow structures must not flow values");
} else if (curr->is<LocalSet>()) {
verify(!curr->type.isConcrete(), "tees are not allowed, only sets");
} else {
for (auto* child : ChildIterator(curr)) {
verify(Properties::isConstantExpression(child) ||
child->is<LocalGet>() || child->is<Unreachable>(),
"instructions must only have constant expressions, local.get, "
"or unreachable as children");
}
}
}
void verify(bool condition, const char* message) {
if (!condition) {
Fatal() << "IR must be flat: run --flatten beforehand (" << message
<< ", in " << getFunction()->name << ')';
}
}
};
VerifyFlatness verifier;
verifier.walkFunction(func);
verifier.setFunction(func);
verifier.verify(!func->body->type.isConcrete(),
"function bodies must not flow values");
}
inline void verifyFlatness(Module* module) {
struct VerifyFlatness
: public WalkerPass<
PostWalker<VerifyFlatness, UnifiedExpressionVisitor<VerifyFlatness>>> {
bool isFunctionParallel() override { return true; }
VerifyFlatness* create() override { return new VerifyFlatness(); }
void doVisitFunction(Function* func) { verifyFlatness(func); }
};
PassRunner runner(module);
VerifyFlatness().run(&runner, module);
}
} // namespace Flat
} // namespace wasm
#endif // wasm_ir_flat_h
|