diff options
author | Alon Zakai <alonzakai@gmail.com> | 2019-04-16 10:00:19 -0700 |
---|---|---|
committer | GitHub <noreply@github.com> | 2019-04-16 10:00:19 -0700 |
commit | 324238cc44e51c65637d29a938c435248d384154 (patch) | |
tree | fcd7a35442c720385ff5746a3de83a7123e04be8 /src/ir/flat.h | |
parent | cb2d63586c08a3dd194d2b733ceb3f5051c081f8 (diff) | |
download | binaryen-324238cc44e51c65637d29a938c435248d384154.tar.gz binaryen-324238cc44e51c65637d29a938c435248d384154.tar.bz2 binaryen-324238cc44e51c65637d29a938c435248d384154.zip |
Verify flat IR where it is expected, and give a nice error (#2009)
Fixes #2007 #2008
Diffstat (limited to 'src/ir/flat.h')
-rw-r--r-- | src/ir/flat.h | 120 |
1 files changed, 120 insertions, 0 deletions
diff --git a/src/ir/flat.h b/src/ir/flat.h new file mode 100644 index 000000000..ed6178ef5 --- /dev/null +++ b/src/ir/flat.h @@ -0,0 +1,120 @@ +/* + * 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 "wasm-traversal.h" +#include "ir/iteration.h" + +namespace wasm { + +namespace Flat { + +inline bool isControlFlowStructure(Expression* curr) { + return curr->is<Block>() || curr->is<If>() || curr->is<Loop>(); +} + +inline void verifyFlatness(Function* func) { + struct VerifyFlatness : public PostWalker<VerifyFlatness, UnifiedExpressionVisitor<VerifyFlatness>> { + void visitExpression(Expression* curr) { + if (isControlFlowStructure(curr)) { + verify(!isConcreteType(curr->type), "control flow structures must not flow values"); + } else if (curr->is<SetLocal>()) { + verify(!isConcreteType(curr->type), "tees are not allowed, only sets"); + } else { + for (auto* child : ChildIterator(curr)) { + verify(child->is<Const>() || child->is<GetLocal>() || child->is<Unreachable>(), + "instructions must only have const, 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(!isConcreteType(func->body->type), "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); + runner.setIsNested(true); + runner.add<VerifyFlatness>(); + runner.run(); +} + +} // namespace Fkat + +} // namespace wasm + +#endif // wasm_ir_flat_h |