diff options
author | Jérôme Vouillon <jerome.vouillon@gmail.com> | 2023-09-21 20:26:53 +0200 |
---|---|---|
committer | GitHub <noreply@github.com> | 2023-09-21 11:26:53 -0700 |
commit | a35af4b1e9461b12fd7993b4fd249bd39d73945d (patch) | |
tree | 5643c36c5dac5acead49a93c9a2007c3a99e0b91 /src/passes/Heap2Local.cpp | |
parent | 199811942d5f88d1d54158c9d7d5d5026cb1accb (diff) | |
download | binaryen-a35af4b1e9461b12fd7993b4fd249bd39d73945d.tar.gz binaryen-a35af4b1e9461b12fd7993b4fd249bd39d73945d.tar.bz2 binaryen-a35af4b1e9461b12fd7993b4fd249bd39d73945d.zip |
Make heap2local work through casts (#5952)
E.g.
(local $x (ref eq)
...
(local.set $x
(struct.new $float
...
)
)
(struct.get $float 0
(ref.cast (ref $float)
(local.get $x)
)
)
This PR allows us to use heap2local, ignoring the passing cast.
This is similar to existing handling of ref.as_non_null.
Diffstat (limited to 'src/passes/Heap2Local.cpp')
-rw-r--r-- | src/passes/Heap2Local.cpp | 28 |
1 files changed, 26 insertions, 2 deletions
diff --git a/src/passes/Heap2Local.cpp b/src/passes/Heap2Local.cpp index 84037e0dd..e027f5a46 100644 --- a/src/passes/Heap2Local.cpp +++ b/src/passes/Heap2Local.cpp @@ -430,6 +430,18 @@ struct Heap2LocalOptimizer { replaceCurrent(curr->value); } + void visitRefCast(RefCast* curr) { + if (!reached.count(curr)) { + return; + } + + // It is safe to optimize out this RefCast, since we proved it + // contains our allocation and we have checked that the type of + // the allocation is a subtype of the type of the cast, and so + // cannot trap. + replaceCurrent(curr->ref); + } + void visitStructSet(StructSet* curr) { if (!reached.count(curr)) { return; @@ -526,7 +538,7 @@ struct Heap2LocalOptimizer { return; } - switch (getParentChildInteraction(parent, child)) { + switch (getParentChildInteraction(allocation, parent, child)) { case ParentChildInteraction::Escapes: { // If the parent may let us escape then we are done. return; @@ -586,7 +598,8 @@ struct Heap2LocalOptimizer { rewriter.applyOptimization(); } - ParentChildInteraction getParentChildInteraction(Expression* parent, + ParentChildInteraction getParentChildInteraction(StructNew* allocation, + Expression* parent, Expression* child) { // If there is no parent then we are the body of the function, and that // means we escape by flowing to the caller. @@ -595,6 +608,7 @@ struct Heap2LocalOptimizer { } struct Checker : public Visitor<Checker> { + StructNew* allocation; Expression* child; // Assume escaping (or some other problem we cannot analyze) unless we are @@ -647,6 +661,15 @@ struct Heap2LocalOptimizer { } } + void visitRefCast(RefCast* curr) { + // As it is our allocation that flows through here, we need to + // check that the cast will not trap, so that we can continue + // to (hopefully) optimize this allocation. + if (Type::isSubType(allocation->type, curr->type)) { + escapes = false; + } + } + // GC operations. void visitStructSet(StructSet* curr) { // The reference does not escape (but the value is stored to memory and @@ -664,6 +687,7 @@ struct Heap2LocalOptimizer { // TODO Array and I31 operations } checker; + checker.allocation = allocation; checker.child = child; checker.visit(parent); |