summaryrefslogtreecommitdiff
path: root/src
diff options
context:
space:
mode:
authorMax Graey <maxgraey@gmail.com>2021-07-22 00:10:48 +0300
committerGitHub <noreply@github.com>2021-07-21 14:10:48 -0700
commited97c68294a8d95c57972eae22906cc0bea1aae8 (patch)
tree3b90c42bc96153b137a683e0edc6cf529b453f55 /src
parent1e79c53ec1ad3d80db3354f15919331fdfa7ed28 (diff)
downloadbinaryen-ed97c68294a8d95c57972eae22906cc0bea1aae8.tar.gz
binaryen-ed97c68294a8d95c57972eae22906cc0bea1aae8.tar.bz2
binaryen-ed97c68294a8d95c57972eae22906cc0bea1aae8.zip
[Optimize Instructions] Combine reinterprets, loads and stores (#4006)
Fixes #3973 Loads: f32.reinterpret_i32(i32.load(x)) => f32.load(x) f64.reinterpret_i64(i64.load(x)) => f64.load(x) i32.reinterpret_f32(f32.load(x)) => i32.load(x) i64.reinterpret_f64(f64.load(x)) => i64.load(x) Stores: f32.store(y, f32.reinterpret_i32(x)) => i32.store(y, x) f64.store(y, f64.reinterpret_i64(x)) => i64.store(y, x) i32.store(y, i32.reinterpret_f32(x)) => f32.store(y, x) i64.store(y, i64.reinterpret_f64(x)) => f64.store(y, x) Also optimize reinterprets that are undone: i32.reinterpret_f32(f32.reinterpret_i32(x)) => x i64.reinterpret_f64(f64.reinterpret_i64(x)) => x f32.reinterpret_i32(i32.reinterpret_f32(x)) => x f64.reinterpret_i64(i64.reinterpret_f64(x)) => x
Diffstat (limited to 'src')
-rw-r--r--src/ir/abstract.h5
-rw-r--r--src/passes/OptimizeInstructions.cpp32
2 files changed, 37 insertions, 0 deletions
diff --git a/src/ir/abstract.h b/src/ir/abstract.h
index e5dfd57e0..2308c47ba 100644
--- a/src/ir/abstract.h
+++ b/src/ir/abstract.h
@@ -67,6 +67,11 @@ inline bool hasAnyShift(BinaryOp op) {
op == RotRInt64;
}
+inline bool hasAnyReinterpret(UnaryOp op) {
+ return op == ReinterpretInt32 || op == ReinterpretInt64 ||
+ op == ReinterpretFloat32 || op == ReinterpretFloat64;
+}
+
// Provide a wasm type and an abstract op and get the concrete one. For example,
// you can provide i32 and Add and receive the specific opcode for a 32-bit
// addition, AddInt32. If the op does not exist, it returns Invalid.
diff --git a/src/passes/OptimizeInstructions.cpp b/src/passes/OptimizeInstructions.cpp
index 59b1130d1..9f095b333 100644
--- a/src/passes/OptimizeInstructions.cpp
+++ b/src/passes/OptimizeInstructions.cpp
@@ -856,6 +856,30 @@ struct OptimizeInstructions
}
}
+ if (Abstract::hasAnyReinterpret(curr->op)) {
+ // i32.reinterpret_f32(f32.reinterpret_i32(x)) => x
+ // i64.reinterpret_f64(f64.reinterpret_i64(x)) => x
+ // f32.reinterpret_i32(i32.reinterpret_f32(x)) => x
+ // f64.reinterpret_i64(i64.reinterpret_f64(x)) => x
+ if (auto* inner = curr->value->dynCast<Unary>()) {
+ if (Abstract::hasAnyReinterpret(inner->op)) {
+ if (inner->value->type == curr->type) {
+ return replaceCurrent(inner->value);
+ }
+ }
+ }
+ // f32.reinterpret_i32(i32.load(x)) => f32.load(x)
+ // f64.reinterpret_i64(i64.load(x)) => f64.load(x)
+ // i32.reinterpret_f32(f32.load(x)) => i32.load(x)
+ // i64.reinterpret_f64(f64.load(x)) => i64.load(x)
+ if (auto* load = curr->value->dynCast<Load>()) {
+ if (!load->isAtomic && load->bytes == curr->type.getByteSize()) {
+ load->type = curr->type;
+ return replaceCurrent(load);
+ }
+ }
+ }
+
if (curr->op == EqZInt32) {
if (auto* inner = curr->value->dynCast<Binary>()) {
// Try to invert a relational operation using De Morgan's law
@@ -1016,6 +1040,14 @@ struct OptimizeInstructions
// instead of wrapping to 32, just store some of the bits in the i64
curr->valueType = Type::i64;
curr->value = unary->value;
+ } else if (!curr->isAtomic && Abstract::hasAnyReinterpret(unary->op) &&
+ curr->bytes == curr->valueType.getByteSize()) {
+ // f32.store(y, f32.reinterpret_i32(x)) => i32.store(y, x)
+ // f64.store(y, f64.reinterpret_i64(x)) => i64.store(y, x)
+ // i32.store(y, i32.reinterpret_f32(x)) => f32.store(y, x)
+ // i64.store(y, i64.reinterpret_f64(x)) => f64.store(y, x)
+ curr->valueType = unary->value->type;
+ curr->value = unary->value;
}
}
}