diff options
author | Alon Zakai <azakai@google.com> | 2021-07-23 14:40:24 -0700 |
---|---|---|
committer | GitHub <noreply@github.com> | 2021-07-23 14:40:24 -0700 |
commit | 0b0054342af5c73de62ddf0603619292443c505c (patch) | |
tree | a8cbf83258546d83bc6e07294f88db10a305ccdb /src/passes/Precompute.cpp | |
parent | d0d6ea3c201c52118c12c218c97f44efd6252542 (diff) | |
download | binaryen-0b0054342af5c73de62ddf0603619292443c505c.tar.gz binaryen-0b0054342af5c73de62ddf0603619292443c505c.tar.bz2 binaryen-0b0054342af5c73de62ddf0603619292443c505c.zip |
[Wasm GC] Fix Heap2Local + non-nullable locals (#4017)
Given
(local $x (ref $foo))
(local.set $x ..)
(.. (local.get $x))
If we remove the local.set but not the get, then we end up with
(local $x (ref $foo))
(.. (local.get $x))
It looks like the local.get reads the initial value of a non-nullable local,
which is not allowed.
In practice, this would crash in precompute-propagate which would
try to propagate the initial value to the get. Add an assertion there with
a clear message, as until we have full validation of non-nullable locals
(and the spec for that is in flux), that pass is where bugs will end up
being noticed.
To fix this, replace the get as well. We can replace it with a null
for simplicity; it will never be used anyhow.
This also uncovered a small bug with reached not containing all
the things we reached - it was missing local.gets.
Diffstat (limited to 'src/passes/Precompute.cpp')
-rw-r--r-- | src/passes/Precompute.cpp | 6 |
1 files changed, 4 insertions, 2 deletions
diff --git a/src/passes/Precompute.cpp b/src/passes/Precompute.cpp index a832c9e85..44227a50f 100644 --- a/src/passes/Precompute.cpp +++ b/src/passes/Precompute.cpp @@ -350,8 +350,10 @@ private: Literals curr; if (set == nullptr) { if (getFunction()->isVar(get->index)) { - curr = - Literal::makeZeros(getFunction()->getLocalType(get->index)); + auto localType = getFunction()->getLocalType(get->index); + assert(!localType.isNonNullable() && + "Non-nullable locals must not use the default value"); + curr = Literal::makeZeros(localType); } else { // it's a param, so it's hopeless values = {}; |