summaryrefslogtreecommitdiff
path: root/src/passes/OptimizeInstructions.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'src/passes/OptimizeInstructions.cpp')
-rw-r--r--src/passes/OptimizeInstructions.cpp22
1 files changed, 22 insertions, 0 deletions
diff --git a/src/passes/OptimizeInstructions.cpp b/src/passes/OptimizeInstructions.cpp
index 1a474be2c..81d0aef76 100644
--- a/src/passes/OptimizeInstructions.cpp
+++ b/src/passes/OptimizeInstructions.cpp
@@ -2501,11 +2501,33 @@ private:
// Ignore extraneous things and compare them syntactically. We can also
// look at the full fallthrough for both sides now.
left = Properties::getFallthrough(left, passOptions, *getModule());
+ auto* originalRight = right;
right = Properties::getFallthrough(right, passOptions, *getModule());
if (!ExpressionAnalyzer::equal(left, right)) {
return false;
}
+ // We must also not have non-fallthrough effects that invalidate us, such as
+ // this situation:
+ //
+ // (local.get $x)
+ // (block
+ // (local.set $x ..)
+ // (local.get $x)
+ // )
+ //
+ // The fallthroughs are identical, but the set may cause us to read a
+ // different value.
+ if (originalRight != right) {
+ // TODO: We could be more precise here and ignore right itself in
+ // originalRightEffects.
+ auto originalRightEffects = effects(originalRight);
+ auto rightEffects = effects(right);
+ if (originalRightEffects.invalidates(rightEffects)) {
+ return false;
+ }
+ }
+
// To be equal, they must also be known to return the same result
// deterministically.
return !Properties::isGenerative(left);