Disallow branching to the entry block.

Functions that would otherwise start with a loop should start with a
separate ebb which just branches to the header of the loop.
This commit is contained in:
Dan Gohman
2017-10-09 14:59:08 -07:00
parent 893a6716c6
commit 6aeeaebbd3
5 changed files with 133 additions and 116 deletions

View File

@@ -2,30 +2,33 @@ test licm
function %simple_loop(i32) -> i32 { function %simple_loop(i32) -> i32 {
ebb1(v0: i32): ebb0(v0: i32):
v1 = iconst.i32 1 jump ebb1(v0)
v2 = iconst.i32 2
v3 = iadd v1, v2
brz v0, ebb2(v0)
v4 = isub v0, v1
jump ebb1(v4)
ebb2(v5: i32): ebb1(v1: i32):
return v5 v2 = iconst.i32 1
v3 = iconst.i32 2
v4 = iadd v2, v3
brz v1, ebb2(v1)
v5 = isub v1, v2
jump ebb1(v5)
ebb2(v6: i32):
return v6
} }
; sameln: function %simple_loop ; sameln: function %simple_loop
; nextln: ebb2(v6: i32):
; nextln: v1 = iconst.i32 1
; nextln: v2 = iconst.i32 2
; nextln: v3 = iadd v1, v2
; nextln: jump ebb0(v6)
; nextln:
; nextln: ebb0(v0: i32): ; nextln: ebb0(v0: i32):
; nextln: brz v0, ebb1(v0) ; nextln: v2 = iconst.i32 1
; nextln: v4 = isub v0, v1 ; nextln: v3 = iconst.i32 2
; nextln: jump ebb0(v4) ; nextln: v4 = iadd v2, v3
; nextln: jump ebb1(v0)
; nextln: ; nextln:
; nextln: ebb1(v5: i32): ; nextln: ebb1(v1: i32):
; nextln: return v5 ; nextln: brz v1, ebb2(v1)
; nextln: v5 = isub v1, v2
; nextln: jump ebb1(v5)
; nextln:
; nextln: ebb2(v6: i32):
; nextln: return v6
; nextln: } ; nextln: }

View File

