summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorAlon Zakai <alonzakai@gmail.com>2017-07-18 11:22:08 -0700
committerGitHub <noreply@github.com>2017-07-18 11:22:08 -0700
commit0b2122d49a28f199ef7fde247e7e7a14829fa96e (patch)
treedef77ee01192fd27fca6cec61fa82c4525873ddf
parent275c5ebafdf2860edb965d322bdad4c3e3717bea (diff)
parent2daf000f471afdcf21f784f83440bb4bdc1a9de9 (diff)
downloadbinaryen-0b2122d49a28f199ef7fde247e7e7a14829fa96e.tar.gz
binaryen-0b2122d49a28f199ef7fde247e7e7a14829fa96e.tar.bz2
binaryen-0b2122d49a28f199ef7fde247e7e7a14829fa96e.zip
Merge pull request #1100 from WebAssembly/fuzz-4
Yet more fuzz fixes
-rw-r--r--src/passes/OptimizeInstructions.cpp2
-rw-r--r--src/passes/RemoveUnusedBrs.cpp38
-rw-r--r--src/passes/Vacuum.cpp1
-rw-r--r--src/wasm-builder.h22
-rw-r--r--test/passes/optimize-instructions.txt13
-rw-r--r--test/passes/optimize-instructions.wast16
-rw-r--r--test/passes/remove-unused-brs.txt40
-rw-r--r--test/passes/remove-unused-brs.wast18
-rw-r--r--test/passes/remove-unused-names_remove-unused-brs_vacuum.txt18
-rw-r--r--test/passes/remove-unused-names_remove-unused-brs_vacuum.wast18
-rw-r--r--test/passes/vacuum.txt4
-rw-r--r--test/passes/vacuum.wast23
12 files changed, 180 insertions, 33 deletions
diff --git a/src/passes/OptimizeInstructions.cpp b/src/passes/OptimizeInstructions.cpp
index c6006d712..de6f96ccc 100644
--- a/src/passes/OptimizeInstructions.cpp
+++ b/src/passes/OptimizeInstructions.cpp
@@ -453,7 +453,7 @@ struct OptimizeInstructions : public WalkerPass<PostWalker<OptimizeInstructions,
if ((count > 0 && count < 32 - bits) || (constSignBit && count == 0)) {
// mixed or [zero upper const bits with sign bit set]; the compared values can never be identical, so
// force something definitely impossible even after zext
- assert(bits < 31);
+ assert(bits < 32);
c->value = Literal(int32_t(0x80000000));
// TODO: if no side effects, we can just replace it all with 1 or 0
} else {
diff --git a/src/passes/RemoveUnusedBrs.cpp b/src/passes/RemoveUnusedBrs.cpp
index b725de901..9fed3e4a5 100644
--- a/src/passes/RemoveUnusedBrs.cpp
+++ b/src/passes/RemoveUnusedBrs.cpp
@@ -226,17 +226,51 @@ struct RemoveUnusedBrs : public WalkerPass<PostWalker<RemoveUnusedBrs>> {
// we need the ifTrue to break, so it cannot reach the code we want to move
if (ExpressionAnalyzer::obviouslyDoesNotFlowOut(iff->ifTrue)) {
iff->ifFalse = builder.stealSlice(block, i + 1, list.size());
+ iff->finalize();
+ block->finalize();
return true;
}
} else {
// this is already an if-else. if one side is a dead end, we can append to the other, if
// there is no returned value to concern us
assert(!isConcreteWasmType(iff->type)); // can't be, since in the middle of a block
+
+ // ensures the first node is a block, if it isn't already, and merges in the second,
+ // either as a single element or, if a block, by appending to the first block. this
+ // keeps the order of operations in place, that is, the appended element will be
+ // executed after the first node's elements
+ auto blockifyMerge = [&](Expression* any, Expression* append) -> Block* {
+ Block* block = nullptr;
+ if (any) block = any->dynCast<Block>();
+ // if the first isn't a block, or it's a block with a name (so we might
+ // branch to the end, and so can't append to it, we might skip that code!)
+ // then make a new block
+ if (!block || block->name.is()) {
+ block = builder.makeBlock(any);
+ } else {
+ assert(!isConcreteWasmType(block->type));
+ }
+ auto* other = append->dynCast<Block>();
+ if (!other) {
+ block->list.push_back(append);
+ } else {
+ for (auto* item : other->list) {
+ block->list.push_back(item);
+ }
+ }
+ block->finalize();
+ return block;
+ };
+
if (ExpressionAnalyzer::obviouslyDoesNotFlowOut(iff->ifTrue)) {
- iff->ifFalse = builder.blockifyMerge(iff->ifFalse, builder.stealSlice(block, i + 1, list.size()));
+ iff->ifFalse = blockifyMerge(iff->ifFalse, builder.stealSlice(block, i + 1, list.size()));
+ iff->finalize();
+ block->finalize();
return true;
} else if (ExpressionAnalyzer::obviouslyDoesNotFlowOut(iff->ifFalse)) {
- iff->ifTrue = builder.blockifyMerge(iff->ifTrue, builder.stealSlice(block, i + 1, list.size()));
+ iff->ifTrue = blockifyMerge(iff->ifTrue, builder.stealSlice(block, i + 1, list.size()));
+ iff->finalize();
+ block->finalize();
return true;
}
}
diff --git a/src/passes/Vacuum.cpp b/src/passes/Vacuum.cpp
index 3f27fb36a..bb6c7c296 100644
--- a/src/passes/Vacuum.cpp
+++ b/src/passes/Vacuum.cpp
@@ -225,6 +225,7 @@ struct Vacuum : public WalkerPass<PostWalker<Vacuum>> {
child = curr->ifFalse;
typeUpdater.noteRecursiveRemoval(curr->ifTrue);
} else {
+ typeUpdater.noteRecursiveRemoval(curr);
ExpressionManipulator::nop(curr);
return;
}
diff --git a/src/wasm-builder.h b/src/wasm-builder.h
index 41aaf6766..f702342d2 100644
--- a/src/wasm-builder.h
+++ b/src/wasm-builder.h
@@ -315,28 +315,6 @@ public:
return block;
}
- // ensures the first node is a block, if it isn't already, and merges in the second,
- // either as a single element or, if a block, by appending to the first block
- Block* blockifyMerge(Expression* any, Expression* append) {
- Block* block = nullptr;
- if (any) block = any->dynCast<Block>();
- if (!block) {
- block = makeBlock(any);
- } else {
- assert(!isConcreteWasmType(block->type));
- }
- auto* other = append->dynCast<Block>();
- if (!other) {
- block->list.push_back(append);
- } else {
- for (auto* item : other->list) {
- block->list.push_back(item);
- }
- }
- block->finalize();
- return block;
- }
-
// a helper for the common pattern of a sequence of two expressions. Similar to
// blockify, but does *not* reuse a block if the first is one.
Block* makeSequence(Expression* left, Expression* right) {
diff --git a/test/passes/optimize-instructions.txt b/test/passes/optimize-instructions.txt
index 8b42fd2ad..604c9c3ea 100644
--- a/test/passes/optimize-instructions.txt
+++ b/test/passes/optimize-instructions.txt
@@ -1993,4 +1993,17 @@
)
)
)
+ (func $sign-ext-1-and-ne (type $2) (result i32)
+ (select
+ (i32.ne
+ (i32.and
+ (call $sign-ext-1-and-ne)
+ (i32.const 2147483647)
+ )
+ (i32.const -2147483648)
+ )
+ (i32.const 2)
+ (i32.const 1)
+ )
+ )
)
diff --git a/test/passes/optimize-instructions.wast b/test/passes/optimize-instructions.wast
index 6d1fd60ed..cf8275412 100644
--- a/test/passes/optimize-instructions.wast
+++ b/test/passes/optimize-instructions.wast
@@ -2417,4 +2417,20 @@
)
)
)
+ (func $sign-ext-1-and-ne (result i32)
+ (select
+ (i32.ne
+ (i32.const 1333788672)
+ (i32.shr_s
+ (i32.shl
+ (call $sign-ext-1-and-ne)
+ (i32.const 1)
+ )
+ (i32.const 1)
+ )
+ )
+ (i32.const 2)
+ (i32.const 1)
+ )
+ )
)
diff --git a/test/passes/remove-unused-brs.txt b/test/passes/remove-unused-brs.txt
index 07b4575ce..417bd3728 100644
--- a/test/passes/remove-unused-brs.txt
+++ b/test/passes/remove-unused-brs.txt
@@ -5,6 +5,7 @@
(type $3 (func (param i32 i32) (result i32)))
(type $4 (func (param i32 i32)))
(type $5 (func (param f32 i32 f32 i32 i32 f64 f32) (result i32)))
+ (type $6 (func (param i32) (result i64)))
(memory $0 256 256)
(func $b0-yes (type $0) (param $i1 i32)
(block $topmost
@@ -539,11 +540,13 @@
(block $out59
(if
(i32.const 0)
- (block $block61
- (drop
- (i32.const 1)
+ (block
+ (block $block61
+ (drop
+ (i32.const 1)
+ )
+ (call $loops)
)
- (call $loops)
(br $in58)
)
(nop)
@@ -584,11 +587,13 @@
(block $out69
(if
(i32.const 0)
- (block $block71
- (drop
- (i32.const 1)
+ (block
+ (block $block71
+ (drop
+ (i32.const 1)
+ )
+ (call $loops)
)
- (call $loops)
(drop
(i32.const 102)
)
@@ -1011,4 +1016,23 @@
(i32.const 1935947830)
)
)
+ (func $unexitable-loops-result (type $6) (param $0 i32) (result i64)
+ (loop $label$0
+ (loop $label$1
+ (br_if $label$0
+ (i32.load8_s
+ (i32.const 201460482)
+ )
+ )
+ (block
+ (block $label$3
+ (br_if $label$1
+ (get_local $0)
+ )
+ )
+ (br $label$1)
+ )
+ )
+ )
+ )
)
diff --git a/test/passes/remove-unused-brs.wast b/test/passes/remove-unused-brs.wast
index 04ebdac33..d694bd4d0 100644
--- a/test/passes/remove-unused-brs.wast
+++ b/test/passes/remove-unused-brs.wast
@@ -901,5 +901,23 @@
(i32.const 1935947830)
)
)
+ (func $unexitable-loops-result (param $0 i32) (result i64)
+ (loop $label$0
+ (loop $label$1
+ (if
+ (i32.load8_s
+ (i32.const 201460482)
+ )
+ (br $label$0)
+ (block $label$3
+ (br_if $label$3
+ (get_local $0)
+ )
+ )
+ )
+ (br $label$1)
+ )
+ )
+ )
)
diff --git a/test/passes/remove-unused-names_remove-unused-brs_vacuum.txt b/test/passes/remove-unused-names_remove-unused-brs_vacuum.txt
index d8ce15491..4eeb1ee97 100644
--- a/test/passes/remove-unused-names_remove-unused-brs_vacuum.txt
+++ b/test/passes/remove-unused-names_remove-unused-brs_vacuum.txt
@@ -22,6 +22,7 @@
(type $FUNCSIG$vii (func (param i32 i32)))
(type $FUNCSIG$iiii (func (param i32 i32 i32) (result i32)))
(type $FUNCSIG$id (func (param f64) (result i32)))
+ (type $23 (func (param i32) (result i64)))
(import "env" "DYNAMICTOP_PTR" (global $import$0 i32))
(import "env" "tempDoublePtr" (global $import$1 i32))
(import "env" "ABORT" (global $import$2 i32))
@@ -113,4 +114,21 @@
(func $23 (type $0) (param $0 i32) (param $1 i32) (param $2 i32) (result i32)
(unreachable)
)
+ (func $unexitable-loops-result (type $23) (param $0 i32) (result i64)
+ (loop $label$0
+ (loop $label$1
+ (br_if $label$0
+ (i32.load8_s
+ (i32.const 201460482)
+ )
+ )
+ (block
+ (br_if $label$1
+ (get_local $0)
+ )
+ (br $label$1)
+ )
+ )
+ )
+ )
)
diff --git a/test/passes/remove-unused-names_remove-unused-brs_vacuum.wast b/test/passes/remove-unused-names_remove-unused-brs_vacuum.wast
index 5a0ec5114..12f664a36 100644
--- a/test/passes/remove-unused-names_remove-unused-brs_vacuum.wast
+++ b/test/passes/remove-unused-names_remove-unused-brs_vacuum.wast
@@ -127,5 +127,23 @@
(func $23 (param i32) (param i32) (param i32) (result i32)
(unreachable)
)
+ (func $unexitable-loops-result (param $0 i32) (result i64)
+ (loop $label$0
+ (loop $label$1
+ (if
+ (i32.load8_s
+ (i32.const 201460482)
+ )
+ (br $label$0)
+ (block $label$3
+ (br_if $label$3
+ (get_local $0)
+ )
+ )
+ )
+ (br $label$1)
+ )
+ )
+ )
)
diff --git a/test/passes/vacuum.txt b/test/passes/vacuum.txt
index f3b06af1a..011fb99f5 100644
--- a/test/passes/vacuum.txt
+++ b/test/passes/vacuum.txt
@@ -301,4 +301,8 @@
)
)
)
+ (func $nop-if-type-changes (type $0)
+ (local $0 i32)
+ (nop)
+ )
)
diff --git a/test/passes/vacuum.wast b/test/passes/vacuum.wast
index db4bf6749..3b5b22e9c 100644
--- a/test/passes/vacuum.wast
+++ b/test/passes/vacuum.wast
@@ -637,4 +637,27 @@
(i32.const 1579493952)
)
)
+ (func $nop-if-type-changes (type $0)
+ (local $0 i32)
+ (block $label$0
+ (if
+ (i32.eqz
+ (get_local $0)
+ )
+ (block $label$1
+ (block
+ (if ;; we nop this if, which has a type change for block $label$1, no more brs to it
+ (i32.const 0)
+ (br_if $label$1
+ (i32.const 1717966400)
+ )
+ )
+ (drop
+ (br $label$0)
+ )
+ )
+ )
+ )
+ )
+ )
)