summaryrefslogtreecommitdiff
path: root/src/ir/linear-execution.h
diff options
context:
space:
mode:
authorAlon Zakai <azakai@google.com>2023-08-08 11:10:33 -0700
committerGitHub <noreply@github.com>2023-08-08 11:10:33 -0700
commitf1e92f99867b646aebd8a4f9b35c4972301e6469 (patch)
treeaaac08ab6c8b824c3e86086fae92a75ce89360f7 /src/ir/linear-execution.h
parentd26d2146c3f58116d4ae7751e3bbfffeb1455412 (diff)
downloadbinaryen-f1e92f99867b646aebd8a4f9b35c4972301e6469.tar.gz
binaryen-f1e92f99867b646aebd8a4f9b35c4972301e6469.tar.bz2
binaryen-f1e92f99867b646aebd8a4f9b35c4972301e6469.zip
LinearExecutionTraversal: Add an option to connect adjacent code, use in SimplifyLocals (#5860)
This addresses most of the minor regression from the correctness fix in #5857. That PR makes us consider calls as branching, but in some cases it is ok to ignore that branching (see the comment in the code here), which this PR allows as an option. This undoes one test change from that PR, showing it undoes the regression for SimplifyLocals. More tests are added to cover this specifically as well.
Diffstat (limited to 'src/ir/linear-execution.h')
-rw-r--r--src/ir/linear-execution.h47
1 files changed, 42 insertions, 5 deletions
diff --git a/src/ir/linear-execution.h b/src/ir/linear-execution.h
index cee039b37..b7020f7fb 100644
--- a/src/ir/linear-execution.h
+++ b/src/ir/linear-execution.h
@@ -44,14 +44,49 @@ struct LinearExecutionWalker : public PostWalker<SubType, VisitorType> {
self->noteNonLinear(*currp);
}
+ // Optionally, we can connect adjacent basic blocks. "Adjacent" here means
+ // that the first branches to the second, and that there is no other code in
+ // between them. As a result, the first dominates the second, but it might not
+ // reach it.
+ //
+ // For example, a call may branch if exceptions are enabled, but if this
+ // option is flipped on then we will *not* call doNoteNonLinear on the call:
+ //
+ // ..A..
+ // call();
+ // ..B..
+ //
+ // As a result, we'd consider A and B to be together. Another example is an
+ // if:
+ //
+ // ..A..
+ // if
+ // ..B..
+ // else
+ // ..C..
+ // end
+ //
+ // Here we will connect A and B, but *not* A and C (there is code in between)
+ // or B and C (they do not branch to each other).
+ //
+ // As the if case shows, this can be useful for cases where we want to look at
+ // dominated blocks with their dominator, but it only handles the trivial
+ // adjacent cases of such dominance. Passes should generally uses a full CFG
+ // and dominator tree for this, but this option does help some very common
+ // cases (calls, if without an else) and it has very low overhead (we still
+ // only do a simple postorder walk on the IR, no CFG is constructed, etc.).
+ bool connectAdjacentBlocks = false;
+
static void scan(SubType* self, Expression** currp) {
Expression* curr = *currp;
auto handleCall = [&](bool isReturn) {
- // Control is nonlinear if we return, or if EH is enabled or may be.
- if (isReturn || !self->getModule() ||
- self->getModule()->features.hasExceptionHandling()) {
- self->pushTask(SubType::doNoteNonLinear, currp);
+ if (!self->connectAdjacentBlocks) {
+ // Control is nonlinear if we return, or if EH is enabled or may be.
+ if (isReturn || !self->getModule() ||
+ self->getModule()->features.hasExceptionHandling()) {
+ self->pushTask(SubType::doNoteNonLinear, currp);
+ }
}
// Scan the children normally.
@@ -78,7 +113,9 @@ struct LinearExecutionWalker : public PostWalker<SubType, VisitorType> {
self->maybePushTask(SubType::scan, &curr->cast<If>()->ifFalse);
self->pushTask(SubType::doNoteNonLinear, currp);
self->pushTask(SubType::scan, &curr->cast<If>()->ifTrue);
- self->pushTask(SubType::doNoteNonLinear, currp);
+ if (!self->connectAdjacentBlocks) {
+ self->pushTask(SubType::doNoteNonLinear, currp);
+ }
self->pushTask(SubType::scan, &curr->cast<If>()->condition);
break;
}