summaryrefslogtreecommitdiff
path: root/src
diff options
context:
space:
mode:
authorAlon Zakai <alonzakai@gmail.com>2016-05-24 12:02:06 -0700
committerAlon Zakai <alonzakai@gmail.com>2016-05-24 15:50:26 -0700
commitf1384f6c98765de6ac9777ae44661c1a713a6e11 (patch)
tree6bf190afe424af9a7bd566db397ab7aaf45b8d89 /src
parentab05c684e570951c39e081cc15fa659296445186 (diff)
downloadbinaryen-f1384f6c98765de6ac9777ae44661c1a713a6e11.tar.gz
binaryen-f1384f6c98765de6ac9777ae44661c1a713a6e11.tar.bz2
binaryen-f1384f6c98765de6ac9777ae44661c1a713a6e11.zip
move blocks outside in merge-blocks so that they can be merged later
Diffstat (limited to 'src')
-rw-r--r--src/passes/MergeBlocks.cpp124
1 files changed, 124 insertions, 0 deletions
diff --git a/src/passes/MergeBlocks.cpp b/src/passes/MergeBlocks.cpp
index d26817e18..5794e1379 100644
--- a/src/passes/MergeBlocks.cpp
+++ b/src/passes/MergeBlocks.cpp
@@ -17,9 +17,53 @@
//
// Merges blocks to their parents.
//
+// We also restructure blocks in order to enable such merging. For
+// example,
+//
+// (i32.store
+// (block
+// (call $foo)
+// (i32.load (i32.const 100))
+// )
+// (i32.const 0)
+// )
+//
+// can be transformed into
+//
+// (block
+// (call $foo)
+// (i32.store
+// (block
+// (i32.load (i32.const 100))
+// )
+// (i32.const 0)
+// )
+// )
+//
+// after which the internal block can go away, and
+// the new external block might be mergeable. This is always
+// worth it if the internal block ends up with 1 item.
+// For the second operand,
+//
+// (i32.store
+// (i32.const 100)
+// (block
+// (call $foo)
+// (i32.load (i32.const 200))
+// )
+// )
+//
+// The order of operations requires that the first execute
+// before. We can do the same operation, but only if the
+// first has no side effects, or the code we are moving out
+// has no side effects.
+// If we can do this to both operands, we can generate a
+// single outside block.
+//
#include <wasm.h>
#include <pass.h>
+#include <ast_utils.h>
namespace wasm {
@@ -50,6 +94,86 @@ struct MergeBlocks : public WalkerPass<PostWalker<MergeBlocks, Visitor<MergeBloc
}
}
}
+
+ Block* optimize(Expression* curr, Expression*& child, Block* outer = nullptr, Expression** dependency1 = nullptr, Expression** dependency2 = nullptr) {
+ if (!child) return outer;
+ if (dependency1 && *dependency1 && EffectAnalyzer(*dependency1).hasSideEffects()) return outer;
+ if (dependency2 && *dependency2 && EffectAnalyzer(*dependency2).hasSideEffects()) return outer;
+ if (auto* block = child->dynCast<Block>()) {
+ if (block->list.size() >= 2) {
+ child = block->list.back();
+ if (outer == nullptr) {
+ // reuse the block, move it out
+ block->list.back() = curr;
+ block->finalize(); // last block element was our input, and is now our output, which may differ TODO optimize
+ replaceCurrent(block);
+ return block;
+ } else {
+ // append to an existing outer block
+ assert(outer->list.back() == curr);
+ outer->list.pop_back();
+ for (Index i = 0; i < block->list.size() - 1; i++) {
+ outer->list.push_back(block->list[i]);
+ }
+ outer->list.push_back(curr);
+ }
+ }
+ }
+ return outer;
+ }
+
+ void visitUnary(Unary* curr) {
+ optimize(curr, curr->value);
+ }
+ void visitSetLocal(SetLocal* curr) {
+ optimize(curr, curr->value);
+ }
+ void visitLoad(Load* curr) {
+ optimize(curr, curr->ptr);
+ }
+ void visitReturn(Return* curr) {
+ optimize(curr, curr->value);
+ }
+
+ void visitBinary(Binary* curr) {
+ optimize(curr, curr->right, optimize(curr, curr->left), &curr->left);
+ }
+ void visitStore(Store* curr) {
+ optimize(curr, curr->value, optimize(curr, curr->ptr), &curr->ptr);
+ }
+
+ void visitSelect(Select* curr) {
+ optimize(curr, curr->condition, optimize(curr, curr->ifFalse, optimize(curr, curr->ifTrue), &curr->ifTrue), &curr->ifTrue, &curr->ifFalse);
+ }
+
+ void visitBreak(Break* curr) {
+ optimize(curr, curr->condition, optimize(curr, curr->value), &curr->value);
+ }
+ void visitSwitch(Switch* curr) {
+ optimize(curr, curr->condition, optimize(curr, curr->value), &curr->value);
+ }
+
+ template<typename T>
+ void handleCall(T* curr, Block* outer = nullptr) {
+ for (Index i = 0; i < curr->operands.size(); i++) {
+ outer = optimize(curr, curr->operands[i], outer);
+ if (EffectAnalyzer(curr->operands[i]).hasSideEffects()) return;
+ }
+ }
+
+ void visitCall(Call* curr) {
+ handleCall(curr);
+ }
+
+ void visitCallImport(CallImport* curr) {
+ handleCall(curr);
+ }
+
+ void visitCallIndirect(CallIndirect* curr) {
+ auto* outer = optimize(curr, curr->target);
+ if (EffectAnalyzer(curr->target).hasSideEffects()) return;
+ handleCall(curr, outer);
+ }
};
static RegisterPass<MergeBlocks> registerPass("merge-blocks", "merges blocks to their parents");