diff options
author | Alex Crichton <alex@alexcrichton.com> | 2018-05-29 18:19:19 -0500 |
---|---|---|
committer | Alon Zakai <alonzakai@gmail.com> | 2018-05-29 16:19:19 -0700 |
commit | 706b3f6c6cc95d688a5f97e9d82e15df6d04d54d (patch) | |
tree | 00de50f981c1f500ab324c9beb0c51741674ad80 /src/emscripten-optimizer | |
parent | 91b90b76ea870cf96b2853b54cbe42ff46e387bf (diff) | |
download | binaryen-706b3f6c6cc95d688a5f97e9d82e15df6d04d54d.tar.gz binaryen-706b3f6c6cc95d688a5f97e9d82e15df6d04d54d.tar.bz2 binaryen-706b3f6c6cc95d688a5f97e9d82e15df6d04d54d.zip |
wasm2asm: Fix and enable a large number of spec tests (#1558)
* Import `abort` from the environment
* Add passing spec tests
* Bind the abort function
* wasm2asm: Fix name collisions
Currently function names and local names can collide in namespaces, causing
buggy results when a function intends to call another function but ends up using
a local value as the target!
This fix was required to enable the `fac` spec test
* wasm2asm: Get multiple modules in one file working
The spec tests seem to have multiple modules defined in some tests and the
invocations all use the most recently defined module. This commit updates the
`--allow-asserts` mode of wasm2asm to work with this mode of tests, enabling us
to enable more spec tests for wasm2asm.
* wasm2asm: Enable the float_literals spec test
This needed to be modified to account for how JS engines don't work with NaN
bits the same way, but it's otherwise largely the same test. Additionally it
turns out that asm.js doesn't accept either `Infinity` or `NaN` ambient globals
so they needed to get imported through the `global` variable rather than defined
as literals in code
* wasm2asm: Fix function pointer invocations
This commit fixes invocations of functions through function pointers as
previously the table names on lookup and definition were mismatched. Both tables
now go through signature-based namification rather than athe name of the type
itself.
Overall this enables a slew of spec tests
* wasm2asm: Enable the left-to-right spec test
There were two small bugs in the order of evaluation of operators with
wasm2asm. The `select` instruction would sometimes evaluate the condition first
when it was supposed to be last. Similarly a `call_indirect` instruction would
evaluate the function pointer first when it was supposed to be evaluated last.
The `select` instruction case was a relatively small fix but the one for
`call_indirect` was a bit more pessimized to generate some temporaries.
Hopefully if this becomes up a problem it can be tightened up.
* wasm2asm: Fix signed load promotions of 64-bit ints
This commit enables the `endianness` spec test which revealed a bug in 64-bit
loads from smaller sizes which were signed. Previously the upper bits of the
64-bit number were all set to zero but the fix was for signed loads to have all
the upper bits match the highest bit of the low 32 bits that we load.
* wasm2asm: Enable the `stack` spec test
Internally the spec test uses a mixture of the s-expression syntax and the wat
syntax, so this is copied over into the `wasm2asm` folder after going through
`wat2wasm` to ensure it's consistent for binaryen.
* wasm2asm: Fix unaligned loads/stores of floats
Replace these operations in `RemoveNonJSOps` by using reinterpretation to
translate floats to integers and then use the existing code for unaligned
loads/stores of integers.
* wasm2asm: Fix a tricky grow_memory codegen bug
This commit fixes a tricky codegen bug found in the `grow_memory` instruction.
Specifically if you stored the result of `grow_memory` immediately into memory
it would look like:
HEAP32[..] = __wasm_grow_memory(..);
Here though it looks like JS evaluates the destination *before* the grow
function is called, but the grow function will invalidate the destination!
Furthermore this is actually generalizable to all function calls:
HEAP32[..] = foo(..);
Because any function could transitively call `grow_memory`. This commit fixes
the issue by ensuring that store instructions are always considered statements,
unconditionally evaluating the value into a temporary and then storing that into
the destination. While a bit of a pessmimization for now it should hopefully fix
the bug here.
* wasm2asm: Handle offsets in tables
This commit fixes initializing tables whose elements have an initial offset.
This should hopefully help fix some more Rust code which has all function
pointers offset by default!
* Update tests
* Tweak * location on types
* Rename entries of NameScope and document fromName
* Comment on lowercase names
* Update compiled JS
* Update js test output expectation
* Rename NameScope::Global to NameScope::Top
* Switch to `enum class`
* Switch to `Fatal()`
* Add TODO for when asm.js is no longer generated
Diffstat (limited to 'src/emscripten-optimizer')
-rw-r--r-- | src/emscripten-optimizer/simple_ast.h | 20 |
1 files changed, 14 insertions, 6 deletions
diff --git a/src/emscripten-optimizer/simple_ast.h b/src/emscripten-optimizer/simple_ast.h index 64aab7708..237d9c1b5 100644 --- a/src/emscripten-optimizer/simple_ast.h +++ b/src/emscripten-optimizer/simple_ast.h @@ -836,17 +836,25 @@ struct JSPrinter { } static char* numToString(double d, bool finalize=true) { + // If this number is NaN or infinite then things are a bit tricky. In JS we + // want to eventually use `NaN` and/or `Infinity`, but neither of those + // identifiers are valid in asm.js. Instead we have to explicitly import + // `NaN` and `Infinity` from the global environment, and those names are + // bound locally in an asm function as `nan` and `infinity`. + // + // TODO: the JS names of `NaN` and `Infinity` should be used once literal + // asm.js code isn't generated any more if (std::isnan(d)) { if (std::signbit(d)) { - return (char*) "-NaN"; + return (char*) "-nan"; } else { - return (char*) "NaN"; + return (char*) "nan"; } } else if (!std::isfinite(d)) { if (std::signbit(d)) { - return (char*) "-Infinity"; + return (char*) "-infinity"; } else { - return (char*) "Infinity"; + return (char*) "infinity"; } } bool neg = d < 0; @@ -1059,8 +1067,8 @@ struct JSPrinter { ensure(1); // we temporarily append a 0 char *curr = buffer + last; // ensure might invalidate buffer[used] = 0; - if (strstr(curr, "Infinity")) return; - if (strstr(curr, "NaN")) return; + if (strstr(curr, "infinity")) return; + if (strstr(curr, "nan")) return; if (strchr(curr, '.')) return; // already a decimal point, all good char *e = strchr(curr, 'e'); if (!e) { |