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:
committed by
Benjamin Bouvier
parent
c7bc1b7c56
commit
641771ac6a
@@ -1,4 +1,5 @@
|
|||||||
test licm
|
test licm
|
||||||
|
target riscv32
|
||||||
|
|
||||||
function %simple_loop(i32) -> i32 {
|
function %simple_loop(i32) -> i32 {
|
||||||
|
|
||||||
|
|||||||
@@ -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):
|
||||||
|
|||||||
36
cranelift/filetests/licm/encoding.clif
Normal file
36
cranelift/filetests/licm/encoding.clif
Normal 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: }
|
||||||
@@ -1,4 +1,5 @@
|
|||||||
test licm
|
test licm
|
||||||
|
target riscv32
|
||||||
|
|
||||||
function %multiple_blocks(i32) -> i32 {
|
function %multiple_blocks(i32) -> i32 {
|
||||||
|
|
||||||
|
|||||||
@@ -1,4 +1,5 @@
|
|||||||
test licm
|
test licm
|
||||||
|
target riscv32
|
||||||
|
|
||||||
function %nested_loops(i32) -> i32 {
|
function %nested_loops(i32) -> i32 {
|
||||||
|
|
||||||
|
|||||||
@@ -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
|
||||||
|
|||||||
@@ -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.
|
||||||
|
|||||||
@@ -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();
|
||||||
|
|||||||
@@ -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();
|
||||||
|
|||||||
Reference in New Issue
Block a user