diff options
author | Alon Zakai <azakai@google.com> | 2022-11-01 17:12:29 -0700 |
---|---|---|
committer | GitHub <noreply@github.com> | 2022-11-02 00:12:29 +0000 |
commit | 25e5aa67bd16f277ad32d42bfbbf7cd130ddf028 (patch) | |
tree | 6156f359bfc057a6e36b38ba7512d2ba317c98a3 /test/lit/passes | |
parent | a030cc7f69a00dc0e699b8b580224126650c738b (diff) | |
download | binaryen-25e5aa67bd16f277ad32d42bfbbf7cd130ddf028.tar.gz binaryen-25e5aa67bd16f277ad32d42bfbbf7cd130ddf028.tar.bz2 binaryen-25e5aa67bd16f277ad32d42bfbbf7cd130ddf028.zip |
ReorderGlobals pass (#4904)
This sorts globals by their usage (and respecting dependencies). If the module
has very many globals then using smaller LEBs can matter.
If there are fewer than 128 globals then we cannot reduce size, and the pass
exits early (so this pass will not slow down MVP builds, which usually have just
1 global, the stack pointer). But with wasm GC it is common to use globals for
vtables etc., and often there is a very large number of them.
Diffstat (limited to 'test/lit/passes')
-rw-r--r-- | test/lit/passes/reorder-globals.wast | 294 |
1 files changed, 294 insertions, 0 deletions
diff --git a/test/lit/passes/reorder-globals.wast b/test/lit/passes/reorder-globals.wast new file mode 100644 index 000000000..bdfd03d8c --- /dev/null +++ b/test/lit/passes/reorder-globals.wast @@ -0,0 +1,294 @@ +;; NOTE: Assertions have been generated by update_lit_checks.py and should not be edited. +;; RUN: foreach %s %t wasm-opt --reorder-globals-always -S -o - | filecheck %s +;; RUN: foreach %s %t wasm-opt --reorder-globals-always --roundtrip -S -o - | filecheck %s + +;; Also check roundtripping here, so verify we don't end up emitting invalid +;; binaries. + +;; Global $b has more uses, so it should be sorted first. +(module + + ;; CHECK: (global $b i32 (i32.const 20)) + + ;; CHECK: (global $a i32 (i32.const 10)) + (global $a i32 (i32.const 10)) + (global $b i32 (i32.const 20)) + + ;; CHECK: (func $uses + ;; CHECK-NEXT: (drop + ;; CHECK-NEXT: (global.get $b) + ;; CHECK-NEXT: ) + ;; CHECK-NEXT: ) + (func $uses + (drop + (global.get $b) + ) + ) +) + +;; As above, but now with global.sets. Again $b should be sorted first. +(module + + ;; CHECK: (global $b (mut i32) (i32.const 20)) + + ;; CHECK: (global $a (mut i32) (i32.const 10)) + (global $a (mut i32) (i32.const 10)) + (global $b (mut i32) (i32.const 20)) + + ;; CHECK: (func $uses + ;; CHECK-NEXT: (global.set $b + ;; CHECK-NEXT: (i32.const 30) + ;; CHECK-NEXT: ) + ;; CHECK-NEXT: (global.set $b + ;; CHECK-NEXT: (i32.const 40) + ;; CHECK-NEXT: ) + ;; CHECK-NEXT: (drop + ;; CHECK-NEXT: (global.get $a) + ;; CHECK-NEXT: ) + ;; CHECK-NEXT: ) + (func $uses + (global.set $b + (i32.const 30) + ) + (global.set $b + (i32.const 40) + ) + (drop + (global.get $a) + ) + ) +) + +;; As above, but flipped so now $a has more, and should remain first. +(module + ;; CHECK: (global $a (mut i32) (i32.const 10)) + (global $a (mut i32) (i32.const 10)) + ;; CHECK: (global $b (mut i32) (i32.const 20)) + (global $b (mut i32) (i32.const 20)) + + ;; CHECK: (func $uses + ;; CHECK-NEXT: (global.set $a + ;; CHECK-NEXT: (i32.const 30) + ;; CHECK-NEXT: ) + ;; CHECK-NEXT: (global.set $a + ;; CHECK-NEXT: (i32.const 40) + ;; CHECK-NEXT: ) + ;; CHECK-NEXT: (drop + ;; CHECK-NEXT: (global.get $b) + ;; CHECK-NEXT: ) + ;; CHECK-NEXT: ) + (func $uses + (global.set $a + (i32.const 30) + ) + (global.set $a + (i32.const 40) + ) + (drop + (global.get $b) + ) + ) +) + +;; $b has more uses, but it depends on $a and cannot be sorted before it. +(module + ;; CHECK: (global $a i32 (i32.const 10)) + (global $a i32 (i32.const 10)) + ;; CHECK: (global $b i32 (global.get $a)) + (global $b i32 (global.get $a)) + + ;; CHECK: (func $uses + ;; CHECK-NEXT: (drop + ;; CHECK-NEXT: (global.get $b) + ;; CHECK-NEXT: ) + ;; CHECK-NEXT: ) + (func $uses + (drop + (global.get $b) + ) + ) +) + +;; $c has more uses, but it depends on $b and $a and cannot be sorted before +;; them. Likewise $b cannot be before $a. +(module + ;; CHECK: (global $a i32 (i32.const 10)) + (global $a i32 (i32.const 10)) + ;; CHECK: (global $b i32 (global.get $a)) + (global $b i32 (global.get $a)) + ;; CHECK: (global $c i32 (global.get $b)) + (global $c i32 (global.get $b)) + + ;; CHECK: (func $uses + ;; CHECK-NEXT: (drop + ;; CHECK-NEXT: (global.get $b) + ;; CHECK-NEXT: ) + ;; CHECK-NEXT: (drop + ;; CHECK-NEXT: (global.get $c) + ;; CHECK-NEXT: ) + ;; CHECK-NEXT: (drop + ;; CHECK-NEXT: (global.get $c) + ;; CHECK-NEXT: ) + ;; CHECK-NEXT: ) + (func $uses + (drop + (global.get $b) + ) + (drop + (global.get $c) + ) + (drop + (global.get $c) + ) + ) +) + +;; As above, but without dependencies, so now $c is first and then $b. +(module + + + ;; CHECK: (global $c i32 (i32.const 30)) + + ;; CHECK: (global $b i32 (i32.const 20)) + + ;; CHECK: (global $a i32 (i32.const 10)) + (global $a i32 (i32.const 10)) + (global $b i32 (i32.const 20)) + (global $c i32 (i32.const 30)) + + ;; CHECK: (func $uses + ;; CHECK-NEXT: (drop + ;; CHECK-NEXT: (global.get $b) + ;; CHECK-NEXT: ) + ;; CHECK-NEXT: (drop + ;; CHECK-NEXT: (global.get $c) + ;; CHECK-NEXT: ) + ;; CHECK-NEXT: (drop + ;; CHECK-NEXT: (global.get $c) + ;; CHECK-NEXT: ) + ;; CHECK-NEXT: ) + (func $uses + (drop + (global.get $b) + ) + (drop + (global.get $c) + ) + (drop + (global.get $c) + ) + ) +) + +;; As above, but a mixed case: $b depends on $a but $c has no dependencies. $c +;; can be first. +(module + + ;; CHECK: (global $c i32 (i32.const 30)) + + ;; CHECK: (global $a i32 (i32.const 10)) + (global $a i32 (i32.const 10)) + ;; CHECK: (global $b i32 (global.get $a)) + (global $b i32 (global.get $a)) + (global $c i32 (i32.const 30)) + + ;; CHECK: (func $uses + ;; CHECK-NEXT: (drop + ;; CHECK-NEXT: (global.get $b) + ;; CHECK-NEXT: ) + ;; CHECK-NEXT: (drop + ;; CHECK-NEXT: (global.get $c) + ;; CHECK-NEXT: ) + ;; CHECK-NEXT: (drop + ;; CHECK-NEXT: (global.get $c) + ;; CHECK-NEXT: ) + ;; CHECK-NEXT: ) + (func $uses + (drop + (global.get $b) + ) + (drop + (global.get $c) + ) + (drop + (global.get $c) + ) + ) +) + +;; Another mixed case, now with $c depending on $b. $b can be before $a. +(module + + + ;; CHECK: (global $b i32 (i32.const 20)) + + ;; CHECK: (global $c i32 (global.get $b)) + + ;; CHECK: (global $a i32 (i32.const 10)) + (global $a i32 (i32.const 10)) + (global $b i32 (i32.const 20)) + (global $c i32 (global.get $b)) + + ;; CHECK: (func $uses + ;; CHECK-NEXT: (drop + ;; CHECK-NEXT: (global.get $b) + ;; CHECK-NEXT: ) + ;; CHECK-NEXT: (drop + ;; CHECK-NEXT: (global.get $c) + ;; CHECK-NEXT: ) + ;; CHECK-NEXT: (drop + ;; CHECK-NEXT: (global.get $c) + ;; CHECK-NEXT: ) + ;; CHECK-NEXT: ) + (func $uses + (drop + (global.get $b) + ) + (drop + (global.get $c) + ) + (drop + (global.get $c) + ) + ) +) + +;; $b has more uses, but $a is an import and must remain first. +(module + ;; CHECK: (import "a" "b" (global $a i32)) + (import "a" "b" (global $a i32)) + ;; CHECK: (global $b i32 (i32.const 10)) + (global $b i32 (i32.const 10)) + + ;; CHECK: (func $uses + ;; CHECK-NEXT: (drop + ;; CHECK-NEXT: (global.get $b) + ;; CHECK-NEXT: ) + ;; CHECK-NEXT: ) + (func $uses + (drop + (global.get $b) + ) + ) +) + +;; As above, but with a and b's names flipped, to check that the names do not +;; matter, and we keep imports first. +(module + ;; CHECK: (import "a" "b" (global $b i32)) + (import "a" "b" (global $b i32)) + + ;; CHECK: (global $a i32 (i32.const 10)) + (global $a i32 (i32.const 10)) + + ;; CHECK: (func $uses + ;; CHECK-NEXT: (drop + ;; CHECK-NEXT: (global.get $a) + ;; CHECK-NEXT: ) + ;; CHECK-NEXT: ) + (func $uses + (drop + (global.get $a) + ) + ) +) |