@@ -1,81 +1,83 @@
test licm test licm
function %complex(i32) -> i32 { function %complex(i32) -> i32 native {
ebb0(v0: i32): ebb0(v0: i32):
v1 = iconst.i32 1 jump ebb1(v0)
v19 = iconst.i32 4
v2 = iadd v1, v0
brz v0, ebb1(v1)
jump ebb3(v2)
ebb1(v3: i32): ebb1(v1: i32):
v4 = iconst.i32 2 v2 = iconst.i32 1
v5 = iadd v3, v2 v3 = iconst.i32 4
v6 = iadd v4, v0 v4 = iadd v2, v1
jump ebb2(v6) brz v1, ebb2(v2)
jump ebb4(v4)
ebb2(v7: i32): ebb2(v5: i32):
v8 = iadd v7, v3 v6 = iconst.i32 2
v9 = iadd v0, v2 v7 = iadd v5, v4
brz v0, ebb1(v7) v8 = iadd v6, v1
jump ebb5(v8) jump ebb3(v8)
ebb3(v10: i32): ebb3(v9: i32):
v11 = iconst.i32 3 v10 = iadd v9, v5
v12 = iadd v10, v11 v11 = iadd.i32 v1, v4
v13 = iadd v2, v11 brz.i32 v1, ebb2(v9)
jump ebb4(v11) jump ebb6(v10)
ebb4(v14: i32): ebb4(v12: i32):
v15 = iadd v12, v2 v13 = iconst.i32 3
brz v0, ebb3(v14) v14 = iadd v12, v13
jump ebb5(v14) v15 = iadd.i32 v4, v13
jump ebb5(v13)
ebb5(v16: i32): ebb5(v16: i32):
v17 = iadd v16, v1 v17 = iadd.i32 v14, v4
v18 = iadd v1, v19 brz.i32 v1, ebb4(v16)
brz v0, ebb0(v18) jump ebb6(v16)
return v17
ebb6(v18: i32):
v19 = iadd v18, v2
v20 = iadd.i32 v2, v3
brz.i32 v1, ebb1(v20)
return v19
} }
; sameln: function %complex ; sameln: function %complex
; nextln: ebb6(v20: i32):
; nextln: v1 = iconst.i32 1
; nextln: v2 = iconst.i32 4
; nextln: v5 = iconst.i32 2
; nextln: v12 = iconst.i32 3
; nextln: v19 = iadd v1, v2
; nextln: jump ebb0(v20)
; nextln:
; nextln: ebb0(v0: i32): ; nextln: ebb0(v0: i32):
; nextln: v3 = iadd.i32 v1, v0 ; nextln: v2 = iconst.i32 1
; nextln: v7 = iadd.i32 v5, v0 ; nextln: v3 = iconst.i32 4
; nextln: v10 = iadd v0, v3 ; nextln: v6 = iconst.i32 2
; nextln: brz v0, ebb1(v1) ; nextln: v13 = iconst.i32 3
; nextln: v14 = iadd v3, v12 ; nextln: v20 = iadd v2, v3
; nextln: jump ebb3(v3) ; nextln: jump ebb1(v0)
; nextln: ; nextln:
; nextln: ebb1(v4: i32): ; nextln: ebb1(v1: i32):
; nextln: v6 = iadd v4, v3 ; nextln: v4 = iadd.i32 v2, v1
; nextln: jump ebb2(v7) ; nextln: v8 = iadd.i32 v6, v1
; nextln: v11 = iadd v1, v4
; nextln: brz v1, ebb2(v2)
; nextln: v15 = iadd v4, v13
; nextln: jump ebb4(v4)
; nextln: ; nextln:
; nextln: ebb2(v8: i32): ; nextln: ebb2(v5: i32):
; nextln: v9 = iadd v8, v4 ; nextln: v7 = iadd v5, v4
; nextln: brz.i32 v0, ebb1(v8) ; nextln: jump ebb3(v8)
; nextln: jump ebb5(v9)
; nextln: ; nextln:
; nextln: ebb3(v11: i32): ; nextln: ebb3(v9: i32):
; nextln: v13 = iadd v11, v12 ; nextln: v10 = iadd v9, v5
; nextln: jump ebb4(v12) ; nextln: brz.i32 v1, ebb2(v9)
; nextln: jump ebb6(v10)
; nextln: ; nextln:
; nextln: ebb4(v15: i32): ; nextln: ebb4(v12: i32):
; nextln: v16 = iadd.i32 v13, v3 ; nextln: v14 = iadd v12, v13
; nextln: brz.i32 v0, ebb3(v15) ; nextln: jump ebb5(v13)
; nextln: jump ebb5(v15)
; nextln: ; nextln:
; nextln: ebb5(v17: i32): ; nextln: ebb5(v16: i32):
; nextln: v18 = iadd v17, v1 ; nextln: v17 = iadd.i32 v14, v4
; nextln: brz.i32 v0, ebb0(v19) ; nextln: brz.i32 v1, ebb4(v16)
; nextln: return v18 ; nextln: jump ebb6(v16)
; nextln:
; nextln: ebb6(v18: i32):
; nextln: v19 = iadd v18, v2
; nextln: brz.i32 v1, ebb1(v20)
; nextln: return v19
; nextln: } ; nextln: }

View File

