1use crate::entity::{self, PrimaryMap, SecondaryMap};
4use crate::ir;
5use crate::ir::builder::ReplaceBuilder;
6use crate::ir::dynamic_type::{DynamicTypeData, DynamicTypes};
7use crate::ir::instructions::{CallInfo, InstructionData};
8use crate::ir::pcc::Fact;
9use crate::ir::user_stack_maps::{UserStackMapEntry, UserStackMapEntryVec};
10use crate::ir::{
11 Block, BlockArg, BlockCall, ConstantData, ConstantPool, DynamicType, ExceptionTables,
12 ExtFuncData, FuncRef, Immediate, Inst, JumpTables, RelSourceLoc, SigRef, Signature, Type,
13 Value, ValueLabelAssignments, ValueList, ValueListPool, types,
14};
15use crate::packed_option::ReservedValue;
16use crate::write::write_operands;
17use core::fmt;
18use core::iter;
19use core::mem;
20use core::ops::{Index, IndexMut};
21use core::u16;
22
23use alloc::collections::BTreeMap;
24#[cfg(feature = "enable-serde")]
25use serde_derive::{Deserialize, Serialize};
26use smallvec::SmallVec;
27
28#[derive(Clone, PartialEq, Hash)]
30#[cfg_attr(feature = "enable-serde", derive(Serialize, Deserialize))]
31pub struct Insts(PrimaryMap<Inst, InstructionData>);
32
33impl Index<Inst> for Insts {
35 type Output = InstructionData;
36
37 fn index(&self, inst: Inst) -> &InstructionData {
38 self.0.index(inst)
39 }
40}
41
42impl IndexMut<Inst> for Insts {
44 fn index_mut(&mut self, inst: Inst) -> &mut InstructionData {
45 self.0.index_mut(inst)
46 }
47}
48
49#[derive(Clone, PartialEq, Hash)]
51#[cfg_attr(feature = "enable-serde", derive(Serialize, Deserialize))]
52pub struct Blocks(PrimaryMap<Block, BlockData>);
53
54impl Blocks {
55 pub fn add(&mut self) -> Block {
57 self.0.push(BlockData::new())
58 }
59
60 pub fn len(&self) -> usize {
65 self.0.len()
66 }
67
68 pub fn reserve(&mut self, additional: usize) {
71 self.0.reserve(additional);
72 }
73
74 pub fn is_valid(&self, block: Block) -> bool {
76 self.0.is_valid(block)
77 }
78
79 pub fn iter(&self) -> impl Iterator<Item = Block> {
84 self.0.keys()
85 }
86}
87
88impl Index<Block> for Blocks {
89 type Output = BlockData;
90
91 fn index(&self, block: Block) -> &BlockData {
92 &self.0[block]
93 }
94}
95
96impl IndexMut<Block> for Blocks {
97 fn index_mut(&mut self, block: Block) -> &mut BlockData {
98 &mut self.0[block]
99 }
100}
101
102#[derive(Clone, PartialEq, Hash)]
110#[cfg_attr(feature = "enable-serde", derive(Serialize, Deserialize))]
111pub struct DataFlowGraph {
112 pub insts: Insts,
116
117 results: SecondaryMap<Inst, ValueList>,
122
123 user_stack_maps: alloc::collections::BTreeMap<Inst, UserStackMapEntryVec>,
125
126 pub blocks: Blocks,
131
132 pub dynamic_types: DynamicTypes,
134
135 pub value_lists: ValueListPool,
143
144 values: PrimaryMap<Value, ValueDataPacked>,
146
147 pub facts: SecondaryMap<Value, Option<Fact>>,
149
150 pub signatures: PrimaryMap<SigRef, Signature>,
153
154 pub ext_funcs: PrimaryMap<FuncRef, ExtFuncData>,
156
157 pub values_labels: Option<BTreeMap<Value, ValueLabelAssignments>>,
159
160 pub constants: ConstantPool,
162
163 pub immediates: PrimaryMap<Immediate, ConstantData>,
165
166 pub jump_tables: JumpTables,
168
169 pub exception_tables: ExceptionTables,
171}
172
173impl DataFlowGraph {
174 pub fn new() -> Self {
176 Self {
177 insts: Insts(PrimaryMap::new()),
178 results: SecondaryMap::new(),
179 user_stack_maps: alloc::collections::BTreeMap::new(),
180 blocks: Blocks(PrimaryMap::new()),
181 dynamic_types: DynamicTypes::new(),
182 value_lists: ValueListPool::new(),
183 values: PrimaryMap::new(),
184 facts: SecondaryMap::new(),
185 signatures: PrimaryMap::new(),
186 ext_funcs: PrimaryMap::new(),
187 values_labels: None,
188 constants: ConstantPool::new(),
189 immediates: PrimaryMap::new(),
190 jump_tables: JumpTables::new(),
191 exception_tables: ExceptionTables::new(),
192 }
193 }
194
195 pub fn clear(&mut self) {
197 self.insts.0.clear();
198 self.results.clear();
199 self.user_stack_maps.clear();
200 self.blocks.0.clear();
201 self.dynamic_types.clear();
202 self.value_lists.clear();
203 self.values.clear();
204 self.signatures.clear();
205 self.ext_funcs.clear();
206 self.values_labels = None;
207 self.constants.clear();
208 self.immediates.clear();
209 self.jump_tables.clear();
210 self.facts.clear();
211 }
212
213 pub fn num_insts(&self) -> usize {
218 self.insts.0.len()
219 }
220
221 pub fn inst_is_valid(&self, inst: Inst) -> bool {
223 self.insts.0.is_valid(inst)
224 }
225
226 pub fn num_blocks(&self) -> usize {
231 self.blocks.len()
232 }
233
234 pub fn block_is_valid(&self, block: Block) -> bool {
236 self.blocks.is_valid(block)
237 }
238
239 pub fn block_call<'a>(
241 &mut self,
242 block: Block,
243 args: impl IntoIterator<Item = &'a BlockArg>,
244 ) -> BlockCall {
245 BlockCall::new(block, args.into_iter().copied(), &mut self.value_lists)
246 }
247
248 pub fn num_values(&self) -> usize {
250 self.values.len()
251 }
252
253 pub fn values_and_defs(&self) -> impl Iterator<Item = (Value, ValueDef)> + '_ {
255 self.values().map(|value| (value, self.value_def(value)))
256 }
257
258 pub fn collect_debug_info(&mut self) {
260 if self.values_labels.is_none() {
261 self.values_labels = Some(Default::default());
262 }
263 }
264
265 pub fn add_value_label_alias(&mut self, to_alias: Value, from: RelSourceLoc, value: Value) {
268 if let Some(values_labels) = self.values_labels.as_mut() {
269 values_labels.insert(to_alias, ir::ValueLabelAssignments::Alias { from, value });
270 }
271 }
272}
273
274fn maybe_resolve_aliases(
279 values: &PrimaryMap<Value, ValueDataPacked>,
280 value: Value,
281) -> Option<Value> {
282 let mut v = value;
283
284 for _ in 0..=values.len() {
286 if let ValueData::Alias { original, .. } = ValueData::from(values[v]) {
287 v = original;
288 } else {
289 return Some(v);
290 }
291 }
292
293 None
294}
295
296fn resolve_aliases(values: &PrimaryMap<Value, ValueDataPacked>, value: Value) -> Value {
300 if let Some(v) = maybe_resolve_aliases(values, value) {
301 v
302 } else {
303 panic!("Value alias loop detected for {value}");
304 }
305}
306
307pub struct Values<'a> {
309 inner: entity::Iter<'a, Value, ValueDataPacked>,
310}
311
312fn valid_valuedata(data: ValueDataPacked) -> bool {
314 let data = ValueData::from(data);
315 if let ValueData::Alias {
316 ty: types::INVALID,
317 original,
318 } = data
319 {
320 if original == Value::reserved_value() {
321 return false;
322 }
323 }
324 true
325}
326
327impl<'a> Iterator for Values<'a> {
328 type Item = Value;
329
330 fn next(&mut self) -> Option<Self::Item> {
331 self.inner
332 .by_ref()
333 .find(|kv| valid_valuedata(*kv.1))
334 .map(|kv| kv.0)
335 }
336
337 fn size_hint(&self) -> (usize, Option<usize>) {
338 self.inner.size_hint()
339 }
340}
341
342impl ExactSizeIterator for Values<'_> {
343 fn len(&self) -> usize {
344 self.inner.len()
345 }
346}
347
348impl DataFlowGraph {
352 fn make_value(&mut self, data: ValueData) -> Value {
354 self.values.push(data.into())
355 }
356
357 pub fn len_values(&self) -> usize {
359 self.values.len()
360 }
361
362 pub fn values<'a>(&'a self) -> Values<'a> {
364 Values {
365 inner: self.values.iter(),
366 }
367 }
368
369 pub fn value_is_valid(&self, v: Value) -> bool {
371 self.values.is_valid(v)
372 }
373
374 pub fn value_is_real(&self, value: Value) -> bool {
376 self.value_is_valid(value) && !matches!(self.values[value].into(), ValueData::Alias { .. })
379 }
380
381 pub fn value_type(&self, v: Value) -> Type {
383 self.values[v].ty()
384 }
385
386 pub fn value_def(&self, v: Value) -> ValueDef {
391 match ValueData::from(self.values[v]) {
392 ValueData::Inst { inst, num, .. } => ValueDef::Result(inst, num as usize),
393 ValueData::Param { block, num, .. } => ValueDef::Param(block, num as usize),
394 ValueData::Alias { original, .. } => {
395 self.value_def(self.resolve_aliases(original))
398 }
399 ValueData::Union { x, y, .. } => ValueDef::Union(x, y),
400 }
401 }
402
403 pub fn value_is_attached(&self, v: Value) -> bool {
410 use self::ValueData::*;
411 match ValueData::from(self.values[v]) {
412 Inst { inst, num, .. } => Some(&v) == self.inst_results(inst).get(num as usize),
413 Param { block, num, .. } => Some(&v) == self.block_params(block).get(num as usize),
414 Alias { .. } => false,
415 Union { .. } => false,
416 }
417 }
418
419 pub fn resolve_aliases(&self, value: Value) -> Value {
423 resolve_aliases(&self.values, value)
424 }
425
426 pub fn resolve_all_aliases(&mut self) {
429 let invalid_value = ValueDataPacked::from(ValueData::Alias {
430 ty: types::INVALID,
431 original: Value::reserved_value(),
432 });
433
434 for mut src in self.values.keys() {
439 let value_data = self.values[src];
440 if value_data == invalid_value {
441 continue;
442 }
443 if let ValueData::Alias { mut original, .. } = value_data.into() {
444 let resolved = ValueDataPacked::from(ValueData::Alias {
447 ty: types::INVALID,
448 original: resolve_aliases(&self.values, original),
449 });
450 loop {
454 self.values[src] = resolved;
455 src = original;
456 if let ValueData::Alias { original: next, .. } = self.values[src].into() {
457 original = next;
458 } else {
459 break;
460 }
461 }
462 }
463 }
464
465 for inst in self.insts.0.values_mut() {
470 inst.map_values(
471 &mut self.value_lists,
472 &mut self.jump_tables,
473 &mut self.exception_tables,
474 |arg| {
475 if let ValueData::Alias { original, .. } = self.values[arg].into() {
476 original
477 } else {
478 arg
479 }
480 },
481 );
482 }
483
484 for value in self.facts.keys() {
497 if let ValueData::Alias { original, .. } = self.values[value].into() {
498 if let Some(new_fact) = self.facts[value].take() {
499 match &mut self.facts[original] {
500 Some(old_fact) => *old_fact = Fact::intersect(old_fact, &new_fact),
501 old_fact => *old_fact = Some(new_fact),
502 }
503 }
504 }
505 }
506
507 if let Some(values_labels) = &mut self.values_labels {
510 values_labels.retain(|&k, _| !matches!(self.values[k].into(), ValueData::Alias { .. }));
513
514 for value_label in values_labels.values_mut() {
517 if let ValueLabelAssignments::Alias { value, .. } = value_label {
518 if let ValueData::Alias { original, .. } = self.values[*value].into() {
519 *value = original;
520 }
521 }
522 }
523 }
524
525 for value in self.values.values_mut() {
530 if let ValueData::Alias { .. } = ValueData::from(*value) {
531 *value = invalid_value;
532 }
533 }
534 }
535
536 pub fn change_to_alias(&mut self, dest: Value, src: Value) {
543 debug_assert!(!self.value_is_attached(dest));
544 let original = self.resolve_aliases(src);
547 debug_assert_ne!(
548 dest, original,
549 "Aliasing {dest} to {src} would create a loop"
550 );
551 let ty = self.value_type(original);
552 debug_assert_eq!(
553 self.value_type(dest),
554 ty,
555 "Aliasing {} to {} would change its type {} to {}",
556 dest,
557 src,
558 self.value_type(dest),
559 ty
560 );
561 debug_assert_ne!(ty, types::INVALID);
562
563 self.values[dest] = ValueData::Alias { ty, original }.into();
564 }
565
566 pub fn replace_with_aliases(&mut self, dest_inst: Inst, original_inst: Inst) {
576 debug_assert_ne!(
577 dest_inst, original_inst,
578 "Replacing {dest_inst} with itself would create a loop"
579 );
580
581 let dest_results = self.results[dest_inst].as_slice(&self.value_lists);
582 let original_results = self.results[original_inst].as_slice(&self.value_lists);
583
584 debug_assert_eq!(
585 dest_results.len(),
586 original_results.len(),
587 "Replacing {dest_inst} with {original_inst} would produce a different number of results."
588 );
589
590 for (&dest, &original) in dest_results.iter().zip(original_results) {
591 let ty = self.value_type(original);
592 debug_assert_eq!(
593 self.value_type(dest),
594 ty,
595 "Aliasing {} to {} would change its type {} to {}",
596 dest,
597 original,
598 self.value_type(dest),
599 ty
600 );
601 debug_assert_ne!(ty, types::INVALID);
602
603 self.values[dest] = ValueData::Alias { ty, original }.into();
604 }
605
606 self.clear_results(dest_inst);
607 }
608
609 pub fn user_stack_map_entries(&self, inst: Inst) -> Option<&[UserStackMapEntry]> {
611 self.user_stack_maps.get(&inst).map(|es| &**es)
612 }
613
614 pub fn append_user_stack_map_entry(&mut self, inst: Inst, entry: UserStackMapEntry) {
620 let opcode = self.insts[inst].opcode();
621 assert!(opcode.is_safepoint());
622 self.user_stack_maps.entry(inst).or_default().push(entry);
623 }
624
625 pub fn append_user_stack_map_entries(
631 &mut self,
632 inst: Inst,
633 entries: impl IntoIterator<Item = UserStackMapEntry>,
634 ) {
635 for entry in entries {
636 self.append_user_stack_map_entry(inst, entry);
637 }
638 }
639
640 pub(crate) fn take_user_stack_map_entries(
643 &mut self,
644 inst: Inst,
645 ) -> Option<UserStackMapEntryVec> {
646 self.user_stack_maps.remove(&inst)
647 }
648}
649
650#[derive(Clone, Copy, Debug, PartialEq, Eq)]
652pub enum ValueDef {
653 Result(Inst, usize),
655 Param(Block, usize),
657 Union(Value, Value),
659}
660
661impl ValueDef {
662 pub fn unwrap_inst(&self) -> Inst {
664 self.inst().expect("Value is not an instruction result")
665 }
666
667 pub fn inst(&self) -> Option<Inst> {
669 match *self {
670 Self::Result(inst, _) => Some(inst),
671 _ => None,
672 }
673 }
674
675 pub fn unwrap_block(&self) -> Block {
677 match *self {
678 Self::Param(block, _) => block,
679 _ => panic!("Value is not a block parameter"),
680 }
681 }
682
683 pub fn num(self) -> usize {
688 match self {
689 Self::Result(_, n) | Self::Param(_, n) => n,
690 Self::Union(_, _) => 0,
691 }
692 }
693}
694
695#[derive(Clone, Debug, PartialEq, Hash)]
697#[cfg_attr(feature = "enable-serde", derive(Serialize, Deserialize))]
698enum ValueData {
699 Inst { ty: Type, num: u16, inst: Inst },
701
702 Param { ty: Type, num: u16, block: Block },
704
705 Alias { ty: Type, original: Value },
709
710 Union { ty: Type, x: Value, y: Value },
714}
715
716#[derive(Clone, Copy, Debug, PartialEq, Hash)]
729#[cfg_attr(feature = "enable-serde", derive(Serialize, Deserialize))]
730#[repr(Rust, packed)]
731struct ValueDataPacked {
732 x: u32,
733 y: u32,
734 flags_and_type: u16,
735}
736
737impl ValueDataPacked {
738 const TYPE_SHIFT: u8 = 0;
739 const TYPE_BITS: u8 = 14;
740 const TAG_SHIFT: u8 = Self::TYPE_SHIFT + Self::TYPE_BITS;
741 const TAG_BITS: u8 = 2;
742
743 const TAG_INST: u16 = 0;
744 const TAG_PARAM: u16 = 1;
745 const TAG_ALIAS: u16 = 2;
746 const TAG_UNION: u16 = 3;
747
748 fn make(tag: u16, ty: Type, x: u32, y: u32) -> ValueDataPacked {
749 debug_assert!(tag < (1 << Self::TAG_BITS));
750 debug_assert!(ty.repr() < (1 << Self::TYPE_BITS));
751
752 ValueDataPacked {
753 x,
754 y,
755 flags_and_type: (tag << Self::TAG_SHIFT) | (ty.repr() << Self::TYPE_SHIFT),
756 }
757 }
758
759 #[inline(always)]
760 fn field(self, shift: u8, bits: u8) -> u16 {
761 (self.flags_and_type >> shift) & ((1 << bits) - 1)
762 }
763
764 #[inline(always)]
765 fn ty(self) -> Type {
766 let ty = self.field(ValueDataPacked::TYPE_SHIFT, ValueDataPacked::TYPE_BITS);
767 Type::from_repr(ty)
768 }
769
770 #[inline(always)]
771 fn set_type(&mut self, ty: Type) {
772 self.flags_and_type &= !(((1 << Self::TYPE_BITS) - 1) << Self::TYPE_SHIFT);
773 self.flags_and_type |= ty.repr() << Self::TYPE_SHIFT;
774 }
775}
776
777impl From<ValueData> for ValueDataPacked {
778 fn from(data: ValueData) -> Self {
779 match data {
780 ValueData::Inst { ty, num, inst } => {
781 Self::make(Self::TAG_INST, ty, num.into(), inst.as_bits())
782 }
783 ValueData::Param { ty, num, block } => {
784 Self::make(Self::TAG_PARAM, ty, num.into(), block.as_bits())
785 }
786 ValueData::Alias { ty, original } => {
787 Self::make(Self::TAG_ALIAS, ty, 0, original.as_bits())
788 }
789 ValueData::Union { ty, x, y } => {
790 Self::make(Self::TAG_UNION, ty, x.as_bits(), y.as_bits())
791 }
792 }
793 }
794}
795
796impl From<ValueDataPacked> for ValueData {
797 fn from(data: ValueDataPacked) -> Self {
798 let tag = data.field(ValueDataPacked::TAG_SHIFT, ValueDataPacked::TAG_BITS);
799 let ty = data.field(ValueDataPacked::TYPE_SHIFT, ValueDataPacked::TYPE_BITS);
800
801 let ty = Type::from_repr(ty);
802 match tag {
803 ValueDataPacked::TAG_INST => ValueData::Inst {
804 ty,
805 num: u16::try_from(data.x).expect("Inst result num should fit in u16"),
806 inst: Inst::from_bits(data.y),
807 },
808 ValueDataPacked::TAG_PARAM => ValueData::Param {
809 ty,
810 num: u16::try_from(data.x).expect("Blockparam index should fit in u16"),
811 block: Block::from_bits(data.y),
812 },
813 ValueDataPacked::TAG_ALIAS => ValueData::Alias {
814 ty,
815 original: Value::from_bits(data.y),
816 },
817 ValueDataPacked::TAG_UNION => ValueData::Union {
818 ty,
819 x: Value::from_bits(data.x),
820 y: Value::from_bits(data.y),
821 },
822 _ => panic!("Invalid tag {tag} in ValueDataPacked"),
823 }
824 }
825}
826
827impl DataFlowGraph {
830 pub fn make_inst(&mut self, data: InstructionData) -> Inst {
838 let n = self.num_insts() + 1;
839 self.results.resize(n);
840 self.insts.0.push(data)
841 }
842
843 pub fn make_dynamic_ty(&mut self, data: DynamicTypeData) -> DynamicType {
845 self.dynamic_types.push(data)
846 }
847
848 pub fn display_inst<'a>(&'a self, inst: Inst) -> DisplayInst<'a> {
850 DisplayInst(self, inst)
851 }
852
853 pub fn display_value_inst(&self, value: Value) -> DisplayInst<'_> {
858 match self.value_def(value) {
859 ir::ValueDef::Result(inst, _) => self.display_inst(inst),
860 ir::ValueDef::Param(_, _) => panic!("value is not defined by an instruction"),
861 ir::ValueDef::Union(_, _) => panic!("value is a union of two other values"),
862 }
863 }
864
865 pub fn inst_values<'dfg>(
867 &'dfg self,
868 inst: Inst,
869 ) -> impl DoubleEndedIterator<Item = Value> + 'dfg {
870 self.inst_args(inst)
871 .iter()
872 .copied()
873 .chain(
874 self.insts[inst]
875 .branch_destination(&self.jump_tables, &self.exception_tables)
876 .into_iter()
877 .flat_map(|branch| {
878 branch
879 .args(&self.value_lists)
880 .filter_map(|arg| arg.as_value())
881 }),
882 )
883 .chain(
884 self.insts[inst]
885 .exception_table()
886 .into_iter()
887 .flat_map(|et| self.exception_tables[et].contexts()),
888 )
889 }
890
891 pub fn map_inst_values<F>(&mut self, inst: Inst, body: F)
893 where
894 F: FnMut(Value) -> Value,
895 {
896 self.insts[inst].map_values(
897 &mut self.value_lists,
898 &mut self.jump_tables,
899 &mut self.exception_tables,
900 body,
901 );
902 }
903
904 pub fn overwrite_inst_values<I>(&mut self, inst: Inst, mut values: I)
908 where
909 I: Iterator<Item = Value>,
910 {
911 self.insts[inst].map_values(
912 &mut self.value_lists,
913 &mut self.jump_tables,
914 &mut self.exception_tables,
915 |_| values.next().unwrap(),
916 );
917 }
918
919 pub fn inst_args(&self, inst: Inst) -> &[Value] {
921 self.insts[inst].arguments(&self.value_lists)
922 }
923
924 pub fn inst_args_mut(&mut self, inst: Inst) -> &mut [Value] {
926 self.insts[inst].arguments_mut(&mut self.value_lists)
927 }
928
929 pub fn inst_fixed_args(&self, inst: Inst) -> &[Value] {
931 let num_fixed_args = self.insts[inst]
932 .opcode()
933 .constraints()
934 .num_fixed_value_arguments();
935 &self.inst_args(inst)[..num_fixed_args]
936 }
937
938 pub fn inst_fixed_args_mut(&mut self, inst: Inst) -> &mut [Value] {
940 let num_fixed_args = self.insts[inst]
941 .opcode()
942 .constraints()
943 .num_fixed_value_arguments();
944 &mut self.inst_args_mut(inst)[..num_fixed_args]
945 }
946
947 pub fn inst_variable_args(&self, inst: Inst) -> &[Value] {
949 let num_fixed_args = self.insts[inst]
950 .opcode()
951 .constraints()
952 .num_fixed_value_arguments();
953 &self.inst_args(inst)[num_fixed_args..]
954 }
955
956 pub fn inst_variable_args_mut(&mut self, inst: Inst) -> &mut [Value] {
958 let num_fixed_args = self.insts[inst]
959 .opcode()
960 .constraints()
961 .num_fixed_value_arguments();
962 &mut self.inst_args_mut(inst)[num_fixed_args..]
963 }
964
965 pub fn make_inst_results(&mut self, inst: Inst, ctrl_typevar: Type) -> usize {
978 self.make_inst_results_reusing(inst, ctrl_typevar, iter::empty())
979 }
980
981 pub fn make_inst_results_reusing<I>(
987 &mut self,
988 inst: Inst,
989 ctrl_typevar: Type,
990 reuse: I,
991 ) -> usize
992 where
993 I: Iterator<Item = Option<Value>>,
994 {
995 self.clear_results(inst);
996
997 let mut reuse = reuse.fuse();
998 let result_tys: SmallVec<[_; 16]> = self.inst_result_types(inst, ctrl_typevar).collect();
999
1000 for (expected, &ty) in result_tys.iter().enumerate() {
1001 let num = u16::try_from(expected).expect("Result value index should fit in u16");
1002 let value_data = ValueData::Inst { ty, num, inst };
1003 let v = if let Some(Some(v)) = reuse.next() {
1004 debug_assert_eq!(self.value_type(v), ty, "Reused {ty} is wrong type");
1005 debug_assert!(!self.value_is_attached(v));
1006 self.values[v] = value_data.into();
1007 v
1008 } else {
1009 self.make_value(value_data)
1010 };
1011 let actual = self.results[inst].push(v, &mut self.value_lists);
1012 debug_assert_eq!(expected, actual);
1013 }
1014
1015 result_tys.len()
1016 }
1017
1018 pub fn replace(&mut self, inst: Inst) -> ReplaceBuilder<'_> {
1020 ReplaceBuilder::new(self, inst)
1021 }
1022
1023 pub fn clear_results(&mut self, inst: Inst) {
1028 self.results[inst].clear(&mut self.value_lists)
1029 }
1030
1031 pub fn replace_result(&mut self, old_value: Value, new_type: Type) -> Value {
1039 let (num, inst) = match ValueData::from(self.values[old_value]) {
1040 ValueData::Inst { num, inst, .. } => (num, inst),
1041 _ => panic!("{old_value} is not an instruction result value"),
1042 };
1043 let new_value = self.make_value(ValueData::Inst {
1044 ty: new_type,
1045 num,
1046 inst,
1047 });
1048 let num = num as usize;
1049 let attached = mem::replace(
1050 self.results[inst]
1051 .get_mut(num, &mut self.value_lists)
1052 .expect("Replacing detached result"),
1053 new_value,
1054 );
1055 debug_assert_eq!(
1056 attached,
1057 old_value,
1058 "{} wasn't detached from {}",
1059 old_value,
1060 self.display_inst(inst)
1061 );
1062 new_value
1063 }
1064
1065 pub fn clone_inst(&mut self, inst: Inst) -> Inst {
1068 let inst_data = self.insts[inst];
1070 let inst_data = inst_data.deep_clone(&mut self.value_lists);
1074 let new_inst = self.make_inst(inst_data);
1075 let ctrl_typevar = self.ctrl_typevar(inst);
1077 let num_results = self.make_inst_results(new_inst, ctrl_typevar);
1079 for i in 0..num_results {
1081 let old_result = self.inst_results(inst)[i];
1082 let new_result = self.inst_results(new_inst)[i];
1083 self.facts[new_result] = self.facts[old_result].clone();
1084 }
1085 new_inst
1086 }
1087
1088 pub fn first_result(&self, inst: Inst) -> Value {
1092 self.results[inst]
1093 .first(&self.value_lists)
1094 .unwrap_or_else(|| panic!("{inst} has no results"))
1095 }
1096
1097 pub fn has_results(&self, inst: Inst) -> bool {
1099 !self.results[inst].is_empty()
1100 }
1101
1102 pub fn inst_results(&self, inst: Inst) -> &[Value] {
1104 self.results[inst].as_slice(&self.value_lists)
1105 }
1106
1107 pub fn inst_results_list(&self, inst: Inst) -> ValueList {
1109 self.results[inst]
1110 }
1111
1112 pub fn union(&mut self, x: Value, y: Value) -> Value {
1114 let ty = self.value_type(x);
1116 debug_assert_eq!(ty, self.value_type(y));
1117 self.make_value(ValueData::Union { ty, x, y })
1118 }
1119
1120 pub fn call_signature(&self, inst: Inst) -> Option<SigRef> {
1123 match self.insts[inst].analyze_call(&self.value_lists, &self.exception_tables) {
1124 CallInfo::NotACall => None,
1125 CallInfo::Direct(f, _) => Some(self.ext_funcs[f].signature),
1126 CallInfo::DirectWithSig(_, s, _) => Some(s),
1127 CallInfo::Indirect(s, _) => Some(s),
1128 }
1129 }
1130
1131 fn non_tail_call_or_try_call_signature(&self, inst: Inst) -> Option<SigRef> {
1135 let sig = self.call_signature(inst)?;
1136 match self.insts[inst].opcode() {
1137 ir::Opcode::ReturnCall | ir::Opcode::ReturnCallIndirect => None,
1138 ir::Opcode::TryCall | ir::Opcode::TryCallIndirect => None,
1139 _ => Some(sig),
1140 }
1141 }
1142
1143 pub(crate) fn num_expected_results_for_verifier(&self, inst: Inst) -> usize {
1146 match self.non_tail_call_or_try_call_signature(inst) {
1147 Some(sig) => self.signatures[sig].returns.len(),
1148 None => {
1149 let constraints = self.insts[inst].opcode().constraints();
1150 constraints.num_fixed_results()
1151 }
1152 }
1153 }
1154
1155 pub fn inst_result_types<'a>(
1157 &'a self,
1158 inst: Inst,
1159 ctrl_typevar: Type,
1160 ) -> impl iter::ExactSizeIterator<Item = Type> + 'a {
1161 return match self.non_tail_call_or_try_call_signature(inst) {
1162 Some(sig) => InstResultTypes::Signature(self, sig, 0),
1163 None => {
1164 let constraints = self.insts[inst].opcode().constraints();
1165 InstResultTypes::Constraints(constraints, ctrl_typevar, 0)
1166 }
1167 };
1168
1169 enum InstResultTypes<'a> {
1170 Signature(&'a DataFlowGraph, SigRef, usize),
1171 Constraints(ir::instructions::OpcodeConstraints, Type, usize),
1172 }
1173
1174 impl Iterator for InstResultTypes<'_> {
1175 type Item = Type;
1176
1177 fn next(&mut self) -> Option<Type> {
1178 match self {
1179 InstResultTypes::Signature(dfg, sig, i) => {
1180 let param = dfg.signatures[*sig].returns.get(*i)?;
1181 *i += 1;
1182 Some(param.value_type)
1183 }
1184 InstResultTypes::Constraints(constraints, ctrl_ty, i) => {
1185 if *i < constraints.num_fixed_results() {
1186 let ty = constraints.result_type(*i, *ctrl_ty);
1187 *i += 1;
1188 Some(ty)
1189 } else {
1190 None
1191 }
1192 }
1193 }
1194 }
1195
1196 fn size_hint(&self) -> (usize, Option<usize>) {
1197 let len = match self {
1198 InstResultTypes::Signature(dfg, sig, i) => {
1199 dfg.signatures[*sig].returns.len() - *i
1200 }
1201 InstResultTypes::Constraints(constraints, _, i) => {
1202 constraints.num_fixed_results() - *i
1203 }
1204 };
1205 (len, Some(len))
1206 }
1207 }
1208
1209 impl ExactSizeIterator for InstResultTypes<'_> {}
1210 }
1211
1212 pub fn compute_result_type(
1220 &self,
1221 inst: Inst,
1222 result_idx: usize,
1223 ctrl_typevar: Type,
1224 ) -> Option<Type> {
1225 self.inst_result_types(inst, ctrl_typevar).nth(result_idx)
1226 }
1227
1228 pub fn ctrl_typevar(&self, inst: Inst) -> Type {
1230 let constraints = self.insts[inst].opcode().constraints();
1231
1232 if !constraints.is_polymorphic() {
1233 types::INVALID
1234 } else if constraints.requires_typevar_operand() {
1235 self.value_type(
1238 self.insts[inst]
1239 .typevar_operand(&self.value_lists)
1240 .unwrap_or_else(|| {
1241 panic!(
1242 "Instruction format for {:?} doesn't have a designated operand",
1243 self.insts[inst]
1244 )
1245 }),
1246 )
1247 } else {
1248 self.value_type(self.first_result(inst))
1249 }
1250 }
1251}
1252
1253impl DataFlowGraph {
1255 pub fn make_block(&mut self) -> Block {
1257 self.blocks.add()
1258 }
1259
1260 pub fn num_block_params(&self, block: Block) -> usize {
1262 self.blocks[block].params(&self.value_lists).len()
1263 }
1264
1265 pub fn block_params(&self, block: Block) -> &[Value] {
1267 self.blocks[block].params(&self.value_lists)
1268 }
1269
1270 pub fn block_param_types(&self, block: Block) -> impl Iterator<Item = Type> + '_ {
1272 self.block_params(block).iter().map(|&v| self.value_type(v))
1273 }
1274
1275 pub fn append_block_param(&mut self, block: Block, ty: Type) -> Value {
1277 let param = self.values.next_key();
1278 let num = self.blocks[block].params.push(param, &mut self.value_lists);
1279 debug_assert!(num <= u16::MAX as usize, "Too many parameters on block");
1280 self.make_value(ValueData::Param {
1281 ty,
1282 num: num as u16,
1283 block,
1284 })
1285 }
1286
1287 pub fn swap_remove_block_param(&mut self, val: Value) -> usize {
1296 let (block, num) =
1297 if let ValueData::Param { num, block, .. } = ValueData::from(self.values[val]) {
1298 (block, num)
1299 } else {
1300 panic!("{val} must be a block parameter");
1301 };
1302 self.blocks[block]
1303 .params
1304 .swap_remove(num as usize, &mut self.value_lists);
1305 if let Some(last_arg_val) = self.blocks[block]
1306 .params
1307 .get(num as usize, &self.value_lists)
1308 {
1309 let mut last_arg_data = ValueData::from(self.values[last_arg_val]);
1311 if let ValueData::Param { num: old_num, .. } = &mut last_arg_data {
1312 *old_num = num;
1313 self.values[last_arg_val] = last_arg_data.into();
1314 } else {
1315 panic!("{last_arg_val} should be a Block parameter");
1316 }
1317 }
1318 num as usize
1319 }
1320
1321 pub fn remove_block_param(&mut self, val: Value) {
1324 let (block, num) =
1325 if let ValueData::Param { num, block, .. } = ValueData::from(self.values[val]) {
1326 (block, num)
1327 } else {
1328 panic!("{val} must be a block parameter");
1329 };
1330 self.blocks[block]
1331 .params
1332 .remove(num as usize, &mut self.value_lists);
1333 for index in num..(self.num_block_params(block) as u16) {
1334 let packed = &mut self.values[self.blocks[block]
1335 .params
1336 .get(index as usize, &self.value_lists)
1337 .unwrap()];
1338 let mut data = ValueData::from(*packed);
1339 match &mut data {
1340 ValueData::Param { num, .. } => {
1341 *num -= 1;
1342 *packed = data.into();
1343 }
1344 _ => panic!(
1345 "{} must be a block parameter",
1346 self.blocks[block]
1347 .params
1348 .get(index as usize, &self.value_lists)
1349 .unwrap()
1350 ),
1351 }
1352 }
1353 }
1354
1355 pub fn attach_block_param(&mut self, block: Block, param: Value) {
1361 debug_assert!(!self.value_is_attached(param));
1362 let num = self.blocks[block].params.push(param, &mut self.value_lists);
1363 debug_assert!(num <= u16::MAX as usize, "Too many parameters on block");
1364 let ty = self.value_type(param);
1365 self.values[param] = ValueData::Param {
1366 ty,
1367 num: num as u16,
1368 block,
1369 }
1370 .into();
1371 }
1372
1373 pub fn replace_block_param(&mut self, old_value: Value, new_type: Type) -> Value {
1383 let (block, num) =
1385 if let ValueData::Param { num, block, .. } = ValueData::from(self.values[old_value]) {
1386 (block, num)
1387 } else {
1388 panic!("{old_value} must be a block parameter");
1389 };
1390 let new_arg = self.make_value(ValueData::Param {
1391 ty: new_type,
1392 num,
1393 block,
1394 });
1395
1396 self.blocks[block]
1397 .params
1398 .as_mut_slice(&mut self.value_lists)[num as usize] = new_arg;
1399 new_arg
1400 }
1401
1402 pub fn detach_block_params(&mut self, block: Block) -> ValueList {
1408 self.blocks[block].params.take()
1409 }
1410
1411 pub fn detach_inst_results(&mut self, inst: Inst) {
1417 self.results[inst].clear(&mut self.value_lists);
1418 }
1419
1420 pub fn merge_facts(&mut self, a: Value, b: Value) {
1424 let a = self.resolve_aliases(a);
1425 let b = self.resolve_aliases(b);
1426 match (&self.facts[a], &self.facts[b]) {
1427 (Some(a), Some(b)) if a == b => { }
1428 (None, None) => { }
1429 (Some(a), None) => {
1430 self.facts[b] = Some(a.clone());
1431 }
1432 (None, Some(b)) => {
1433 self.facts[a] = Some(b.clone());
1434 }
1435 (Some(a_fact), Some(b_fact)) => {
1436 assert_eq!(self.value_type(a), self.value_type(b));
1437 let merged = Fact::intersect(a_fact, b_fact);
1438 crate::trace!(
1439 "facts merge on {} and {}: {:?}, {:?} -> {:?}",
1440 a,
1441 b,
1442 a_fact,
1443 b_fact,
1444 merged,
1445 );
1446 self.facts[a] = Some(merged.clone());
1447 self.facts[b] = Some(merged);
1448 }
1449 }
1450 }
1451}
1452
1453#[derive(Clone, PartialEq, Hash)]
1459#[cfg_attr(feature = "enable-serde", derive(Serialize, Deserialize))]
1460pub struct BlockData {
1461 params: ValueList,
1463}
1464
1465impl BlockData {
1466 fn new() -> Self {
1467 Self {
1468 params: ValueList::new(),
1469 }
1470 }
1471
1472 pub fn params<'a>(&self, pool: &'a ValueListPool) -> &'a [Value] {
1474 self.params.as_slice(pool)
1475 }
1476}
1477
1478pub struct DisplayInst<'a>(&'a DataFlowGraph, Inst);
1480
1481impl<'a> fmt::Display for DisplayInst<'a> {
1482 fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
1483 let dfg = self.0;
1484 let inst = self.1;
1485
1486 if let Some((first, rest)) = dfg.inst_results(inst).split_first() {
1487 write!(f, "{first}")?;
1488 for v in rest {
1489 write!(f, ", {v}")?;
1490 }
1491 write!(f, " = ")?;
1492 }
1493
1494 let typevar = dfg.ctrl_typevar(inst);
1495 if typevar.is_invalid() {
1496 write!(f, "{}", dfg.insts[inst].opcode())?;
1497 } else {
1498 write!(f, "{}.{}", dfg.insts[inst].opcode(), typevar)?;
1499 }
1500 write_operands(f, dfg, inst)
1501 }
1502}
1503
1504impl DataFlowGraph {
1506 #[cold]
1509 fn set_value_type_for_parser(&mut self, v: Value, t: Type) {
1510 assert_eq!(
1511 self.value_type(v),
1512 types::INVALID,
1513 "this function is only for assigning types to previously invalid values"
1514 );
1515 self.values[v].set_type(t);
1516 }
1517
1518 pub fn check_dynamic_type(&mut self, ty: Type) -> Option<Type> {
1520 debug_assert!(ty.is_dynamic_vector());
1521 if self
1522 .dynamic_types
1523 .values()
1524 .any(|dyn_ty_data| dyn_ty_data.concrete().unwrap() == ty)
1525 {
1526 Some(ty)
1527 } else {
1528 None
1529 }
1530 }
1531
1532 #[cold]
1536 pub fn make_inst_results_for_parser(
1537 &mut self,
1538 inst: Inst,
1539 ctrl_typevar: Type,
1540 reuse: &[Value],
1541 ) -> usize {
1542 let mut reuse_iter = reuse.iter().copied();
1543 let result_tys: SmallVec<[_; 16]> = self.inst_result_types(inst, ctrl_typevar).collect();
1544 for ty in result_tys {
1545 if ty.is_dynamic_vector() {
1546 self.check_dynamic_type(ty)
1547 .unwrap_or_else(|| panic!("Use of undeclared dynamic type: {ty}"));
1548 }
1549 if let Some(v) = reuse_iter.next() {
1550 self.set_value_type_for_parser(v, ty);
1551 }
1552 }
1553
1554 self.make_inst_results_reusing(inst, ctrl_typevar, reuse.iter().map(|x| Some(*x)))
1555 }
1556
1557 #[cold]
1561 pub fn append_block_param_for_parser(&mut self, block: Block, ty: Type, val: Value) {
1562 let num = self.blocks[block].params.push(val, &mut self.value_lists);
1563 assert!(num <= u16::MAX as usize, "Too many parameters on block");
1564 self.values[val] = ValueData::Param {
1565 ty,
1566 num: num as u16,
1567 block,
1568 }
1569 .into();
1570 }
1571
1572 #[cold]
1575 pub fn make_value_alias_for_serialization(&mut self, src: Value, dest: Value) {
1576 assert_ne!(src, Value::reserved_value());
1577 assert_ne!(dest, Value::reserved_value());
1578
1579 let ty = if self.values.is_valid(src) {
1580 self.value_type(src)
1581 } else {
1582 types::INVALID
1585 };
1586 let data = ValueData::Alias { ty, original: src };
1587 self.values[dest] = data.into();
1588 }
1589
1590 #[cold]
1594 pub fn value_alias_dest_for_serialization(&self, v: Value) -> Option<Value> {
1595 if let ValueData::Alias { original, .. } = ValueData::from(self.values[v]) {
1596 Some(original)
1597 } else {
1598 None
1599 }
1600 }
1601
1602 #[cold]
1605 pub fn set_alias_type_for_parser(&mut self, v: Value) -> bool {
1606 if let Some(resolved) = maybe_resolve_aliases(&self.values, v) {
1607 let old_ty = self.value_type(v);
1608 let new_ty = self.value_type(resolved);
1609 if old_ty == types::INVALID {
1610 self.set_value_type_for_parser(v, new_ty);
1611 } else {
1612 assert_eq!(old_ty, new_ty);
1613 }
1614 true
1615 } else {
1616 false
1617 }
1618 }
1619
1620 #[cold]
1623 pub fn make_invalid_value_for_parser(&mut self) {
1624 let data = ValueData::Alias {
1625 ty: types::INVALID,
1626 original: Value::reserved_value(),
1627 };
1628 self.make_value(data);
1629 }
1630
1631 #[cold]
1634 pub fn value_is_valid_for_parser(&self, v: Value) -> bool {
1635 if !self.value_is_valid(v) {
1636 return false;
1637 }
1638 if let ValueData::Alias { ty, .. } = ValueData::from(self.values[v]) {
1639 ty != types::INVALID
1640 } else {
1641 true
1642 }
1643 }
1644}
1645
1646#[cfg(test)]
1647mod tests {
1648 use super::*;
1649 use crate::cursor::{Cursor, FuncCursor};
1650 use crate::ir::{Function, Opcode, TrapCode};
1651 use alloc::string::ToString;
1652
1653 #[test]
1654 fn make_inst() {
1655 let mut dfg = DataFlowGraph::new();
1656
1657 let idata = InstructionData::UnaryImm {
1658 opcode: Opcode::Iconst,
1659 imm: 0.into(),
1660 };
1661 let inst = dfg.make_inst(idata);
1662
1663 dfg.make_inst_results(inst, types::I32);
1664 assert_eq!(inst.to_string(), "inst0");
1665 assert_eq!(dfg.display_inst(inst).to_string(), "v0 = iconst.i32 0");
1666
1667 {
1669 let immdfg = &dfg;
1670 let ins = &immdfg.insts[inst];
1671 assert_eq!(ins.opcode(), Opcode::Iconst);
1672 }
1673
1674 let val = dfg.first_result(inst);
1676 assert_eq!(dfg.inst_results(inst), &[val]);
1677
1678 assert_eq!(dfg.value_def(val), ValueDef::Result(inst, 0));
1679 assert_eq!(dfg.value_type(val), types::I32);
1680
1681 assert!(dfg.value_is_attached(val));
1683 let v2 = dfg.replace_result(val, types::F64);
1684 assert!(!dfg.value_is_attached(val));
1685 assert!(dfg.value_is_attached(v2));
1686 assert_eq!(dfg.inst_results(inst), &[v2]);
1687 assert_eq!(dfg.value_def(v2), ValueDef::Result(inst, 0));
1688 assert_eq!(dfg.value_type(v2), types::F64);
1689 }
1690
1691 #[test]
1692 fn no_results() {
1693 let mut dfg = DataFlowGraph::new();
1694
1695 let idata = InstructionData::Trap {
1696 opcode: Opcode::Trap,
1697 code: TrapCode::unwrap_user(1),
1698 };
1699 let inst = dfg.make_inst(idata);
1700 assert_eq!(dfg.display_inst(inst).to_string(), "trap user1");
1701
1702 assert_eq!(dfg.inst_results(inst), &[]);
1704 }
1705
1706 #[test]
1707 fn block() {
1708 let mut dfg = DataFlowGraph::new();
1709
1710 let block = dfg.make_block();
1711 assert_eq!(block.to_string(), "block0");
1712 assert_eq!(dfg.num_block_params(block), 0);
1713 assert_eq!(dfg.block_params(block), &[]);
1714 assert!(dfg.detach_block_params(block).is_empty());
1715 assert_eq!(dfg.num_block_params(block), 0);
1716 assert_eq!(dfg.block_params(block), &[]);
1717
1718 let arg1 = dfg.append_block_param(block, types::F32);
1719 assert_eq!(arg1.to_string(), "v0");
1720 assert_eq!(dfg.num_block_params(block), 1);
1721 assert_eq!(dfg.block_params(block), &[arg1]);
1722
1723 let arg2 = dfg.append_block_param(block, types::I16);
1724 assert_eq!(arg2.to_string(), "v1");
1725 assert_eq!(dfg.num_block_params(block), 2);
1726 assert_eq!(dfg.block_params(block), &[arg1, arg2]);
1727
1728 assert_eq!(dfg.value_def(arg1), ValueDef::Param(block, 0));
1729 assert_eq!(dfg.value_def(arg2), ValueDef::Param(block, 1));
1730 assert_eq!(dfg.value_type(arg1), types::F32);
1731 assert_eq!(dfg.value_type(arg2), types::I16);
1732
1733 let vlist = dfg.detach_block_params(block);
1735 assert_eq!(dfg.num_block_params(block), 0);
1736 assert_eq!(dfg.block_params(block), &[]);
1737 assert_eq!(vlist.as_slice(&dfg.value_lists), &[arg1, arg2]);
1738 dfg.attach_block_param(block, arg2);
1739 let arg3 = dfg.append_block_param(block, types::I32);
1740 dfg.attach_block_param(block, arg1);
1741 assert_eq!(dfg.block_params(block), &[arg2, arg3, arg1]);
1742 }
1743
1744 #[test]
1745 fn replace_block_params() {
1746 let mut dfg = DataFlowGraph::new();
1747
1748 let block = dfg.make_block();
1749 let arg1 = dfg.append_block_param(block, types::F32);
1750
1751 let new1 = dfg.replace_block_param(arg1, types::I64);
1752 assert_eq!(dfg.value_type(arg1), types::F32);
1753 assert_eq!(dfg.value_type(new1), types::I64);
1754 assert_eq!(dfg.block_params(block), &[new1]);
1755
1756 dfg.attach_block_param(block, arg1);
1757 assert_eq!(dfg.block_params(block), &[new1, arg1]);
1758
1759 let new2 = dfg.replace_block_param(arg1, types::I8);
1760 assert_eq!(dfg.value_type(arg1), types::F32);
1761 assert_eq!(dfg.value_type(new2), types::I8);
1762 assert_eq!(dfg.block_params(block), &[new1, new2]);
1763
1764 dfg.attach_block_param(block, arg1);
1765 assert_eq!(dfg.block_params(block), &[new1, new2, arg1]);
1766
1767 let new3 = dfg.replace_block_param(new2, types::I16);
1768 assert_eq!(dfg.value_type(new1), types::I64);
1769 assert_eq!(dfg.value_type(new2), types::I8);
1770 assert_eq!(dfg.value_type(new3), types::I16);
1771 assert_eq!(dfg.block_params(block), &[new1, new3, arg1]);
1772 }
1773
1774 #[test]
1775 fn swap_remove_block_params() {
1776 let mut dfg = DataFlowGraph::new();
1777
1778 let block = dfg.make_block();
1779 let arg1 = dfg.append_block_param(block, types::F32);
1780 let arg2 = dfg.append_block_param(block, types::F32);
1781 let arg3 = dfg.append_block_param(block, types::F32);
1782 assert_eq!(dfg.block_params(block), &[arg1, arg2, arg3]);
1783
1784 dfg.swap_remove_block_param(arg1);
1785 assert_eq!(dfg.value_is_attached(arg1), false);
1786 assert_eq!(dfg.value_is_attached(arg2), true);
1787 assert_eq!(dfg.value_is_attached(arg3), true);
1788 assert_eq!(dfg.block_params(block), &[arg3, arg2]);
1789 dfg.swap_remove_block_param(arg2);
1790 assert_eq!(dfg.value_is_attached(arg2), false);
1791 assert_eq!(dfg.value_is_attached(arg3), true);
1792 assert_eq!(dfg.block_params(block), &[arg3]);
1793 dfg.swap_remove_block_param(arg3);
1794 assert_eq!(dfg.value_is_attached(arg3), false);
1795 assert_eq!(dfg.block_params(block), &[]);
1796 }
1797
1798 #[test]
1799 fn aliases() {
1800 use crate::ir::InstBuilder;
1801 use crate::ir::condcodes::IntCC;
1802
1803 let mut func = Function::new();
1804 let block0 = func.dfg.make_block();
1805 let mut pos = FuncCursor::new(&mut func);
1806 pos.insert_block(block0);
1807
1808 let v1 = pos.ins().iconst(types::I32, 42);
1810
1811 assert_eq!(pos.func.dfg.resolve_aliases(v1), v1);
1813
1814 let arg0 = pos.func.dfg.append_block_param(block0, types::I32);
1815 let (s, c) = pos.ins().uadd_overflow(v1, arg0);
1816 let iadd = match pos.func.dfg.value_def(s) {
1817 ValueDef::Result(i, 0) => i,
1818 _ => panic!(),
1819 };
1820
1821 pos.func.stencil.dfg.results[iadd].remove(1, &mut pos.func.stencil.dfg.value_lists);
1823
1824 pos.func.dfg.replace(iadd).iadd(v1, arg0);
1826 let c2 = pos.ins().icmp(IntCC::Equal, s, v1);
1827 pos.func.dfg.change_to_alias(c, c2);
1828
1829 assert_eq!(pos.func.dfg.resolve_aliases(c2), c2);
1830 assert_eq!(pos.func.dfg.resolve_aliases(c), c2);
1831 }
1832
1833 #[test]
1834 fn cloning() {
1835 use crate::ir::InstBuilder;
1836
1837 let mut func = Function::new();
1838 let mut sig = Signature::new(crate::isa::CallConv::SystemV);
1839 sig.params.push(ir::AbiParam::new(types::I32));
1840 let sig = func.import_signature(sig);
1841 let block0 = func.dfg.make_block();
1842 let mut pos = FuncCursor::new(&mut func);
1843 pos.insert_block(block0);
1844 let v1 = pos.ins().iconst(types::I32, 0);
1845 let v2 = pos.ins().iconst(types::I32, 1);
1846 let call_inst = pos.ins().call_indirect(sig, v1, &[v1]);
1847 let func = pos.func;
1848
1849 let call_inst_dup = func.dfg.clone_inst(call_inst);
1850 func.dfg.inst_args_mut(call_inst)[0] = v2;
1851 assert_eq!(v1, func.dfg.inst_args(call_inst_dup)[0]);
1852 }
1853}