diff options
author | Alon Zakai <alonzakai@gmail.com> | 2018-11-28 16:45:34 -0800 |
---|---|---|
committer | GitHub <noreply@github.com> | 2018-11-28 16:45:34 -0800 |
commit | 0b425638bdafe7be7ed914943de931b491881e23 (patch) | |
tree | aaccb37a19e922bd29cd0ab5ec96075896b59f50 /src | |
parent | 2de7b22f2b245ab2cf4efffab500a88511972bd8 (diff) | |
download | binaryen-0b425638bdafe7be7ed914943de931b491881e23.tar.gz binaryen-0b425638bdafe7be7ed914943de931b491881e23.tar.bz2 binaryen-0b425638bdafe7be7ed914943de931b491881e23.zip |
LocalCSE fuzz fix: invalidate the set operations too (#1778)
We invalidated based on effects of set values, but not of the sets themselves. Without that, a set could be overridden by something irrelevant and we thought we could still reuse the old value.
Before this PR, the testcase would have the last set's value be optimized into a get, incorrectly.
Diffstat (limited to 'src')
-rw-r--r-- | src/passes/LocalCSE.cpp | 25 |
1 files changed, 23 insertions, 2 deletions
diff --git a/src/passes/LocalCSE.cpp b/src/passes/LocalCSE.cpp index a458937db..12f0d9d93 100644 --- a/src/passes/LocalCSE.cpp +++ b/src/passes/LocalCSE.cpp @@ -92,14 +92,35 @@ struct LocalCSE : public WalkerPass<LinearExecutionWalker<LocalCSE>> { equivalences.clear(); } - void checkInvalidations(EffectAnalyzer& effects) { + // Checks invalidations due to a set of effects. Also optionally receive + // an expression that was just post-visited, and that also needs to be + // taken into account. + void checkInvalidations(EffectAnalyzer& effects, Expression* curr = nullptr) { // TODO: this is O(bad) std::vector<HashedExpression> invalidated; for (auto& sinkable : usables) { + // Check invalidations of the values we may want to use. if (effects.invalidates(sinkable.second.effects)) { invalidated.push_back(sinkable.first); } } + if (curr) { + // If we are a set, we have more to check: each of the usable + // values was from a set, and we did not consider the set in + // the loop above - just the values. So here we must check that + // sets do not interfere. (Note that due to flattening we + // have no risk of tees etc.) + if (auto* set = curr->dynCast<SetLocal>()) { + for (auto& sinkable : usables) { + // Check if the index is the same. Make sure to ignore + // our own value, which we may have just added! + if (sinkable.second.index == set->index && + sinkable.second.value != set->value) { + invalidated.push_back(sinkable.first); + } + } + } + } for (auto index : invalidated) { usables.erase(index); } @@ -129,7 +150,7 @@ struct LocalCSE : public WalkerPass<LinearExecutionWalker<LocalCSE>> { EffectAnalyzer effects(self->getPassOptions()); if (effects.checkPost(curr)) { - self->checkInvalidations(effects); + self->checkInvalidations(effects, curr); } self->expressionStack.pop_back(); |