@@ -3,50 +3,53 @@ test licm
function %nested_loops(i32) -> i32 { function %nested_loops(i32) -> i32 {
ebb0(v0: i32): ebb0(v0: i32):
v1 = iconst.i32 1 jump ebb1(v0)
v2 = iconst.i32 2
v3 = iadd v1, v2
v4 = isub v0, v1
jump ebb1(v4,v4)
ebb1(v10: i32,v11: i32): ebb1(v1: i32):
brz v11, ebb2(v10) v2 = iconst.i32 1
v3 = iconst.i32 2
v4 = iadd v2, v3
v5 = isub v1, v2
jump ebb2(v5, v5)
ebb2(v10: i32, v11: i32):
brz v11, ebb3(v10)
v12 = iconst.i32 1 v12 = iconst.i32 1
v15 = iadd v12, v4 v15 = iadd v12, v5
v13 = isub v11, v12 v13 = isub v11, v12
jump ebb1(v10,v13) jump ebb2(v10,v13)
ebb2(v20: i32): ebb3(v20: i32):
brz v20, ebb3(v20) brz v20, ebb4(v20)
jump ebb0(v20) jump ebb1(v20)
ebb3(v30: i32): ebb4(v30: i32):
return v30 return v30
} }
; sameln:function %nested_loops(i32) -> i32 { ; sameln:function %nested_loops(i32) -> i32 {
; nextln: ebb4(v12: i32):
; nextln: v1 = iconst.i32 1
; nextln: v2 = iconst.i32 2
; nextln: v3 = iadd v1, v2
; nextln: v7 = iconst.i32 1
; nextln: jump ebb0(v12)
; nextln:
; nextln: ebb0(v0: i32): ; nextln: ebb0(v0: i32):
; nextln: v4 = isub v0, v1 ; nextln: v2 = iconst.i32 1
; nextln: v8 = iadd.i32 v7, v4 ; nextln: v3 = iconst.i32 2
; nextln: jump ebb1(v4, v4) ; nextln: v4 = iadd v2, v3
; nextln: v8 = iconst.i32 1
; nextln: jump ebb1(v0)
; nextln: ; nextln:
; nextln: ebb1(v5: i32, v6: i32): ; nextln: ebb1(v1: i32):
; nextln: brz v6, ebb2(v5) ; nextln: v5 = isub v1, v2
; nextln: v9 = isub v6, v7 ; nextln: v9 = iadd.i32 v8, v5
; nextln: jump ebb1(v5, v9) ; nextln: jump ebb2(v5, v5)
; nextln: ; nextln:
; nextln: ebb2(v10: i32): ; nextln: ebb2(v6: i32, v7: i32):
; nextln: brz v10, ebb3(v10) ; nextln: brz v7, ebb3(v6)
; nextln: jump ebb0(v10) ; nextln: v10 = isub v7, v8
; nextln: jump ebb2(v6, v10)
; nextln: ; nextln:
; nextln: ebb3(v11: i32): ; nextln: ebb3(v11: i32):
; nextln: return v11 ; nextln: brz v11, ebb4(v11)
; nextln: jump ebb1(v11)
; nextln:
; nextln: ebb4(v12: i32):
; nextln: return v12
; nextln: } ; nextln: }

View File

@@ -597,11 +597,15 @@ mod test {
#[test] #[test]
fn renumbering() { fn renumbering() {
let mut func = Function::new(); let mut func = Function::new();
let entry = func.dfg.make_ebb();
let ebb0 = func.dfg.make_ebb(); let ebb0 = func.dfg.make_ebb();
let ebb100 = func.dfg.make_ebb(); let ebb100 = func.dfg.make_ebb();
let mut cur = FuncCursor::new(&mut func); let mut cur = FuncCursor::new(&mut func);
cur.insert_ebb(entry);
cur.ins().jump(ebb0, &[]);
cur.insert_ebb(ebb0); cur.insert_ebb(ebb0);
let cond = cur.ins().iconst(I32, 0); let cond = cur.ins().iconst(I32, 0);
let inst2 = cur.ins().brz(cond, ebb0, &[]); let inst2 = cur.ins().brz(cond, ebb0, &[]);

View File

@@ -14,6 +14,7 @@
//! - The instruction format must match the opcode. //! - The instruction format must match the opcode.
//! - All result values must be created for multi-valued instructions. //! - All result values must be created for multi-valued instructions.
//! - All referenced entities must exist. (Values, EBBs, stack slots, ...) //! - All referenced entities must exist. (Values, EBBs, stack slots, ...)
//! - Instructions must not reference (eg. branch to) the entry block.
//! //!
//! SSA form //! SSA form
//! //!
@@ -352,10 +353,14 @@ impl<'a> Verifier<'a> {
fn verify_ebb(&self, inst: Inst, e: Ebb) -> Result { fn verify_ebb(&self, inst: Inst, e: Ebb) -> Result {
if !self.func.dfg.ebb_is_valid(e) || !self.func.layout.is_ebb_inserted(e) { if !self.func.dfg.ebb_is_valid(e) || !self.func.layout.is_ebb_inserted(e) {
err!(inst, "invalid ebb reference {}", e) return err!(inst, "invalid ebb reference {}", e);
} else {
Ok(())
} }
if let Some(entry_block) = self.func.layout.entry_block() {
if e == entry_block {
return err!(inst, "invalid reference to entry ebb {}", e);
}
}
Ok(())
} }
fn verify_sig_ref(&self, inst: Inst, s: SigRef) -> Result { fn verify_sig_ref(&self, inst: Inst, s: SigRef) -> Result {