summaryrefslogtreecommitdiff
path: root/src/ir/ordering.h
diff options
context:
space:
mode:
authorAlon Zakai <azakai@google.com>2022-09-06 13:56:50 -0700
committerGitHub <noreply@github.com>2022-09-06 13:56:50 -0700
commit16f72aae58b6c856d405d6d608efd436549ea678 (patch)
treed2b55e6cef5128719f1b736e67a96891a97749f9 /src/ir/ordering.h
parentbd5422c8f13265cf89bdf49f30a04b882ee91d0e (diff)
downloadbinaryen-16f72aae58b6c856d405d6d608efd436549ea678.tar.gz
binaryen-16f72aae58b6c856d405d6d608efd436549ea678.tar.bz2
binaryen-16f72aae58b6c856d405d6d608efd436549ea678.zip
[Wasm GC] Fix GlobalTypeOptimization fuzz bug on replacing unreachable struct.set (#5021)
We replaced an unreachable struct.set with something reachable, which can break validation in corner cases.
Diffstat (limited to 'src/ir/ordering.h')
-rw-r--r--src/ir/ordering.h14
1 files changed, 14 insertions, 0 deletions
diff --git a/src/ir/ordering.h b/src/ir/ordering.h
index ed2c00ee2..bc8d69055 100644
--- a/src/ir/ordering.h
+++ b/src/ir/ordering.h
@@ -34,6 +34,11 @@ namespace wasm {
//
// (temp = first, second, temp)
//
+// The first expression is assumed to not be unreachable (otherwise, there is no
+// value to get the result of). If the second is unreachable, this returns
+// something with type unreachable (that avoids returning something with a
+// concrete type, which might replace something with unreachable type - we want
+// to keep the type the same, in most cases).
inline Expression* getResultOfFirst(Expression* first,
Expression* second,
Function* func,
@@ -43,6 +48,15 @@ inline Expression* getResultOfFirst(Expression* first,
Builder builder(*wasm);
+ if (second->type == Type::unreachable) {
+ // No value is actually consumed here. Emit something with unreachable type.
+ // (Note that if we continued to the canReorder code after us, and emitted
+ // second followed by first, then the block would have a concrete type due
+ // to the last element having such a type - which would not have unreachable
+ // type.)
+ return builder.makeSequence(builder.makeDrop(first), second);
+ }
+
if (EffectAnalyzer::canReorder(passOptions, *wasm, first, second)) {
return builder.makeSequence(second, first);
}