summaryrefslogtreecommitdiff
path: root/src/passes/DeNaN.cpp
diff options
context:
space:
mode:
authorAlon Zakai <azakai@google.com>2020-05-27 09:18:36 -0700
committerGitHub <noreply@github.com>2020-05-27 09:18:36 -0700
commitc7f18b7fb34e2464f7a7beb31d8c8363e8597902 (patch)
tree71e6a0a4d7659f08c1644c22103ec66be4bfdfaa /src/passes/DeNaN.cpp
parent616463d1f33804f639d6c32ad284e7280a0a7b71 (diff)
downloadbinaryen-c7f18b7fb34e2464f7a7beb31d8c8363e8597902.tar.gz
binaryen-c7f18b7fb34e2464f7a7beb31d8c8363e8597902.tar.bz2
binaryen-c7f18b7fb34e2464f7a7beb31d8c8363e8597902.zip
DeNaN pass (#2877)
This moves the fuzzer de-NaN logic out into a separate pass. This is cleaner and also better since the old way would de-NaN once, but then the reducer could generate code with nans. The new way lets us de-NaN while reducing.
Diffstat (limited to 'src/passes/DeNaN.cpp')
-rw-r--r--src/passes/DeNaN.cpp96
1 files changed, 96 insertions, 0 deletions
diff --git a/src/passes/DeNaN.cpp b/src/passes/DeNaN.cpp
new file mode 100644
index 000000000..044c13c86
--- /dev/null
+++ b/src/passes/DeNaN.cpp
@@ -0,0 +1,96 @@
+/*
+ * Copyright 2020 WebAssembly Community Group participants
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+//
+// Instrument the wasm to convert NaN values at runtime into 0s. That is, every
+// operation that might produce a NaN will go through a helper function which
+// filters out NaNs (replacing them with 0). This ensures that NaNs are never
+// consumed by any instructions, which is useful when fuzzing between VMs that
+// differ on wasm's nondeterminism around NaNs.
+//
+
+#include "pass.h"
+#include "wasm-builder.h"
+#include "wasm.h"
+
+namespace wasm {
+
+struct DeNaN : public WalkerPass<
+ ControlFlowWalker<DeNaN, UnifiedExpressionVisitor<DeNaN>>> {
+ void visitExpression(Expression* expr) {
+ // If the expression returns a floating-point value, ensure it is not a
+ // NaN. If we can do this at compile time, do it now, which is useful for
+ // initializations of global (which we can't do a function call in).
+ Builder builder(*getModule());
+ Expression* replacement = nullptr;
+ auto* c = expr->dynCast<Const>();
+ if (expr->type == Type::f32) {
+ if (c && c->value.isNaN()) {
+ replacement = builder.makeConst(Literal(float(0)));
+ } else {
+ replacement = builder.makeCall("deNan32", {expr}, Type::f32);
+ }
+ } else if (expr->type == Type::f64) {
+ if (c && c->value.isNaN()) {
+ replacement = builder.makeConst(Literal(double(0)));
+ } else {
+ replacement = builder.makeCall("deNan64", {expr}, Type::f64);
+ }
+ }
+ if (replacement) {
+ // We can't do this outside of a function, like in a global initializer,
+ // where a call would be illegal.
+ if (replacement->is<Const>() || getFunction()) {
+ replaceCurrent(replacement);
+ } else {
+ std::cerr << "warning: cannot de-nan outside of function context\n";
+ }
+ }
+ }
+
+ void visitModule(Module* module) {
+ // Add helper functions.
+ Builder builder(*module);
+ auto add = [&](Name name, Type type, Literal literal, BinaryOp op) {
+ auto* func = new Function;
+ func->name = name;
+ func->sig = Signature(type, type);
+ // Compare the value to itself to check if it is a NaN, and return 0 if
+ // so:
+ //
+ // (if (result f*)
+ // (f*.eq
+ // (local.get $0)
+ // (local.get $0)
+ // )
+ // (local.get $0)
+ // (f*.const 0)
+ // )
+ func->body = builder.makeIf(
+ builder.makeBinary(
+ op, builder.makeLocalGet(0, type), builder.makeLocalGet(0, type)),
+ builder.makeLocalGet(0, type),
+ builder.makeConst(literal));
+ module->addFunction(func);
+ };
+ add("deNan32", Type::f32, Literal(float(0)), EqFloat32);
+ add("deNan64", Type::f64, Literal(double(0)), EqFloat64);
+ }
+};
+
+Pass* createDeNaNPass() { return new DeNaN(); }
+
+} // namespace wasm