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::user_stack_maps::{UserStackMapEntry, UserStackMapEntryVec};
9use crate::ir::{
10 Block, BlockArg, BlockCall, ConstantData, ConstantPool, DynamicType, ExceptionTables,
11 ExtFuncData, FuncRef, Immediate, Inst, JumpTables, RelSourceLoc, SigRef, Signature, Type,
12 Value, ValueLabelAssignments, ValueList, ValueListPool, types,
13};
14use crate::packed_option::ReservedValue;
15use crate::write::write_operands;
16use core::fmt;
17use core::iter;
18use core::mem;
19use core::ops::{Index, IndexMut};
20use core::u16;
21
22use alloc::collections::BTreeMap;
23#[cfg(feature = "enable-serde")]
24use serde_derive::{Deserialize, Serialize};
25use smallvec::SmallVec;
26
27#[derive(Clone, PartialEq, Hash)]
29#[cfg_attr(feature = "enable-serde", derive(Serialize, Deserialize))]
30pub struct Insts(PrimaryMap<Inst, InstructionData>);
31
32impl Index<Inst> for Insts {
34 type Output = InstructionData;
35
36 fn index(&self, inst: Inst) -> &InstructionData {
37 self.0.index(inst)
38 }
39}
40
41impl IndexMut<Inst> for Insts {
43 fn index_mut(&mut self, inst: Inst) -> &mut InstructionData {
44 self.0.index_mut(inst)
45 }
46}
47
48#[derive(Clone, PartialEq, Hash)]
50#[cfg_attr(feature = "enable-serde", derive(Serialize, Deserialize))]
51pub struct Blocks(PrimaryMap<Block, BlockData>);
52
53impl Blocks {
54 pub fn add(&mut self) -> Block {
56 self.0.push(BlockData::new())
57 }
58
59 pub fn len(&self) -> usize {
64 self.0.len()
65 }
66
67 pub fn reserve(&mut self, additional: usize) {
70 self.0.reserve(additional);
71 }
72
73 pub fn is_valid(&self, block: Block) -> bool {
75 self.0.is_valid(block)
76 }
77
78 pub fn iter(&self) -> impl Iterator<Item = Block> {
83 self.0.keys()
84 }
85}
86
87impl Index<Block> for Blocks {
88 type Output = BlockData;
89
90 fn index(&self, block: Block) -> &BlockData {
91 &self.0[block]
92 }
93}
94
95impl IndexMut<Block> for Blocks {
96 fn index_mut(&mut self, block: Block) -> &mut BlockData {
97 &mut self.0[block]
98 }
99}
100
101#[derive(Clone, PartialEq, Hash)]
109#[cfg_attr(feature = "enable-serde", derive(Serialize, Deserialize))]
110pub struct DataFlowGraph {
111 pub insts: Insts,
115
116 results: SecondaryMap<Inst, ValueList>,
121
122 user_stack_maps: alloc::collections::BTreeMap<Inst, UserStackMapEntryVec>,
124
125 pub blocks: Blocks,
130
131 pub dynamic_types: DynamicTypes,
133
134 pub value_lists: ValueListPool,
142
143 values: PrimaryMap<Value, ValueDataPacked>,
145
146 pub signatures: PrimaryMap<SigRef, Signature>,
149
150 pub ext_funcs: PrimaryMap<FuncRef, ExtFuncData>,
152
153 pub values_labels: Option<BTreeMap<Value, ValueLabelAssignments>>,
155
156 pub constants: ConstantPool,
158
159 pub immediates: PrimaryMap<Immediate, ConstantData>,
161
162 pub jump_tables: JumpTables,
164
165 pub exception_tables: ExceptionTables,
167}
168
169impl DataFlowGraph {
170 pub fn new() -> Self {
172 Self {
173 insts: Insts(PrimaryMap::new()),
174 results: SecondaryMap::new(),
175 user_stack_maps: alloc::collections::BTreeMap::new(),
176 blocks: Blocks(PrimaryMap::new()),
177 dynamic_types: DynamicTypes::new(),
178 value_lists: ValueListPool::new(),
179 values: PrimaryMap::new(),
180 signatures: PrimaryMap::new(),
181 ext_funcs: PrimaryMap::new(),
182 values_labels: None,
183 constants: ConstantPool::new(),
184 immediates: PrimaryMap::new(),
185 jump_tables: JumpTables::new(),
186 exception_tables: ExceptionTables::new(),
187 }
188 }
189
190 pub fn clear(&mut self) {
192 self.insts.0.clear();
193 self.results.clear();
194 self.user_stack_maps.clear();
195 self.blocks.0.clear();
196 self.dynamic_types.clear();
197 self.value_lists.clear();
198 self.values.clear();
199 self.signatures.clear();
200 self.ext_funcs.clear();
201 self.values_labels = None;
202 self.constants.clear();
203 self.immediates.clear();
204 self.jump_tables.clear();
205 }
206
207 pub fn num_insts(&self) -> usize {
212 self.insts.0.len()
213 }
214
215 pub fn inst_is_valid(&self, inst: Inst) -> bool {
217 self.insts.0.is_valid(inst)
218 }
219
220 pub fn num_blocks(&self) -> usize {
225 self.blocks.len()
226 }
227
228 pub fn block_is_valid(&self, block: Block) -> bool {
230 self.blocks.is_valid(block)
231 }
232
233 pub fn block_call<'a>(
235 &mut self,
236 block: Block,
237 args: impl IntoIterator<Item = &'a BlockArg>,
238 ) -> BlockCall {
239 BlockCall::new(block, args.into_iter().copied(), &mut self.value_lists)
240 }
241
242 pub fn num_values(&self) -> usize {
244 self.values.len()
245 }
246
247 pub fn values_and_defs(&self) -> impl Iterator<Item = (Value, ValueDef)> + '_ {
249 self.values().map(|value| (value, self.value_def(value)))
250 }
251
252 pub fn collect_debug_info(&mut self) {
254 if self.values_labels.is_none() {
255 self.values_labels = Some(Default::default());
256 }
257 }
258
259 pub fn add_value_label_alias(&mut self, to_alias: Value, from: RelSourceLoc, value: Value) {
262 if let Some(values_labels) = self.values_labels.as_mut() {
263 values_labels.insert(to_alias, ir::ValueLabelAssignments::Alias { from, value });
264 }
265 }
266}
267
268fn maybe_resolve_aliases(
273 values: &PrimaryMap<Value, ValueDataPacked>,
274 value: Value,
275) -> Option<Value> {
276 let mut v = value;
277
278 for _ in 0..=values.len() {
280 if let ValueData::Alias { original, .. } = ValueData::from(values[v]) {
281 v = original;
282 } else {
283 return Some(v);
284 }
285 }
286
287 None
288}
289
290fn resolve_aliases(values: &PrimaryMap<Value, ValueDataPacked>, value: Value) -> Value {
294 if let Some(v) = maybe_resolve_aliases(values, value) {
295 v
296 } else {
297 panic!("Value alias loop detected for {value}");
298 }
299}
300
301pub struct Values<'a> {
303 inner: entity::Iter<'a, Value, ValueDataPacked>,
304}
305
306fn valid_valuedata(data: ValueDataPacked) -> bool {
308 let data = ValueData::from(data);
309 if let ValueData::Alias {
310 ty: types::INVALID,
311 original,
312 } = data
313 {
314 if original == Value::reserved_value() {
315 return false;
316 }
317 }
318 true
319}
320
321impl<'a> Iterator for Values<'a> {
322 type Item = Value;
323
324 fn next(&mut self) -> Option<Self::Item> {
325 self.inner
326 .by_ref()
327 .find(|kv| valid_valuedata(*kv.1))
328 .map(|kv| kv.0)
329 }
330
331 fn size_hint(&self) -> (usize, Option<usize>) {
332 self.inner.size_hint()
333 }
334}
335
336impl ExactSizeIterator for Values<'_> {
337 fn len(&self) -> usize {
338 self.inner.len()
339 }
340}
341
342impl DataFlowGraph {
346 fn make_value(&mut self, data: ValueData) -> Value {
348 self.values.push(data.into())
349 }
350
351 pub fn len_values(&self) -> usize {
353 self.values.len()
354 }
355
356 pub fn values<'a>(&'a self) -> Values<'a> {
358 Values {
359 inner: self.values.iter(),
360 }
361 }
362
363 pub fn value_is_valid(&self, v: Value) -> bool {
365 self.values.is_valid(v)
366 }
367
368 pub fn value_is_real(&self, value: Value) -> bool {
370 self.value_is_valid(value) && !matches!(self.values[value].into(), ValueData::Alias { .. })
373 }
374
375 pub fn value_is_alias(&self, v: Value) -> bool {
377 match ValueData::from(self.values[v]) {
378 ValueData::Alias { .. } => true,
379 ValueData::Inst { .. } | ValueData::Param { .. } | ValueData::Union { .. } => false,
380 }
381 }
382
383 pub fn value_type(&self, v: Value) -> Type {
385 self.values[v].ty()
386 }
387
388 pub fn value_def(&self, v: Value) -> ValueDef {
393 match ValueData::from(self.values[v]) {
394 ValueData::Inst { inst, num, .. } => ValueDef::Result(inst, num as usize),
395 ValueData::Param { block, num, .. } => ValueDef::Param(block, num as usize),
396 ValueData::Alias { original, .. } => {
397 self.value_def(self.resolve_aliases(original))
400 }
401 ValueData::Union { x, y, .. } => ValueDef::Union(x, y),
402 }
403 }
404
405 pub fn value_is_attached(&self, v: Value) -> bool {
412 use self::ValueData::*;
413 match ValueData::from(self.values[v]) {
414 Inst { inst, num, .. } => Some(&v) == self.inst_results(inst).get(num as usize),
415 Param { block, num, .. } => Some(&v) == self.block_params(block).get(num as usize),
416 Alias { .. } => false,
417 Union { .. } => false,
418 }
419 }
420
421 pub fn resolve_aliases(&self, value: Value) -> Value {
425 resolve_aliases(&self.values, value)
426 }
427
428 pub fn resolve_all_aliases(&mut self) {
431 let invalid_value = ValueDataPacked::from(ValueData::Alias {
432 ty: types::INVALID,
433 original: Value::reserved_value(),
434 });
435
436 for mut src in self.values.keys() {
441 let value_data = self.values[src];
442 if value_data == invalid_value {
443 continue;
444 }
445 if let ValueData::Alias { mut original, .. } = value_data.into() {
446 let resolved = ValueDataPacked::from(ValueData::Alias {
449 ty: types::INVALID,
450 original: resolve_aliases(&self.values, original),
451 });
452 loop {
456 self.values[src] = resolved;
457 src = original;
458 if let ValueData::Alias { original: next, .. } = self.values[src].into() {
459 original = next;
460 } else {
461 break;
462 }
463 }
464 }
465 }
466
467 for inst in self.insts.0.values_mut() {
472 inst.map_values(
473 &mut self.value_lists,
474 &mut self.jump_tables,
475 &mut self.exception_tables,
476 |arg| {
477 if let ValueData::Alias { original, .. } = self.values[arg].into() {
478 original
479 } else {
480 arg
481 }
482 },
483 );
484 }
485
486 if let Some(values_labels) = &mut self.values_labels {
497 values_labels.retain(|&k, _| !matches!(self.values[k].into(), ValueData::Alias { .. }));
500
501 for value_label in values_labels.values_mut() {
504 if let ValueLabelAssignments::Alias { value, .. } = value_label {
505 if let ValueData::Alias { original, .. } = self.values[*value].into() {
506 *value = original;
507 }
508 }
509 }
510 }
511
512 for value in self.values.values_mut() {
517 if let ValueData::Alias { .. } = ValueData::from(*value) {
518 *value = invalid_value;
519 }
520 }
521 }
522
523 pub fn change_to_alias(&mut self, dest: Value, src: Value) {
530 debug_assert!(!self.value_is_attached(dest));
531 let original = self.resolve_aliases(src);
534 debug_assert_ne!(
535 dest, original,
536 "Aliasing {dest} to {src} would create a loop"
537 );
538 let ty = self.value_type(original);
539 debug_assert_eq!(
540 self.value_type(dest),
541 ty,
542 "Aliasing {} to {} would change its type {} to {}",
543 dest,
544 src,
545 self.value_type(dest),
546 ty
547 );
548 debug_assert_ne!(ty, types::INVALID);
549
550 self.values[dest] = ValueData::Alias { ty, original }.into();
551 }
552
553 pub fn replace_with_aliases(&mut self, dest_inst: Inst, original_inst: Inst) {
563 debug_assert_ne!(
564 dest_inst, original_inst,
565 "Replacing {dest_inst} with itself would create a loop"
566 );
567
568 let dest_results = self.results[dest_inst].as_slice(&self.value_lists);
569 let original_results = self.results[original_inst].as_slice(&self.value_lists);
570
571 debug_assert_eq!(
572 dest_results.len(),
573 original_results.len(),
574 "Replacing {dest_inst} with {original_inst} would produce a different number of results."
575 );
576
577 for (&dest, &original) in dest_results.iter().zip(original_results) {
578 let ty = self.value_type(original);
579 debug_assert_eq!(
580 self.value_type(dest),
581 ty,
582 "Aliasing {} to {} would change its type {} to {}",
583 dest,
584 original,
585 self.value_type(dest),
586 ty
587 );
588 debug_assert_ne!(ty, types::INVALID);
589
590 self.values[dest] = ValueData::Alias { ty, original }.into();
591 }
592
593 self.clear_results(dest_inst);
594 }
595
596 pub fn user_stack_map_entries(&self, inst: Inst) -> Option<&[UserStackMapEntry]> {
598 self.user_stack_maps.get(&inst).map(|es| &**es)
599 }
600
601 pub fn append_user_stack_map_entry(&mut self, inst: Inst, entry: UserStackMapEntry) {
607 let opcode = self.insts[inst].opcode();
608 assert!(opcode.is_safepoint());
609 self.user_stack_maps.entry(inst).or_default().push(entry);
610 }
611
612 pub fn append_user_stack_map_entries(
618 &mut self,
619 inst: Inst,
620 entries: impl IntoIterator<Item = UserStackMapEntry>,
621 ) {
622 for entry in entries {
623 self.append_user_stack_map_entry(inst, entry);
624 }
625 }
626
627 pub(crate) fn take_user_stack_map_entries(
630 &mut self,
631 inst: Inst,
632 ) -> Option<UserStackMapEntryVec> {
633 self.user_stack_maps.remove(&inst)
634 }
635}
636
637#[derive(Clone, Copy, Debug, PartialEq, Eq)]
639pub enum ValueDef {
640 Result(Inst, usize),
642 Param(Block, usize),
644 Union(Value, Value),
646}
647
648impl ValueDef {
649 pub fn unwrap_inst(&self) -> Inst {
651 self.inst().expect("Value is not an instruction result")
652 }
653
654 pub fn inst(&self) -> Option<Inst> {
656 match *self {
657 Self::Result(inst, _) => Some(inst),
658 _ => None,
659 }
660 }
661
662 pub fn unwrap_block(&self) -> Block {
664 match *self {
665 Self::Param(block, _) => block,
666 _ => panic!("Value is not a block parameter"),
667 }
668 }
669
670 pub fn num(self) -> usize {
675 match self {
676 Self::Result(_, n) | Self::Param(_, n) => n,
677 Self::Union(_, _) => 0,
678 }
679 }
680}
681
682#[derive(Clone, Debug, PartialEq, Hash)]
684#[cfg_attr(feature = "enable-serde", derive(Serialize, Deserialize))]
685enum ValueData {
686 Inst { ty: Type, num: u16, inst: Inst },
688
689 Param { ty: Type, num: u16, block: Block },
691
692 Alias { ty: Type, original: Value },
696
697 Union { ty: Type, x: Value, y: Value },
701}
702
703#[derive(Clone, Copy, Debug, PartialEq, Hash)]
716#[cfg_attr(feature = "enable-serde", derive(Serialize, Deserialize))]
717#[repr(Rust, packed)]
718struct ValueDataPacked {
719 x: u32,
720 y: u32,
721 flags_and_type: u16,
722}
723
724impl ValueDataPacked {
725 const TYPE_SHIFT: u8 = 0;
726 const TYPE_BITS: u8 = 14;
727 const TAG_SHIFT: u8 = Self::TYPE_SHIFT + Self::TYPE_BITS;
728 const TAG_BITS: u8 = 2;
729
730 const TAG_INST: u16 = 0;
731 const TAG_PARAM: u16 = 1;
732 const TAG_ALIAS: u16 = 2;
733 const TAG_UNION: u16 = 3;
734
735 fn make(tag: u16, ty: Type, x: u32, y: u32) -> ValueDataPacked {
736 debug_assert!(tag < (1 << Self::TAG_BITS));
737 debug_assert!(ty.repr() < (1 << Self::TYPE_BITS));
738
739 ValueDataPacked {
740 x,
741 y,
742 flags_and_type: (tag << Self::TAG_SHIFT) | (ty.repr() << Self::TYPE_SHIFT),
743 }
744 }
745
746 #[inline(always)]
747 fn field(self, shift: u8, bits: u8) -> u16 {
748 (self.flags_and_type >> shift) & ((1 << bits) - 1)
749 }
750
751 #[inline(always)]
752 fn ty(self) -> Type {
753 let ty = self.field(ValueDataPacked::TYPE_SHIFT, ValueDataPacked::TYPE_BITS);
754 Type::from_repr(ty)
755 }
756
757 #[inline(always)]
758 fn set_type(&mut self, ty: Type) {
759 self.flags_and_type &= !(((1 << Self::TYPE_BITS) - 1) << Self::TYPE_SHIFT);
760 self.flags_and_type |= ty.repr() << Self::TYPE_SHIFT;
761 }
762}
763
764impl From<ValueData> for ValueDataPacked {
765 fn from(data: ValueData) -> Self {
766 match data {
767 ValueData::Inst { ty, num, inst } => {
768 Self::make(Self::TAG_INST, ty, num.into(), inst.as_bits())
769 }
770 ValueData::Param { ty, num, block } => {
771 Self::make(Self::TAG_PARAM, ty, num.into(), block.as_bits())
772 }
773 ValueData::Alias { ty, original } => {
774 Self::make(Self::TAG_ALIAS, ty, 0, original.as_bits())
775 }
776 ValueData::Union { ty, x, y } => {
777 Self::make(Self::TAG_UNION, ty, x.as_bits(), y.as_bits())
778 }
779 }
780 }
781}
782
783impl From<ValueDataPacked> for ValueData {
784 fn from(data: ValueDataPacked) -> Self {
785 let tag = data.field(ValueDataPacked::TAG_SHIFT, ValueDataPacked::TAG_BITS);
786 let ty = data.field(ValueDataPacked::TYPE_SHIFT, ValueDataPacked::TYPE_BITS);
787
788 let ty = Type::from_repr(ty);
789 match tag {
790 ValueDataPacked::TAG_INST => ValueData::Inst {
791 ty,
792 num: u16::try_from(data.x).expect("Inst result num should fit in u16"),
793 inst: Inst::from_bits(data.y),
794 },
795 ValueDataPacked::TAG_PARAM => ValueData::Param {
796 ty,
797 num: u16::try_from(data.x).expect("Blockparam index should fit in u16"),
798 block: Block::from_bits(data.y),
799 },
800 ValueDataPacked::TAG_ALIAS => ValueData::Alias {
801 ty,
802 original: Value::from_bits(data.y),
803 },
804 ValueDataPacked::TAG_UNION => ValueData::Union {
805 ty,
806 x: Value::from_bits(data.x),
807 y: Value::from_bits(data.y),
808 },
809 _ => panic!("Invalid tag {tag} in ValueDataPacked"),
810 }
811 }
812}
813
814impl DataFlowGraph {
817 pub fn make_inst(&mut self, data: InstructionData) -> Inst {
825 let n = self.num_insts() + 1;
826 self.results.resize(n);
827 self.insts.0.push(data)
828 }
829
830 pub fn make_dynamic_ty(&mut self, data: DynamicTypeData) -> DynamicType {
832 self.dynamic_types.push(data)
833 }
834
835 pub fn display_inst<'a>(&'a self, inst: Inst) -> DisplayInst<'a> {
837 DisplayInst(self, inst)
838 }
839
840 pub fn display_value_inst(&self, value: Value) -> DisplayInst<'_> {
845 match self.value_def(value) {
846 ir::ValueDef::Result(inst, _) => self.display_inst(inst),
847 ir::ValueDef::Param(_, _) => panic!("value is not defined by an instruction"),
848 ir::ValueDef::Union(_, _) => panic!("value is a union of two other values"),
849 }
850 }
851
852 pub fn inst_values<'dfg>(
854 &'dfg self,
855 inst: Inst,
856 ) -> impl DoubleEndedIterator<Item = Value> + 'dfg {
857 self.inst_args(inst)
858 .iter()
859 .copied()
860 .chain(
861 self.insts[inst]
862 .branch_destination(&self.jump_tables, &self.exception_tables)
863 .into_iter()
864 .flat_map(|branch| {
865 branch
866 .args(&self.value_lists)
867 .filter_map(|arg| arg.as_value())
868 }),
869 )
870 .chain(
871 self.insts[inst]
872 .exception_table()
873 .into_iter()
874 .flat_map(|et| self.exception_tables[et].contexts()),
875 )
876 }
877
878 pub fn map_inst_values<F>(&mut self, inst: Inst, body: F)
880 where
881 F: FnMut(Value) -> Value,
882 {
883 self.insts[inst].map_values(
884 &mut self.value_lists,
885 &mut self.jump_tables,
886 &mut self.exception_tables,
887 body,
888 );
889 }
890
891 pub fn overwrite_inst_values<I>(&mut self, inst: Inst, mut values: I)
895 where
896 I: Iterator<Item = Value>,
897 {
898 self.insts[inst].map_values(
899 &mut self.value_lists,
900 &mut self.jump_tables,
901 &mut self.exception_tables,
902 |_| values.next().unwrap(),
903 );
904 }
905
906 pub fn inst_args(&self, inst: Inst) -> &[Value] {
908 self.insts[inst].arguments(&self.value_lists)
909 }
910
911 pub fn inst_args_mut(&mut self, inst: Inst) -> &mut [Value] {
913 self.insts[inst].arguments_mut(&mut self.value_lists)
914 }
915
916 pub fn inst_fixed_args(&self, inst: Inst) -> &[Value] {
918 let num_fixed_args = self.insts[inst]
919 .opcode()
920 .constraints()
921 .num_fixed_value_arguments();
922 &self.inst_args(inst)[..num_fixed_args]
923 }
924
925 pub fn inst_fixed_args_mut(&mut self, inst: Inst) -> &mut [Value] {
927 let num_fixed_args = self.insts[inst]
928 .opcode()
929 .constraints()
930 .num_fixed_value_arguments();
931 &mut self.inst_args_mut(inst)[..num_fixed_args]
932 }
933
934 pub fn inst_variable_args(&self, inst: Inst) -> &[Value] {
936 let num_fixed_args = self.insts[inst]
937 .opcode()
938 .constraints()
939 .num_fixed_value_arguments();
940 &self.inst_args(inst)[num_fixed_args..]
941 }
942
943 pub fn inst_variable_args_mut(&mut self, inst: Inst) -> &mut [Value] {
945 let num_fixed_args = self.insts[inst]
946 .opcode()
947 .constraints()
948 .num_fixed_value_arguments();
949 &mut self.inst_args_mut(inst)[num_fixed_args..]
950 }
951
952 pub fn make_inst_results(&mut self, inst: Inst, ctrl_typevar: Type) -> usize {
965 self.make_inst_results_reusing(inst, ctrl_typevar, iter::empty())
966 }
967
968 pub fn make_inst_results_reusing<I>(
974 &mut self,
975 inst: Inst,
976 ctrl_typevar: Type,
977 reuse: I,
978 ) -> usize
979 where
980 I: Iterator<Item = Option<Value>>,
981 {
982 self.clear_results(inst);
983
984 let mut reuse = reuse.fuse();
985 let result_tys: SmallVec<[_; 16]> = self.inst_result_types(inst, ctrl_typevar).collect();
986
987 for (expected, &ty) in result_tys.iter().enumerate() {
988 let num = u16::try_from(expected).expect("Result value index should fit in u16");
989 let value_data = ValueData::Inst { ty, num, inst };
990 let v = if let Some(Some(v)) = reuse.next() {
991 debug_assert_eq!(self.value_type(v), ty, "Reused {ty} is wrong type");
992 debug_assert!(!self.value_is_attached(v));
993 self.values[v] = value_data.into();
994 v
995 } else {
996 self.make_value(value_data)
997 };
998 let actual = self.results[inst].push(v, &mut self.value_lists);
999 debug_assert_eq!(expected, actual);
1000 }
1001
1002 result_tys.len()
1003 }
1004
1005 pub fn replace(&mut self, inst: Inst) -> ReplaceBuilder<'_> {
1007 ReplaceBuilder::new(self, inst)
1008 }
1009
1010 pub fn clear_results(&mut self, inst: Inst) {
1015 self.results[inst].clear(&mut self.value_lists)
1016 }
1017
1018 pub fn replace_result(&mut self, old_value: Value, new_type: Type) -> Value {
1026 let (num, inst) = match ValueData::from(self.values[old_value]) {
1027 ValueData::Inst { num, inst, .. } => (num, inst),
1028 _ => panic!("{old_value} is not an instruction result value"),
1029 };
1030 let new_value = self.make_value(ValueData::Inst {
1031 ty: new_type,
1032 num,
1033 inst,
1034 });
1035 let num = num as usize;
1036 let attached = mem::replace(
1037 self.results[inst]
1038 .get_mut(num, &mut self.value_lists)
1039 .expect("Replacing detached result"),
1040 new_value,
1041 );
1042 debug_assert_eq!(
1043 attached,
1044 old_value,
1045 "{} wasn't detached from {}",
1046 old_value,
1047 self.display_inst(inst)
1048 );
1049 new_value
1050 }
1051
1052 pub fn clone_inst(&mut self, inst: Inst) -> Inst {
1055 let inst_data = self.insts[inst];
1057 let inst_data = inst_data.deep_clone(&mut self.value_lists);
1061 let new_inst = self.make_inst(inst_data);
1062 let ctrl_typevar = self.ctrl_typevar(inst);
1064 self.make_inst_results(new_inst, ctrl_typevar);
1066 new_inst
1067 }
1068
1069 pub fn first_result(&self, inst: Inst) -> Value {
1073 self.results[inst]
1074 .first(&self.value_lists)
1075 .unwrap_or_else(|| panic!("{inst} has no results"))
1076 }
1077
1078 pub fn has_results(&self, inst: Inst) -> bool {
1080 !self.results[inst].is_empty()
1081 }
1082
1083 pub fn inst_results(&self, inst: Inst) -> &[Value] {
1085 self.results[inst].as_slice(&self.value_lists)
1086 }
1087
1088 pub fn inst_results_list(&self, inst: Inst) -> ValueList {
1090 self.results[inst]
1091 }
1092
1093 pub fn union(&mut self, x: Value, y: Value) -> Value {
1095 let ty = self.value_type(x);
1097 debug_assert_eq!(ty, self.value_type(y));
1098 self.make_value(ValueData::Union { ty, x, y })
1099 }
1100
1101 pub fn call_signature(&self, inst: Inst) -> Option<SigRef> {
1104 match self.insts[inst].analyze_call(&self.value_lists, &self.exception_tables) {
1105 CallInfo::NotACall => None,
1106 CallInfo::Direct(f, _) => Some(self.ext_funcs[f].signature),
1107 CallInfo::DirectWithSig(_, s, _) => Some(s),
1108 CallInfo::Indirect(s, _) => Some(s),
1109 }
1110 }
1111
1112 fn non_tail_call_or_try_call_signature(&self, inst: Inst) -> Option<SigRef> {
1116 let sig = self.call_signature(inst)?;
1117 match self.insts[inst].opcode() {
1118 ir::Opcode::ReturnCall | ir::Opcode::ReturnCallIndirect => None,
1119 ir::Opcode::TryCall | ir::Opcode::TryCallIndirect => None,
1120 _ => Some(sig),
1121 }
1122 }
1123
1124 pub(crate) fn num_expected_results_for_verifier(&self, inst: Inst) -> usize {
1127 match self.non_tail_call_or_try_call_signature(inst) {
1128 Some(sig) => self.signatures[sig].returns.len(),
1129 None => {
1130 let constraints = self.insts[inst].opcode().constraints();
1131 constraints.num_fixed_results()
1132 }
1133 }
1134 }
1135
1136 pub fn inst_result_types<'a>(
1138 &'a self,
1139 inst: Inst,
1140 ctrl_typevar: Type,
1141 ) -> impl iter::ExactSizeIterator<Item = Type> + 'a {
1142 return match self.non_tail_call_or_try_call_signature(inst) {
1143 Some(sig) => InstResultTypes::Signature(self, sig, 0),
1144 None => {
1145 let constraints = self.insts[inst].opcode().constraints();
1146 InstResultTypes::Constraints(constraints, ctrl_typevar, 0)
1147 }
1148 };
1149
1150 enum InstResultTypes<'a> {
1151 Signature(&'a DataFlowGraph, SigRef, usize),
1152 Constraints(ir::instructions::OpcodeConstraints, Type, usize),
1153 }
1154
1155 impl Iterator for InstResultTypes<'_> {
1156 type Item = Type;
1157
1158 fn next(&mut self) -> Option<Type> {
1159 match self {
1160 InstResultTypes::Signature(dfg, sig, i) => {
1161 let param = dfg.signatures[*sig].returns.get(*i)?;
1162 *i += 1;
1163 Some(param.value_type)
1164 }
1165 InstResultTypes::Constraints(constraints, ctrl_ty, i) => {
1166 if *i < constraints.num_fixed_results() {
1167 let ty = constraints.result_type(*i, *ctrl_ty);
1168 *i += 1;
1169 Some(ty)
1170 } else {
1171 None
1172 }
1173 }
1174 }
1175 }
1176
1177 fn size_hint(&self) -> (usize, Option<usize>) {
1178 let len = match self {
1179 InstResultTypes::Signature(dfg, sig, i) => {
1180 dfg.signatures[*sig].returns.len() - *i
1181 }
1182 InstResultTypes::Constraints(constraints, _, i) => {
1183 constraints.num_fixed_results() - *i
1184 }
1185 };
1186 (len, Some(len))
1187 }
1188 }
1189
1190 impl ExactSizeIterator for InstResultTypes<'_> {}
1191 }
1192
1193 pub fn compute_result_type(
1201 &self,
1202 inst: Inst,
1203 result_idx: usize,
1204 ctrl_typevar: Type,
1205 ) -> Option<Type> {
1206 self.inst_result_types(inst, ctrl_typevar).nth(result_idx)
1207 }
1208
1209 pub fn ctrl_typevar(&self, inst: Inst) -> Type {
1211 let constraints = self.insts[inst].opcode().constraints();
1212
1213 if !constraints.is_polymorphic() {
1214 types::INVALID
1215 } else if constraints.requires_typevar_operand() {
1216 self.value_type(
1219 self.insts[inst]
1220 .typevar_operand(&self.value_lists)
1221 .unwrap_or_else(|| {
1222 panic!(
1223 "Instruction format for {:?} doesn't have a designated operand",
1224 self.insts[inst]
1225 )
1226 }),
1227 )
1228 } else {
1229 self.value_type(self.first_result(inst))
1230 }
1231 }
1232}
1233
1234impl DataFlowGraph {
1236 pub fn make_block(&mut self) -> Block {
1238 self.blocks.add()
1239 }
1240
1241 pub fn num_block_params(&self, block: Block) -> usize {
1243 self.blocks[block].params(&self.value_lists).len()
1244 }
1245
1246 pub fn block_params(&self, block: Block) -> &[Value] {
1248 self.blocks[block].params(&self.value_lists)
1249 }
1250
1251 pub fn block_param_types(&self, block: Block) -> impl Iterator<Item = Type> + '_ {
1253 self.block_params(block).iter().map(|&v| self.value_type(v))
1254 }
1255
1256 pub fn append_block_param(&mut self, block: Block, ty: Type) -> Value {
1258 let param = self.values.next_key();
1259 let num = self.blocks[block].params.push(param, &mut self.value_lists);
1260 debug_assert!(num <= u16::MAX as usize, "Too many parameters on block");
1261 self.make_value(ValueData::Param {
1262 ty,
1263 num: num as u16,
1264 block,
1265 })
1266 }
1267
1268 pub fn swap_remove_block_param(&mut self, val: Value) -> usize {
1277 let (block, num) =
1278 if let ValueData::Param { num, block, .. } = ValueData::from(self.values[val]) {
1279 (block, num)
1280 } else {
1281 panic!("{val} must be a block parameter");
1282 };
1283 self.blocks[block]
1284 .params
1285 .swap_remove(num as usize, &mut self.value_lists);
1286 if let Some(last_arg_val) = self.blocks[block]
1287 .params
1288 .get(num as usize, &self.value_lists)
1289 {
1290 let mut last_arg_data = ValueData::from(self.values[last_arg_val]);
1292 if let ValueData::Param { num: old_num, .. } = &mut last_arg_data {
1293 *old_num = num;
1294 self.values[last_arg_val] = last_arg_data.into();
1295 } else {
1296 panic!("{last_arg_val} should be a Block parameter");
1297 }
1298 }
1299 num as usize
1300 }
1301
1302 pub fn remove_block_param(&mut self, val: Value) {
1305 let (block, num) =
1306 if let ValueData::Param { num, block, .. } = ValueData::from(self.values[val]) {
1307 (block, num)
1308 } else {
1309 panic!("{val} must be a block parameter");
1310 };
1311 self.blocks[block]
1312 .params
1313 .remove(num as usize, &mut self.value_lists);
1314 for index in num..(self.num_block_params(block) as u16) {
1315 let packed = &mut self.values[self.blocks[block]
1316 .params
1317 .get(index as usize, &self.value_lists)
1318 .unwrap()];
1319 let mut data = ValueData::from(*packed);
1320 match &mut data {
1321 ValueData::Param { num, .. } => {
1322 *num -= 1;
1323 *packed = data.into();
1324 }
1325 _ => panic!(
1326 "{} must be a block parameter",
1327 self.blocks[block]
1328 .params
1329 .get(index as usize, &self.value_lists)
1330 .unwrap()
1331 ),
1332 }
1333 }
1334 }
1335
1336 pub fn attach_block_param(&mut self, block: Block, param: Value) {
1342 debug_assert!(!self.value_is_attached(param));
1343 let num = self.blocks[block].params.push(param, &mut self.value_lists);
1344 debug_assert!(num <= u16::MAX as usize, "Too many parameters on block");
1345 let ty = self.value_type(param);
1346 self.values[param] = ValueData::Param {
1347 ty,
1348 num: num as u16,
1349 block,
1350 }
1351 .into();
1352 }
1353
1354 pub fn replace_block_param(&mut self, old_value: Value, new_type: Type) -> Value {
1364 let (block, num) =
1366 if let ValueData::Param { num, block, .. } = ValueData::from(self.values[old_value]) {
1367 (block, num)
1368 } else {
1369 panic!("{old_value} must be a block parameter");
1370 };
1371 let new_arg = self.make_value(ValueData::Param {
1372 ty: new_type,
1373 num,
1374 block,
1375 });
1376
1377 self.blocks[block]
1378 .params
1379 .as_mut_slice(&mut self.value_lists)[num as usize] = new_arg;
1380 new_arg
1381 }
1382
1383 pub fn detach_block_params(&mut self, block: Block) -> ValueList {
1389 self.blocks[block].params.take()
1390 }
1391
1392 pub fn detach_inst_results(&mut self, inst: Inst) {
1398 self.results[inst].clear(&mut self.value_lists);
1399 }
1400}
1401
1402#[derive(Clone, PartialEq, Hash)]
1408#[cfg_attr(feature = "enable-serde", derive(Serialize, Deserialize))]
1409pub struct BlockData {
1410 params: ValueList,
1412}
1413
1414impl BlockData {
1415 fn new() -> Self {
1416 Self {
1417 params: ValueList::new(),
1418 }
1419 }
1420
1421 pub fn params<'a>(&self, pool: &'a ValueListPool) -> &'a [Value] {
1423 self.params.as_slice(pool)
1424 }
1425}
1426
1427pub struct DisplayInst<'a>(&'a DataFlowGraph, Inst);
1429
1430impl<'a> fmt::Display for DisplayInst<'a> {
1431 fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
1432 let dfg = self.0;
1433 let inst = self.1;
1434
1435 if let Some((first, rest)) = dfg.inst_results(inst).split_first() {
1436 write!(f, "{first}")?;
1437 for v in rest {
1438 write!(f, ", {v}")?;
1439 }
1440 write!(f, " = ")?;
1441 }
1442
1443 let typevar = dfg.ctrl_typevar(inst);
1444 if typevar.is_invalid() {
1445 write!(f, "{}", dfg.insts[inst].opcode())?;
1446 } else {
1447 write!(f, "{}.{}", dfg.insts[inst].opcode(), typevar)?;
1448 }
1449 write_operands(f, dfg, inst)
1450 }
1451}
1452
1453impl DataFlowGraph {
1455 #[cold]
1458 fn set_value_type_for_parser(&mut self, v: Value, t: Type) {
1459 assert_eq!(
1460 self.value_type(v),
1461 types::INVALID,
1462 "this function is only for assigning types to previously invalid values"
1463 );
1464 self.values[v].set_type(t);
1465 }
1466
1467 pub fn check_dynamic_type(&mut self, ty: Type) -> Option<Type> {
1469 debug_assert!(ty.is_dynamic_vector());
1470 if self
1471 .dynamic_types
1472 .values()
1473 .any(|dyn_ty_data| dyn_ty_data.concrete().unwrap() == ty)
1474 {
1475 Some(ty)
1476 } else {
1477 None
1478 }
1479 }
1480
1481 #[cold]
1485 pub fn make_inst_results_for_parser(
1486 &mut self,
1487 inst: Inst,
1488 ctrl_typevar: Type,
1489 reuse: &[Value],
1490 ) -> usize {
1491 let mut reuse_iter = reuse.iter().copied();
1492 let result_tys: SmallVec<[_; 16]> = self.inst_result_types(inst, ctrl_typevar).collect();
1493 for ty in result_tys {
1494 if ty.is_dynamic_vector() {
1495 self.check_dynamic_type(ty)
1496 .unwrap_or_else(|| panic!("Use of undeclared dynamic type: {ty}"));
1497 }
1498 if let Some(v) = reuse_iter.next() {
1499 self.set_value_type_for_parser(v, ty);
1500 }
1501 }
1502
1503 self.make_inst_results_reusing(inst, ctrl_typevar, reuse.iter().map(|x| Some(*x)))
1504 }
1505
1506 #[cold]
1510 pub fn append_block_param_for_parser(&mut self, block: Block, ty: Type, val: Value) {
1511 let num = self.blocks[block].params.push(val, &mut self.value_lists);
1512 assert!(num <= u16::MAX as usize, "Too many parameters on block");
1513 self.values[val] = ValueData::Param {
1514 ty,
1515 num: num as u16,
1516 block,
1517 }
1518 .into();
1519 }
1520
1521 #[cold]
1524 pub fn make_value_alias_for_serialization(&mut self, src: Value, dest: Value) {
1525 assert_ne!(src, Value::reserved_value());
1526 assert_ne!(dest, Value::reserved_value());
1527
1528 let ty = if self.values.is_valid(src) {
1529 self.value_type(src)
1530 } else {
1531 types::INVALID
1534 };
1535 let data = ValueData::Alias { ty, original: src };
1536 self.values[dest] = data.into();
1537 }
1538
1539 #[cold]
1543 pub fn value_alias_dest_for_serialization(&self, v: Value) -> Option<Value> {
1544 if let ValueData::Alias { original, .. } = ValueData::from(self.values[v]) {
1545 Some(original)
1546 } else {
1547 None
1548 }
1549 }
1550
1551 #[cold]
1554 pub fn set_alias_type_for_parser(&mut self, v: Value) -> bool {
1555 if let Some(resolved) = maybe_resolve_aliases(&self.values, v) {
1556 let old_ty = self.value_type(v);
1557 let new_ty = self.value_type(resolved);
1558 if old_ty == types::INVALID {
1559 self.set_value_type_for_parser(v, new_ty);
1560 } else {
1561 assert_eq!(old_ty, new_ty);
1562 }
1563 true
1564 } else {
1565 false
1566 }
1567 }
1568
1569 #[cold]
1572 pub fn make_invalid_value_for_parser(&mut self) {
1573 let data = ValueData::Alias {
1574 ty: types::INVALID,
1575 original: Value::reserved_value(),
1576 };
1577 self.make_value(data);
1578 }
1579
1580 #[cold]
1583 pub fn value_is_valid_for_parser(&self, v: Value) -> bool {
1584 if !self.value_is_valid(v) {
1585 return false;
1586 }
1587 if let ValueData::Alias { ty, .. } = ValueData::from(self.values[v]) {
1588 ty != types::INVALID
1589 } else {
1590 true
1591 }
1592 }
1593}
1594
1595#[cfg(test)]
1596mod tests {
1597 use super::*;
1598 use crate::cursor::{Cursor, FuncCursor};
1599 use crate::ir::{Function, Opcode, TrapCode};
1600 use alloc::string::ToString;
1601
1602 #[test]
1603 fn make_inst() {
1604 let mut dfg = DataFlowGraph::new();
1605
1606 let idata = InstructionData::UnaryImm {
1607 opcode: Opcode::Iconst,
1608 imm: 0.into(),
1609 };
1610 let inst = dfg.make_inst(idata);
1611
1612 dfg.make_inst_results(inst, types::I32);
1613 assert_eq!(inst.to_string(), "inst0");
1614 assert_eq!(dfg.display_inst(inst).to_string(), "v0 = iconst.i32 0");
1615
1616 {
1618 let immdfg = &dfg;
1619 let ins = &immdfg.insts[inst];
1620 assert_eq!(ins.opcode(), Opcode::Iconst);
1621 }
1622
1623 let val = dfg.first_result(inst);
1625 assert_eq!(dfg.inst_results(inst), &[val]);
1626
1627 assert_eq!(dfg.value_def(val), ValueDef::Result(inst, 0));
1628 assert_eq!(dfg.value_type(val), types::I32);
1629
1630 assert!(dfg.value_is_attached(val));
1632 let v2 = dfg.replace_result(val, types::F64);
1633 assert!(!dfg.value_is_attached(val));
1634 assert!(dfg.value_is_attached(v2));
1635 assert_eq!(dfg.inst_results(inst), &[v2]);
1636 assert_eq!(dfg.value_def(v2), ValueDef::Result(inst, 0));
1637 assert_eq!(dfg.value_type(v2), types::F64);
1638 }
1639
1640 #[test]
1641 fn no_results() {
1642 let mut dfg = DataFlowGraph::new();
1643
1644 let idata = InstructionData::Trap {
1645 opcode: Opcode::Trap,
1646 code: TrapCode::unwrap_user(1),
1647 };
1648 let inst = dfg.make_inst(idata);
1649 assert_eq!(dfg.display_inst(inst).to_string(), "trap user1");
1650
1651 assert_eq!(dfg.inst_results(inst), &[]);
1653 }
1654
1655 #[test]
1656 fn block() {
1657 let mut dfg = DataFlowGraph::new();
1658
1659 let block = dfg.make_block();
1660 assert_eq!(block.to_string(), "block0");
1661 assert_eq!(dfg.num_block_params(block), 0);
1662 assert_eq!(dfg.block_params(block), &[]);
1663 assert!(dfg.detach_block_params(block).is_empty());
1664 assert_eq!(dfg.num_block_params(block), 0);
1665 assert_eq!(dfg.block_params(block), &[]);
1666
1667 let arg1 = dfg.append_block_param(block, types::F32);
1668 assert_eq!(arg1.to_string(), "v0");
1669 assert_eq!(dfg.num_block_params(block), 1);
1670 assert_eq!(dfg.block_params(block), &[arg1]);
1671
1672 let arg2 = dfg.append_block_param(block, types::I16);
1673 assert_eq!(arg2.to_string(), "v1");
1674 assert_eq!(dfg.num_block_params(block), 2);
1675 assert_eq!(dfg.block_params(block), &[arg1, arg2]);
1676
1677 assert_eq!(dfg.value_def(arg1), ValueDef::Param(block, 0));
1678 assert_eq!(dfg.value_def(arg2), ValueDef::Param(block, 1));
1679 assert_eq!(dfg.value_type(arg1), types::F32);
1680 assert_eq!(dfg.value_type(arg2), types::I16);
1681
1682 let vlist = dfg.detach_block_params(block);
1684 assert_eq!(dfg.num_block_params(block), 0);
1685 assert_eq!(dfg.block_params(block), &[]);
1686 assert_eq!(vlist.as_slice(&dfg.value_lists), &[arg1, arg2]);
1687 dfg.attach_block_param(block, arg2);
1688 let arg3 = dfg.append_block_param(block, types::I32);
1689 dfg.attach_block_param(block, arg1);
1690 assert_eq!(dfg.block_params(block), &[arg2, arg3, arg1]);
1691 }
1692
1693 #[test]
1694 fn replace_block_params() {
1695 let mut dfg = DataFlowGraph::new();
1696
1697 let block = dfg.make_block();
1698 let arg1 = dfg.append_block_param(block, types::F32);
1699
1700 let new1 = dfg.replace_block_param(arg1, types::I64);
1701 assert_eq!(dfg.value_type(arg1), types::F32);
1702 assert_eq!(dfg.value_type(new1), types::I64);
1703 assert_eq!(dfg.block_params(block), &[new1]);
1704
1705 dfg.attach_block_param(block, arg1);
1706 assert_eq!(dfg.block_params(block), &[new1, arg1]);
1707
1708 let new2 = dfg.replace_block_param(arg1, types::I8);
1709 assert_eq!(dfg.value_type(arg1), types::F32);
1710 assert_eq!(dfg.value_type(new2), types::I8);
1711 assert_eq!(dfg.block_params(block), &[new1, new2]);
1712
1713 dfg.attach_block_param(block, arg1);
1714 assert_eq!(dfg.block_params(block), &[new1, new2, arg1]);
1715
1716 let new3 = dfg.replace_block_param(new2, types::I16);
1717 assert_eq!(dfg.value_type(new1), types::I64);
1718 assert_eq!(dfg.value_type(new2), types::I8);
1719 assert_eq!(dfg.value_type(new3), types::I16);
1720 assert_eq!(dfg.block_params(block), &[new1, new3, arg1]);
1721 }
1722
1723 #[test]
1724 fn swap_remove_block_params() {
1725 let mut dfg = DataFlowGraph::new();
1726
1727 let block = dfg.make_block();
1728 let arg1 = dfg.append_block_param(block, types::F32);
1729 let arg2 = dfg.append_block_param(block, types::F32);
1730 let arg3 = dfg.append_block_param(block, types::F32);
1731 assert_eq!(dfg.block_params(block), &[arg1, arg2, arg3]);
1732
1733 dfg.swap_remove_block_param(arg1);
1734 assert_eq!(dfg.value_is_attached(arg1), false);
1735 assert_eq!(dfg.value_is_attached(arg2), true);
1736 assert_eq!(dfg.value_is_attached(arg3), true);
1737 assert_eq!(dfg.block_params(block), &[arg3, arg2]);
1738 dfg.swap_remove_block_param(arg2);
1739 assert_eq!(dfg.value_is_attached(arg2), false);
1740 assert_eq!(dfg.value_is_attached(arg3), true);
1741 assert_eq!(dfg.block_params(block), &[arg3]);
1742 dfg.swap_remove_block_param(arg3);
1743 assert_eq!(dfg.value_is_attached(arg3), false);
1744 assert_eq!(dfg.block_params(block), &[]);
1745 }
1746
1747 #[test]
1748 fn aliases() {
1749 use crate::ir::InstBuilder;
1750 use crate::ir::condcodes::IntCC;
1751
1752 let mut func = Function::new();
1753 let block0 = func.dfg.make_block();
1754 let mut pos = FuncCursor::new(&mut func);
1755 pos.insert_block(block0);
1756
1757 let v1 = pos.ins().iconst(types::I32, 42);
1759
1760 assert_eq!(pos.func.dfg.resolve_aliases(v1), v1);
1762
1763 let arg0 = pos.func.dfg.append_block_param(block0, types::I32);
1764 let (s, c) = pos.ins().uadd_overflow(v1, arg0);
1765 let iadd = match pos.func.dfg.value_def(s) {
1766 ValueDef::Result(i, 0) => i,
1767 _ => panic!(),
1768 };
1769
1770 pos.func.stencil.dfg.results[iadd].remove(1, &mut pos.func.stencil.dfg.value_lists);
1772
1773 pos.func.dfg.replace(iadd).iadd(v1, arg0);
1775 let c2 = pos.ins().icmp(IntCC::Equal, s, v1);
1776 pos.func.dfg.change_to_alias(c, c2);
1777
1778 assert_eq!(pos.func.dfg.resolve_aliases(c2), c2);
1779 assert_eq!(pos.func.dfg.resolve_aliases(c), c2);
1780 }
1781
1782 #[test]
1783 fn cloning() {
1784 use crate::ir::InstBuilder;
1785
1786 let mut func = Function::new();
1787 let mut sig = Signature::new(crate::isa::CallConv::SystemV);
1788 sig.params.push(ir::AbiParam::new(types::I32));
1789 let sig = func.import_signature(sig);
1790 let block0 = func.dfg.make_block();
1791 let mut pos = FuncCursor::new(&mut func);
1792 pos.insert_block(block0);
1793 let v1 = pos.ins().iconst(types::I32, 0);
1794 let v2 = pos.ins().iconst(types::I32, 1);
1795 let call_inst = pos.ins().call_indirect(sig, v1, &[v1]);
1796 let func = pos.func;
1797
1798 let call_inst_dup = func.dfg.clone_inst(call_inst);
1799 func.dfg.inst_args_mut(call_inst)[0] = v2;
1800 assert_eq!(v1, func.dfg.inst_args(call_inst_dup)[0]);
1801 }
1802}