forked from vyperlang/vyper
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
feat[venom]: add store elimination pass (vyperlang#4021)
Venom uses the `store` instruction to load a variable or literal to another variable. This commit adds a new pass `StoreElimination` that forwards the rhs for `store`s to their uses, eliminating the `store` instruction in the process. --------- Co-authored-by: Charles Cooper <[email protected]>
- Loading branch information
1 parent
ea1e4ea
commit c276c94
Showing
3 changed files
with
55 additions
and
22 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,49 @@ | ||
from vyper.venom.analysis.cfg import CFGAnalysis | ||
from vyper.venom.analysis.dfg import DFGAnalysis | ||
from vyper.venom.analysis.dominators import DominatorTreeAnalysis | ||
from vyper.venom.analysis.liveness import LivenessAnalysis | ||
from vyper.venom.basicblock import IRVariable | ||
from vyper.venom.passes.base_pass import IRPass | ||
|
||
|
||
class StoreElimination(IRPass): | ||
""" | ||
This pass forwards variables to their uses though `store` instructions, | ||
and removes the `store` instruction. | ||
""" | ||
|
||
def run_pass(self): | ||
self.analyses_cache.request_analysis(CFGAnalysis) | ||
dfg = self.analyses_cache.request_analysis(DFGAnalysis) | ||
|
||
for var, inst in dfg.outputs.items(): | ||
if inst.opcode != "store": | ||
continue | ||
if not isinstance(inst.operands[0], IRVariable): | ||
continue | ||
if inst.operands[0].name in ["%ret_ofst", "%ret_size"]: | ||
continue | ||
if inst.output.name in ["%ret_ofst", "%ret_size"]: | ||
continue | ||
self._process_store(dfg, inst, var, inst.operands[0]) | ||
|
||
self.analyses_cache.invalidate_analysis(DominatorTreeAnalysis) | ||
self.analyses_cache.invalidate_analysis(LivenessAnalysis) | ||
self.analyses_cache.invalidate_analysis(DFGAnalysis) | ||
|
||
def _process_store(self, dfg, inst, var, new_var): | ||
""" | ||
Process store instruction. If the variable is only used by a load instruction, | ||
forward the variable to the load instruction. | ||
""" | ||
uses = dfg.get_uses(var) | ||
|
||
if any([inst.opcode == "phi" for inst in uses]): | ||
return | ||
|
||
for use_inst in uses: | ||
for i, operand in enumerate(use_inst.operands): | ||
if operand == var: | ||
use_inst.operands[i] = new_var | ||
|
||
inst.parent.remove_instruction(inst) |