1use crate::Variable;
11use alloc::vec::Vec;
12use core::mem;
13use cranelift_codegen::cursor::{Cursor, FuncCursor};
14use cranelift_codegen::entity::{EntityList, EntitySet, ListPool, SecondaryMap};
15use cranelift_codegen::ir::immediates::{Ieee16, Ieee32, Ieee64, Ieee128};
16use cranelift_codegen::ir::types::{F16, F32, F64, F128, I64, I128};
17use cranelift_codegen::ir::{Block, Function, Inst, InstBuilder, Type, Value};
18use cranelift_codegen::packed_option::PackedOption;
19
20#[derive(Default)]
34pub struct SSABuilder {
35 variables: SecondaryMap<Variable, SecondaryMap<Block, PackedOption<Value>>>,
39
40 ssa_blocks: SecondaryMap<Block, SSABlockData>,
43
44 calls: Vec<Call>,
46 results: Vec<Value>,
48
49 side_effects: SideEffects,
51
52 visited: EntitySet<Block>,
54
55 variable_pool: ListPool<Variable>,
57
58 inst_pool: ListPool<Inst>,
60}
61
62#[derive(Default)]
64pub struct SideEffects {
65 pub instructions_added_to_blocks: Vec<Block>,
70}
71
72impl SideEffects {
73 fn is_empty(&self) -> bool {
74 let Self {
75 instructions_added_to_blocks,
76 } = self;
77 instructions_added_to_blocks.is_empty()
78 }
79}
80
81#[derive(Clone)]
82enum Sealed {
83 No {
84 undef_variables: EntityList<Variable>,
86 },
87 Yes,
88}
89
90impl Default for Sealed {
91 fn default() -> Self {
92 Sealed::No {
93 undef_variables: EntityList::new(),
94 }
95 }
96}
97
98#[derive(Clone, Default)]
99struct SSABlockData {
100 predecessors: EntityList<Inst>,
102 sealed: Sealed,
104 single_predecessor: PackedOption<Block>,
106}
107
108impl SSABuilder {
109 pub fn clear(&mut self) {
112 self.variables.clear();
113 self.ssa_blocks.clear();
114 self.variable_pool.clear();
115 self.inst_pool.clear();
116 debug_assert!(self.calls.is_empty());
117 debug_assert!(self.results.is_empty());
118 debug_assert!(self.side_effects.is_empty());
119 }
120
121 pub fn is_empty(&self) -> bool {
123 self.variables.is_empty()
124 && self.ssa_blocks.is_empty()
125 && self.calls.is_empty()
126 && self.results.is_empty()
127 && self.side_effects.is_empty()
128 }
129}
130
131enum Call {
133 UseVar(Inst),
134 FinishPredecessorsLookup(Value, Block),
135}
136
137fn emit_zero(ty: Type, mut cur: FuncCursor) -> Value {
139 match ty {
140 I128 => {
141 let zero = cur.ins().iconst(I64, 0);
142 cur.ins().uextend(I128, zero)
143 }
144 ty if ty.is_int() => cur.ins().iconst(ty, 0),
145 F16 => cur.ins().f16const(Ieee16::with_bits(0)),
146 F32 => cur.ins().f32const(Ieee32::with_bits(0)),
147 F64 => cur.ins().f64const(Ieee64::with_bits(0)),
148 F128 => {
149 let zero = cur.func.dfg.constants.insert(Ieee128::with_bits(0).into());
150 cur.ins().f128const(zero)
151 }
152 ty if ty.is_vector() => match ty.lane_type() {
153 scalar_ty if scalar_ty.is_int() => {
154 let zero = cur
155 .func
156 .dfg
157 .constants
158 .insert(vec![0; ty.bytes().try_into().unwrap()].into());
159 cur.ins().vconst(ty, zero)
160 }
161 F16 => {
162 let scalar = cur.ins().f16const(Ieee16::with_bits(0));
163 cur.ins().splat(ty, scalar)
164 }
165 F32 => {
166 let scalar = cur.ins().f32const(Ieee32::with_bits(0));
167 cur.ins().splat(ty, scalar)
168 }
169 F64 => {
170 let scalar = cur.ins().f64const(Ieee64::with_bits(0));
171 cur.ins().splat(ty, scalar)
172 }
173 F128 => {
174 let zero = cur.func.dfg.constants.insert(Ieee128::with_bits(0).into());
175 let scalar = cur.ins().f128const(zero);
176 cur.ins().splat(ty, scalar)
177 }
178 _ => panic!("unimplemented scalar type: {ty:?}"),
179 },
180 ty => panic!("unimplemented type: {ty:?}"),
181 }
182}
183
184impl SSABuilder {
204 pub fn values_for_var(&self, var: Variable) -> impl Iterator<Item = Value> + '_ {
207 self.variables[var].values().filter_map(|v| v.expand())
208 }
209
210 pub fn def_var(&mut self, var: Variable, val: Value, block: Block) {
214 self.variables[var][block] = PackedOption::from(val);
215 }
216
217 pub fn use_var(
225 &mut self,
226 func: &mut Function,
227 var: Variable,
228 ty: Type,
229 block: Block,
230 ) -> (Value, SideEffects) {
231 debug_assert!(self.calls.is_empty());
232 debug_assert!(self.results.is_empty());
233 debug_assert!(self.side_effects.is_empty());
234
235 self.use_var_nonlocal(func, var, ty, block);
237 let value = self.run_state_machine(func, var, ty);
238
239 let side_effects = mem::take(&mut self.side_effects);
240 (value, side_effects)
241 }
242
243 fn use_var_nonlocal(&mut self, func: &mut Function, var: Variable, ty: Type, mut block: Block) {
247 if let Some(val) = self.variables[var][block].expand() {
250 self.results.push(val);
251 return;
252 }
253
254 let (val, from) = self.find_var(func, var, ty, block);
258
259 let var_defs = &mut self.variables[var];
278 while block != from {
279 debug_assert!(var_defs[block].is_none());
280 var_defs[block] = PackedOption::from(val);
281 block = self.ssa_blocks[block].single_predecessor.unwrap();
282 }
283 }
284
285 fn find_var(
309 &mut self,
310 func: &mut Function,
311 var: Variable,
312 ty: Type,
313 mut block: Block,
314 ) -> (Value, Block) {
315 self.visited.clear();
317 let var_defs = &mut self.variables[var];
318 while let Some(pred) = self.ssa_blocks[block].single_predecessor.expand() {
319 if !self.visited.insert(block) {
320 break;
321 }
322 block = pred;
323 if let Some(val) = var_defs[block].expand() {
324 self.results.push(val);
325 return (val, block);
326 }
327 }
328
329 let val = func.dfg.append_block_param(block, ty);
332 var_defs[block] = PackedOption::from(val);
333
334 match &mut self.ssa_blocks[block].sealed {
340 Sealed::Yes => self.begin_predecessors_lookup(val, block),
343 Sealed::No { undef_variables } => {
344 undef_variables.push(var, &mut self.variable_pool);
345 self.results.push(val);
346 }
347 }
348 (val, block)
349 }
350
351 pub fn declare_block(&mut self, block: Block) {
355 let _ = &mut self.ssa_blocks[block];
359 }
360
361 pub fn declare_block_predecessor(&mut self, block: Block, inst: Inst) {
371 debug_assert!(!self.is_sealed(block));
372 self.ssa_blocks[block]
373 .predecessors
374 .push(inst, &mut self.inst_pool);
375 }
376
377 pub fn remove_block_predecessor(&mut self, block: Block, inst: Inst) {
382 debug_assert!(!self.is_sealed(block));
383 let data = &mut self.ssa_blocks[block];
384 let pred = data
385 .predecessors
386 .as_slice(&self.inst_pool)
387 .iter()
388 .position(|&branch| branch == inst)
389 .expect("the predecessor you are trying to remove is not declared");
390 data.predecessors.swap_remove(pred, &mut self.inst_pool);
391 }
392
393 pub fn seal_block(&mut self, block: Block, func: &mut Function) -> SideEffects {
401 debug_assert!(
402 !self.is_sealed(block),
403 "Attempting to seal {block} which is already sealed."
404 );
405 self.seal_one_block(block, func);
406 mem::take(&mut self.side_effects)
407 }
408
409 pub fn seal_all_blocks(&mut self, func: &mut Function) -> SideEffects {
416 for block in self.ssa_blocks.keys() {
420 self.seal_one_block(block, func);
421 }
422 mem::take(&mut self.side_effects)
423 }
424
425 fn seal_one_block(&mut self, block: Block, func: &mut Function) {
427 let mut undef_variables =
430 match mem::replace(&mut self.ssa_blocks[block].sealed, Sealed::Yes) {
431 Sealed::No { undef_variables } => undef_variables,
432 Sealed::Yes => return,
433 };
434 let ssa_params = undef_variables.len(&self.variable_pool);
435
436 let predecessors = self.predecessors(block);
437 if predecessors.len() == 1 {
438 let pred = func.layout.inst_block(predecessors[0]).unwrap();
439 self.ssa_blocks[block].single_predecessor = PackedOption::from(pred);
440 }
441
442 for idx in 0..ssa_params {
446 let var = undef_variables.get(idx, &self.variable_pool).unwrap();
447
448 let block_params = func.dfg.block_params(block);
453
454 let val = block_params[block_params.len() - (ssa_params - idx)];
459
460 debug_assert!(self.calls.is_empty());
461 debug_assert!(self.results.is_empty());
462 self.begin_predecessors_lookup(val, block);
465 self.run_state_machine(func, var, func.dfg.value_type(val));
466 }
467
468 undef_variables.clear(&mut self.variable_pool);
469 }
470
471 fn begin_predecessors_lookup(&mut self, sentinel: Value, dest_block: Block) {
486 self.calls
487 .push(Call::FinishPredecessorsLookup(sentinel, dest_block));
488 self.calls.extend(
490 self.ssa_blocks[dest_block]
491 .predecessors
492 .as_slice(&self.inst_pool)
493 .iter()
494 .rev()
495 .copied()
496 .map(Call::UseVar),
497 );
498 }
499
500 fn finish_predecessors_lookup(
503 &mut self,
504 func: &mut Function,
505 sentinel: Value,
506 dest_block: Block,
507 ) -> Value {
508 let num_predecessors = self.predecessors(dest_block).len();
515 let results = self.results.drain(self.results.len() - num_predecessors..);
517
518 let pred_val = {
519 let mut iter = results
520 .as_slice()
521 .iter()
522 .map(|&val| func.dfg.resolve_aliases(val))
523 .filter(|&val| val != sentinel);
524 if let Some(val) = iter.next() {
525 if iter.all(|other| other == val) {
528 Some(val)
529 } else {
530 None
531 }
532 } else {
533 if !func.layout.is_block_inserted(dest_block) {
537 func.layout.append_block(dest_block);
538 }
539 self.side_effects
540 .instructions_added_to_blocks
541 .push(dest_block);
542 let zero = emit_zero(
543 func.dfg.value_type(sentinel),
544 FuncCursor::new(func).at_first_insertion_point(dest_block),
545 );
546 Some(zero)
547 }
548 };
549
550 if let Some(pred_val) = pred_val {
551 func.dfg.remove_block_param(sentinel);
556 func.dfg.change_to_alias(sentinel, pred_val);
557 pred_val
558 } else {
559 let mut preds = self.ssa_blocks[dest_block].predecessors;
562 let dfg = &mut func.stencil.dfg;
563 for (idx, &val) in results.as_slice().iter().enumerate() {
564 let pred = preds.get_mut(idx, &mut self.inst_pool).unwrap();
565 let branch = *pred;
566
567 let dests = dfg.insts[branch]
568 .branch_destination_mut(&mut dfg.jump_tables, &mut dfg.exception_tables);
569 assert!(
570 !dests.is_empty(),
571 "you have declared a non-branch instruction as a predecessor to a block!"
572 );
573 for block in dests {
574 if block.block(&dfg.value_lists) == dest_block {
575 block.append_argument(val, &mut dfg.value_lists);
576 }
577 }
578 }
579 sentinel
580 }
581 }
582
583 fn predecessors(&self, block: Block) -> &[Inst] {
585 self.ssa_blocks[block]
586 .predecessors
587 .as_slice(&self.inst_pool)
588 }
589
590 pub fn has_any_predecessors(&self, block: Block) -> bool {
592 !self.predecessors(block).is_empty()
593 }
594
595 pub fn is_sealed(&self, block: Block) -> bool {
597 matches!(self.ssa_blocks[block].sealed, Sealed::Yes)
598 }
599
600 fn run_state_machine(&mut self, func: &mut Function, var: Variable, ty: Type) -> Value {
606 while let Some(call) = self.calls.pop() {
608 match call {
609 Call::UseVar(branch) => {
610 let block = func.layout.inst_block(branch).unwrap();
611 self.use_var_nonlocal(func, var, ty, block);
612 }
613 Call::FinishPredecessorsLookup(sentinel, dest_block) => {
614 let val = self.finish_predecessors_lookup(func, sentinel, dest_block);
615 self.results.push(val);
616 }
617 }
618 }
619 debug_assert_eq!(self.results.len(), 1);
620 self.results.pop().unwrap()
621 }
622}
623
624#[cfg(test)]
625mod tests {
626 use crate::Variable;
627 use crate::ssa::SSABuilder;
628 use cranelift_codegen::cursor::{Cursor, FuncCursor};
629 use cranelift_codegen::entity::EntityRef;
630 use cranelift_codegen::ir;
631 use cranelift_codegen::ir::types::*;
632 use cranelift_codegen::ir::{Function, Inst, InstBuilder, JumpTableData, Opcode};
633 use cranelift_codegen::settings;
634 use cranelift_codegen::verify_function;
635
636 #[test]
637 fn simple_block() {
638 let mut func = Function::new();
639 let mut ssa = SSABuilder::default();
640 let block0 = func.dfg.make_block();
641 ssa.declare_block(block0);
649 let x_var = Variable::new(0);
650 let x_ssa = {
651 let mut cur = FuncCursor::new(&mut func);
652 cur.insert_block(block0);
653 cur.ins().iconst(I32, 1)
654 };
655 ssa.def_var(x_var, x_ssa, block0);
656 let y_var = Variable::new(1);
657 let y_ssa = {
658 let mut cur = FuncCursor::new(&mut func).at_bottom(block0);
659 cur.ins().iconst(I32, 2)
660 };
661 ssa.def_var(y_var, y_ssa, block0);
662 assert_eq!(ssa.use_var(&mut func, x_var, I32, block0).0, x_ssa);
663 assert_eq!(ssa.use_var(&mut func, y_var, I32, block0).0, y_ssa);
664
665 let z_var = Variable::new(2);
666 let x_use1 = ssa.use_var(&mut func, x_var, I32, block0).0;
667 let y_use1 = ssa.use_var(&mut func, y_var, I32, block0).0;
668 let z1_ssa = {
669 let mut cur = FuncCursor::new(&mut func).at_bottom(block0);
670 cur.ins().iadd(x_use1, y_use1)
671 };
672 ssa.def_var(z_var, z1_ssa, block0);
673 assert_eq!(ssa.use_var(&mut func, z_var, I32, block0).0, z1_ssa);
674
675 let x_use2 = ssa.use_var(&mut func, x_var, I32, block0).0;
676 let z_use1 = ssa.use_var(&mut func, z_var, I32, block0).0;
677 let z2_ssa = {
678 let mut cur = FuncCursor::new(&mut func).at_bottom(block0);
679 cur.ins().iadd(x_use2, z_use1)
680 };
681 ssa.def_var(z_var, z2_ssa, block0);
682 assert_eq!(ssa.use_var(&mut func, z_var, I32, block0).0, z2_ssa);
683 }
684
685 #[test]
686 fn sequence_of_blocks() {
687 let mut func = Function::new();
688 let mut ssa = SSABuilder::default();
689 let block0 = func.dfg.make_block();
690 let block1 = func.dfg.make_block();
691 let block2 = func.dfg.make_block();
692 {
704 let mut cur = FuncCursor::new(&mut func);
705 cur.insert_block(block0);
706 cur.insert_block(block1);
707 cur.insert_block(block2);
708 }
709
710 ssa.declare_block(block0);
712 ssa.seal_block(block0, &mut func);
713 let x_var = Variable::new(0);
714 let x_ssa = {
715 let mut cur = FuncCursor::new(&mut func).at_bottom(block0);
716 cur.ins().iconst(I32, 1)
717 };
718 ssa.def_var(x_var, x_ssa, block0);
719 let y_var = Variable::new(1);
720 let y_ssa = {
721 let mut cur = FuncCursor::new(&mut func).at_bottom(block0);
722 cur.ins().iconst(I32, 2)
723 };
724 ssa.def_var(y_var, y_ssa, block0);
725 let z_var = Variable::new(2);
726 let x_use1 = ssa.use_var(&mut func, x_var, I32, block0).0;
727 let y_use1 = ssa.use_var(&mut func, y_var, I32, block0).0;
728 let z1_ssa = {
729 let mut cur = FuncCursor::new(&mut func).at_bottom(block0);
730 cur.ins().iadd(x_use1, y_use1)
731 };
732 ssa.def_var(z_var, z1_ssa, block0);
733 let y_use2 = ssa.use_var(&mut func, y_var, I32, block0).0;
734 let brif_block0_block2_block1: Inst = {
735 let mut cur = FuncCursor::new(&mut func).at_bottom(block0);
736 cur.ins().brif(y_use2, block2, &[], block1, &[])
737 };
738
739 assert_eq!(ssa.use_var(&mut func, x_var, I32, block0).0, x_ssa);
740 assert_eq!(ssa.use_var(&mut func, y_var, I32, block0).0, y_ssa);
741 assert_eq!(ssa.use_var(&mut func, z_var, I32, block0).0, z1_ssa);
742
743 ssa.declare_block(block1);
745 ssa.declare_block_predecessor(block1, brif_block0_block2_block1);
746 ssa.seal_block(block1, &mut func);
747
748 let x_use2 = ssa.use_var(&mut func, x_var, I32, block1).0;
749 let z_use1 = ssa.use_var(&mut func, z_var, I32, block1).0;
750 let z2_ssa = {
751 let mut cur = FuncCursor::new(&mut func).at_bottom(block1);
752 cur.ins().iadd(x_use2, z_use1)
753 };
754 ssa.def_var(z_var, z2_ssa, block1);
755 let jump_block1_block2: Inst = {
756 let mut cur = FuncCursor::new(&mut func).at_bottom(block1);
757 cur.ins().jump(block2, &[])
758 };
759
760 assert_eq!(x_use2, x_ssa);
761 assert_eq!(z_use1, z1_ssa);
762 assert_eq!(ssa.use_var(&mut func, z_var, I32, block1).0, z2_ssa);
763
764 ssa.declare_block(block2);
766 ssa.declare_block_predecessor(block2, brif_block0_block2_block1);
767 ssa.declare_block_predecessor(block2, jump_block1_block2);
768 ssa.seal_block(block2, &mut func);
769 let x_use3 = ssa.use_var(&mut func, x_var, I32, block2).0;
770 let y_use3 = ssa.use_var(&mut func, y_var, I32, block2).0;
771 let y2_ssa = {
772 let mut cur = FuncCursor::new(&mut func).at_bottom(block2);
773 cur.ins().iadd(x_use3, y_use3)
774 };
775 ssa.def_var(y_var, y2_ssa, block2);
776
777 assert_eq!(x_ssa, x_use3);
778 assert_eq!(y_ssa, y_use3);
779 match func.dfg.insts[brif_block0_block2_block1] {
780 ir::InstructionData::Brif {
781 blocks: [block_then, block_else],
782 ..
783 } => {
784 assert_eq!(block_then.block(&func.dfg.value_lists), block2);
785 assert_eq!(block_then.args(&func.dfg.value_lists).len(), 0);
786 assert_eq!(block_else.block(&func.dfg.value_lists), block1);
787 assert_eq!(block_else.args(&func.dfg.value_lists).len(), 0);
788 }
789 _ => assert!(false),
790 };
791 match func.dfg.insts[brif_block0_block2_block1] {
792 ir::InstructionData::Brif {
793 blocks: [block_then, block_else],
794 ..
795 } => {
796 assert_eq!(block_then.block(&func.dfg.value_lists), block2);
797 assert_eq!(block_then.args(&func.dfg.value_lists).len(), 0);
798 assert_eq!(block_else.block(&func.dfg.value_lists), block1);
799 assert_eq!(block_else.args(&func.dfg.value_lists).len(), 0);
800 }
801 _ => assert!(false),
802 };
803 match func.dfg.insts[jump_block1_block2] {
804 ir::InstructionData::Jump {
805 destination: dest, ..
806 } => {
807 assert_eq!(dest.block(&func.dfg.value_lists), block2);
808 assert_eq!(dest.args(&func.dfg.value_lists).len(), 0);
809 }
810 _ => assert!(false),
811 };
812 }
813
814 #[test]
815 fn program_with_loop() {
816 let mut func = Function::new();
817 let mut ssa = SSABuilder::default();
818 let block0 = func.dfg.make_block();
819 let block1 = func.dfg.make_block();
820 let block2 = func.dfg.make_block();
821 let block3 = func.dfg.make_block();
822 {
823 let mut cur = FuncCursor::new(&mut func);
824 cur.insert_block(block0);
825 cur.insert_block(block1);
826 cur.insert_block(block2);
827 cur.insert_block(block3);
828 }
829 ssa.declare_block(block0);
847 ssa.seal_block(block0, &mut func);
848 let x_var = Variable::new(0);
849 let x1 = {
850 let mut cur = FuncCursor::new(&mut func).at_bottom(block0);
851 cur.ins().iconst(I32, 1)
852 };
853 ssa.def_var(x_var, x1, block0);
854 let y_var = Variable::new(1);
855 let y1 = {
856 let mut cur = FuncCursor::new(&mut func).at_bottom(block0);
857 cur.ins().iconst(I32, 2)
858 };
859 ssa.def_var(y_var, y1, block0);
860 let z_var = Variable::new(2);
861 let x2 = ssa.use_var(&mut func, x_var, I32, block0).0;
862 let y2 = ssa.use_var(&mut func, y_var, I32, block0).0;
863 let z1 = {
864 let mut cur = FuncCursor::new(&mut func).at_bottom(block0);
865 cur.ins().iadd(x2, y2)
866 };
867 ssa.def_var(z_var, z1, block0);
868 let jump_block0_block1 = {
869 let mut cur = FuncCursor::new(&mut func).at_bottom(block0);
870 cur.ins().jump(block1, &[])
871 };
872 assert_eq!(ssa.use_var(&mut func, x_var, I32, block0).0, x1);
873 assert_eq!(ssa.use_var(&mut func, y_var, I32, block0).0, y1);
874 assert_eq!(x2, x1);
875 assert_eq!(y2, y1);
876
877 ssa.declare_block(block1);
879 ssa.declare_block_predecessor(block1, jump_block0_block1);
880 let z2 = ssa.use_var(&mut func, z_var, I32, block1).0;
881 let y3 = ssa.use_var(&mut func, y_var, I32, block1).0;
882 let z3 = {
883 let mut cur = FuncCursor::new(&mut func).at_bottom(block1);
884 cur.ins().iadd(z2, y3)
885 };
886 ssa.def_var(z_var, z3, block1);
887 let y4 = ssa.use_var(&mut func, y_var, I32, block1).0;
888 assert_eq!(y4, y3);
889 let brif_block1_block3_block2 = {
890 let mut cur = FuncCursor::new(&mut func).at_bottom(block1);
891 cur.ins().brif(y4, block3, &[], block2, &[])
892 };
893
894 ssa.declare_block(block2);
896 ssa.declare_block_predecessor(block2, brif_block1_block3_block2);
897 ssa.seal_block(block2, &mut func);
898 let z4 = ssa.use_var(&mut func, z_var, I32, block2).0;
899 assert_eq!(z4, z3);
900 let x3 = ssa.use_var(&mut func, x_var, I32, block2).0;
901 let z5 = {
902 let mut cur = FuncCursor::new(&mut func).at_bottom(block2);
903 cur.ins().isub(z4, x3)
904 };
905 ssa.def_var(z_var, z5, block2);
906 let y5 = ssa.use_var(&mut func, y_var, I32, block2).0;
907 assert_eq!(y5, y3);
908 {
909 let mut cur = FuncCursor::new(&mut func).at_bottom(block2);
910 cur.ins().return_(&[y5])
911 };
912
913 ssa.declare_block(block3);
915 ssa.declare_block_predecessor(block3, brif_block1_block3_block2);
916 ssa.seal_block(block3, &mut func);
917 let y6 = ssa.use_var(&mut func, y_var, I32, block3).0;
918 assert_eq!(y6, y3);
919 let x4 = ssa.use_var(&mut func, x_var, I32, block3).0;
920 assert_eq!(x4, x3);
921 let y7 = {
922 let mut cur = FuncCursor::new(&mut func).at_bottom(block3);
923 cur.ins().isub(y6, x4)
924 };
925 ssa.def_var(y_var, y7, block3);
926 let jump_block3_block1 = {
927 let mut cur = FuncCursor::new(&mut func).at_bottom(block3);
928 cur.ins().jump(block1, &[])
929 };
930
931 ssa.declare_block_predecessor(block1, jump_block3_block1);
933 ssa.seal_block(block1, &mut func);
934 assert_eq!(func.dfg.block_params(block1)[0], z2);
935 assert_eq!(func.dfg.block_params(block1)[1], y3);
936 assert_eq!(func.dfg.resolve_aliases(x3), x1);
937 }
938
939 #[test]
940 fn br_table_with_args() {
941 let mut func = Function::new();
958 let mut ssa = SSABuilder::default();
959 let block0 = func.dfg.make_block();
960 let block1 = func.dfg.make_block();
961 let block2 = func.dfg.make_block();
962 {
963 let mut cur = FuncCursor::new(&mut func);
964 cur.insert_block(block0);
965 cur.insert_block(block1);
966 cur.insert_block(block2);
967 }
968
969 let x1 = {
971 let mut cur = FuncCursor::new(&mut func).at_bottom(block0);
972 cur.ins().iconst(I32, 1)
973 };
974 ssa.declare_block(block0);
975 ssa.seal_block(block0, &mut func);
976 let x_var = Variable::new(0);
977 ssa.def_var(x_var, x1, block0);
978 ssa.use_var(&mut func, x_var, I32, block0).0;
979 let br_table = {
980 let jump_table = JumpTableData::new(
981 func.dfg.block_call(block2, &[]),
982 &[
983 func.dfg.block_call(block2, &[]),
984 func.dfg.block_call(block1, &[]),
985 ],
986 );
987 let jt = func.create_jump_table(jump_table);
988 let mut cur = FuncCursor::new(&mut func).at_bottom(block0);
989 cur.ins().br_table(x1, jt)
990 };
991
992 ssa.declare_block(block1);
994 ssa.declare_block_predecessor(block1, br_table);
995 ssa.seal_block(block1, &mut func);
996 let x2 = {
997 let mut cur = FuncCursor::new(&mut func).at_bottom(block1);
998 cur.ins().iconst(I32, 2)
999 };
1000 ssa.def_var(x_var, x2, block1);
1001 let jump_block1_block2 = {
1002 let mut cur = FuncCursor::new(&mut func).at_bottom(block1);
1003 cur.ins().jump(block2, &[])
1004 };
1005
1006 ssa.declare_block(block2);
1008 ssa.declare_block_predecessor(block2, jump_block1_block2);
1009 ssa.declare_block_predecessor(block2, br_table);
1010 ssa.seal_block(block2, &mut func);
1011 let x3 = ssa.use_var(&mut func, x_var, I32, block2).0;
1012 let x4 = {
1013 let mut cur = FuncCursor::new(&mut func).at_bottom(block2);
1014 cur.ins().iadd_imm(x3, 1)
1015 };
1016 ssa.def_var(x_var, x4, block2);
1017 {
1018 let mut cur = FuncCursor::new(&mut func).at_bottom(block2);
1019 cur.ins().return_(&[])
1020 };
1021
1022 let flags = settings::Flags::new(settings::builder());
1023 match verify_function(&func, &flags) {
1024 Ok(()) => {}
1025 Err(_errors) => {
1026 #[cfg(feature = "std")]
1027 panic!("{}", _errors);
1028 #[cfg(not(feature = "std"))]
1029 panic!("function failed to verify");
1030 }
1031 }
1032 }
1033
1034 #[test]
1035 fn undef_values_reordering() {
1036 let mut func = Function::new();
1048 let mut ssa = SSABuilder::default();
1049 let block0 = func.dfg.make_block();
1050 let block1 = func.dfg.make_block();
1051 {
1052 let mut cur = FuncCursor::new(&mut func);
1053 cur.insert_block(block0);
1054 cur.insert_block(block1);
1055 }
1056
1057 ssa.declare_block(block0);
1059 let x_var = Variable::new(0);
1060 ssa.seal_block(block0, &mut func);
1061 let x1 = {
1062 let mut cur = FuncCursor::new(&mut func).at_bottom(block0);
1063 cur.ins().iconst(I32, 0)
1064 };
1065 ssa.def_var(x_var, x1, block0);
1066 let y_var = Variable::new(1);
1067 let y1 = {
1068 let mut cur = FuncCursor::new(&mut func).at_bottom(block0);
1069 cur.ins().iconst(I32, 1)
1070 };
1071 ssa.def_var(y_var, y1, block0);
1072 let z_var = Variable::new(2);
1073 let z1 = {
1074 let mut cur = FuncCursor::new(&mut func).at_bottom(block0);
1075 cur.ins().iconst(I32, 2)
1076 };
1077 ssa.def_var(z_var, z1, block0);
1078 let jump_block0_block1 = {
1079 let mut cur = FuncCursor::new(&mut func).at_bottom(block0);
1080 cur.ins().jump(block1, &[])
1081 };
1082
1083 ssa.declare_block(block1);
1085 ssa.declare_block_predecessor(block1, jump_block0_block1);
1086 let z2 = ssa.use_var(&mut func, z_var, I32, block1).0;
1087 assert_eq!(func.dfg.block_params(block1)[0], z2);
1088 let x2 = ssa.use_var(&mut func, x_var, I32, block1).0;
1089 assert_eq!(func.dfg.block_params(block1)[1], x2);
1090 let x3 = {
1091 let mut cur = FuncCursor::new(&mut func).at_bottom(block1);
1092 cur.ins().iadd(x2, z2)
1093 };
1094 ssa.def_var(x_var, x3, block1);
1095 let x4 = ssa.use_var(&mut func, x_var, I32, block1).0;
1096 let y3 = ssa.use_var(&mut func, y_var, I32, block1).0;
1097 assert_eq!(func.dfg.block_params(block1)[2], y3);
1098 let y4 = {
1099 let mut cur = FuncCursor::new(&mut func).at_bottom(block1);
1100 cur.ins().isub(y3, x4)
1101 };
1102 ssa.def_var(y_var, y4, block1);
1103 let jump_block1_block1 = {
1104 let mut cur = FuncCursor::new(&mut func).at_bottom(block1);
1105 cur.ins().jump(block1, &[])
1106 };
1107 ssa.declare_block_predecessor(block1, jump_block1_block1);
1108 ssa.seal_block(block1, &mut func);
1109 assert_eq!(func.dfg.block_params(block1)[1], y3);
1112 assert_eq!(func.dfg.block_params(block1)[0], x2);
1113 }
1114
1115 #[test]
1116 fn undef() {
1117 let mut func = Function::new();
1119 let mut ssa = SSABuilder::default();
1120 let block0 = func.dfg.make_block();
1121 ssa.declare_block(block0);
1122 ssa.seal_block(block0, &mut func);
1123 let i32_var = Variable::new(0);
1124 let f32_var = Variable::new(1);
1125 let f64_var = Variable::new(2);
1126 let i8_var = Variable::new(3);
1127 let f32x4_var = Variable::new(4);
1128 ssa.use_var(&mut func, i32_var, I32, block0);
1129 ssa.use_var(&mut func, f32_var, F32, block0);
1130 ssa.use_var(&mut func, f64_var, F64, block0);
1131 ssa.use_var(&mut func, i8_var, I8, block0);
1132 ssa.use_var(&mut func, f32x4_var, F32X4, block0);
1133 assert_eq!(func.dfg.num_block_params(block0), 0);
1134 }
1135
1136 #[test]
1137 fn undef_in_entry() {
1138 let mut func = Function::new();
1141 let mut ssa = SSABuilder::default();
1142 let block0 = func.dfg.make_block();
1143 ssa.declare_block(block0);
1144 ssa.seal_block(block0, &mut func);
1145 let x_var = Variable::new(0);
1146 assert_eq!(func.dfg.num_block_params(block0), 0);
1147 ssa.use_var(&mut func, x_var, I32, block0);
1148 assert_eq!(func.dfg.num_block_params(block0), 0);
1149 assert_eq!(
1150 func.dfg.insts[func.layout.first_inst(block0).unwrap()].opcode(),
1151 Opcode::Iconst
1152 );
1153 }
1154
1155 #[test]
1156 fn undef_in_entry_sealed_after() {
1157 let mut func = Function::new();
1161 let mut ssa = SSABuilder::default();
1162 let block0 = func.dfg.make_block();
1163 ssa.declare_block(block0);
1164 let x_var = Variable::new(0);
1165 assert_eq!(func.dfg.num_block_params(block0), 0);
1166 ssa.use_var(&mut func, x_var, I32, block0);
1167 assert_eq!(func.dfg.num_block_params(block0), 1);
1168 ssa.seal_block(block0, &mut func);
1169 assert_eq!(func.dfg.num_block_params(block0), 0);
1170 assert_eq!(
1171 func.dfg.insts[func.layout.first_inst(block0).unwrap()].opcode(),
1172 Opcode::Iconst
1173 );
1174 }
1175
1176 #[test]
1177 fn unreachable_use() {
1178 let mut func = Function::new();
1184 let mut ssa = SSABuilder::default();
1185 let block0 = func.dfg.make_block();
1186 let block1 = func.dfg.make_block();
1187 {
1188 let mut cur = FuncCursor::new(&mut func);
1189 cur.insert_block(block0);
1190 cur.insert_block(block1);
1191 }
1192
1193 ssa.declare_block(block0);
1195 ssa.seal_block(block0, &mut func);
1196 {
1197 let mut cur = FuncCursor::new(&mut func).at_bottom(block0);
1198 cur.ins().return_(&[]);
1199 }
1200
1201 ssa.declare_block(block1);
1203 {
1204 let mut cur = FuncCursor::new(&mut func).at_bottom(block1);
1205 let x_var = Variable::new(0);
1206 let x_val = ssa.use_var(&mut cur.func, x_var, I32, block1).0;
1207 let brif = cur.ins().brif(x_val, block1, &[], block1, &[]);
1208 ssa.declare_block_predecessor(block1, brif);
1209 }
1210 ssa.seal_block(block1, &mut func);
1211
1212 let flags = settings::Flags::new(settings::builder());
1213 match verify_function(&func, &flags) {
1214 Ok(()) => {}
1215 Err(_errors) => {
1216 #[cfg(feature = "std")]
1217 panic!("{}", _errors);
1218 #[cfg(not(feature = "std"))]
1219 panic!("function failed to verify");
1220 }
1221 }
1222 }
1223
1224 #[test]
1225 fn unreachable_use_with_multiple_preds() {
1226 let mut func = Function::new();
1234 let mut ssa = SSABuilder::default();
1235 let block0 = func.dfg.make_block();
1236 let block1 = func.dfg.make_block();
1237 let block2 = func.dfg.make_block();
1238 {
1239 let mut cur = FuncCursor::new(&mut func);
1240 cur.insert_block(block0);
1241 cur.insert_block(block1);
1242 cur.insert_block(block2);
1243 }
1244
1245 ssa.declare_block(block0);
1247 ssa.seal_block(block0, &mut func);
1248 {
1249 let mut cur = FuncCursor::new(&mut func).at_bottom(block0);
1250 cur.ins().return_(&[]);
1251 }
1252
1253 ssa.declare_block(block1);
1255 let brif = {
1256 let mut cur = FuncCursor::new(&mut func).at_bottom(block1);
1257 let x_var = Variable::new(0);
1258 let x_val = ssa.use_var(&mut cur.func, x_var, I32, block1).0;
1259 cur.ins().brif(x_val, block2, &[], block1, &[])
1260 };
1261
1262 ssa.declare_block(block2);
1264 ssa.declare_block_predecessor(block1, brif);
1265 ssa.declare_block_predecessor(block2, brif);
1266 ssa.seal_block(block2, &mut func);
1267 let jump_block2_block1 = {
1268 let mut cur = FuncCursor::new(&mut func).at_bottom(block2);
1269 cur.ins().jump(block1, &[])
1270 };
1271
1272 ssa.declare_block_predecessor(block1, jump_block2_block1);
1274 ssa.seal_block(block1, &mut func);
1275 let flags = settings::Flags::new(settings::builder());
1276 match verify_function(&func, &flags) {
1277 Ok(()) => {}
1278 Err(_errors) => {
1279 #[cfg(feature = "std")]
1280 panic!("{}", _errors);
1281 #[cfg(not(feature = "std"))]
1282 panic!("function failed to verify");
1283 }
1284 }
1285 }
1286
1287 #[test]
1288 fn reassign_with_predecessor_loop_hangs() {
1289 let mut func = Function::new();
1301 let mut ssa = SSABuilder::default();
1302 let block0 = func.dfg.make_block();
1303 let block1 = func.dfg.make_block();
1304 let block2 = func.dfg.make_block();
1305 let var0 = Variable::new(0);
1306
1307 {
1308 let mut cur = FuncCursor::new(&mut func);
1309 for block in [block0, block1, block2] {
1310 cur.insert_block(block);
1311 ssa.declare_block(block);
1312 }
1313 }
1314
1315 {
1317 let mut cur = FuncCursor::new(&mut func).at_bottom(block0);
1318
1319 let var0_iconst = cur.ins().iconst(I32, 0);
1320 ssa.def_var(var0, var0_iconst, block0);
1321
1322 cur.ins().return_(&[]);
1323 }
1324
1325 {
1327 let mut cur = FuncCursor::new(&mut func).at_bottom(block1);
1328
1329 let jump = cur.ins().jump(block2, &[]);
1330 ssa.declare_block_predecessor(block2, jump);
1331 }
1332
1333 {
1335 let mut cur = FuncCursor::new(&mut func).at_bottom(block2);
1336
1337 let _ = ssa.use_var(&mut cur.func, var0, I32, block2).0;
1338 let var0_iconst = cur.ins().iconst(I32, 1);
1339 ssa.def_var(var0, var0_iconst, block2);
1340
1341 let jump = cur.ins().jump(block1, &[]);
1342 ssa.declare_block_predecessor(block1, jump);
1343 }
1344
1345 ssa.seal_all_blocks(&mut func);
1347 }
1348}