diff options
Diffstat (limited to 'src/passes/OptimizeInstructions.cpp')
-rw-r--r-- | src/passes/OptimizeInstructions.cpp | 22 |
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); |