diff options
author | Alon Zakai <alonzakai@gmail.com> | 2018-09-05 13:44:43 -0700 |
---|---|---|
committer | GitHub <noreply@github.com> | 2018-09-05 13:44:43 -0700 |
commit | a0cbc9dbbfcfd5bb254da904c19e3f6b0a1716c1 (patch) | |
tree | e06fcb231cb49378e6dc9b20e289c8f69a7a1898 /src/mixed_arena.h | |
parent | 8900ceb7e1adf6b91374c236a30ad7f20b980106 (diff) | |
download | binaryen-a0cbc9dbbfcfd5bb254da904c19e3f6b0a1716c1.tar.gz binaryen-a0cbc9dbbfcfd5bb254da904c19e3f6b0a1716c1.tar.bz2 binaryen-a0cbc9dbbfcfd5bb254da904c19e3f6b0a1716c1.zip |
DeadArgumentElimination Pass (#1641)
This adds a pass to remove unnecessary call arguments in an LTO-like manner, that is:
* If a parameter is not actually used in a function, we don't need to send anything, and can remove it from the function's declaration. Concretely,
(func $a (param $x i32)
..no uses of $x..
)
(func $b
(call $a (..))
)
=>
(func $a
..no uses of $x..
)
(func $b
(call $a)
)
And
* If a parameter is only ever sent the same constant value, we can just set that constant value in the function (which then means that the values sent from the outside are no longer used, as in the previous point). Concretely,
(func $a (param $x i32)
..may use $x..
)
(func $b
(call $a (i32.const 1))
(call $a (i32.const 1))
)
=>
(func $a
(local $x i32)
(set_local $x (i32.const 1)
..may use $x..
)
(func $b
(call $a)
(call $a)
)
How much this helps depends on the codebase obviously, but sometimes it is pretty useful. For example, it shrinks 0.72% on Unity and 0.37% on Mono. Note that those numbers include not just the optimization itself, but the other optimizations it then enables - in particular the second point from earlier leads to inlining a constant value, which often allows constant propagation, and also removing parameters may enable more duplicate function elimination, etc. - which explains how this can shrink Unity by almost 1%.
Implementation is pretty straightforward, but there is some work to make the heavy part of the pass parallel, and a bunch of corner cases to avoid (can't change a function that is exported or in the table, etc.). Like the Inlining pass, there is both a standard and an "optimizing" version of this pass - the latter also optimizes the functions it changes, as like Inlining, it's useful to not need to re-run all function optimizations on the whole module.
Diffstat (limited to 'src/mixed_arena.h')
-rw-r--r-- | src/mixed_arena.h | 4 |
1 files changed, 4 insertions, 0 deletions
diff --git a/src/mixed_arena.h b/src/mixed_arena.h index a1c51b5eb..90203842e 100644 --- a/src/mixed_arena.h +++ b/src/mixed_arena.h @@ -223,6 +223,10 @@ public: usedElements -= size; } + void erase(Iterator it) { + erase(it, it + 1); + } + void clear() { usedElements = 0; } |