Skip to main content

rustc_middle/mir/
visit.rs

1//! # The MIR Visitor
2//!
3//! ## Overview
4//!
5//! There are two visitors, one for immutable and one for mutable references,
6//! but both are generated by the `make_mir_visitor` macro.
7//! The code is written according to the following conventions:
8//!
9//! - introduce a `visit_foo` and a `super_foo` method for every MIR type
10//! - `visit_foo`, by default, calls `super_foo`
11//! - `super_foo`, by default, destructures the `foo` and calls `visit_foo`
12//!
13//! This allows you to override `visit_foo` for types you are
14//! interested in, and invoke (within that method call)
15//! `self.super_foo` to get the default behavior. Just as in an OO
16//! language, you should never call `super` methods ordinarily except
17//! in that circumstance.
18//!
19//! For the most part, we do not destructure things external to the
20//! MIR, e.g., types, spans, etc, but simply visit them and stop. This
21//! avoids duplication with other visitors like `TypeFoldable`.
22//!
23//! ## Updating
24//!
25//! The code is written in a very deliberate style intended to minimize
26//! the chance of things being overlooked. You'll notice that we always
27//! use pattern matching to reference fields and we ensure that all
28//! matches are exhaustive.
29//!
30//! For example, the `super_basic_block_data` method begins like this:
31//!
32//! ```ignore (pseudo-rust)
33//! fn super_basic_block_data(
34//!     &mut self,
35//!     block: BasicBlock,
36//!     data: & $($mutability)? BasicBlockData<'tcx>
37//! ) {
38//!     let BasicBlockData {
39//!         statements,
40//!         terminator,
41//!         is_cleanup: _
42//!     } = *data;
43//!
44//!     for statement in statements {
45//!         self.visit_statement(block, statement);
46//!     }
47//!
48//!     ...
49//! }
50//! ```
51//!
52//! Here we used `let BasicBlockData { <fields> } = *data` deliberately,
53//! rather than writing `data.statements` in the body. This is because if one
54//! adds a new field to `BasicBlockData`, one will be forced to revise this code,
55//! and hence one will (hopefully) invoke the correct visit methods (if any).
56//!
57//! For this to work, ALL MATCHES MUST BE EXHAUSTIVE IN FIELDS AND VARIANTS.
58//! That means you never write `..` to skip over fields, nor do you write `_`
59//! to skip over variants in a `match`.
60//!
61//! The only place that `_` is acceptable is to match a field (or
62//! variant argument) that does not require visiting, as in
63//! `is_cleanup` above.
64
65use 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            // Override these, and call `self.super_xxx` to revert back to the
72            // default behavior.
73
74            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            /// This is called for every constant in the MIR body and every `required_consts`
185            /// (i.e., including consts that have been dead-code-eliminated).
186            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            // The `super_xxx` methods comprise the default behavior and are
287            // not meant to be overridden.
288
289            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                            // FIXME(eddyb) use a better `TyContext` here.
368                            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                        // `return` logically moves from the return place `_0`. Note that the place
533                        // cannot be changed by any visitor, though.
534                        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                        // Nothing to visit
681                    }
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: _, // no visit method for this
952                    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            // Convenience methods
998
999            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                        // This converts the borrowed projection into `Cow::Owned(_)` and returns a
1144                        // clone of the projection so we can mutate and reintern later.
1145                        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                // ^ Only change the context if it is a real use, not a "use" in debuginfo.
1224                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                            // ^ Only change the context if it is a real use, not a "use" in debuginfo.
1265                            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(_, _) => {}
        }
    }
    /// This is called for every constant in the MIR body and every `required_consts`
    /// (i.e., including consts that have been dead-code-eliminated).
    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,
        }
    }
    /// This is called for every constant in the MIR body and every `required_consts`
    /// (i.e., including consts that have been dead-code-eliminated).
    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/// Extra information passed to `visit_ty` and friends to give context
1285/// about where the type etc appears.
1286#[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        /// The index of the local variable we are visiting.
1290        local: Local,
1291
1292        /// The source location where this local variable was declared.
1293        source_info: SourceInfo,
1294    },
1295
1296    /// The inferred type of a user type annotation.
1297    UserTy(Span),
1298
1299    /// The return type of the function.
1300    ReturnTy(SourceInfo),
1301
1302    YieldTy(SourceInfo),
1303
1304    ResumeTy(SourceInfo),
1305
1306    /// A type found at some location.
1307    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    /// Being inspected in some way, like loading a len.
1313    Inspect,
1314    /// Consumed as part of an operand.
1315    Copy,
1316    /// Consumed as part of an operand.
1317    Move,
1318    /// Shared borrow.
1319    SharedBorrow,
1320    /// A fake borrow.
1321    /// FIXME: do we need to distinguish shallow and deep fake borrows? In fact, do we need to
1322    /// distinguish fake and normal deep borrows?
1323    FakeBorrow,
1324    /// `&raw const`.
1325    RawBorrow,
1326    /// PlaceMention statement.
1327    ///
1328    /// This statement is executed as a check that the `Place` is live without reading from it,
1329    /// so it must be considered as a non-mutating use.
1330    PlaceMention,
1331    /// Used as base for another place, e.g., `x` in `x.y`. Will not mutate the place.
1332    /// For example, the projection `x.y` is not marked as a mutation in these cases:
1333    /// ```ignore (illustrative)
1334    /// z = x.y;
1335    /// f(&x.y);
1336    /// ```
1337    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    /// Appears as LHS of an assignment.
1343    Store,
1344    /// Appears on `SetDiscriminant`
1345    SetDiscriminant,
1346    /// Output operand of an inline assembly block.
1347    AsmOutput,
1348    /// Destination of a call.
1349    Call,
1350    /// Destination of a yield.
1351    Yield,
1352    /// Being dropped.
1353    Drop,
1354    /// Mutable borrow.
1355    Borrow,
1356    /// `&raw mut`.
1357    RawBorrow,
1358    /// Used as base for another place, e.g., `x` in `x.y`. Could potentially mutate the place.
1359    /// For example, the projection `x.y` is marked as a mutation in these cases:
1360    /// ```ignore (illustrative)
1361    /// x.y = ...;
1362    /// f(&mut x.y);
1363    /// ```
1364    Projection,
1365    /// Retagging, a "Stacked Borrows" shadow state operation
1366    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    /// Starting a storage live range.
1372    StorageLive,
1373    /// Ending a storage live range.
1374    StorageDead,
1375    /// User type annotation assertions for NLL.
1376    AscribeUserTy(ty::Variance),
1377    /// The data of a user variable, for debug info.
1378    VarDebugInfo,
1379    /// A `BackwardIncompatibleDropHint` statement, meant for edition 2024 lints.
1380    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    /// Returns `true` if this place context represents a drop.
1392    #[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    /// Returns `true` if this place context represents a borrow, excluding fake borrows
1398    /// (which are an artifact of borrowck and not actually borrows in runtime MIR).
1399    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    /// Returns `true` if this place context represents an address-of.
1408    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    /// Returns `true` if this place context may be used to know the address of the given place.
1417    #[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    /// Returns `true` if this place context represents a storage live or storage dead marker.
1435    #[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    /// Returns `true` if this place context represents a use that potentially changes the value.
1444    #[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    /// Returns `true` if this place context represents a use.
1450    #[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    /// Returns `true` if this place context represents an assignment statement.
1456    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    /// The variance of a place in the given context.
1468    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
1485/// Small utility to visit places and locals without manually implementing a full visitor.
1486pub 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}