summaryrefslogtreecommitdiff
path: root/src
diff options
context:
space:
mode:
authorAlon Zakai (kripken) <alonzakai@gmail.com>2017-05-20 11:55:30 -0700
committerAlon Zakai (kripken) <alonzakai@gmail.com>2017-05-20 15:08:47 -0700
commit1462c0dbfa9980481058e17d2aac317420e45acc (patch)
treeaf0b2b5bbad0ae233af464d43dde73d3b5f76f9c /src
parent9e0958982d2044949746c2d8290dbc0368546ebf (diff)
downloadbinaryen-1462c0dbfa9980481058e17d2aac317420e45acc.tar.gz
binaryen-1462c0dbfa9980481058e17d2aac317420e45acc.tar.bz2
binaryen-1462c0dbfa9980481058e17d2aac317420e45acc.zip
use TypeUpdater in vacuum
Diffstat (limited to 'src')
-rw-r--r--src/passes/Vacuum.cpp63
-rw-r--r--src/wasm-validator.h21
2 files changed, 48 insertions, 36 deletions
diff --git a/src/passes/Vacuum.cpp b/src/passes/Vacuum.cpp
index 1f820ec9d..4e47ad9a3 100644
--- a/src/passes/Vacuum.cpp
+++ b/src/passes/Vacuum.cpp
@@ -23,6 +23,7 @@
#include <ast_utils.h>
#include <wasm-builder.h>
#include <ast/block-utils.h>
+#include <ast/type-updating.h>
namespace wasm {
@@ -31,7 +32,20 @@ struct Vacuum : public WalkerPass<PostWalker<Vacuum>> {
Pass* create() override { return new Vacuum; }
- bool needRefinalize = false;
+ TypeUpdater typeUpdater;
+
+ Expression* replaceCurrent(Expression* expression) {
+ auto* old = getCurrent();
+ WalkerPass<PostWalker<Vacuum>>::replaceCurrent(expression);
+ // also update the type updater
+ typeUpdater.noteReplacement(old, expression);
+ return expression;
+ }
+
+ void doWalkFunction(Function* func) {
+ typeUpdater.walk(func->body);
+ walk(func->body);
+ }
// returns nullptr if curr is dead, curr if it must stay as is, or another node if it can be replaced
Expression* optimize(Expression* curr, bool resultUsed) {
@@ -143,40 +157,39 @@ struct Vacuum : public WalkerPass<PostWalker<Vacuum>> {
int skip = 0;
auto& list = curr->list;
size_t size = list.size();
- bool needResize = false;
for (size_t z = 0; z < size; z++) {
- auto* optimized = optimize(list[z], z == size - 1 && isConcreteWasmType(curr->type));
+ auto* child = list[z];
+ auto* optimized = optimize(child, z == size - 1 && isConcreteWasmType(curr->type));
if (!optimized) {
+ typeUpdater.noteRecursiveRemoval(child);
skip++;
- needResize = true;
} else {
- if (optimized != list[z]) {
+ if (optimized != child) {
+ typeUpdater.noteReplacement(child, optimized);
list[z] = optimized;
}
if (skip > 0) {
list[z - skip] = list[z];
+ list[z] = nullptr;
}
- // if this is an unconditional br, the rest is dead code
- Break* br = list[z - skip]->dynCast<Break>();
- Switch* sw = list[z - skip]->dynCast<Switch>();
- if ((br && !br->condition) || sw) {
- auto* last = list.back();
- list.resize(z - skip + 1);
- // if we removed the last one, and it was a return value, it must be returned
- if (list.back() != last && isConcreteWasmType(last->type)) {
- list.push_back(last);
+ // if this is unreachable, the rest is dead code
+ if (list[z - skip]->type == unreachable && z < size - 1) {
+ for (Index i = z - skip + 1; i < list.size(); i++) {
+ auto* remove = list[i];
+ if (remove) {
+ typeUpdater.noteRecursiveRemoval(remove);
+ }
}
- needResize = false;
- needRefinalize = true;
+ list.resize(z - skip + 1);
+ typeUpdater.maybeUpdateTypeToUnreachable(curr);
+ skip = 0; // nothing more to do on the list
break;
}
}
}
- if (needResize) {
+ if (skip > 0) {
list.resize(size - skip);
- // resizing means we drop elements, which may include breaks, which may
- // render blocks unreachable now
- needRefinalize = true;
+ typeUpdater.maybeUpdateTypeToUnreachable(curr);
}
// the block may now be a trivial one that we can get rid of and just leave its contents
replaceCurrent(BlockUtils::simplifyToContents(curr, this));
@@ -198,13 +211,6 @@ struct Vacuum : public WalkerPass<PostWalker<Vacuum>> {
}
}
replaceCurrent(child);
- if (curr->type != child->type) {
- // e.g., if (1) unreachable is none => unreachable
- // or if i32 (1) unreachable else 10 is i32 => unreachable
- // in which cases we must update our parents.
- // we must do this now, so that our parents see valid data
- ReFinalize().walk(getFunction()->body);
- }
return;
}
if (curr->ifFalse) {
@@ -307,9 +313,6 @@ struct Vacuum : public WalkerPass<PostWalker<Vacuum>> {
}
void visitFunction(Function* curr) {
- if (needRefinalize) {
- ReFinalize().walk(curr->body);
- }
auto* optimized = optimize(curr->body, curr->result != none);
if (optimized) {
curr->body = optimized;
diff --git a/src/wasm-validator.h b/src/wasm-validator.h
index 699e5910a..aa404099e 100644
--- a/src/wasm-validator.h
+++ b/src/wasm-validator.h
@@ -408,17 +408,26 @@ public:
switch (curr->op) {
case ClzInt32:
case CtzInt32:
- case PopcntInt32:
+ case PopcntInt32: {
+ shouldBeEqual(curr->value->type, i32, curr, "i32 unary value type must be correct");
+ break;
+ }
+ case ClzInt64:
+ case CtzInt64:
+ case PopcntInt64: {
+ shouldBeEqual(curr->value->type, i64, curr, "i64 unary value type must be correct");
+ break;
+ }
case NegFloat32:
case AbsFloat32:
case CeilFloat32:
case FloorFloat32:
case TruncFloat32:
case NearestFloat32:
- case SqrtFloat32:
- case ClzInt64:
- case CtzInt64:
- case PopcntInt64:
+ case SqrtFloat32: {
+ shouldBeEqual(curr->value->type, f32, curr, "f32 unary value type must be correct");
+ break;
+ }
case NegFloat64:
case AbsFloat64:
case CeilFloat64:
@@ -426,7 +435,7 @@ public:
case TruncFloat64:
case NearestFloat64:
case SqrtFloat64: {
- shouldBeEqual(curr->value->type, curr->type, curr, "non-conversion unaries must return the same type");
+ shouldBeEqual(curr->value->type, f64, curr, "f64 unary value type must be correct");
break;
}
case EqZInt32: {