summaryrefslogtreecommitdiff
path: root/src/emscripten-optimizer
diff options
context:
space:
mode:
authorAlex Crichton <alex@alexcrichton.com>2018-05-29 18:19:19 -0500
committerAlon Zakai <alonzakai@gmail.com>2018-05-29 16:19:19 -0700
commit706b3f6c6cc95d688a5f97e9d82e15df6d04d54d (patch)
tree00de50f981c1f500ab324c9beb0c51741674ad80 /src/emscripten-optimizer
parent91b90b76ea870cf96b2853b54cbe42ff46e387bf (diff)
downloadbinaryen-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.h20
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) {