Add encodings to the instructions in LICM's generated preheaders.

When LICM generates a preheader, it needs to add encodings to the
instructions it generates, since it runs after legalization.
This commit is contained in:
Dan Gohman
2018-11-12 05:13:43 -08:00
committed by Benjamin Bouvier
parent c7bc1b7c56
commit 641771ac6a
9 changed files with 54 additions and 8 deletions

View File

@@ -1,4 +1,5 @@
test licm test licm
target riscv32
function %simple_loop(i32) -> i32 { function %simple_loop(i32) -> i32 {

View File

@@ -1,4 +1,5 @@
test licm test licm
target riscv32
function %complex(i32) -> i32 system_v { function %complex(i32) -> i32 system_v {
ebb0(v0: i32): ebb0(v0: i32):

View File

@@ -0,0 +1,36 @@
test licm
target riscv32
; Ensure that instructions emitted by LICM get encodings.
function %simple_loop(i32) -> i32 {
ebb0(v0: i32):
[UJ#1b] jump ebb1(v0)
ebb1(v1: i32):
[Iz#04,%x0] v2 = iconst.i32 1
[Iz#04,%x1] v3 = iconst.i32 2
[R#0c,%x2] v4 = iadd v2, v3
[SBzero#18] brz v1, ebb2(v1)
[R#200c,%x5] v5 = isub v1, v2
[UJ#1b] jump ebb1(v5)
ebb2(v6: i32):
[Iret#19] return v6
}
; check: function
; nextln: ebb0(v0: i32):
; nextln: [Iz#04,%x0] v2 = iconst.i32 1
; nextln: [Iz#04,%x1] v3 = iconst.i32 2
; nextln: [R#0c,%x2] v4 = iadd v2, v3
; nextln: [UJ#1b] jump ebb1(v0)
; nextln:
; nextln: ebb1(v1: i32):
; nextln: [SBzero#18] brz v1, ebb2(v1)
; nextln: [R#200c,%x5] v5 = isub v1, v2
; nextln: [UJ#1b] jump ebb1(v5)
; nextln:
; nextln: ebb2(v6: i32):
; nextln: [Iret#19] return v6
; nextln: }

View File

@@ -1,4 +1,5 @@
test licm test licm
target riscv32
function %multiple_blocks(i32) -> i32 { function %multiple_blocks(i32) -> i32 {

View File

@@ -1,4 +1,5 @@
test licm test licm
target riscv32
function %nested_loops(i32) -> i32 { function %nested_loops(i32) -> i32 {

View File

@@ -1,4 +1,5 @@
test licm test licm
target riscv32
function %other_side_effects(i32) -> i32 { function %other_side_effects(i32) -> i32 {
@@ -6,9 +7,9 @@ ebb0(v0: i32):
jump ebb1(v0) jump ebb1(v0)
ebb1(v1: i32): ebb1(v1: i32):
regmove.i32 v0, %10 -> %20 regmove.i32 v0, %x10 -> %x20
; check: ebb1(v1: i32): ; check: ebb1(v1: i32):
; check: regmove.i32 v0, %10 -> %20 ; check: regmove.i32 v0, %x10 -> %x20
v2 = iconst.i32 1 v2 = iconst.i32 1
brz v1, ebb2(v1) brz v1, ebb2(v1)
v5 = isub v1, v2 v5 = isub v1, v2

View File

@@ -275,14 +275,15 @@ impl Context {
} }
/// Perform LICM on the function. /// Perform LICM on the function.
pub fn licm<'a, FOI: Into<FlagsOrIsa<'a>>>(&mut self, fisa: FOI) -> CodegenResult<()> { pub fn licm(&mut self, isa: &TargetIsa) -> CodegenResult<()> {
do_licm( do_licm(
isa,
&mut self.func, &mut self.func,
&mut self.cfg, &mut self.cfg,
&mut self.domtree, &mut self.domtree,
&mut self.loop_analysis, &mut self.loop_analysis,
); );
self.verify_if(fisa) self.verify_if(isa)
} }
/// Perform unreachable code elimination. /// Perform unreachable code elimination.

View File

@@ -1,11 +1,12 @@
//! A Loop Invariant Code Motion optimization pass //! A Loop Invariant Code Motion optimization pass
use cursor::{Cursor, FuncCursor}; use cursor::{Cursor, FuncCursor, EncCursor};
use dominator_tree::DominatorTree; use dominator_tree::DominatorTree;
use entity::{EntityList, ListPool}; use entity::{EntityList, ListPool};
use flowgraph::{BasicBlock, ControlFlowGraph}; use flowgraph::{BasicBlock, ControlFlowGraph};
use fx::FxHashSet; use fx::FxHashSet;
use ir::{DataFlowGraph, Ebb, Function, Inst, InstBuilder, Layout, Opcode, Type, Value}; use ir::{DataFlowGraph, Ebb, Function, Inst, InstBuilder, Layout, Opcode, Type, Value};
use isa::TargetIsa;
use loop_analysis::{Loop, LoopAnalysis}; use loop_analysis::{Loop, LoopAnalysis};
use std::vec::Vec; use std::vec::Vec;
use timing; use timing;
@@ -14,6 +15,7 @@ use timing;
/// loop-invariant instructions out of them. /// loop-invariant instructions out of them.
/// Changes the CFG and domtree in-place during the operation. /// Changes the CFG and domtree in-place during the operation.
pub fn do_licm( pub fn do_licm(
isa: &TargetIsa,
func: &mut Function, func: &mut Function,
cfg: &mut ControlFlowGraph, cfg: &mut ControlFlowGraph,
domtree: &mut DominatorTree, domtree: &mut DominatorTree,
@@ -36,7 +38,7 @@ pub fn do_licm(
match has_pre_header(&func.layout, cfg, domtree, loop_analysis.loop_header(lp)) { match has_pre_header(&func.layout, cfg, domtree, loop_analysis.loop_header(lp)) {
None => { None => {
let pre_header = let pre_header =
create_pre_header(loop_analysis.loop_header(lp), func, cfg, domtree); create_pre_header(isa, loop_analysis.loop_header(lp), func, cfg, domtree);
pos = FuncCursor::new(func).at_last_inst(pre_header); pos = FuncCursor::new(func).at_last_inst(pre_header);
} }
// If there is a natural pre-header we insert new instructions just before the // If there is a natural pre-header we insert new instructions just before the
@@ -60,6 +62,7 @@ pub fn do_licm(
// Insert a pre-header before the header, modifying the function layout and CFG to reflect it. // Insert a pre-header before the header, modifying the function layout and CFG to reflect it.
// A jump instruction to the header is placed at the end of the pre-header. // A jump instruction to the header is placed at the end of the pre-header.
fn create_pre_header( fn create_pre_header(
isa: &TargetIsa,
header: Ebb, header: Ebb,
func: &mut Function, func: &mut Function,
cfg: &mut ControlFlowGraph, cfg: &mut ControlFlowGraph,
@@ -87,7 +90,7 @@ fn create_pre_header(
} }
} }
{ {
let mut pos = FuncCursor::new(func).at_top(header); let mut pos = EncCursor::new(func, isa).at_top(header);
// Inserts the pre-header at the right place in the layout. // Inserts the pre-header at the right place in the layout.
pos.insert_ebb(pre_header); pos.insert_ebb(pre_header);
pos.next_inst(); pos.next_inst();

View File

@@ -33,12 +33,13 @@ impl SubTest for TestLICM {
} }
fn run(&self, func: Cow<Function>, context: &Context) -> SubtestResult<()> { fn run(&self, func: Cow<Function>, context: &Context) -> SubtestResult<()> {
let isa = context.isa.expect("LICM needs an ISA");
let mut comp_ctx = cranelift_codegen::Context::for_function(func.into_owned()); let mut comp_ctx = cranelift_codegen::Context::for_function(func.into_owned());
comp_ctx.flowgraph(); comp_ctx.flowgraph();
comp_ctx.compute_loop_analysis(); comp_ctx.compute_loop_analysis();
comp_ctx comp_ctx
.licm(context.flags_or_isa()) .licm(isa)
.map_err(|e| pretty_error(&comp_ctx.func, context.isa, Into::into(e)))?; .map_err(|e| pretty_error(&comp_ctx.func, context.isa, Into::into(e)))?;
let text = comp_ctx.func.display(context.isa).to_string(); let text = comp_ctx.func.display(context.isa).to_string();