summaryrefslogtreecommitdiff
path: root/src/ir/flat.h
diff options
context:
space:
mode:
authorAlon Zakai <azakai@google.com>2020-05-27 13:30:12 -0700
committerGitHub <noreply@github.com>2020-05-27 13:30:12 -0700
commitdfe473e6af0a31cad7a7b26f5dead358d9bbf536 (patch)
tree6dddc00894823055802c1ed4f137473840f0f2a7 /src/ir/flat.h
parentc7f18b7fb34e2464f7a7beb31d8c8363e8597902 (diff)
downloadbinaryen-dfe473e6af0a31cad7a7b26f5dead358d9bbf536.tar.gz
binaryen-dfe473e6af0a31cad7a7b26f5dead358d9bbf536.tar.bz2
binaryen-dfe473e6af0a31cad7a7b26f5dead358d9bbf536.zip
Flat IR: local.set's value should not be a control flow (#2589)
Diffstat (limited to 'src/ir/flat.h')
-rw-r--r--src/ir/flat.h14
1 files changed, 10 insertions, 4 deletions
diff --git a/src/ir/flat.h b/src/ir/flat.h
index d54d4771f..706ca5a86 100644
--- a/src/ir/flat.h
+++ b/src/ir/flat.h
@@ -45,11 +45,14 @@
// 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
+// 2. Disallow control flow (block, loop, if, and try) 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.
+// 4. local.set cannot have an operand that is control flow (control flow with
+// values is prohibited already, but e.g. a block ending in unreachable,
+// which can normally be nested, is also disallowed).
//
#ifndef wasm_ir_flat_h
@@ -72,8 +75,11 @@ inline void verifyFlatness(Function* func) {
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 if (auto* set = curr->dynCast<LocalSet>()) {
+ verify(!set->isTee() || set->type == Type::unreachable,
+ "tees are not allowed, only sets");
+ verify(!Properties::isControlFlowStructure(set->value),
+ "set values cannot be control flow");
} else {
for (auto* child : ChildIterator(curr)) {
verify(Properties::isConstantExpression(child) ||