summaryrefslogtreecommitdiff
path: root/src
diff options
context:
space:
mode:
authorAlon Zakai <alonzakai@gmail.com>2018-11-28 16:45:34 -0800
committerGitHub <noreply@github.com>2018-11-28 16:45:34 -0800
commit0b425638bdafe7be7ed914943de931b491881e23 (patch)
treeaaccb37a19e922bd29cd0ab5ec96075896b59f50 /src
parent2de7b22f2b245ab2cf4efffab500a88511972bd8 (diff)
downloadbinaryen-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.cpp25
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();