1use crate::mir::*;
66use crate::ty::CanonicalUserTypeAnnotation;
67
68macro_rules! make_mir_visitor {
69 ($visitor_trait_name:ident, $($mutability:ident)?) => {
70 pub trait $visitor_trait_name<'tcx> {
71 fn visit_body(
75 &mut self,
76 body: &$($mutability)? Body<'tcx>,
77 ) {
78 self.super_body(body);
79 }
80
81 extra_body_methods!($($mutability)?);
82
83 fn visit_basic_block_data(
84 &mut self,
85 block: BasicBlock,
86 data: & $($mutability)? BasicBlockData<'tcx>,
87 ) {
88 self.super_basic_block_data(block, data);
89 }
90
91 fn visit_source_scope_data(
92 &mut self,
93 scope_data: & $($mutability)? SourceScopeData<'tcx>,
94 ) {
95 self.super_source_scope_data(scope_data);
96 }
97
98 fn visit_statement_debuginfo(
99 &mut self,
100 stmt_debuginfo: & $($mutability)? StmtDebugInfo<'tcx>,
101 location: Location
102 ) {
103 self.super_statement_debuginfo(stmt_debuginfo, location);
104 }
105
106 fn visit_statement(
107 &mut self,
108 statement: & $($mutability)? Statement<'tcx>,
109 location: Location,
110 ) {
111 self.super_statement(statement, location);
112 }
113
114 fn visit_assign(
115 &mut self,
116 place: & $($mutability)? Place<'tcx>,
117 rvalue: & $($mutability)? Rvalue<'tcx>,
118 location: Location,
119 ) {
120 self.super_assign(place, rvalue, location);
121 }
122
123 fn visit_terminator(
124 &mut self,
125 terminator: & $($mutability)? Terminator<'tcx>,
126 location: Location,
127 ) {
128 self.super_terminator(terminator, location);
129 }
130
131 fn visit_assert_message(
132 &mut self,
133 msg: & $($mutability)? AssertMessage<'tcx>,
134 location: Location,
135 ) {
136 self.super_assert_message(msg, location);
137 }
138
139 fn visit_rvalue(
140 &mut self,
141 rvalue: & $($mutability)? Rvalue<'tcx>,
142 location: Location,
143 ) {
144 self.super_rvalue(rvalue, location);
145 }
146
147 fn visit_operand(
148 &mut self,
149 operand: & $($mutability)? Operand<'tcx>,
150 location: Location,
151 ) {
152 self.super_operand(operand, location);
153 }
154
155 fn visit_ascribe_user_ty(
156 &mut self,
157 place: & $($mutability)? Place<'tcx>,
158 variance: $(& $mutability)? ty::Variance,
159 user_ty: & $($mutability)? UserTypeProjection,
160 location: Location,
161 ) {
162 self.super_ascribe_user_ty(place, variance, user_ty, location);
163 }
164
165 fn visit_coverage(
166 &mut self,
167 kind: & $($mutability)? coverage::CoverageKind,
168 location: Location,
169 ) {
170 self.super_coverage(kind, location);
171 }
172
173 fn visit_place(
174 &mut self,
175 place: & $($mutability)? Place<'tcx>,
176 context: PlaceContext,
177 location: Location,
178 ) {
179 self.super_place(place, context, location);
180 }
181
182 visit_place_fns!($($mutability)?);
183
184 fn visit_const_operand(
187 &mut self,
188 constant: & $($mutability)? ConstOperand<'tcx>,
189 location: Location,
190 ) {
191 self.super_const_operand(constant, location);
192 }
193
194 fn visit_ty_const(
195 &mut self,
196 ct: $( & $mutability)? ty::Const<'tcx>,
197 location: Location,
198 ) {
199 self.super_ty_const(ct, location);
200 }
201
202 fn visit_span(
203 &mut self,
204 span: $(& $mutability)? Span,
205 ) {
206 self.super_span(span);
207 }
208
209 fn visit_source_info(
210 &mut self,
211 source_info: & $($mutability)? SourceInfo,
212 ) {
213 self.super_source_info(source_info);
214 }
215
216 fn visit_ty(
217 &mut self,
218 ty: $(& $mutability)? Ty<'tcx>,
219 _: TyContext,
220 ) {
221 self.super_ty(ty);
222 }
223
224 fn visit_user_type_projection(
225 &mut self,
226 ty: & $($mutability)? UserTypeProjection,
227 ) {
228 self.super_user_type_projection(ty);
229 }
230
231 fn visit_user_type_annotation(
232 &mut self,
233 index: UserTypeAnnotationIndex,
234 ty: & $($mutability)? CanonicalUserTypeAnnotation<'tcx>,
235 ) {
236 self.super_user_type_annotation(index, ty);
237 }
238
239 fn visit_region(
240 &mut self,
241 region: $(& $mutability)? ty::Region<'tcx>,
242 _: Location,
243 ) {
244 self.super_region(region);
245 }
246
247 fn visit_args(
248 &mut self,
249 args: & $($mutability)? GenericArgsRef<'tcx>,
250 _: Location,
251 ) {
252 self.super_args(args);
253 }
254
255 fn visit_local_decl(
256 &mut self,
257 local: Local,
258 local_decl: & $($mutability)? LocalDecl<'tcx>,
259 ) {
260 self.super_local_decl(local, local_decl);
261 }
262
263 fn visit_var_debug_info(
264 &mut self,
265 var_debug_info: & $($mutability)* VarDebugInfo<'tcx>,
266 ) {
267 self.super_var_debug_info(var_debug_info);
268 }
269
270 fn visit_local(
271 &mut self,
272 local: $(& $mutability)? Local,
273 context: PlaceContext,
274 location: Location,
275 ) {
276 self.super_local(local, context, location)
277 }
278
279 fn visit_source_scope(
280 &mut self,
281 scope: $(& $mutability)? SourceScope,
282 ) {
283 self.super_source_scope(scope);
284 }
285
286 fn super_body(
290 &mut self,
291 body: &$($mutability)? Body<'tcx>,
292 ) {
293 super_body!(self, body, $($mutability, true)?);
294 }
295
296 fn super_basic_block_data(
297 &mut self,
298 block: BasicBlock,
299 data: & $($mutability)? BasicBlockData<'tcx>)
300 {
301 let BasicBlockData {
302 statements,
303 after_last_stmt_debuginfos,
304 terminator,
305 is_cleanup: _
306 } = data;
307
308 let mut index = 0;
309 for statement in statements {
310 let location = Location { block, statement_index: index };
311 self.visit_statement(statement, location);
312 index += 1;
313 }
314
315 let location = Location { block, statement_index: index };
316 for debuginfo in after_last_stmt_debuginfos as & $($mutability)? [_] {
317 self.visit_statement_debuginfo(debuginfo, location);
318 }
319 if let Some(terminator) = terminator {
320 self.visit_terminator(terminator, location);
321 }
322 }
323
324 fn super_source_scope_data(
325 &mut self,
326 scope_data: & $($mutability)? SourceScopeData<'tcx>,
327 ) {
328 let SourceScopeData {
329 span,
330 parent_scope,
331 inlined,
332 inlined_parent_scope,
333 local_data: _,
334 } = scope_data;
335
336 self.visit_span($(& $mutability)? *span);
337 if let Some(parent_scope) = parent_scope {
338 self.visit_source_scope($(& $mutability)? *parent_scope);
339 }
340 if let Some((callee, callsite_span)) = inlined {
341 let location = Location::START;
342
343 self.visit_span($(& $mutability)? *callsite_span);
344
345 let ty::Instance { def: callee_def, args: callee_args } = callee;
346 match callee_def {
347 ty::InstanceKind::Item(_def_id) => {}
348
349 ty::InstanceKind::Intrinsic(_def_id)
350 | ty::InstanceKind::VTableShim(_def_id)
351 | ty::InstanceKind::ReifyShim(_def_id, _)
352 | ty::InstanceKind::Virtual(_def_id, _)
353 | ty::InstanceKind::ThreadLocalShim(_def_id)
354 | ty::InstanceKind::ClosureOnceShim { call_once: _def_id, track_caller: _ }
355 | ty::InstanceKind::ConstructCoroutineInClosureShim {
356 coroutine_closure_def_id: _def_id,
357 receiver_by_ref: _,
358 }
359 | ty::InstanceKind::DropGlue(_def_id, None) => {}
360
361 ty::InstanceKind::FnPtrShim(_def_id, ty)
362 | ty::InstanceKind::DropGlue(_def_id, Some(ty))
363 | ty::InstanceKind::CloneShim(_def_id, ty)
364 | ty::InstanceKind::FnPtrAddrShim(_def_id, ty)
365 | ty::InstanceKind::AsyncDropGlue(_def_id, ty)
366 | ty::InstanceKind::AsyncDropGlueCtorShim(_def_id, ty) => {
367 self.visit_ty($(& $mutability)? *ty, TyContext::Location(location));
369 }
370 ty::InstanceKind::FutureDropPollShim(_def_id, proxy_ty, impl_ty) => {
371 self.visit_ty($(& $mutability)? *proxy_ty, TyContext::Location(location));
372 self.visit_ty($(& $mutability)? *impl_ty, TyContext::Location(location));
373 }
374 }
375 self.visit_args(callee_args, location);
376 }
377 if let Some(inlined_parent_scope) = inlined_parent_scope {
378 self.visit_source_scope($(& $mutability)? *inlined_parent_scope);
379 }
380 }
381
382 fn super_statement_debuginfo(
383 &mut self,
384 stmt_debuginfo: & $($mutability)? StmtDebugInfo<'tcx>,
385 location: Location
386 ) {
387 match stmt_debuginfo {
388 StmtDebugInfo::AssignRef(local, place) => {
389 self.visit_local(
390 $(& $mutability)? *local,
391 PlaceContext::NonUse(NonUseContext::VarDebugInfo),
392 location
393 );
394 self.visit_place(
395 place,
396 PlaceContext::NonUse(NonUseContext::VarDebugInfo),
397 location
398 );
399 },
400 StmtDebugInfo::InvalidAssign(local) => {
401 self.visit_local(
402 $(& $mutability)? *local,
403 PlaceContext::NonUse(NonUseContext::VarDebugInfo),
404 location
405 );
406 }
407 }
408 }
409
410 fn super_statement(
411 &mut self,
412 statement: & $($mutability)? Statement<'tcx>,
413 location: Location
414 ) {
415 let Statement { source_info, kind, debuginfos } = statement;
416
417 self.visit_source_info(source_info);
418 for debuginfo in debuginfos as & $($mutability)? [_] {
419 self.visit_statement_debuginfo(debuginfo, location);
420 }
421 match kind {
422 StatementKind::Assign((place, rvalue)) => {
423 self.visit_assign(place, rvalue, location);
424 }
425 StatementKind::FakeRead((_, place)) => {
426 self.visit_place(
427 place,
428 PlaceContext::NonMutatingUse(NonMutatingUseContext::Inspect),
429 location
430 );
431 }
432 StatementKind::SetDiscriminant { place, .. } => {
433 self.visit_place(
434 place,
435 PlaceContext::MutatingUse(MutatingUseContext::SetDiscriminant),
436 location
437 );
438 }
439 StatementKind::StorageLive(local) => {
440 self.visit_local(
441 $(& $mutability)? *local,
442 PlaceContext::NonUse(NonUseContext::StorageLive),
443 location
444 );
445 }
446 StatementKind::StorageDead(local) => {
447 self.visit_local(
448 $(& $mutability)? *local,
449 PlaceContext::NonUse(NonUseContext::StorageDead),
450 location
451 );
452 }
453 StatementKind::PlaceMention(place) => {
454 self.visit_place(
455 place,
456 PlaceContext::NonMutatingUse(NonMutatingUseContext::PlaceMention),
457 location
458 );
459 }
460 StatementKind::AscribeUserType((place, user_ty), variance) => {
461 self.visit_ascribe_user_ty(
462 place,
463 $(& $mutability)? *variance,
464 user_ty,
465 location
466 );
467 }
468 StatementKind::Coverage(coverage) => {
469 self.visit_coverage(
470 coverage,
471 location
472 )
473 }
474 StatementKind::Intrinsic(intrinsic) => {
475 match intrinsic {
476 NonDivergingIntrinsic::Assume(op) => self.visit_operand(op, location),
477 NonDivergingIntrinsic::CopyNonOverlapping(CopyNonOverlapping {
478 src,
479 dst,
480 count
481 }) => {
482 self.visit_operand(src, location);
483 self.visit_operand(dst, location);
484 self.visit_operand(count, location);
485 }
486 }
487 }
488 StatementKind::BackwardIncompatibleDropHint { place, .. } => {
489 self.visit_place(
490 place,
491 PlaceContext::NonUse(NonUseContext::BackwardIncompatibleDropHint),
492 location
493 );
494 }
495 StatementKind::ConstEvalCounter => {}
496 StatementKind::Nop => {}
497 }
498 }
499
500 fn super_assign(
501 &mut self,
502 place: &$($mutability)? Place<'tcx>,
503 rvalue: &$($mutability)? Rvalue<'tcx>,
504 location: Location
505 ) {
506 self.visit_place(
507 place,
508 PlaceContext::MutatingUse(MutatingUseContext::Store),
509 location
510 );
511 self.visit_rvalue(rvalue, location);
512 }
513
514 fn super_terminator(
515 &mut self,
516 terminator: &$($mutability)? Terminator<'tcx>,
517 location: Location
518 ) {
519 let Terminator { source_info, kind } = terminator;
520
521 self.visit_source_info(source_info);
522 match kind {
523 TerminatorKind::Goto { .. }
524 | TerminatorKind::UnwindResume
525 | TerminatorKind::UnwindTerminate(_)
526 | TerminatorKind::CoroutineDrop
527 | TerminatorKind::Unreachable
528 | TerminatorKind::FalseEdge { .. }
529 | TerminatorKind::FalseUnwind { .. } => {}
530
531 TerminatorKind::Return => {
532 let $($mutability)? local = RETURN_PLACE;
535 self.visit_local(
536 $(& $mutability)? local,
537 PlaceContext::NonMutatingUse(NonMutatingUseContext::Move),
538 location,
539 );
540
541 assert_eq!(
542 local,
543 RETURN_PLACE,
544 "`MutVisitor` tried to mutate return place of `return` terminator"
545 );
546 }
547
548 TerminatorKind::SwitchInt { discr, targets: _ } => {
549 self.visit_operand(discr, location);
550 }
551
552 TerminatorKind::Drop {
553 place,
554 target: _,
555 unwind: _,
556 replace: _,
557 drop: _,
558 async_fut,
559 } => {
560 self.visit_place(
561 place,
562 PlaceContext::MutatingUse(MutatingUseContext::Drop),
563 location
564 );
565 if let Some(async_fut) = async_fut {
566 self.visit_local(
567 $(&$mutability)? *async_fut,
568 PlaceContext::MutatingUse(MutatingUseContext::Borrow),
569 location
570 );
571 }
572 }
573
574 TerminatorKind::Call {
575 func,
576 args,
577 destination,
578 target: _,
579 unwind: _,
580 call_source: _,
581 fn_span,
582 } => {
583 self.visit_span($(& $mutability)? *fn_span);
584 self.visit_operand(func, location);
585 for arg in args {
586 self.visit_operand(&$($mutability)? arg.node, location);
587 }
588 self.visit_place(
589 destination,
590 PlaceContext::MutatingUse(MutatingUseContext::Call),
591 location
592 );
593 }
594
595 TerminatorKind::TailCall { func, args, fn_span } => {
596 self.visit_span($(& $mutability)? *fn_span);
597 self.visit_operand(func, location);
598 for arg in args {
599 self.visit_operand(&$($mutability)? arg.node, location);
600 }
601 },
602
603 TerminatorKind::Assert { cond, expected: _, msg, target: _, unwind: _ } => {
604 self.visit_operand(cond, location);
605 self.visit_assert_message(msg, location);
606 }
607
608 TerminatorKind::Yield { value, resume: _, resume_arg, drop: _ } => {
609 self.visit_operand(value, location);
610 self.visit_place(
611 resume_arg,
612 PlaceContext::MutatingUse(MutatingUseContext::Yield),
613 location,
614 );
615 }
616
617 TerminatorKind::InlineAsm {
618 asm_macro: _,
619 template: _,
620 operands,
621 options: _,
622 line_spans: _,
623 targets: _,
624 unwind: _,
625 } => {
626 for op in operands {
627 match op {
628 InlineAsmOperand::In { value, .. } => {
629 self.visit_operand(value, location);
630 }
631 InlineAsmOperand::Out { place: Some(place), .. } => {
632 self.visit_place(
633 place,
634 PlaceContext::MutatingUse(MutatingUseContext::AsmOutput),
635 location,
636 );
637 }
638 InlineAsmOperand::InOut { in_value, out_place, .. } => {
639 self.visit_operand(in_value, location);
640 if let Some(out_place) = out_place {
641 self.visit_place(
642 out_place,
643 PlaceContext::MutatingUse(MutatingUseContext::AsmOutput),
644 location,
645 );
646 }
647 }
648 InlineAsmOperand::Const { value }
649 | InlineAsmOperand::SymFn { value } => {
650 self.visit_const_operand(value, location);
651 }
652 InlineAsmOperand::Out { place: None, .. }
653 | InlineAsmOperand::SymStatic { def_id: _ }
654 | InlineAsmOperand::Label { target_index: _ } => {}
655 }
656 }
657 }
658 }
659 }
660
661 fn super_assert_message(
662 &mut self,
663 msg: & $($mutability)? AssertMessage<'tcx>,
664 location: Location
665 ) {
666 use crate::mir::AssertKind::*;
667 match msg {
668 BoundsCheck { len, index } => {
669 self.visit_operand(len, location);
670 self.visit_operand(index, location);
671 }
672 Overflow(_, l, r) => {
673 self.visit_operand(l, location);
674 self.visit_operand(r, location);
675 }
676 OverflowNeg(op) | DivisionByZero(op) | RemainderByZero(op) | InvalidEnumConstruction(op) => {
677 self.visit_operand(op, location);
678 }
679 ResumedAfterReturn(_) | ResumedAfterPanic(_) | NullPointerDereference | ResumedAfterDrop(_) => {
680 }
682 MisalignedPointerDereference { required, found } => {
683 self.visit_operand(required, location);
684 self.visit_operand(found, location);
685 }
686 }
687 }
688
689 fn super_rvalue(
690 &mut self,
691 rvalue: & $($mutability)? Rvalue<'tcx>,
692 location: Location
693 ) {
694 match rvalue {
695 Rvalue::Use(operand, _with_retag) => {
696 self.visit_operand(operand, location);
697 }
698
699 Rvalue::Repeat(value, ct) => {
700 self.visit_operand(value, location);
701 self.visit_ty_const($(&$mutability)? *ct, location);
702 }
703
704 Rvalue::ThreadLocalRef(_) => {}
705
706 Rvalue::Ref(r, bk, path) => {
707 self.visit_region($(& $mutability)? *r, location);
708 let ctx = match bk {
709 BorrowKind::Shared => PlaceContext::NonMutatingUse(
710 NonMutatingUseContext::SharedBorrow
711 ),
712 BorrowKind::Fake(_) => PlaceContext::NonMutatingUse(
713 NonMutatingUseContext::FakeBorrow
714 ),
715 BorrowKind::Mut { .. } =>
716 PlaceContext::MutatingUse(MutatingUseContext::Borrow),
717 };
718 self.visit_place(path, ctx, location);
719 }
720
721 Rvalue::Reborrow(target, mutability, place) => {
722 self.visit_ty($(& $mutability)? *target, TyContext::Location(location));
723 self.visit_place(
724 place,
725 match mutability {
726 Mutability::Not => PlaceContext::NonMutatingUse(NonMutatingUseContext::SharedBorrow),
727 Mutability::Mut => PlaceContext::MutatingUse(MutatingUseContext::Borrow),
728 },
729 location
730 );
731 }
732
733 Rvalue::CopyForDeref(place) => {
734 self.visit_place(
735 place,
736 PlaceContext::NonMutatingUse(NonMutatingUseContext::Inspect),
737 location
738 );
739 }
740
741 Rvalue::RawPtr(m, path) => {
742 let ctx = match m {
743 RawPtrKind::Mut => PlaceContext::MutatingUse(
744 MutatingUseContext::RawBorrow
745 ),
746 RawPtrKind::Const => PlaceContext::NonMutatingUse(
747 NonMutatingUseContext::RawBorrow
748 ),
749 RawPtrKind::FakeForPtrMetadata => PlaceContext::NonMutatingUse(
750 NonMutatingUseContext::Inspect
751 ),
752 };
753 self.visit_place(path, ctx, location);
754 }
755
756 Rvalue::Cast(_cast_kind, operand, ty) => {
757 self.visit_operand(operand, location);
758 self.visit_ty($(& $mutability)? *ty, TyContext::Location(location));
759 }
760
761 Rvalue::BinaryOp(_bin_op, (lhs, rhs)) => {
762 self.visit_operand(lhs, location);
763 self.visit_operand(rhs, location);
764 }
765
766 Rvalue::UnaryOp(_un_op, op) => {
767 self.visit_operand(op, location);
768 }
769
770 Rvalue::Discriminant(place) => {
771 self.visit_place(
772 place,
773 PlaceContext::NonMutatingUse(NonMutatingUseContext::Inspect),
774 location
775 );
776 }
777
778 Rvalue::Aggregate(kind, operands) => {
779 let kind = &$($mutability)? **kind;
780 match kind {
781 AggregateKind::Array(ty) => {
782 self.visit_ty($(& $mutability)? *ty, TyContext::Location(location));
783 }
784 AggregateKind::Tuple => {}
785 AggregateKind::Adt(
786 _adt_def,
787 _variant_index,
788 args,
789 _user_args,
790 _active_field_index
791 ) => {
792 self.visit_args(args, location);
793 }
794 AggregateKind::Closure(_, closure_args) => {
795 self.visit_args(closure_args, location);
796 }
797 AggregateKind::Coroutine(_, coroutine_args) => {
798 self.visit_args(coroutine_args, location);
799 }
800 AggregateKind::CoroutineClosure(_, coroutine_closure_args) => {
801 self.visit_args(coroutine_closure_args, location);
802 }
803 AggregateKind::RawPtr(ty, _) => {
804 self.visit_ty($(& $mutability)? *ty, TyContext::Location(location));
805 }
806 }
807
808 for operand in operands {
809 self.visit_operand(operand, location);
810 }
811 }
812
813 Rvalue::WrapUnsafeBinder(op, ty) => {
814 self.visit_operand(op, location);
815 self.visit_ty($(& $mutability)? *ty, TyContext::Location(location));
816 }
817
818
819 }
820 }
821
822 fn super_operand(
823 &mut self,
824 operand: & $($mutability)? Operand<'tcx>,
825 location: Location
826 ) {
827 match operand {
828 Operand::Copy(place) => {
829 self.visit_place(
830 place,
831 PlaceContext::NonMutatingUse(NonMutatingUseContext::Copy),
832 location
833 );
834 }
835 Operand::Move(place) => {
836 self.visit_place(
837 place,
838 PlaceContext::NonMutatingUse(NonMutatingUseContext::Move),
839 location
840 );
841 }
842 Operand::Constant(constant) => {
843 self.visit_const_operand(constant, location);
844 }
845 Operand::RuntimeChecks(_) => {}
846 }
847 }
848
849 fn super_ascribe_user_ty(
850 &mut self,
851 place: & $($mutability)? Place<'tcx>,
852 variance: $(& $mutability)? ty::Variance,
853 user_ty: & $($mutability)? UserTypeProjection,
854 location: Location)
855 {
856 self.visit_place(
857 place,
858 PlaceContext::NonUse(
859 NonUseContext::AscribeUserTy($(* &$mutability *)? variance)
860 ),
861 location
862 );
863 self.visit_user_type_projection(user_ty);
864 }
865
866 fn super_coverage(
867 &mut self,
868 _kind: & $($mutability)? coverage::CoverageKind,
869 _location: Location
870 ) {
871 }
872
873 fn super_local_decl(
874 &mut self,
875 local: Local,
876 local_decl: & $($mutability)? LocalDecl<'tcx>
877 ) {
878 let LocalDecl {
879 mutability: _,
880 ty,
881 user_ty,
882 source_info,
883 local_info: _,
884 } = local_decl;
885
886 self.visit_source_info(source_info);
887
888 self.visit_ty($(& $mutability)? *ty, TyContext::LocalDecl {
889 local,
890 source_info: *source_info,
891 });
892 if let Some(user_ty) = user_ty {
893 for user_ty in & $($mutability)? user_ty.contents {
894 self.visit_user_type_projection(user_ty);
895 }
896 }
897 }
898
899 fn super_local(
900 &mut self,
901 _local: $(& $mutability)? Local,
902 _context: PlaceContext,
903 _location: Location,
904 ) {
905 }
906
907 fn super_var_debug_info(
908 &mut self,
909 var_debug_info: & $($mutability)? VarDebugInfo<'tcx>
910 ) {
911 let VarDebugInfo {
912 name: _,
913 source_info,
914 composite,
915 value,
916 argument_index: _,
917 } = var_debug_info;
918
919 self.visit_source_info(source_info);
920 let location = Location::START;
921 if let Some(VarDebugInfoFragment {
922 ty,
923 projection
924 }) = composite {
925 self.visit_ty($(& $mutability)? *ty, TyContext::Location(location));
926 for elem in projection {
927 let ProjectionElem::Field(_, ty) = elem else { bug!() };
928 self.visit_ty($(& $mutability)? *ty, TyContext::Location(location));
929 }
930 }
931 match value {
932 VarDebugInfoContents::Const(c) => self.visit_const_operand(c, location),
933 VarDebugInfoContents::Place(place) =>
934 self.visit_place(
935 place,
936 PlaceContext::NonUse(NonUseContext::VarDebugInfo),
937 location
938 ),
939 }
940 }
941
942 fn super_source_scope(&mut self, _scope: $(& $mutability)? SourceScope) {}
943
944 fn super_const_operand(
945 &mut self,
946 constant: & $($mutability)? ConstOperand<'tcx>,
947 location: Location
948 ) {
949 let ConstOperand {
950 span,
951 user_ty: _, const_,
953 } = constant;
954
955 self.visit_span($(& $mutability)? *span);
956 match const_ {
957 Const::Ty(_, ct) => self.visit_ty_const($(&$mutability)? *ct, location),
958 Const::Val(_, ty) | Const::Unevaluated(_, ty) => {
959 self.visit_ty($(& $mutability)? *ty, TyContext::Location(location));
960 }
961 }
962 }
963
964 fn super_ty_const(
965 &mut self,
966 _ct: $(& $mutability)? ty::Const<'tcx>,
967 _location: Location,
968 ) {
969 }
970
971 fn super_span(&mut self, _span: $(& $mutability)? Span) {}
972
973 fn super_source_info(&mut self, source_info: & $($mutability)? SourceInfo) {
974 let SourceInfo { span, scope } = source_info;
975
976 self.visit_span($(& $mutability)? *span);
977 self.visit_source_scope($(& $mutability)? *scope);
978 }
979
980 fn super_user_type_projection(&mut self, _ty: & $($mutability)? UserTypeProjection) {}
981
982 fn super_user_type_annotation(
983 &mut self,
984 _index: UserTypeAnnotationIndex,
985 ty: & $($mutability)? CanonicalUserTypeAnnotation<'tcx>,
986 ) {
987 self.visit_span($(& $mutability)? ty.span);
988 self.visit_ty($(& $mutability)? ty.inferred_ty, TyContext::UserTy(ty.span));
989 }
990
991 fn super_ty(&mut self, _ty: $(& $mutability)? Ty<'tcx>) {}
992
993 fn super_region(&mut self, _region: $(& $mutability)? ty::Region<'tcx>) {}
994
995 fn super_args(&mut self, _args: & $($mutability)? GenericArgsRef<'tcx>) {}
996
997 fn visit_location(
1000 &mut self,
1001 body: &$($mutability)? Body<'tcx>,
1002 location: Location
1003 ) {
1004 let basic_block =
1005 & $($mutability)? basic_blocks!(body, $($mutability, true)?)[location.block];
1006 if basic_block.statements.len() == location.statement_index {
1007 if let Some(ref $($mutability)? terminator) = basic_block.terminator {
1008 self.visit_terminator(terminator, location)
1009 }
1010 } else {
1011 let statement = & $($mutability)?
1012 basic_block.statements[location.statement_index];
1013 self.visit_statement(statement, location)
1014 }
1015 }
1016 }
1017 }
1018}
1019
1020macro_rules! basic_blocks {
1021 ($body:ident, mut, true) => {
1022 $body.basic_blocks.as_mut()
1023 };
1024 ($body:ident, mut, false) => {
1025 $body.basic_blocks.as_mut_preserves_cfg()
1026 };
1027 ($body:ident,) => {
1028 $body.basic_blocks
1029 };
1030}
1031
1032macro_rules! basic_blocks_iter {
1033 ($body:ident, mut, $invalidate:tt) => {
1034 basic_blocks!($body, mut, $invalidate).iter_enumerated_mut()
1035 };
1036 ($body:ident,) => {
1037 basic_blocks!($body,).iter_enumerated()
1038 };
1039}
1040
1041macro_rules! extra_body_methods {
1042 (mut) => {
1043 fn visit_body_preserves_cfg(&mut self, body: &mut Body<'tcx>) {
1044 self.super_body_preserves_cfg(body);
1045 }
1046
1047 fn super_body_preserves_cfg(&mut self, body: &mut Body<'tcx>) {
1048 super_body!(self, body, mut, false);
1049 }
1050 };
1051 () => {};
1052}
1053
1054macro_rules! super_body {
1055 ($self:ident, $body:ident, $($mutability:ident, $invalidate:tt)?) => {
1056 let span = $body.span;
1057 if let Some(coroutine) = &$($mutability)? $body.coroutine {
1058 if let Some(yield_ty) = $(& $mutability)? coroutine.yield_ty {
1059 $self.visit_ty(
1060 yield_ty,
1061 TyContext::YieldTy(SourceInfo::outermost(span))
1062 );
1063 }
1064 if let Some(resume_ty) = $(& $mutability)? coroutine.resume_ty {
1065 $self.visit_ty(
1066 resume_ty,
1067 TyContext::ResumeTy(SourceInfo::outermost(span))
1068 );
1069 }
1070 }
1071
1072 for var_debug_info in &$($mutability)? $body.var_debug_info {
1073 $self.visit_var_debug_info(var_debug_info);
1074 }
1075
1076 for (bb, data) in basic_blocks_iter!($body, $($mutability, $invalidate)?) {
1077 $self.visit_basic_block_data(bb, data);
1078 }
1079
1080 for scope in &$($mutability)? $body.source_scopes {
1081 $self.visit_source_scope_data(scope);
1082 }
1083
1084 $self.visit_ty(
1085 $(& $mutability)? $body.return_ty(),
1086 TyContext::ReturnTy(SourceInfo::outermost($body.span))
1087 );
1088
1089 for local in $body.local_decls.indices() {
1090 $self.visit_local_decl(local, & $($mutability)? $body.local_decls[local]);
1091 }
1092
1093 #[allow(unused_macro_rules)]
1094 macro_rules! type_annotations {
1095 (mut) => ($body.user_type_annotations.iter_enumerated_mut());
1096 () => ($body.user_type_annotations.iter_enumerated());
1097 }
1098
1099 for (index, annotation) in type_annotations!($($mutability)?) {
1100 $self.visit_user_type_annotation(
1101 index, annotation
1102 );
1103 }
1104
1105 $self.visit_span($(& $mutability)? $body.span);
1106
1107 if let Some(required_consts) = &$($mutability)? $body.required_consts {
1108 for const_ in required_consts {
1109 let location = Location::START;
1110 $self.visit_const_operand(const_, location);
1111 }
1112 }
1113 }
1114}
1115
1116macro_rules! visit_place_fns {
1117 (mut) => {
1118 fn tcx<'a>(&'a self) -> TyCtxt<'tcx>;
1119
1120 fn super_place(
1121 &mut self,
1122 place: &mut Place<'tcx>,
1123 context: PlaceContext,
1124 location: Location,
1125 ) {
1126 self.visit_local(&mut place.local, context, location);
1127
1128 if let Some(new_projection) = self.process_projection(&place.projection, location) {
1129 place.projection = self.tcx().mk_place_elems(&new_projection);
1130 }
1131 }
1132
1133 fn process_projection<'a>(
1134 &mut self,
1135 projection: &'a [PlaceElem<'tcx>],
1136 location: Location,
1137 ) -> Option<Vec<PlaceElem<'tcx>>> {
1138 let mut projection = Cow::Borrowed(projection);
1139
1140 for i in 0..projection.len() {
1141 if let Some(&elem) = projection.get(i) {
1142 if let Some(elem) = self.process_projection_elem(elem, location) {
1143 let vec = projection.to_mut();
1146 vec[i] = elem;
1147 }
1148 }
1149 }
1150
1151 match projection {
1152 Cow::Borrowed(_) => None,
1153 Cow::Owned(vec) => Some(vec),
1154 }
1155 }
1156
1157 fn process_projection_elem(
1158 &mut self,
1159 elem: PlaceElem<'tcx>,
1160 location: Location,
1161 ) -> Option<PlaceElem<'tcx>> {
1162 match elem {
1163 PlaceElem::Index(local) => {
1164 let mut new_local = local;
1165 self.visit_local(
1166 &mut new_local,
1167 PlaceContext::NonMutatingUse(NonMutatingUseContext::Copy),
1168 location,
1169 );
1170
1171 if new_local == local { None } else { Some(PlaceElem::Index(new_local)) }
1172 }
1173 PlaceElem::Field(field, ty) => {
1174 let mut new_ty = ty;
1175 self.visit_ty(&mut new_ty, TyContext::Location(location));
1176 if ty != new_ty { Some(PlaceElem::Field(field, new_ty)) } else { None }
1177 }
1178 PlaceElem::OpaqueCast(ty) => {
1179 let mut new_ty = ty;
1180 self.visit_ty(&mut new_ty, TyContext::Location(location));
1181 if ty != new_ty { Some(PlaceElem::OpaqueCast(new_ty)) } else { None }
1182 }
1183 PlaceElem::UnwrapUnsafeBinder(ty) => {
1184 let mut new_ty = ty;
1185 self.visit_ty(&mut new_ty, TyContext::Location(location));
1186 if ty != new_ty { Some(PlaceElem::UnwrapUnsafeBinder(new_ty)) } else { None }
1187 }
1188 PlaceElem::Deref
1189 | PlaceElem::ConstantIndex { .. }
1190 | PlaceElem::Subslice { .. }
1191 | PlaceElem::Downcast(..) => None,
1192 }
1193 }
1194 };
1195
1196 () => {
1197 fn visit_projection(
1198 &mut self,
1199 place_ref: PlaceRef<'tcx>,
1200 context: PlaceContext,
1201 location: Location,
1202 ) {
1203 self.super_projection(place_ref, context, location);
1204 }
1205
1206 fn visit_projection_elem(
1207 &mut self,
1208 place_ref: PlaceRef<'tcx>,
1209 elem: PlaceElem<'tcx>,
1210 context: PlaceContext,
1211 location: Location,
1212 ) {
1213 self.super_projection_elem(place_ref, elem, context, location);
1214 }
1215
1216 fn super_place(
1217 &mut self,
1218 place: &Place<'tcx>,
1219 mut context: PlaceContext,
1220 location: Location,
1221 ) {
1222 if !place.projection.is_empty() && context.is_use() {
1223 context = if context.is_mutating_use() {
1225 PlaceContext::MutatingUse(MutatingUseContext::Projection)
1226 } else {
1227 PlaceContext::NonMutatingUse(NonMutatingUseContext::Projection)
1228 };
1229 }
1230
1231 self.visit_local(place.local, context, location);
1232
1233 self.visit_projection(place.as_ref(), context, location);
1234 }
1235
1236 fn super_projection(
1237 &mut self,
1238 place_ref: PlaceRef<'tcx>,
1239 context: PlaceContext,
1240 location: Location,
1241 ) {
1242 for (base, elem) in place_ref.iter_projections().rev() {
1243 self.visit_projection_elem(base, elem, context, location);
1244 }
1245 }
1246
1247 fn super_projection_elem(
1248 &mut self,
1249 _place_ref: PlaceRef<'tcx>,
1250 elem: PlaceElem<'tcx>,
1251 context: PlaceContext,
1252 location: Location,
1253 ) {
1254 match elem {
1255 ProjectionElem::OpaqueCast(ty)
1256 | ProjectionElem::Field(_, ty)
1257 | ProjectionElem::UnwrapUnsafeBinder(ty) => {
1258 self.visit_ty(ty, TyContext::Location(location));
1259 }
1260 ProjectionElem::Index(local) => {
1261 self.visit_local(
1262 local,
1263 if context.is_use() {
1264 PlaceContext::NonMutatingUse(NonMutatingUseContext::Copy)
1266 } else {
1267 context
1268 },
1269 location,
1270 );
1271 }
1272 ProjectionElem::Deref
1273 | ProjectionElem::Subslice { from: _, to: _, from_end: _ }
1274 | ProjectionElem::ConstantIndex { offset: _, min_length: _, from_end: _ }
1275 | ProjectionElem::Downcast(_, _) => {}
1276 }
1277 }
1278 };
1279}
1280
1281pub trait Visitor<'tcx> {
fn visit_body(&mut self, body: &Body<'tcx>) { self.super_body(body); }
fn visit_basic_block_data(&mut self, block: BasicBlock,
data: &BasicBlockData<'tcx>) {
self.super_basic_block_data(block, data);
}
fn visit_source_scope_data(&mut self,
scope_data: &SourceScopeData<'tcx>) {
self.super_source_scope_data(scope_data);
}
fn visit_statement_debuginfo(&mut self,
stmt_debuginfo: &StmtDebugInfo<'tcx>, location: Location) {
self.super_statement_debuginfo(stmt_debuginfo, location);
}
fn visit_statement(&mut self, statement: &Statement<'tcx>,
location: Location) {
self.super_statement(statement, location);
}
fn visit_assign(&mut self, place: &Place<'tcx>, rvalue: &Rvalue<'tcx>,
location: Location) {
self.super_assign(place, rvalue, location);
}
fn visit_terminator(&mut self, terminator: &Terminator<'tcx>,
location: Location) {
self.super_terminator(terminator, location);
}
fn visit_assert_message(&mut self, msg: &AssertMessage<'tcx>,
location: Location) {
self.super_assert_message(msg, location);
}
fn visit_rvalue(&mut self, rvalue: &Rvalue<'tcx>, location: Location) {
self.super_rvalue(rvalue, location);
}
fn visit_operand(&mut self, operand: &Operand<'tcx>, location: Location) {
self.super_operand(operand, location);
}
fn visit_ascribe_user_ty(&mut self, place: &Place<'tcx>,
variance: ty::Variance, user_ty: &UserTypeProjection,
location: Location) {
self.super_ascribe_user_ty(place, variance, user_ty, location);
}
fn visit_coverage(&mut self, kind: &coverage::CoverageKind,
location: Location) {
self.super_coverage(kind, location);
}
fn visit_place(&mut self, place: &Place<'tcx>, context: PlaceContext,
location: Location) {
self.super_place(place, context, location);
}
fn visit_projection(&mut self, place_ref: PlaceRef<'tcx>,
context: PlaceContext, location: Location) {
self.super_projection(place_ref, context, location);
}
fn visit_projection_elem(&mut self, place_ref: PlaceRef<'tcx>,
elem: PlaceElem<'tcx>, context: PlaceContext, location: Location) {
self.super_projection_elem(place_ref, elem, context, location);
}
fn super_place(&mut self, place: &Place<'tcx>, mut context: PlaceContext,
location: Location) {
if !place.projection.is_empty() && context.is_use() {
context =
if context.is_mutating_use() {
PlaceContext::MutatingUse(MutatingUseContext::Projection)
} else {
PlaceContext::NonMutatingUse(NonMutatingUseContext::Projection)
};
}
self.visit_local(place.local, context, location);
self.visit_projection(place.as_ref(), context, location);
}
fn super_projection(&mut self, place_ref: PlaceRef<'tcx>,
context: PlaceContext, location: Location) {
for (base, elem) in place_ref.iter_projections().rev() {
self.visit_projection_elem(base, elem, context, location);
}
}
fn super_projection_elem(&mut self, _place_ref: PlaceRef<'tcx>,
elem: PlaceElem<'tcx>, context: PlaceContext, location: Location) {
match elem {
ProjectionElem::OpaqueCast(ty) | ProjectionElem::Field(_, ty) |
ProjectionElem::UnwrapUnsafeBinder(ty) => {
self.visit_ty(ty, TyContext::Location(location));
}
ProjectionElem::Index(local) => {
self.visit_local(local,
if context.is_use() {
PlaceContext::NonMutatingUse(NonMutatingUseContext::Copy)
} else { context }, location);
}
ProjectionElem::Deref | ProjectionElem::Subslice {
from: _, to: _, from_end: _ } |
ProjectionElem::ConstantIndex {
offset: _, min_length: _, from_end: _ } |
ProjectionElem::Downcast(_, _) => {}
}
}
fn visit_const_operand(&mut self, constant: &ConstOperand<'tcx>,
location: Location) {
self.super_const_operand(constant, location);
}
fn visit_ty_const(&mut self, ct: ty::Const<'tcx>, location: Location) {
self.super_ty_const(ct, location);
}
fn visit_span(&mut self, span: Span) { self.super_span(span); }
fn visit_source_info(&mut self, source_info: &SourceInfo) {
self.super_source_info(source_info);
}
fn visit_ty(&mut self, ty: Ty<'tcx>, _: TyContext) { self.super_ty(ty); }
fn visit_user_type_projection(&mut self, ty: &UserTypeProjection) {
self.super_user_type_projection(ty);
}
fn visit_user_type_annotation(&mut self, index: UserTypeAnnotationIndex,
ty: &CanonicalUserTypeAnnotation<'tcx>) {
self.super_user_type_annotation(index, ty);
}
fn visit_region(&mut self, region: ty::Region<'tcx>, _: Location) {
self.super_region(region);
}
fn visit_args(&mut self, args: &GenericArgsRef<'tcx>, _: Location) {
self.super_args(args);
}
fn visit_local_decl(&mut self, local: Local,
local_decl: &LocalDecl<'tcx>) {
self.super_local_decl(local, local_decl);
}
fn visit_var_debug_info(&mut self, var_debug_info: &VarDebugInfo<'tcx>) {
self.super_var_debug_info(var_debug_info);
}
fn visit_local(&mut self, local: Local, context: PlaceContext,
location: Location) {
self.super_local(local, context, location)
}
fn visit_source_scope(&mut self, scope: SourceScope) {
self.super_source_scope(scope);
}
fn super_body(&mut self, body: &Body<'tcx>) {
let span = body.span;
if let Some(coroutine) = &body.coroutine {
if let Some(yield_ty) = coroutine.yield_ty {
self.visit_ty(yield_ty,
TyContext::YieldTy(SourceInfo::outermost(span)));
}
if let Some(resume_ty) = coroutine.resume_ty {
self.visit_ty(resume_ty,
TyContext::ResumeTy(SourceInfo::outermost(span)));
}
}
for var_debug_info in &body.var_debug_info {
self.visit_var_debug_info(var_debug_info);
}
for (bb, data) in body.basic_blocks.iter_enumerated() {
self.visit_basic_block_data(bb, data);
}
for scope in &body.source_scopes {
self.visit_source_scope_data(scope);
}
self.visit_ty(body.return_ty(),
TyContext::ReturnTy(SourceInfo::outermost(body.span)));
for local in body.local_decls.indices() {
self.visit_local_decl(local, &body.local_decls[local]);
}
#[allow(unused_macro_rules)]
macro_rules! type_annotations {
(mut) => (body.user_type_annotations.iter_enumerated_mut()); () =>
(body.user_type_annotations.iter_enumerated());
}
for (index, annotation) in
body.user_type_annotations.iter_enumerated() {
self.visit_user_type_annotation(index, annotation);
}
self.visit_span(body.span);
if let Some(required_consts) = &body.required_consts {
for const_ in required_consts {
let location = Location::START;
self.visit_const_operand(const_, location);
}
};
}
fn super_basic_block_data(&mut self, block: BasicBlock,
data: &BasicBlockData<'tcx>) {
let BasicBlockData {
statements,
after_last_stmt_debuginfos,
terminator,
is_cleanup: _ } = data;
let mut index = 0;
for statement in statements {
let location = Location { block, statement_index: index };
self.visit_statement(statement, location);
index += 1;
}
let location = Location { block, statement_index: index };
for debuginfo in after_last_stmt_debuginfos as &[_] {
self.visit_statement_debuginfo(debuginfo, location);
}
if let Some(terminator) = terminator {
self.visit_terminator(terminator, location);
}
}
fn super_source_scope_data(&mut self,
scope_data: &SourceScopeData<'tcx>) {
let SourceScopeData {
span,
parent_scope,
inlined,
inlined_parent_scope,
local_data: _ } = scope_data;
self.visit_span(*span);
if let Some(parent_scope) = parent_scope {
self.visit_source_scope(*parent_scope);
}
if let Some((callee, callsite_span)) = inlined {
let location = Location::START;
self.visit_span(*callsite_span);
let ty::Instance { def: callee_def, args: callee_args } = callee;
match callee_def {
ty::InstanceKind::Item(_def_id) => {}
ty::InstanceKind::Intrinsic(_def_id) |
ty::InstanceKind::VTableShim(_def_id) |
ty::InstanceKind::ReifyShim(_def_id, _) |
ty::InstanceKind::Virtual(_def_id, _) |
ty::InstanceKind::ThreadLocalShim(_def_id) |
ty::InstanceKind::ClosureOnceShim {
call_once: _def_id, track_caller: _ } |
ty::InstanceKind::ConstructCoroutineInClosureShim {
coroutine_closure_def_id: _def_id, receiver_by_ref: _ } |
ty::InstanceKind::DropGlue(_def_id, None) => {}
ty::InstanceKind::FnPtrShim(_def_id, ty) |
ty::InstanceKind::DropGlue(_def_id, Some(ty)) |
ty::InstanceKind::CloneShim(_def_id, ty) |
ty::InstanceKind::FnPtrAddrShim(_def_id, ty) |
ty::InstanceKind::AsyncDropGlue(_def_id, ty) |
ty::InstanceKind::AsyncDropGlueCtorShim(_def_id, ty) => {
self.visit_ty(*ty, TyContext::Location(location));
}
ty::InstanceKind::FutureDropPollShim(_def_id, proxy_ty,
impl_ty) => {
self.visit_ty(*proxy_ty, TyContext::Location(location));
self.visit_ty(*impl_ty, TyContext::Location(location));
}
}
self.visit_args(callee_args, location);
}
if let Some(inlined_parent_scope) = inlined_parent_scope {
self.visit_source_scope(*inlined_parent_scope);
}
}
fn super_statement_debuginfo(&mut self,
stmt_debuginfo: &StmtDebugInfo<'tcx>, location: Location) {
match stmt_debuginfo {
StmtDebugInfo::AssignRef(local, place) => {
self.visit_local(*local,
PlaceContext::NonUse(NonUseContext::VarDebugInfo),
location);
self.visit_place(place,
PlaceContext::NonUse(NonUseContext::VarDebugInfo),
location);
}
StmtDebugInfo::InvalidAssign(local) => {
self.visit_local(*local,
PlaceContext::NonUse(NonUseContext::VarDebugInfo),
location);
}
}
}
fn super_statement(&mut self, statement: &Statement<'tcx>,
location: Location) {
let Statement { source_info, kind, debuginfos } = statement;
self.visit_source_info(source_info);
for debuginfo in debuginfos as &[_] {
self.visit_statement_debuginfo(debuginfo, location);
}
match kind {
StatementKind::Assign((place, rvalue)) => {
self.visit_assign(place, rvalue, location);
}
StatementKind::FakeRead((_, place)) => {
self.visit_place(place,
PlaceContext::NonMutatingUse(NonMutatingUseContext::Inspect),
location);
}
StatementKind::SetDiscriminant { place, .. } => {
self.visit_place(place,
PlaceContext::MutatingUse(MutatingUseContext::SetDiscriminant),
location);
}
StatementKind::StorageLive(local) => {
self.visit_local(*local,
PlaceContext::NonUse(NonUseContext::StorageLive), location);
}
StatementKind::StorageDead(local) => {
self.visit_local(*local,
PlaceContext::NonUse(NonUseContext::StorageDead), location);
}
StatementKind::PlaceMention(place) => {
self.visit_place(place,
PlaceContext::NonMutatingUse(NonMutatingUseContext::PlaceMention),
location);
}
StatementKind::AscribeUserType((place, user_ty), variance) => {
self.visit_ascribe_user_ty(place, *variance, user_ty,
location);
}
StatementKind::Coverage(coverage) => {
self.visit_coverage(coverage, location)
}
StatementKind::Intrinsic(intrinsic) => {
match intrinsic {
NonDivergingIntrinsic::Assume(op) =>
self.visit_operand(op, location),
NonDivergingIntrinsic::CopyNonOverlapping(CopyNonOverlapping {
src, dst, count }) => {
self.visit_operand(src, location);
self.visit_operand(dst, location);
self.visit_operand(count, location);
}
}
}
StatementKind::BackwardIncompatibleDropHint { place, .. } => {
self.visit_place(place,
PlaceContext::NonUse(NonUseContext::BackwardIncompatibleDropHint),
location);
}
StatementKind::ConstEvalCounter => {}
StatementKind::Nop => {}
}
}
fn super_assign(&mut self, place: &Place<'tcx>, rvalue: &Rvalue<'tcx>,
location: Location) {
self.visit_place(place,
PlaceContext::MutatingUse(MutatingUseContext::Store), location);
self.visit_rvalue(rvalue, location);
}
fn super_terminator(&mut self, terminator: &Terminator<'tcx>,
location: Location) {
let Terminator { source_info, kind } = terminator;
self.visit_source_info(source_info);
match kind {
TerminatorKind::Goto { .. } | TerminatorKind::UnwindResume |
TerminatorKind::UnwindTerminate(_) |
TerminatorKind::CoroutineDrop | TerminatorKind::Unreachable |
TerminatorKind::FalseEdge { .. } |
TerminatorKind::FalseUnwind { .. } => {}
TerminatorKind::Return => {
let local = RETURN_PLACE;
self.visit_local(local,
PlaceContext::NonMutatingUse(NonMutatingUseContext::Move),
location);
match (&local, &RETURN_PLACE) {
(left_val, right_val) => {
if !(*left_val == *right_val) {
let kind = ::core::panicking::AssertKind::Eq;
::core::panicking::assert_failed(kind, &*left_val,
&*right_val,
::core::option::Option::Some(format_args!("`MutVisitor` tried to mutate return place of `return` terminator")));
}
}
};
}
TerminatorKind::SwitchInt { discr, targets: _ } => {
self.visit_operand(discr, location);
}
TerminatorKind::Drop {
place, target: _, unwind: _, replace: _, drop: _, async_fut }
=> {
self.visit_place(place,
PlaceContext::MutatingUse(MutatingUseContext::Drop),
location);
if let Some(async_fut) = async_fut {
self.visit_local(*async_fut,
PlaceContext::MutatingUse(MutatingUseContext::Borrow),
location);
}
}
TerminatorKind::Call {
func,
args,
destination,
target: _,
unwind: _,
call_source: _,
fn_span } => {
self.visit_span(*fn_span);
self.visit_operand(func, location);
for arg in args { self.visit_operand(&arg.node, location); }
self.visit_place(destination,
PlaceContext::MutatingUse(MutatingUseContext::Call),
location);
}
TerminatorKind::TailCall { func, args, fn_span } => {
self.visit_span(*fn_span);
self.visit_operand(func, location);
for arg in args { self.visit_operand(&arg.node, location); }
}
TerminatorKind::Assert {
cond, expected: _, msg, target: _, unwind: _ } => {
self.visit_operand(cond, location);
self.visit_assert_message(msg, location);
}
TerminatorKind::Yield { value, resume: _, resume_arg, drop: _ } =>
{
self.visit_operand(value, location);
self.visit_place(resume_arg,
PlaceContext::MutatingUse(MutatingUseContext::Yield),
location);
}
TerminatorKind::InlineAsm {
asm_macro: _,
template: _,
operands,
options: _,
line_spans: _,
targets: _,
unwind: _ } => {
for op in operands {
match op {
InlineAsmOperand::In { value, .. } => {
self.visit_operand(value, location);
}
InlineAsmOperand::Out { place: Some(place), .. } => {
self.visit_place(place,
PlaceContext::MutatingUse(MutatingUseContext::AsmOutput),
location);
}
InlineAsmOperand::InOut { in_value, out_place, .. } => {
self.visit_operand(in_value, location);
if let Some(out_place) = out_place {
self.visit_place(out_place,
PlaceContext::MutatingUse(MutatingUseContext::AsmOutput),
location);
}
}
InlineAsmOperand::Const { value } |
InlineAsmOperand::SymFn { value } => {
self.visit_const_operand(value, location);
}
InlineAsmOperand::Out { place: None, .. } |
InlineAsmOperand::SymStatic { def_id: _ } |
InlineAsmOperand::Label { target_index: _ } => {}
}
}
}
}
}
fn super_assert_message(&mut self, msg: &AssertMessage<'tcx>,
location: Location) {
use crate::mir::AssertKind::*;
match msg {
BoundsCheck { len, index } => {
self.visit_operand(len, location);
self.visit_operand(index, location);
}
Overflow(_, l, r) => {
self.visit_operand(l, location);
self.visit_operand(r, location);
}
OverflowNeg(op) | DivisionByZero(op) | RemainderByZero(op) |
InvalidEnumConstruction(op) => {
self.visit_operand(op, location);
}
ResumedAfterReturn(_) | ResumedAfterPanic(_) |
NullPointerDereference | ResumedAfterDrop(_) => {}
MisalignedPointerDereference { required, found } => {
self.visit_operand(required, location);
self.visit_operand(found, location);
}
}
}
fn super_rvalue(&mut self, rvalue: &Rvalue<'tcx>, location: Location) {
match rvalue {
Rvalue::Use(operand, _with_retag) => {
self.visit_operand(operand, location);
}
Rvalue::Repeat(value, ct) => {
self.visit_operand(value, location);
self.visit_ty_const(*ct, location);
}
Rvalue::ThreadLocalRef(_) => {}
Rvalue::Ref(r, bk, path) => {
self.visit_region(*r, location);
let ctx =
match bk {
BorrowKind::Shared =>
PlaceContext::NonMutatingUse(NonMutatingUseContext::SharedBorrow),
BorrowKind::Fake(_) =>
PlaceContext::NonMutatingUse(NonMutatingUseContext::FakeBorrow),
BorrowKind::Mut { .. } =>
PlaceContext::MutatingUse(MutatingUseContext::Borrow),
};
self.visit_place(path, ctx, location);
}
Rvalue::Reborrow(target, mutability, place) => {
self.visit_ty(*target, TyContext::Location(location));
self.visit_place(place,
match mutability {
Mutability::Not =>
PlaceContext::NonMutatingUse(NonMutatingUseContext::SharedBorrow),
Mutability::Mut =>
PlaceContext::MutatingUse(MutatingUseContext::Borrow),
}, location);
}
Rvalue::CopyForDeref(place) => {
self.visit_place(place,
PlaceContext::NonMutatingUse(NonMutatingUseContext::Inspect),
location);
}
Rvalue::RawPtr(m, path) => {
let ctx =
match m {
RawPtrKind::Mut =>
PlaceContext::MutatingUse(MutatingUseContext::RawBorrow),
RawPtrKind::Const =>
PlaceContext::NonMutatingUse(NonMutatingUseContext::RawBorrow),
RawPtrKind::FakeForPtrMetadata =>
PlaceContext::NonMutatingUse(NonMutatingUseContext::Inspect),
};
self.visit_place(path, ctx, location);
}
Rvalue::Cast(_cast_kind, operand, ty) => {
self.visit_operand(operand, location);
self.visit_ty(*ty, TyContext::Location(location));
}
Rvalue::BinaryOp(_bin_op, (lhs, rhs)) => {
self.visit_operand(lhs, location);
self.visit_operand(rhs, location);
}
Rvalue::UnaryOp(_un_op, op) => {
self.visit_operand(op, location);
}
Rvalue::Discriminant(place) => {
self.visit_place(place,
PlaceContext::NonMutatingUse(NonMutatingUseContext::Inspect),
location);
}
Rvalue::Aggregate(kind, operands) => {
let kind = &**kind;
match kind {
AggregateKind::Array(ty) => {
self.visit_ty(*ty, TyContext::Location(location));
}
AggregateKind::Tuple => {}
AggregateKind::Adt(_adt_def, _variant_index, args,
_user_args, _active_field_index) => {
self.visit_args(args, location);
}
AggregateKind::Closure(_, closure_args) => {
self.visit_args(closure_args, location);
}
AggregateKind::Coroutine(_, coroutine_args) => {
self.visit_args(coroutine_args, location);
}
AggregateKind::CoroutineClosure(_, coroutine_closure_args)
=> {
self.visit_args(coroutine_closure_args, location);
}
AggregateKind::RawPtr(ty, _) => {
self.visit_ty(*ty, TyContext::Location(location));
}
}
for operand in operands {
self.visit_operand(operand, location);
}
}
Rvalue::WrapUnsafeBinder(op, ty) => {
self.visit_operand(op, location);
self.visit_ty(*ty, TyContext::Location(location));
}
}
}
fn super_operand(&mut self, operand: &Operand<'tcx>, location: Location) {
match operand {
Operand::Copy(place) => {
self.visit_place(place,
PlaceContext::NonMutatingUse(NonMutatingUseContext::Copy),
location);
}
Operand::Move(place) => {
self.visit_place(place,
PlaceContext::NonMutatingUse(NonMutatingUseContext::Move),
location);
}
Operand::Constant(constant) => {
self.visit_const_operand(constant, location);
}
Operand::RuntimeChecks(_) => {}
}
}
fn super_ascribe_user_ty(&mut self, place: &Place<'tcx>,
variance: ty::Variance, user_ty: &UserTypeProjection,
location: Location) {
self.visit_place(place,
PlaceContext::NonUse(NonUseContext::AscribeUserTy(variance)),
location);
self.visit_user_type_projection(user_ty);
}
fn super_coverage(&mut self, _kind: &coverage::CoverageKind,
_location: Location) {}
fn super_local_decl(&mut self, local: Local,
local_decl: &LocalDecl<'tcx>) {
let LocalDecl { mutability: _, ty, user_ty, source_info, local_info: _
} = local_decl;
self.visit_source_info(source_info);
self.visit_ty(*ty,
TyContext::LocalDecl { local, source_info: *source_info });
if let Some(user_ty) = user_ty {
for user_ty in &user_ty.contents {
self.visit_user_type_projection(user_ty);
}
}
}
fn super_local(&mut self, _local: Local, _context: PlaceContext,
_location: Location) {}
fn super_var_debug_info(&mut self, var_debug_info: &VarDebugInfo<'tcx>) {
let VarDebugInfo {
name: _, source_info, composite, value, argument_index: _ } =
var_debug_info;
self.visit_source_info(source_info);
let location = Location::START;
if let Some(VarDebugInfoFragment { ty, projection }) = composite {
self.visit_ty(*ty, TyContext::Location(location));
for elem in projection {
let ProjectionElem::Field(_, ty) =
elem else {
crate::util::bug::bug_fmt(format_args!("impossible case reached"))
};
self.visit_ty(*ty, TyContext::Location(location));
}
}
match value {
VarDebugInfoContents::Const(c) =>
self.visit_const_operand(c, location),
VarDebugInfoContents::Place(place) =>
self.visit_place(place,
PlaceContext::NonUse(NonUseContext::VarDebugInfo),
location),
}
}
fn super_source_scope(&mut self, _scope: SourceScope) {}
fn super_const_operand(&mut self, constant: &ConstOperand<'tcx>,
location: Location) {
let ConstOperand { span, user_ty: _, const_ } = constant;
self.visit_span(*span);
match const_ {
Const::Ty(_, ct) => self.visit_ty_const(*ct, location),
Const::Val(_, ty) | Const::Unevaluated(_, ty) => {
self.visit_ty(*ty, TyContext::Location(location));
}
}
}
fn super_ty_const(&mut self, _ct: ty::Const<'tcx>, _location: Location) {}
fn super_span(&mut self, _span: Span) {}
fn super_source_info(&mut self, source_info: &SourceInfo) {
let SourceInfo { span, scope } = source_info;
self.visit_span(*span);
self.visit_source_scope(*scope);
}
fn super_user_type_projection(&mut self, _ty: &UserTypeProjection) {}
fn super_user_type_annotation(&mut self, _index: UserTypeAnnotationIndex,
ty: &CanonicalUserTypeAnnotation<'tcx>) {
self.visit_span(ty.span);
self.visit_ty(ty.inferred_ty, TyContext::UserTy(ty.span));
}
fn super_ty(&mut self, _ty: Ty<'tcx>) {}
fn super_region(&mut self, _region: ty::Region<'tcx>) {}
fn super_args(&mut self, _args: &GenericArgsRef<'tcx>) {}
fn visit_location(&mut self, body: &Body<'tcx>, location: Location) {
let basic_block = &body.basic_blocks[location.block];
if basic_block.statements.len() == location.statement_index {
if let Some(ref terminator) = basic_block.terminator {
self.visit_terminator(terminator, location)
}
} else {
let statement = &basic_block.statements[location.statement_index];
self.visit_statement(statement, location)
}
}
}make_mir_visitor!(Visitor,);
1282pub trait MutVisitor<'tcx> {
fn visit_body(&mut self, body: &mut Body<'tcx>) { self.super_body(body); }
fn visit_body_preserves_cfg(&mut self, body: &mut Body<'tcx>) {
self.super_body_preserves_cfg(body);
}
fn super_body_preserves_cfg(&mut self, body: &mut Body<'tcx>) {
let span = body.span;
if let Some(coroutine) = &mut body.coroutine {
if let Some(yield_ty) = &mut coroutine.yield_ty {
self.visit_ty(yield_ty,
TyContext::YieldTy(SourceInfo::outermost(span)));
}
if let Some(resume_ty) = &mut coroutine.resume_ty {
self.visit_ty(resume_ty,
TyContext::ResumeTy(SourceInfo::outermost(span)));
}
}
for var_debug_info in &mut body.var_debug_info {
self.visit_var_debug_info(var_debug_info);
}
for (bb, data) in
body.basic_blocks.as_mut_preserves_cfg().iter_enumerated_mut() {
self.visit_basic_block_data(bb, data);
}
for scope in &mut body.source_scopes {
self.visit_source_scope_data(scope);
}
self.visit_ty(&mut body.return_ty(),
TyContext::ReturnTy(SourceInfo::outermost(body.span)));
for local in body.local_decls.indices() {
self.visit_local_decl(local, &mut body.local_decls[local]);
}
#[allow(unused_macro_rules)]
macro_rules! type_annotations {
(mut) => (body.user_type_annotations.iter_enumerated_mut()); () =>
(body.user_type_annotations.iter_enumerated());
}
for (index, annotation) in
body.user_type_annotations.iter_enumerated_mut() {
self.visit_user_type_annotation(index, annotation);
}
self.visit_span(&mut body.span);
if let Some(required_consts) = &mut body.required_consts {
for const_ in required_consts {
let location = Location::START;
self.visit_const_operand(const_, location);
}
};
}
fn visit_basic_block_data(&mut self, block: BasicBlock,
data: &mut BasicBlockData<'tcx>) {
self.super_basic_block_data(block, data);
}
fn visit_source_scope_data(&mut self,
scope_data: &mut SourceScopeData<'tcx>) {
self.super_source_scope_data(scope_data);
}
fn visit_statement_debuginfo(&mut self,
stmt_debuginfo: &mut StmtDebugInfo<'tcx>, location: Location) {
self.super_statement_debuginfo(stmt_debuginfo, location);
}
fn visit_statement(&mut self, statement: &mut Statement<'tcx>,
location: Location) {
self.super_statement(statement, location);
}
fn visit_assign(&mut self, place: &mut Place<'tcx>,
rvalue: &mut Rvalue<'tcx>, location: Location) {
self.super_assign(place, rvalue, location);
}
fn visit_terminator(&mut self, terminator: &mut Terminator<'tcx>,
location: Location) {
self.super_terminator(terminator, location);
}
fn visit_assert_message(&mut self, msg: &mut AssertMessage<'tcx>,
location: Location) {
self.super_assert_message(msg, location);
}
fn visit_rvalue(&mut self, rvalue: &mut Rvalue<'tcx>,
location: Location) {
self.super_rvalue(rvalue, location);
}
fn visit_operand(&mut self, operand: &mut Operand<'tcx>,
location: Location) {
self.super_operand(operand, location);
}
fn visit_ascribe_user_ty(&mut self, place: &mut Place<'tcx>,
variance: &mut ty::Variance, user_ty: &mut UserTypeProjection,
location: Location) {
self.super_ascribe_user_ty(place, variance, user_ty, location);
}
fn visit_coverage(&mut self, kind: &mut coverage::CoverageKind,
location: Location) {
self.super_coverage(kind, location);
}
fn visit_place(&mut self, place: &mut Place<'tcx>, context: PlaceContext,
location: Location) {
self.super_place(place, context, location);
}
fn tcx<'a>(&'a self)
-> TyCtxt<'tcx>;
fn super_place(&mut self, place: &mut Place<'tcx>, context: PlaceContext,
location: Location) {
self.visit_local(&mut place.local, context, location);
if let Some(new_projection) =
self.process_projection(&place.projection, location) {
place.projection = self.tcx().mk_place_elems(&new_projection);
}
}
fn process_projection<'a>(&mut self, projection: &'a [PlaceElem<'tcx>],
location: Location) -> Option<Vec<PlaceElem<'tcx>>> {
let mut projection = Cow::Borrowed(projection);
for i in 0..projection.len() {
if let Some(&elem) = projection.get(i) {
if let Some(elem) =
self.process_projection_elem(elem, location) {
let vec = projection.to_mut();
vec[i] = elem;
}
}
}
match projection {
Cow::Borrowed(_) => None,
Cow::Owned(vec) => Some(vec),
}
}
fn process_projection_elem(&mut self, elem: PlaceElem<'tcx>,
location: Location) -> Option<PlaceElem<'tcx>> {
match elem {
PlaceElem::Index(local) => {
let mut new_local = local;
self.visit_local(&mut new_local,
PlaceContext::NonMutatingUse(NonMutatingUseContext::Copy),
location);
if new_local == local {
None
} else { Some(PlaceElem::Index(new_local)) }
}
PlaceElem::Field(field, ty) => {
let mut new_ty = ty;
self.visit_ty(&mut new_ty, TyContext::Location(location));
if ty != new_ty {
Some(PlaceElem::Field(field, new_ty))
} else { None }
}
PlaceElem::OpaqueCast(ty) => {
let mut new_ty = ty;
self.visit_ty(&mut new_ty, TyContext::Location(location));
if ty != new_ty {
Some(PlaceElem::OpaqueCast(new_ty))
} else { None }
}
PlaceElem::UnwrapUnsafeBinder(ty) => {
let mut new_ty = ty;
self.visit_ty(&mut new_ty, TyContext::Location(location));
if ty != new_ty {
Some(PlaceElem::UnwrapUnsafeBinder(new_ty))
} else { None }
}
PlaceElem::Deref | PlaceElem::ConstantIndex { .. } |
PlaceElem::Subslice { .. } | PlaceElem::Downcast(..) => None,
}
}
fn visit_const_operand(&mut self, constant: &mut ConstOperand<'tcx>,
location: Location) {
self.super_const_operand(constant, location);
}
fn visit_ty_const(&mut self, ct: &mut ty::Const<'tcx>,
location: Location) {
self.super_ty_const(ct, location);
}
fn visit_span(&mut self, span: &mut Span) { self.super_span(span); }
fn visit_source_info(&mut self, source_info: &mut SourceInfo) {
self.super_source_info(source_info);
}
fn visit_ty(&mut self, ty: &mut Ty<'tcx>, _: TyContext) {
self.super_ty(ty);
}
fn visit_user_type_projection(&mut self, ty: &mut UserTypeProjection) {
self.super_user_type_projection(ty);
}
fn visit_user_type_annotation(&mut self, index: UserTypeAnnotationIndex,
ty: &mut CanonicalUserTypeAnnotation<'tcx>) {
self.super_user_type_annotation(index, ty);
}
fn visit_region(&mut self, region: &mut ty::Region<'tcx>, _: Location) {
self.super_region(region);
}
fn visit_args(&mut self, args: &mut GenericArgsRef<'tcx>, _: Location) {
self.super_args(args);
}
fn visit_local_decl(&mut self, local: Local,
local_decl: &mut LocalDecl<'tcx>) {
self.super_local_decl(local, local_decl);
}
fn visit_var_debug_info(&mut self,
var_debug_info: &mut VarDebugInfo<'tcx>) {
self.super_var_debug_info(var_debug_info);
}
fn visit_local(&mut self, local: &mut Local, context: PlaceContext,
location: Location) {
self.super_local(local, context, location)
}
fn visit_source_scope(&mut self, scope: &mut SourceScope) {
self.super_source_scope(scope);
}
fn super_body(&mut self, body: &mut Body<'tcx>) {
let span = body.span;
if let Some(coroutine) = &mut body.coroutine {
if let Some(yield_ty) = &mut coroutine.yield_ty {
self.visit_ty(yield_ty,
TyContext::YieldTy(SourceInfo::outermost(span)));
}
if let Some(resume_ty) = &mut coroutine.resume_ty {
self.visit_ty(resume_ty,
TyContext::ResumeTy(SourceInfo::outermost(span)));
}
}
for var_debug_info in &mut body.var_debug_info {
self.visit_var_debug_info(var_debug_info);
}
for (bb, data) in body.basic_blocks.as_mut().iter_enumerated_mut() {
self.visit_basic_block_data(bb, data);
}
for scope in &mut body.source_scopes {
self.visit_source_scope_data(scope);
}
self.visit_ty(&mut body.return_ty(),
TyContext::ReturnTy(SourceInfo::outermost(body.span)));
for local in body.local_decls.indices() {
self.visit_local_decl(local, &mut body.local_decls[local]);
}
#[allow(unused_macro_rules)]
macro_rules! type_annotations {
(mut) => (body.user_type_annotations.iter_enumerated_mut()); () =>
(body.user_type_annotations.iter_enumerated());
}
for (index, annotation) in
body.user_type_annotations.iter_enumerated_mut() {
self.visit_user_type_annotation(index, annotation);
}
self.visit_span(&mut body.span);
if let Some(required_consts) = &mut body.required_consts {
for const_ in required_consts {
let location = Location::START;
self.visit_const_operand(const_, location);
}
};
}
fn super_basic_block_data(&mut self, block: BasicBlock,
data: &mut BasicBlockData<'tcx>) {
let BasicBlockData {
statements,
after_last_stmt_debuginfos,
terminator,
is_cleanup: _ } = data;
let mut index = 0;
for statement in statements {
let location = Location { block, statement_index: index };
self.visit_statement(statement, location);
index += 1;
}
let location = Location { block, statement_index: index };
for debuginfo in after_last_stmt_debuginfos as &mut [_] {
self.visit_statement_debuginfo(debuginfo, location);
}
if let Some(terminator) = terminator {
self.visit_terminator(terminator, location);
}
}
fn super_source_scope_data(&mut self,
scope_data: &mut SourceScopeData<'tcx>) {
let SourceScopeData {
span,
parent_scope,
inlined,
inlined_parent_scope,
local_data: _ } = scope_data;
self.visit_span(&mut *span);
if let Some(parent_scope) = parent_scope {
self.visit_source_scope(&mut *parent_scope);
}
if let Some((callee, callsite_span)) = inlined {
let location = Location::START;
self.visit_span(&mut *callsite_span);
let ty::Instance { def: callee_def, args: callee_args } = callee;
match callee_def {
ty::InstanceKind::Item(_def_id) => {}
ty::InstanceKind::Intrinsic(_def_id) |
ty::InstanceKind::VTableShim(_def_id) |
ty::InstanceKind::ReifyShim(_def_id, _) |
ty::InstanceKind::Virtual(_def_id, _) |
ty::InstanceKind::ThreadLocalShim(_def_id) |
ty::InstanceKind::ClosureOnceShim {
call_once: _def_id, track_caller: _ } |
ty::InstanceKind::ConstructCoroutineInClosureShim {
coroutine_closure_def_id: _def_id, receiver_by_ref: _ } |
ty::InstanceKind::DropGlue(_def_id, None) => {}
ty::InstanceKind::FnPtrShim(_def_id, ty) |
ty::InstanceKind::DropGlue(_def_id, Some(ty)) |
ty::InstanceKind::CloneShim(_def_id, ty) |
ty::InstanceKind::FnPtrAddrShim(_def_id, ty) |
ty::InstanceKind::AsyncDropGlue(_def_id, ty) |
ty::InstanceKind::AsyncDropGlueCtorShim(_def_id, ty) => {
self.visit_ty(&mut *ty, TyContext::Location(location));
}
ty::InstanceKind::FutureDropPollShim(_def_id, proxy_ty,
impl_ty) => {
self.visit_ty(&mut *proxy_ty,
TyContext::Location(location));
self.visit_ty(&mut *impl_ty, TyContext::Location(location));
}
}
self.visit_args(callee_args, location);
}
if let Some(inlined_parent_scope) = inlined_parent_scope {
self.visit_source_scope(&mut *inlined_parent_scope);
}
}
fn super_statement_debuginfo(&mut self,
stmt_debuginfo: &mut StmtDebugInfo<'tcx>, location: Location) {
match stmt_debuginfo {
StmtDebugInfo::AssignRef(local, place) => {
self.visit_local(&mut *local,
PlaceContext::NonUse(NonUseContext::VarDebugInfo),
location);
self.visit_place(place,
PlaceContext::NonUse(NonUseContext::VarDebugInfo),
location);
}
StmtDebugInfo::InvalidAssign(local) => {
self.visit_local(&mut *local,
PlaceContext::NonUse(NonUseContext::VarDebugInfo),
location);
}
}
}
fn super_statement(&mut self, statement: &mut Statement<'tcx>,
location: Location) {
let Statement { source_info, kind, debuginfos } = statement;
self.visit_source_info(source_info);
for debuginfo in debuginfos as &mut [_] {
self.visit_statement_debuginfo(debuginfo, location);
}
match kind {
StatementKind::Assign((place, rvalue)) => {
self.visit_assign(place, rvalue, location);
}
StatementKind::FakeRead((_, place)) => {
self.visit_place(place,
PlaceContext::NonMutatingUse(NonMutatingUseContext::Inspect),
location);
}
StatementKind::SetDiscriminant { place, .. } => {
self.visit_place(place,
PlaceContext::MutatingUse(MutatingUseContext::SetDiscriminant),
location);
}
StatementKind::StorageLive(local) => {
self.visit_local(&mut *local,
PlaceContext::NonUse(NonUseContext::StorageLive), location);
}
StatementKind::StorageDead(local) => {
self.visit_local(&mut *local,
PlaceContext::NonUse(NonUseContext::StorageDead), location);
}
StatementKind::PlaceMention(place) => {
self.visit_place(place,
PlaceContext::NonMutatingUse(NonMutatingUseContext::PlaceMention),
location);
}
StatementKind::AscribeUserType((place, user_ty), variance) => {
self.visit_ascribe_user_ty(place, &mut *variance, user_ty,
location);
}
StatementKind::Coverage(coverage) => {
self.visit_coverage(coverage, location)
}
StatementKind::Intrinsic(intrinsic) => {
match intrinsic {
NonDivergingIntrinsic::Assume(op) =>
self.visit_operand(op, location),
NonDivergingIntrinsic::CopyNonOverlapping(CopyNonOverlapping {
src, dst, count }) => {
self.visit_operand(src, location);
self.visit_operand(dst, location);
self.visit_operand(count, location);
}
}
}
StatementKind::BackwardIncompatibleDropHint { place, .. } => {
self.visit_place(place,
PlaceContext::NonUse(NonUseContext::BackwardIncompatibleDropHint),
location);
}
StatementKind::ConstEvalCounter => {}
StatementKind::Nop => {}
}
}
fn super_assign(&mut self, place: &mut Place<'tcx>,
rvalue: &mut Rvalue<'tcx>, location: Location) {
self.visit_place(place,
PlaceContext::MutatingUse(MutatingUseContext::Store), location);
self.visit_rvalue(rvalue, location);
}
fn super_terminator(&mut self, terminator: &mut Terminator<'tcx>,
location: Location) {
let Terminator { source_info, kind } = terminator;
self.visit_source_info(source_info);
match kind {
TerminatorKind::Goto { .. } | TerminatorKind::UnwindResume |
TerminatorKind::UnwindTerminate(_) |
TerminatorKind::CoroutineDrop | TerminatorKind::Unreachable |
TerminatorKind::FalseEdge { .. } |
TerminatorKind::FalseUnwind { .. } => {}
TerminatorKind::Return => {
let mut local = RETURN_PLACE;
self.visit_local(&mut local,
PlaceContext::NonMutatingUse(NonMutatingUseContext::Move),
location);
match (&local, &RETURN_PLACE) {
(left_val, right_val) => {
if !(*left_val == *right_val) {
let kind = ::core::panicking::AssertKind::Eq;
::core::panicking::assert_failed(kind, &*left_val,
&*right_val,
::core::option::Option::Some(format_args!("`MutVisitor` tried to mutate return place of `return` terminator")));
}
}
};
}
TerminatorKind::SwitchInt { discr, targets: _ } => {
self.visit_operand(discr, location);
}
TerminatorKind::Drop {
place, target: _, unwind: _, replace: _, drop: _, async_fut }
=> {
self.visit_place(place,
PlaceContext::MutatingUse(MutatingUseContext::Drop),
location);
if let Some(async_fut) = async_fut {
self.visit_local(&mut *async_fut,
PlaceContext::MutatingUse(MutatingUseContext::Borrow),
location);
}
}
TerminatorKind::Call {
func,
args,
destination,
target: _,
unwind: _,
call_source: _,
fn_span } => {
self.visit_span(&mut *fn_span);
self.visit_operand(func, location);
for arg in args {
self.visit_operand(&mut arg.node, location);
}
self.visit_place(destination,
PlaceContext::MutatingUse(MutatingUseContext::Call),
location);
}
TerminatorKind::TailCall { func, args, fn_span } => {
self.visit_span(&mut *fn_span);
self.visit_operand(func, location);
for arg in args {
self.visit_operand(&mut arg.node, location);
}
}
TerminatorKind::Assert {
cond, expected: _, msg, target: _, unwind: _ } => {
self.visit_operand(cond, location);
self.visit_assert_message(msg, location);
}
TerminatorKind::Yield { value, resume: _, resume_arg, drop: _ } =>
{
self.visit_operand(value, location);
self.visit_place(resume_arg,
PlaceContext::MutatingUse(MutatingUseContext::Yield),
location);
}
TerminatorKind::InlineAsm {
asm_macro: _,
template: _,
operands,
options: _,
line_spans: _,
targets: _,
unwind: _ } => {
for op in operands {
match op {
InlineAsmOperand::In { value, .. } => {
self.visit_operand(value, location);
}
InlineAsmOperand::Out { place: Some(place), .. } => {
self.visit_place(place,
PlaceContext::MutatingUse(MutatingUseContext::AsmOutput),
location);
}
InlineAsmOperand::InOut { in_value, out_place, .. } => {
self.visit_operand(in_value, location);
if let Some(out_place) = out_place {
self.visit_place(out_place,
PlaceContext::MutatingUse(MutatingUseContext::AsmOutput),
location);
}
}
InlineAsmOperand::Const { value } |
InlineAsmOperand::SymFn { value } => {
self.visit_const_operand(value, location);
}
InlineAsmOperand::Out { place: None, .. } |
InlineAsmOperand::SymStatic { def_id: _ } |
InlineAsmOperand::Label { target_index: _ } => {}
}
}
}
}
}
fn super_assert_message(&mut self, msg: &mut AssertMessage<'tcx>,
location: Location) {
use crate::mir::AssertKind::*;
match msg {
BoundsCheck { len, index } => {
self.visit_operand(len, location);
self.visit_operand(index, location);
}
Overflow(_, l, r) => {
self.visit_operand(l, location);
self.visit_operand(r, location);
}
OverflowNeg(op) | DivisionByZero(op) | RemainderByZero(op) |
InvalidEnumConstruction(op) => {
self.visit_operand(op, location);
}
ResumedAfterReturn(_) | ResumedAfterPanic(_) |
NullPointerDereference | ResumedAfterDrop(_) => {}
MisalignedPointerDereference { required, found } => {
self.visit_operand(required, location);
self.visit_operand(found, location);
}
}
}
fn super_rvalue(&mut self, rvalue: &mut Rvalue<'tcx>,
location: Location) {
match rvalue {
Rvalue::Use(operand, _with_retag) => {
self.visit_operand(operand, location);
}
Rvalue::Repeat(value, ct) => {
self.visit_operand(value, location);
self.visit_ty_const(&mut *ct, location);
}
Rvalue::ThreadLocalRef(_) => {}
Rvalue::Ref(r, bk, path) => {
self.visit_region(&mut *r, location);
let ctx =
match bk {
BorrowKind::Shared =>
PlaceContext::NonMutatingUse(NonMutatingUseContext::SharedBorrow),
BorrowKind::Fake(_) =>
PlaceContext::NonMutatingUse(NonMutatingUseContext::FakeBorrow),
BorrowKind::Mut { .. } =>
PlaceContext::MutatingUse(MutatingUseContext::Borrow),
};
self.visit_place(path, ctx, location);
}
Rvalue::Reborrow(target, mutability, place) => {
self.visit_ty(&mut *target, TyContext::Location(location));
self.visit_place(place,
match mutability {
Mutability::Not =>
PlaceContext::NonMutatingUse(NonMutatingUseContext::SharedBorrow),
Mutability::Mut =>
PlaceContext::MutatingUse(MutatingUseContext::Borrow),
}, location);
}
Rvalue::CopyForDeref(place) => {
self.visit_place(place,
PlaceContext::NonMutatingUse(NonMutatingUseContext::Inspect),
location);
}
Rvalue::RawPtr(m, path) => {
let ctx =
match m {
RawPtrKind::Mut =>
PlaceContext::MutatingUse(MutatingUseContext::RawBorrow),
RawPtrKind::Const =>
PlaceContext::NonMutatingUse(NonMutatingUseContext::RawBorrow),
RawPtrKind::FakeForPtrMetadata =>
PlaceContext::NonMutatingUse(NonMutatingUseContext::Inspect),
};
self.visit_place(path, ctx, location);
}
Rvalue::Cast(_cast_kind, operand, ty) => {
self.visit_operand(operand, location);
self.visit_ty(&mut *ty, TyContext::Location(location));
}
Rvalue::BinaryOp(_bin_op, (lhs, rhs)) => {
self.visit_operand(lhs, location);
self.visit_operand(rhs, location);
}
Rvalue::UnaryOp(_un_op, op) => {
self.visit_operand(op, location);
}
Rvalue::Discriminant(place) => {
self.visit_place(place,
PlaceContext::NonMutatingUse(NonMutatingUseContext::Inspect),
location);
}
Rvalue::Aggregate(kind, operands) => {
let kind = &mut **kind;
match kind {
AggregateKind::Array(ty) => {
self.visit_ty(&mut *ty, TyContext::Location(location));
}
AggregateKind::Tuple => {}
AggregateKind::Adt(_adt_def, _variant_index, args,
_user_args, _active_field_index) => {
self.visit_args(args, location);
}
AggregateKind::Closure(_, closure_args) => {
self.visit_args(closure_args, location);
}
AggregateKind::Coroutine(_, coroutine_args) => {
self.visit_args(coroutine_args, location);
}
AggregateKind::CoroutineClosure(_, coroutine_closure_args)
=> {
self.visit_args(coroutine_closure_args, location);
}
AggregateKind::RawPtr(ty, _) => {
self.visit_ty(&mut *ty, TyContext::Location(location));
}
}
for operand in operands {
self.visit_operand(operand, location);
}
}
Rvalue::WrapUnsafeBinder(op, ty) => {
self.visit_operand(op, location);
self.visit_ty(&mut *ty, TyContext::Location(location));
}
}
}
fn super_operand(&mut self, operand: &mut Operand<'tcx>,
location: Location) {
match operand {
Operand::Copy(place) => {
self.visit_place(place,
PlaceContext::NonMutatingUse(NonMutatingUseContext::Copy),
location);
}
Operand::Move(place) => {
self.visit_place(place,
PlaceContext::NonMutatingUse(NonMutatingUseContext::Move),
location);
}
Operand::Constant(constant) => {
self.visit_const_operand(constant, location);
}
Operand::RuntimeChecks(_) => {}
}
}
fn super_ascribe_user_ty(&mut self, place: &mut Place<'tcx>,
variance: &mut ty::Variance, user_ty: &mut UserTypeProjection,
location: Location) {
self.visit_place(place,
PlaceContext::NonUse(NonUseContext::AscribeUserTy(*&mut *variance)),
location);
self.visit_user_type_projection(user_ty);
}
fn super_coverage(&mut self, _kind: &mut coverage::CoverageKind,
_location: Location) {}
fn super_local_decl(&mut self, local: Local,
local_decl: &mut LocalDecl<'tcx>) {
let LocalDecl { mutability: _, ty, user_ty, source_info, local_info: _
} = local_decl;
self.visit_source_info(source_info);
self.visit_ty(&mut *ty,
TyContext::LocalDecl { local, source_info: *source_info });
if let Some(user_ty) = user_ty {
for user_ty in &mut user_ty.contents {
self.visit_user_type_projection(user_ty);
}
}
}
fn super_local(&mut self, _local: &mut Local, _context: PlaceContext,
_location: Location) {}
fn super_var_debug_info(&mut self,
var_debug_info: &mut VarDebugInfo<'tcx>) {
let VarDebugInfo {
name: _, source_info, composite, value, argument_index: _ } =
var_debug_info;
self.visit_source_info(source_info);
let location = Location::START;
if let Some(VarDebugInfoFragment { ty, projection }) = composite {
self.visit_ty(&mut *ty, TyContext::Location(location));
for elem in projection {
let ProjectionElem::Field(_, ty) =
elem else {
crate::util::bug::bug_fmt(format_args!("impossible case reached"))
};
self.visit_ty(&mut *ty, TyContext::Location(location));
}
}
match value {
VarDebugInfoContents::Const(c) =>
self.visit_const_operand(c, location),
VarDebugInfoContents::Place(place) =>
self.visit_place(place,
PlaceContext::NonUse(NonUseContext::VarDebugInfo),
location),
}
}
fn super_source_scope(&mut self, _scope: &mut SourceScope) {}
fn super_const_operand(&mut self, constant: &mut ConstOperand<'tcx>,
location: Location) {
let ConstOperand { span, user_ty: _, const_ } = constant;
self.visit_span(&mut *span);
match const_ {
Const::Ty(_, ct) => self.visit_ty_const(&mut *ct, location),
Const::Val(_, ty) | Const::Unevaluated(_, ty) => {
self.visit_ty(&mut *ty, TyContext::Location(location));
}
}
}
fn super_ty_const(&mut self, _ct: &mut ty::Const<'tcx>,
_location: Location) {}
fn super_span(&mut self, _span: &mut Span) {}
fn super_source_info(&mut self, source_info: &mut SourceInfo) {
let SourceInfo { span, scope } = source_info;
self.visit_span(&mut *span);
self.visit_source_scope(&mut *scope);
}
fn super_user_type_projection(&mut self, _ty: &mut UserTypeProjection) {}
fn super_user_type_annotation(&mut self, _index: UserTypeAnnotationIndex,
ty: &mut CanonicalUserTypeAnnotation<'tcx>) {
self.visit_span(&mut ty.span);
self.visit_ty(&mut ty.inferred_ty, TyContext::UserTy(ty.span));
}
fn super_ty(&mut self, _ty: &mut Ty<'tcx>) {}
fn super_region(&mut self, _region: &mut ty::Region<'tcx>) {}
fn super_args(&mut self, _args: &mut GenericArgsRef<'tcx>) {}
fn visit_location(&mut self, body: &mut Body<'tcx>, location: Location) {
let basic_block = &mut body.basic_blocks.as_mut()[location.block];
if basic_block.statements.len() == location.statement_index {
if let Some(ref mut terminator) = basic_block.terminator {
self.visit_terminator(terminator, location)
}
} else {
let statement =
&mut basic_block.statements[location.statement_index];
self.visit_statement(statement, location)
}
}
}make_mir_visitor!(MutVisitor, mut);
1283
1284#[derive(#[automatically_derived]
impl ::core::marker::Copy for TyContext { }Copy, #[automatically_derived]
impl ::core::clone::Clone for TyContext {
#[inline]
fn clone(&self) -> TyContext {
let _: ::core::clone::AssertParamIsClone<Local>;
let _: ::core::clone::AssertParamIsClone<SourceInfo>;
let _: ::core::clone::AssertParamIsClone<Span>;
let _: ::core::clone::AssertParamIsClone<Location>;
*self
}
}Clone, #[automatically_derived]
impl ::core::fmt::Debug for TyContext {
#[inline]
fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
match self {
TyContext::LocalDecl { local: __self_0, source_info: __self_1 } =>
::core::fmt::Formatter::debug_struct_field2_finish(f,
"LocalDecl", "local", __self_0, "source_info", &__self_1),
TyContext::UserTy(__self_0) =>
::core::fmt::Formatter::debug_tuple_field1_finish(f, "UserTy",
&__self_0),
TyContext::ReturnTy(__self_0) =>
::core::fmt::Formatter::debug_tuple_field1_finish(f,
"ReturnTy", &__self_0),
TyContext::YieldTy(__self_0) =>
::core::fmt::Formatter::debug_tuple_field1_finish(f,
"YieldTy", &__self_0),
TyContext::ResumeTy(__self_0) =>
::core::fmt::Formatter::debug_tuple_field1_finish(f,
"ResumeTy", &__self_0),
TyContext::Location(__self_0) =>
::core::fmt::Formatter::debug_tuple_field1_finish(f,
"Location", &__self_0),
}
}
}Debug, #[automatically_derived]
impl ::core::hash::Hash for TyContext {
#[inline]
fn hash<__H: ::core::hash::Hasher>(&self, state: &mut __H) {
let __self_discr = ::core::intrinsics::discriminant_value(self);
::core::hash::Hash::hash(&__self_discr, state);
match self {
TyContext::LocalDecl { local: __self_0, source_info: __self_1 } =>
{
::core::hash::Hash::hash(__self_0, state);
::core::hash::Hash::hash(__self_1, state)
}
TyContext::UserTy(__self_0) =>
::core::hash::Hash::hash(__self_0, state),
TyContext::ReturnTy(__self_0) =>
::core::hash::Hash::hash(__self_0, state),
TyContext::YieldTy(__self_0) =>
::core::hash::Hash::hash(__self_0, state),
TyContext::ResumeTy(__self_0) =>
::core::hash::Hash::hash(__self_0, state),
TyContext::Location(__self_0) =>
::core::hash::Hash::hash(__self_0, state),
}
}
}Hash, #[automatically_derived]
impl ::core::cmp::Eq for TyContext {
#[inline]
#[doc(hidden)]
#[coverage(off)]
fn assert_fields_are_eq(&self) {
let _: ::core::cmp::AssertParamIsEq<Local>;
let _: ::core::cmp::AssertParamIsEq<SourceInfo>;
let _: ::core::cmp::AssertParamIsEq<Span>;
let _: ::core::cmp::AssertParamIsEq<Location>;
}
}Eq, #[automatically_derived]
impl ::core::cmp::PartialEq for TyContext {
#[inline]
fn eq(&self, other: &TyContext) -> bool {
let __self_discr = ::core::intrinsics::discriminant_value(self);
let __arg1_discr = ::core::intrinsics::discriminant_value(other);
__self_discr == __arg1_discr &&
match (self, other) {
(TyContext::LocalDecl { local: __self_0, source_info: __self_1
}, TyContext::LocalDecl {
local: __arg1_0, source_info: __arg1_1 }) =>
__self_0 == __arg1_0 && __self_1 == __arg1_1,
(TyContext::UserTy(__self_0), TyContext::UserTy(__arg1_0)) =>
__self_0 == __arg1_0,
(TyContext::ReturnTy(__self_0), TyContext::ReturnTy(__arg1_0))
=> __self_0 == __arg1_0,
(TyContext::YieldTy(__self_0), TyContext::YieldTy(__arg1_0))
=> __self_0 == __arg1_0,
(TyContext::ResumeTy(__self_0), TyContext::ResumeTy(__arg1_0))
=> __self_0 == __arg1_0,
(TyContext::Location(__self_0), TyContext::Location(__arg1_0))
=> __self_0 == __arg1_0,
_ => unsafe { ::core::intrinsics::unreachable() }
}
}
}PartialEq)]
1287pub enum TyContext {
1288 LocalDecl {
1289 local: Local,
1291
1292 source_info: SourceInfo,
1294 },
1295
1296 UserTy(Span),
1298
1299 ReturnTy(SourceInfo),
1301
1302 YieldTy(SourceInfo),
1303
1304 ResumeTy(SourceInfo),
1305
1306 Location(Location),
1308}
1309
1310#[derive(#[automatically_derived]
impl ::core::marker::Copy for NonMutatingUseContext { }Copy, #[automatically_derived]
impl ::core::clone::Clone for NonMutatingUseContext {
#[inline]
fn clone(&self) -> NonMutatingUseContext { *self }
}Clone, #[automatically_derived]
impl ::core::fmt::Debug for NonMutatingUseContext {
#[inline]
fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
::core::fmt::Formatter::write_str(f,
match self {
NonMutatingUseContext::Inspect => "Inspect",
NonMutatingUseContext::Copy => "Copy",
NonMutatingUseContext::Move => "Move",
NonMutatingUseContext::SharedBorrow => "SharedBorrow",
NonMutatingUseContext::FakeBorrow => "FakeBorrow",
NonMutatingUseContext::RawBorrow => "RawBorrow",
NonMutatingUseContext::PlaceMention => "PlaceMention",
NonMutatingUseContext::Projection => "Projection",
})
}
}Debug, #[automatically_derived]
impl ::core::cmp::PartialEq for NonMutatingUseContext {
#[inline]
fn eq(&self, other: &NonMutatingUseContext) -> bool {
let __self_discr = ::core::intrinsics::discriminant_value(self);
let __arg1_discr = ::core::intrinsics::discriminant_value(other);
__self_discr == __arg1_discr
}
}PartialEq, #[automatically_derived]
impl ::core::cmp::Eq for NonMutatingUseContext {
#[inline]
#[doc(hidden)]
#[coverage(off)]
fn assert_fields_are_eq(&self) {}
}Eq)]
1311pub enum NonMutatingUseContext {
1312 Inspect,
1314 Copy,
1316 Move,
1318 SharedBorrow,
1320 FakeBorrow,
1324 RawBorrow,
1326 PlaceMention,
1331 Projection,
1338}
1339
1340#[derive(#[automatically_derived]
impl ::core::marker::Copy for MutatingUseContext { }Copy, #[automatically_derived]
impl ::core::clone::Clone for MutatingUseContext {
#[inline]
fn clone(&self) -> MutatingUseContext { *self }
}Clone, #[automatically_derived]
impl ::core::fmt::Debug for MutatingUseContext {
#[inline]
fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
::core::fmt::Formatter::write_str(f,
match self {
MutatingUseContext::Store => "Store",
MutatingUseContext::SetDiscriminant => "SetDiscriminant",
MutatingUseContext::AsmOutput => "AsmOutput",
MutatingUseContext::Call => "Call",
MutatingUseContext::Yield => "Yield",
MutatingUseContext::Drop => "Drop",
MutatingUseContext::Borrow => "Borrow",
MutatingUseContext::RawBorrow => "RawBorrow",
MutatingUseContext::Projection => "Projection",
MutatingUseContext::Retag => "Retag",
})
}
}Debug, #[automatically_derived]
impl ::core::cmp::PartialEq for MutatingUseContext {
#[inline]
fn eq(&self, other: &MutatingUseContext) -> bool {
let __self_discr = ::core::intrinsics::discriminant_value(self);
let __arg1_discr = ::core::intrinsics::discriminant_value(other);
__self_discr == __arg1_discr
}
}PartialEq, #[automatically_derived]
impl ::core::cmp::Eq for MutatingUseContext {
#[inline]
#[doc(hidden)]
#[coverage(off)]
fn assert_fields_are_eq(&self) {}
}Eq)]
1341pub enum MutatingUseContext {
1342 Store,
1344 SetDiscriminant,
1346 AsmOutput,
1348 Call,
1350 Yield,
1352 Drop,
1354 Borrow,
1356 RawBorrow,
1358 Projection,
1365 Retag,
1367}
1368
1369#[derive(#[automatically_derived]
impl ::core::marker::Copy for NonUseContext { }Copy, #[automatically_derived]
impl ::core::clone::Clone for NonUseContext {
#[inline]
fn clone(&self) -> NonUseContext {
let _: ::core::clone::AssertParamIsClone<ty::Variance>;
*self
}
}Clone, #[automatically_derived]
impl ::core::fmt::Debug for NonUseContext {
#[inline]
fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
match self {
NonUseContext::StorageLive =>
::core::fmt::Formatter::write_str(f, "StorageLive"),
NonUseContext::StorageDead =>
::core::fmt::Formatter::write_str(f, "StorageDead"),
NonUseContext::AscribeUserTy(__self_0) =>
::core::fmt::Formatter::debug_tuple_field1_finish(f,
"AscribeUserTy", &__self_0),
NonUseContext::VarDebugInfo =>
::core::fmt::Formatter::write_str(f, "VarDebugInfo"),
NonUseContext::BackwardIncompatibleDropHint =>
::core::fmt::Formatter::write_str(f,
"BackwardIncompatibleDropHint"),
}
}
}Debug, #[automatically_derived]
impl ::core::cmp::PartialEq for NonUseContext {
#[inline]
fn eq(&self, other: &NonUseContext) -> bool {
let __self_discr = ::core::intrinsics::discriminant_value(self);
let __arg1_discr = ::core::intrinsics::discriminant_value(other);
__self_discr == __arg1_discr &&
match (self, other) {
(NonUseContext::AscribeUserTy(__self_0),
NonUseContext::AscribeUserTy(__arg1_0)) =>
__self_0 == __arg1_0,
_ => true,
}
}
}PartialEq, #[automatically_derived]
impl ::core::cmp::Eq for NonUseContext {
#[inline]
#[doc(hidden)]
#[coverage(off)]
fn assert_fields_are_eq(&self) {
let _: ::core::cmp::AssertParamIsEq<ty::Variance>;
}
}Eq)]
1370pub enum NonUseContext {
1371 StorageLive,
1373 StorageDead,
1375 AscribeUserTy(ty::Variance),
1377 VarDebugInfo,
1379 BackwardIncompatibleDropHint,
1381}
1382
1383#[derive(#[automatically_derived]
impl ::core::marker::Copy for PlaceContext { }Copy, #[automatically_derived]
impl ::core::clone::Clone for PlaceContext {
#[inline]
fn clone(&self) -> PlaceContext {
let _: ::core::clone::AssertParamIsClone<NonMutatingUseContext>;
let _: ::core::clone::AssertParamIsClone<MutatingUseContext>;
let _: ::core::clone::AssertParamIsClone<NonUseContext>;
*self
}
}Clone, #[automatically_derived]
impl ::core::fmt::Debug for PlaceContext {
#[inline]
fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
match self {
PlaceContext::NonMutatingUse(__self_0) =>
::core::fmt::Formatter::debug_tuple_field1_finish(f,
"NonMutatingUse", &__self_0),
PlaceContext::MutatingUse(__self_0) =>
::core::fmt::Formatter::debug_tuple_field1_finish(f,
"MutatingUse", &__self_0),
PlaceContext::NonUse(__self_0) =>
::core::fmt::Formatter::debug_tuple_field1_finish(f, "NonUse",
&__self_0),
}
}
}Debug, #[automatically_derived]
impl ::core::cmp::PartialEq for PlaceContext {
#[inline]
fn eq(&self, other: &PlaceContext) -> bool {
let __self_discr = ::core::intrinsics::discriminant_value(self);
let __arg1_discr = ::core::intrinsics::discriminant_value(other);
__self_discr == __arg1_discr &&
match (self, other) {
(PlaceContext::NonMutatingUse(__self_0),
PlaceContext::NonMutatingUse(__arg1_0)) =>
__self_0 == __arg1_0,
(PlaceContext::MutatingUse(__self_0),
PlaceContext::MutatingUse(__arg1_0)) =>
__self_0 == __arg1_0,
(PlaceContext::NonUse(__self_0),
PlaceContext::NonUse(__arg1_0)) => __self_0 == __arg1_0,
_ => unsafe { ::core::intrinsics::unreachable() }
}
}
}PartialEq, #[automatically_derived]
impl ::core::cmp::Eq for PlaceContext {
#[inline]
#[doc(hidden)]
#[coverage(off)]
fn assert_fields_are_eq(&self) {
let _: ::core::cmp::AssertParamIsEq<NonMutatingUseContext>;
let _: ::core::cmp::AssertParamIsEq<MutatingUseContext>;
let _: ::core::cmp::AssertParamIsEq<NonUseContext>;
}
}Eq)]
1384pub enum PlaceContext {
1385 NonMutatingUse(NonMutatingUseContext),
1386 MutatingUse(MutatingUseContext),
1387 NonUse(NonUseContext),
1388}
1389
1390impl PlaceContext {
1391 #[inline]
1393 pub fn is_drop(self) -> bool {
1394 #[allow(non_exhaustive_omitted_patterns)] match self {
PlaceContext::MutatingUse(MutatingUseContext::Drop) => true,
_ => false,
}matches!(self, PlaceContext::MutatingUse(MutatingUseContext::Drop))
1395 }
1396
1397 pub fn is_borrow(self) -> bool {
1400 #[allow(non_exhaustive_omitted_patterns)] match self {
PlaceContext::NonMutatingUse(NonMutatingUseContext::SharedBorrow) |
PlaceContext::MutatingUse(MutatingUseContext::Borrow) => true,
_ => false,
}matches!(
1401 self,
1402 PlaceContext::NonMutatingUse(NonMutatingUseContext::SharedBorrow)
1403 | PlaceContext::MutatingUse(MutatingUseContext::Borrow)
1404 )
1405 }
1406
1407 pub fn is_address_of(self) -> bool {
1409 #[allow(non_exhaustive_omitted_patterns)] match self {
PlaceContext::NonMutatingUse(NonMutatingUseContext::RawBorrow) |
PlaceContext::MutatingUse(MutatingUseContext::RawBorrow) => true,
_ => false,
}matches!(
1410 self,
1411 PlaceContext::NonMutatingUse(NonMutatingUseContext::RawBorrow)
1412 | PlaceContext::MutatingUse(MutatingUseContext::RawBorrow)
1413 )
1414 }
1415
1416 #[inline]
1418 pub fn may_observe_address(self) -> bool {
1419 #[allow(non_exhaustive_omitted_patterns)] match self {
PlaceContext::NonMutatingUse(NonMutatingUseContext::SharedBorrow |
NonMutatingUseContext::RawBorrow | NonMutatingUseContext::FakeBorrow)
|
PlaceContext::MutatingUse(MutatingUseContext::Drop |
MutatingUseContext::Borrow | MutatingUseContext::RawBorrow |
MutatingUseContext::AsmOutput) => true,
_ => false,
}matches!(
1420 self,
1421 PlaceContext::NonMutatingUse(
1422 NonMutatingUseContext::SharedBorrow
1423 | NonMutatingUseContext::RawBorrow
1424 | NonMutatingUseContext::FakeBorrow
1425 ) | PlaceContext::MutatingUse(
1426 MutatingUseContext::Drop
1427 | MutatingUseContext::Borrow
1428 | MutatingUseContext::RawBorrow
1429 | MutatingUseContext::AsmOutput
1430 )
1431 )
1432 }
1433
1434 #[inline]
1436 pub fn is_storage_marker(self) -> bool {
1437 #[allow(non_exhaustive_omitted_patterns)] match self {
PlaceContext::NonUse(NonUseContext::StorageLive |
NonUseContext::StorageDead) => true,
_ => false,
}matches!(
1438 self,
1439 PlaceContext::NonUse(NonUseContext::StorageLive | NonUseContext::StorageDead)
1440 )
1441 }
1442
1443 #[inline]
1445 pub fn is_mutating_use(self) -> bool {
1446 #[allow(non_exhaustive_omitted_patterns)] match self {
PlaceContext::MutatingUse(..) => true,
_ => false,
}matches!(self, PlaceContext::MutatingUse(..))
1447 }
1448
1449 #[inline]
1451 pub fn is_use(self) -> bool {
1452 !#[allow(non_exhaustive_omitted_patterns)] match self {
PlaceContext::NonUse(..) => true,
_ => false,
}matches!(self, PlaceContext::NonUse(..))
1453 }
1454
1455 pub fn is_place_assignment(self) -> bool {
1457 #[allow(non_exhaustive_omitted_patterns)] match self {
PlaceContext::MutatingUse(MutatingUseContext::Store |
MutatingUseContext::Call | MutatingUseContext::AsmOutput) => true,
_ => false,
}matches!(
1458 self,
1459 PlaceContext::MutatingUse(
1460 MutatingUseContext::Store
1461 | MutatingUseContext::Call
1462 | MutatingUseContext::AsmOutput,
1463 )
1464 )
1465 }
1466
1467 pub fn ambient_variance(self) -> ty::Variance {
1469 use NonMutatingUseContext::*;
1470 use NonUseContext::*;
1471 match self {
1472 PlaceContext::MutatingUse(_) => ty::Invariant,
1473 PlaceContext::NonUse(
1474 StorageDead | StorageLive | VarDebugInfo | BackwardIncompatibleDropHint,
1475 ) => ty::Invariant,
1476 PlaceContext::NonMutatingUse(
1477 Inspect | Copy | Move | PlaceMention | SharedBorrow | FakeBorrow | RawBorrow
1478 | Projection,
1479 ) => ty::Covariant,
1480 PlaceContext::NonUse(AscribeUserTy(variance)) => variance,
1481 }
1482 }
1483}
1484
1485pub struct VisitPlacesWith<F>(pub F);
1487
1488impl<'tcx, F> Visitor<'tcx> for VisitPlacesWith<F>
1489where
1490 F: FnMut(Place<'tcx>, PlaceContext),
1491{
1492 fn visit_local(&mut self, local: Local, ctxt: PlaceContext, _: Location) {
1493 (self.0)(local.into(), ctxt);
1494 }
1495
1496 fn visit_place(&mut self, place: &Place<'tcx>, ctxt: PlaceContext, location: Location) {
1497 (self.0)(*place, ctxt);
1498 self.visit_projection(place.as_ref(), ctxt, location);
1499 }
1500}