Skip to main content

rustc_middle/mir/
statement.rs

1//! Functionality for statements, operands, places, and things that appear in them.
2
3use std::ops;
4
5use tracing::{debug, instrument};
6
7use super::interpret::GlobalAlloc;
8use super::*;
9use crate::ty::{CoroutineArgsExt, Unnormalized};
10
11///////////////////////////////////////////////////////////////////////////
12// Statements
13
14/// A statement in a basic block, including information about its source code.
15#[derive(#[automatically_derived]
impl<'tcx> ::core::clone::Clone for Statement<'tcx> {
    #[inline]
    fn clone(&self) -> Statement<'tcx> {
        Statement {
            source_info: ::core::clone::Clone::clone(&self.source_info),
            kind: ::core::clone::Clone::clone(&self.kind),
            debuginfos: ::core::clone::Clone::clone(&self.debuginfos),
        }
    }
}Clone, const _: () =
    {
        impl<'tcx, __E: ::rustc_middle::ty::codec::TyEncoder<'tcx>>
            ::rustc_serialize::Encodable<__E> for Statement<'tcx> {
            fn encode(&self, __encoder: &mut __E) {
                match *self {
                    Statement {
                        source_info: ref __binding_0,
                        kind: ref __binding_1,
                        debuginfos: ref __binding_2 } => {
                        ::rustc_serialize::Encodable::<__E>::encode(__binding_0,
                            __encoder);
                        ::rustc_serialize::Encodable::<__E>::encode(__binding_1,
                            __encoder);
                        ::rustc_serialize::Encodable::<__E>::encode(__binding_2,
                            __encoder);
                    }
                }
            }
        }
    };TyEncodable, const _: () =
    {
        impl<'tcx, __D: ::rustc_middle::ty::codec::TyDecoder<'tcx>>
            ::rustc_serialize::Decodable<__D> for Statement<'tcx> {
            fn decode(__decoder: &mut __D) -> Self {
                Statement {
                    source_info: ::rustc_serialize::Decodable::decode(__decoder),
                    kind: ::rustc_serialize::Decodable::decode(__decoder),
                    debuginfos: ::rustc_serialize::Decodable::decode(__decoder),
                }
            }
        }
    };TyDecodable, const _: () =
    {
        impl<'tcx> ::rustc_data_structures::stable_hash::StableHash for
            Statement<'tcx> {
            #[inline]
            fn stable_hash<__Hcx: ::rustc_data_structures::stable_hash::StableHashCtxt>(&self,
                __hcx: &mut __Hcx,
                __hasher:
                    &mut ::rustc_data_structures::stable_hash::StableHasher) {
                match *self {
                    Statement {
                        source_info: ref __binding_0,
                        kind: ref __binding_1,
                        debuginfos: ref __binding_2 } => {
                        { __binding_0.stable_hash(__hcx, __hasher); }
                        { __binding_1.stable_hash(__hcx, __hasher); }
                        { __binding_2.stable_hash(__hcx, __hasher); }
                    }
                }
            }
        }
    };StableHash, const _: () =
    {
        impl<'tcx>
            ::rustc_middle::ty::TypeFoldable<::rustc_middle::ty::TyCtxt<'tcx>>
            for Statement<'tcx> {
            fn try_fold_with<__F: ::rustc_middle::ty::FallibleTypeFolder<::rustc_middle::ty::TyCtxt<'tcx>>>(self,
                __folder: &mut __F) -> Result<Self, __F::Error> {
                Ok(match self {
                        Statement {
                            source_info: __binding_0,
                            kind: __binding_1,
                            debuginfos: __binding_2 } => {
                            Statement {
                                source_info: ::rustc_middle::ty::TypeFoldable::try_fold_with(__binding_0,
                                        __folder)?,
                                kind: ::rustc_middle::ty::TypeFoldable::try_fold_with(__binding_1,
                                        __folder)?,
                                debuginfos: ::rustc_middle::ty::TypeFoldable::try_fold_with(__binding_2,
                                        __folder)?,
                            }
                        }
                    })
            }
            fn fold_with<__F: ::rustc_middle::ty::TypeFolder<::rustc_middle::ty::TyCtxt<'tcx>>>(self,
                __folder: &mut __F) -> Self {
                match self {
                    Statement {
                        source_info: __binding_0,
                        kind: __binding_1,
                        debuginfos: __binding_2 } => {
                        Statement {
                            source_info: ::rustc_middle::ty::TypeFoldable::fold_with(__binding_0,
                                __folder),
                            kind: ::rustc_middle::ty::TypeFoldable::fold_with(__binding_1,
                                __folder),
                            debuginfos: ::rustc_middle::ty::TypeFoldable::fold_with(__binding_2,
                                __folder),
                        }
                    }
                }
            }
        }
    };TypeFoldable, const _: () =
    {
        impl<'tcx>
            ::rustc_middle::ty::TypeVisitable<::rustc_middle::ty::TyCtxt<'tcx>>
            for Statement<'tcx> {
            fn visit_with<__V: ::rustc_middle::ty::TypeVisitor<::rustc_middle::ty::TyCtxt<'tcx>>>(&self,
                __visitor: &mut __V) -> __V::Result {
                match *self {
                    Statement {
                        source_info: ref __binding_0,
                        kind: ref __binding_1,
                        debuginfos: ref __binding_2 } => {
                        {
                            match ::rustc_middle::ty::VisitorResult::branch(::rustc_middle::ty::TypeVisitable::visit_with(__binding_0,
                                        __visitor)) {
                                ::core::ops::ControlFlow::Continue(()) => {}
                                ::core::ops::ControlFlow::Break(r) => {
                                    return ::rustc_middle::ty::VisitorResult::from_residual(r);
                                }
                            }
                        }
                        {
                            match ::rustc_middle::ty::VisitorResult::branch(::rustc_middle::ty::TypeVisitable::visit_with(__binding_1,
                                        __visitor)) {
                                ::core::ops::ControlFlow::Continue(()) => {}
                                ::core::ops::ControlFlow::Break(r) => {
                                    return ::rustc_middle::ty::VisitorResult::from_residual(r);
                                }
                            }
                        }
                        {
                            match ::rustc_middle::ty::VisitorResult::branch(::rustc_middle::ty::TypeVisitable::visit_with(__binding_2,
                                        __visitor)) {
                                ::core::ops::ControlFlow::Continue(()) => {}
                                ::core::ops::ControlFlow::Break(r) => {
                                    return ::rustc_middle::ty::VisitorResult::from_residual(r);
                                }
                            }
                        }
                    }
                }
                <__V::Result as ::rustc_middle::ty::VisitorResult>::output()
            }
        }
    };TypeVisitable)]
16#[non_exhaustive]
17pub struct Statement<'tcx> {
18    pub source_info: SourceInfo,
19    pub kind: StatementKind<'tcx>,
20    /// Some debuginfos appearing before the primary statement.
21    pub debuginfos: StmtDebugInfos<'tcx>,
22}
23
24impl<'tcx> Statement<'tcx> {
25    /// Changes a statement to a nop. This is both faster than deleting instructions and avoids
26    /// invalidating statement indices in `Location`s.
27    pub fn make_nop(&mut self, drop_debuginfo: bool) {
28        if self.kind == StatementKind::Nop {
29            return;
30        }
31        let replaced_stmt = std::mem::replace(&mut self.kind, StatementKind::Nop);
32        if !drop_debuginfo {
33            let Some(debuginfo) = replaced_stmt.as_debuginfo() else {
34                crate::util::bug::bug_fmt(format_args!("debuginfo is not yet supported."))bug!("debuginfo is not yet supported.")
35            };
36            self.debuginfos.push(debuginfo);
37        }
38    }
39
40    pub fn new(source_info: SourceInfo, kind: StatementKind<'tcx>) -> Self {
41        Statement { source_info, kind, debuginfos: StmtDebugInfos::default() }
42    }
43}
44
45impl<'tcx> StatementKind<'tcx> {
46    /// Returns a simple string representation of a `StatementKind` variant, independent of any
47    /// values it might hold (e.g. `StatementKind::Assign` always returns `"Assign"`).
48    pub const fn name(&self) -> &'static str {
49        match self {
50            StatementKind::Assign(..) => "Assign",
51            StatementKind::FakeRead(..) => "FakeRead",
52            StatementKind::SetDiscriminant { .. } => "SetDiscriminant",
53            StatementKind::StorageLive(..) => "StorageLive",
54            StatementKind::StorageDead(..) => "StorageDead",
55            StatementKind::PlaceMention(..) => "PlaceMention",
56            StatementKind::AscribeUserType(..) => "AscribeUserType",
57            StatementKind::Coverage(..) => "Coverage",
58            StatementKind::Intrinsic(..) => "Intrinsic",
59            StatementKind::ConstEvalCounter => "ConstEvalCounter",
60            StatementKind::Nop => "Nop",
61            StatementKind::BackwardIncompatibleDropHint { .. } => "BackwardIncompatibleDropHint",
62        }
63    }
64    pub fn as_assign_mut(&mut self) -> Option<&mut (Place<'tcx>, Rvalue<'tcx>)> {
65        match self {
66            StatementKind::Assign(x) => Some(x),
67            _ => None,
68        }
69    }
70
71    pub fn as_assign(&self) -> Option<&(Place<'tcx>, Rvalue<'tcx>)> {
72        match self {
73            StatementKind::Assign(x) => Some(x),
74            _ => None,
75        }
76    }
77
78    pub fn as_debuginfo(&self) -> Option<StmtDebugInfo<'tcx>> {
79        match self {
80            StatementKind::Assign((place, Rvalue::Ref(_, _, ref_place)))
81                if let Some(local) = place.as_local() =>
82            {
83                Some(StmtDebugInfo::AssignRef(local, *ref_place))
84            }
85            _ => None,
86        }
87    }
88}
89
90///////////////////////////////////////////////////////////////////////////
91// Places
92
93#[derive(#[automatically_derived]
impl<'tcx> ::core::marker::Copy for PlaceTy<'tcx> { }Copy, #[automatically_derived]
impl<'tcx> ::core::clone::Clone for PlaceTy<'tcx> {
    #[inline]
    fn clone(&self) -> PlaceTy<'tcx> {
        let _: ::core::clone::AssertParamIsClone<Ty<'tcx>>;
        let _: ::core::clone::AssertParamIsClone<Option<VariantIdx>>;
        *self
    }
}Clone, #[automatically_derived]
impl<'tcx> ::core::fmt::Debug for PlaceTy<'tcx> {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        ::core::fmt::Formatter::debug_struct_field2_finish(f, "PlaceTy", "ty",
            &self.ty, "variant_index", &&self.variant_index)
    }
}Debug, const _: () =
    {
        impl<'tcx>
            ::rustc_middle::ty::TypeFoldable<::rustc_middle::ty::TyCtxt<'tcx>>
            for PlaceTy<'tcx> {
            fn try_fold_with<__F: ::rustc_middle::ty::FallibleTypeFolder<::rustc_middle::ty::TyCtxt<'tcx>>>(self,
                __folder: &mut __F) -> Result<Self, __F::Error> {
                Ok(match self {
                        PlaceTy { ty: __binding_0, variant_index: __binding_1 } => {
                            PlaceTy {
                                ty: ::rustc_middle::ty::TypeFoldable::try_fold_with(__binding_0,
                                        __folder)?,
                                variant_index: ::rustc_middle::ty::TypeFoldable::try_fold_with(__binding_1,
                                        __folder)?,
                            }
                        }
                    })
            }
            fn fold_with<__F: ::rustc_middle::ty::TypeFolder<::rustc_middle::ty::TyCtxt<'tcx>>>(self,
                __folder: &mut __F) -> Self {
                match self {
                    PlaceTy { ty: __binding_0, variant_index: __binding_1 } => {
                        PlaceTy {
                            ty: ::rustc_middle::ty::TypeFoldable::fold_with(__binding_0,
                                __folder),
                            variant_index: ::rustc_middle::ty::TypeFoldable::fold_with(__binding_1,
                                __folder),
                        }
                    }
                }
            }
        }
    };TypeFoldable, const _: () =
    {
        impl<'tcx>
            ::rustc_middle::ty::TypeVisitable<::rustc_middle::ty::TyCtxt<'tcx>>
            for PlaceTy<'tcx> {
            fn visit_with<__V: ::rustc_middle::ty::TypeVisitor<::rustc_middle::ty::TyCtxt<'tcx>>>(&self,
                __visitor: &mut __V) -> __V::Result {
                match *self {
                    PlaceTy {
                        ty: ref __binding_0, variant_index: ref __binding_1 } => {
                        {
                            match ::rustc_middle::ty::VisitorResult::branch(::rustc_middle::ty::TypeVisitable::visit_with(__binding_0,
                                        __visitor)) {
                                ::core::ops::ControlFlow::Continue(()) => {}
                                ::core::ops::ControlFlow::Break(r) => {
                                    return ::rustc_middle::ty::VisitorResult::from_residual(r);
                                }
                            }
                        }
                        {
                            match ::rustc_middle::ty::VisitorResult::branch(::rustc_middle::ty::TypeVisitable::visit_with(__binding_1,
                                        __visitor)) {
                                ::core::ops::ControlFlow::Continue(()) => {}
                                ::core::ops::ControlFlow::Break(r) => {
                                    return ::rustc_middle::ty::VisitorResult::from_residual(r);
                                }
                            }
                        }
                    }
                }
                <__V::Result as ::rustc_middle::ty::VisitorResult>::output()
            }
        }
    };TypeVisitable)]
94pub struct PlaceTy<'tcx> {
95    pub ty: Ty<'tcx>,
96    /// Downcast to a particular variant of an enum or a coroutine, if included.
97    pub variant_index: Option<VariantIdx>,
98}
99
100// At least on 64 bit systems, `PlaceTy` should not be larger than two or three pointers.
101#[cfg(target_pointer_width = "64")]
102const _: [(); 16] = [(); ::std::mem::size_of::<PlaceTy<'_>>()];rustc_data_structures::static_assert_size!(PlaceTy<'_>, 16);
103
104impl<'tcx> PlaceTy<'tcx> {
105    #[inline]
106    pub fn from_ty(ty: Ty<'tcx>) -> PlaceTy<'tcx> {
107        PlaceTy { ty, variant_index: None }
108    }
109
110    /// `place_ty.field_ty(tcx, f)` computes the type of a given field.
111    ///
112    /// Most clients of `PlaceTy` can instead just extract the relevant type
113    /// directly from their `PlaceElem`, but some instances of `ProjectionElem<V, T>`
114    /// do not carry a `Ty` for `T`.
115    ///
116    /// Note that the resulting type has not been normalized.
117    x;#[instrument(level = "debug", skip(tcx), ret)]
118    pub fn field_ty(
119        tcx: TyCtxt<'tcx>,
120        self_ty: Ty<'tcx>,
121        variant_idx: Option<VariantIdx>,
122        f: FieldIdx,
123    ) -> Unnormalized<'tcx, Ty<'tcx>> {
124        if let Some(variant_index) = variant_idx {
125            match *self_ty.kind() {
126                ty::Adt(adt_def, args) if adt_def.is_enum() => {
127                    adt_def.variant(variant_index).fields[f].ty(tcx, args)
128                }
129                ty::Coroutine(def_id, args) => {
130                    let mut variants = args.as_coroutine().state_tys(def_id, tcx);
131                    let Some(mut variant) = variants.nth(variant_index.into()) else {
132                        bug!("variant {variant_index:?} of coroutine out of range: {self_ty:?}");
133                    };
134
135                    Unnormalized::new_wip(variant.nth(f.index()).unwrap_or_else(|| {
136                        bug!("field {f:?} out of range of variant: {self_ty:?} {variant_idx:?}")
137                    }))
138                }
139                _ => bug!("can't downcast non-adt non-coroutine type: {self_ty:?}"),
140            }
141        } else {
142            match self_ty.kind() {
143                ty::Adt(adt_def, args) if !adt_def.is_enum() => {
144                    adt_def.non_enum_variant().fields[f].ty(tcx, args)
145                }
146                ty::Closure(_, args) => Unnormalized::dummy(
147                    args.as_closure()
148                        .upvar_tys()
149                        .get(f.index())
150                        .copied()
151                        .unwrap_or_else(|| bug!("field {f:?} out of range: {self_ty:?}")),
152                ),
153                ty::CoroutineClosure(_, args) => Unnormalized::dummy(
154                    args.as_coroutine_closure()
155                        .upvar_tys()
156                        .get(f.index())
157                        .copied()
158                        .unwrap_or_else(|| bug!("field {f:?} out of range: {self_ty:?}")),
159                ),
160                // Only prefix fields (upvars and current state) are
161                // accessible without a variant index.
162                ty::Coroutine(_, args) => Unnormalized::dummy(
163                    args.as_coroutine().prefix_tys().get(f.index()).copied().unwrap_or_else(|| {
164                        bug!("field {f:?} out of range of prefixes for {self_ty}")
165                    }),
166                ),
167                ty::Tuple(tys) => Unnormalized::dummy(
168                    tys.get(f.index())
169                        .copied()
170                        .unwrap_or_else(|| bug!("field {f:?} out of range: {self_ty:?}")),
171                ),
172                _ => bug!("can't project out of {self_ty:?}"),
173            }
174        }
175    }
176
177    pub fn multi_projection_ty(
178        self,
179        tcx: TyCtxt<'tcx>,
180        elems: &[PlaceElem<'tcx>],
181    ) -> PlaceTy<'tcx> {
182        elems.iter().fold(self, |place_ty, &elem| place_ty.projection_ty(tcx, elem))
183    }
184
185    /// Convenience wrapper around `projection_ty_core` for `PlaceElem`,
186    /// where we can just use the `Ty` that is already stored inline on
187    /// field projection elems.
188    pub fn projection_ty<V: ::std::fmt::Debug>(
189        self,
190        tcx: TyCtxt<'tcx>,
191        elem: ProjectionElem<V, Ty<'tcx>>,
192    ) -> PlaceTy<'tcx> {
193        self.projection_ty_core(tcx, &elem, |ty| ty, |_, _, _, ty| ty, |ty| ty)
194    }
195
196    /// `place_ty.projection_ty_core(tcx, elem, |...| { ... })`
197    /// projects `place_ty` onto `elem`, returning the appropriate
198    /// `Ty` or downcast variant corresponding to that projection.
199    /// The `handle_field` callback must map a `FieldIdx` to its `Ty`,
200    /// (which should be trivial when `T` = `Ty`).
201    pub fn projection_ty_core<V, T>(
202        self,
203        tcx: TyCtxt<'tcx>,
204        elem: &ProjectionElem<V, T>,
205        // FIXME(#155345): This should take `Unnormalized` as input and only
206        // normalize when actually required.
207        mut structurally_normalize: impl FnMut(Ty<'tcx>) -> Ty<'tcx>,
208        mut handle_field: impl FnMut(Ty<'tcx>, Option<VariantIdx>, FieldIdx, T) -> Ty<'tcx>,
209        mut handle_opaque_cast_and_subtype: impl FnMut(T) -> Ty<'tcx>,
210    ) -> PlaceTy<'tcx>
211    where
212        V: ::std::fmt::Debug,
213        T: ::std::fmt::Debug + Copy,
214    {
215        if self.variant_index.is_some() && !#[allow(non_exhaustive_omitted_patterns)] match elem {
    ProjectionElem::Field(..) => true,
    _ => false,
}matches!(elem, ProjectionElem::Field(..)) {
216            crate::util::bug::bug_fmt(format_args!("cannot use non field projection on downcasted place"))bug!("cannot use non field projection on downcasted place")
217        }
218        let answer = match *elem {
219            ProjectionElem::Deref => {
220                let ty = structurally_normalize(self.ty).builtin_deref(true).unwrap_or_else(|| {
221                    crate::util::bug::bug_fmt(format_args!("deref projection of non-dereferenceable ty {0:?}",
        self))bug!("deref projection of non-dereferenceable ty {:?}", self)
222                });
223                PlaceTy::from_ty(ty)
224            }
225            ProjectionElem::Index(_) | ProjectionElem::ConstantIndex { .. } => {
226                PlaceTy::from_ty(structurally_normalize(self.ty).builtin_index().unwrap())
227            }
228            ProjectionElem::Subslice { from, to, from_end } => {
229                PlaceTy::from_ty(match structurally_normalize(self.ty).kind() {
230                    ty::Slice(..) => self.ty,
231                    ty::Array(inner, _) if !from_end => Ty::new_array(tcx, *inner, to - from),
232                    ty::Array(inner, size) if from_end => {
233                        let size = size
234                            .try_to_target_usize(tcx)
235                            .expect("expected subslice projection on fixed-size array");
236                        let len = size - from - to;
237                        Ty::new_array(tcx, *inner, len)
238                    }
239                    _ => crate::util::bug::bug_fmt(format_args!("cannot subslice non-array type: `{0:?}`",
        self))bug!("cannot subslice non-array type: `{:?}`", self),
240                })
241            }
242            ProjectionElem::Downcast(_name, index) => {
243                PlaceTy { ty: self.ty, variant_index: Some(index) }
244            }
245            ProjectionElem::Field(f, fty) => PlaceTy::from_ty(handle_field(
246                structurally_normalize(self.ty),
247                self.variant_index,
248                f,
249                fty,
250            )),
251            ProjectionElem::OpaqueCast(ty) => PlaceTy::from_ty(handle_opaque_cast_and_subtype(ty)),
252
253            // FIXME(unsafe_binders): Rename `handle_opaque_cast_and_subtype` to be more general.
254            ProjectionElem::UnwrapUnsafeBinder(ty) => {
255                PlaceTy::from_ty(handle_opaque_cast_and_subtype(ty))
256            }
257        };
258        {
    use ::tracing::__macro_support::Callsite as _;
    static __CALLSITE: ::tracing::callsite::DefaultCallsite =
        {
            static META: ::tracing::Metadata<'static> =
                {
                    ::tracing_core::metadata::Metadata::new("event compiler/rustc_middle/src/mir/statement.rs:258",
                        "rustc_middle::mir::statement", ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_middle/src/mir/statement.rs"),
                        ::tracing_core::__macro_support::Option::Some(258u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_middle::mir::statement"),
                        ::tracing_core::field::FieldSet::new(&["message"],
                            ::tracing_core::callsite::Identifier(&__CALLSITE)),
                        ::tracing::metadata::Kind::EVENT)
                };
            ::tracing::callsite::DefaultCallsite::new(&META)
        };
    let enabled =
        ::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
                &&
                ::tracing::Level::DEBUG <=
                    ::tracing::level_filters::LevelFilter::current() &&
            {
                let interest = __CALLSITE.interest();
                !interest.is_never() &&
                    ::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
                        interest)
            };
    if enabled {
        (|value_set: ::tracing::field::ValueSet|
                    {
                        let meta = __CALLSITE.metadata();
                        ::tracing::Event::dispatch(meta, &value_set);
                        ;
                    })({
                #[allow(unused_imports)]
                use ::tracing::field::{debug, display, Value};
                let mut iter = __CALLSITE.metadata().fields().iter();
                __CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
                                    ::tracing::__macro_support::Option::Some(&format_args!("projection_ty self: {0:?} elem: {1:?} yields: {2:?}",
                                                    self, elem, answer) as &dyn Value))])
            });
    } else { ; }
};debug!("projection_ty self: {:?} elem: {:?} yields: {:?}", self, elem, answer);
259        answer
260    }
261}
262
263impl<V, T> ProjectionElem<V, T> {
264    /// Returns `true` if the target of this projection may refer to a different region of memory
265    /// than the base.
266    pub fn is_indirect(&self) -> bool {
267        match self {
268            Self::Deref => true,
269
270            Self::Field(_, _)
271            | Self::Index(_)
272            | Self::OpaqueCast(_)
273            | Self::ConstantIndex { .. }
274            | Self::Subslice { .. }
275            | Self::Downcast(_, _)
276            | Self::UnwrapUnsafeBinder(..) => false,
277        }
278    }
279
280    /// Returns `true` if the target of this projection always refers to the same memory region
281    /// whatever the state of the program.
282    pub fn is_stable_offset(&self) -> bool {
283        match self {
284            Self::Deref | Self::Index(_) => false,
285            Self::Field(_, _)
286            | Self::OpaqueCast(_)
287            | Self::ConstantIndex { .. }
288            | Self::Subslice { .. }
289            | Self::Downcast(_, _)
290            | Self::UnwrapUnsafeBinder(..) => true,
291        }
292    }
293
294    /// Returns `true` if this is a `Downcast` projection with the given `VariantIdx`.
295    pub fn is_downcast_to(&self, v: VariantIdx) -> bool {
296        #[allow(non_exhaustive_omitted_patterns)] match *self {
    Self::Downcast(_, x) if x == v => true,
    _ => false,
}matches!(*self, Self::Downcast(_, x) if x == v)
297    }
298
299    /// Returns `true` if this is a `Field` projection with the given index.
300    pub fn is_field_to(&self, f: FieldIdx) -> bool {
301        #[allow(non_exhaustive_omitted_patterns)] match *self {
    Self::Field(x, _) if x == f => true,
    _ => false,
}matches!(*self, Self::Field(x, _) if x == f)
302    }
303
304    /// Returns `true` if this is accepted inside `VarDebugInfoContents::Place`.
305    pub fn can_use_in_debuginfo(&self) -> bool {
306        match self {
307            Self::ConstantIndex { from_end: false, .. }
308            | Self::Deref
309            | Self::Downcast(_, _)
310            | Self::Field(_, _) => true,
311            Self::ConstantIndex { from_end: true, .. }
312            | Self::Index(_)
313            | Self::OpaqueCast(_)
314            | Self::Subslice { .. } => false,
315
316            // FIXME(unsafe_binders): Figure this out.
317            Self::UnwrapUnsafeBinder(..) => false,
318        }
319    }
320
321    /// Returns the `ProjectionKind` associated to this projection.
322    pub fn kind(self) -> ProjectionKind {
323        self.try_map(|_| Some(()), |_| ()).unwrap()
324    }
325
326    /// Apply functions to types and values in this projection and return the result.
327    pub fn try_map<V2, T2>(
328        self,
329        v: impl FnOnce(V) -> Option<V2>,
330        t: impl FnOnce(T) -> T2,
331    ) -> Option<ProjectionElem<V2, T2>> {
332        Some(match self {
333            ProjectionElem::Deref => ProjectionElem::Deref,
334            ProjectionElem::Downcast(name, read_variant) => {
335                ProjectionElem::Downcast(name, read_variant)
336            }
337            ProjectionElem::Field(f, ty) => ProjectionElem::Field(f, t(ty)),
338            ProjectionElem::ConstantIndex { offset, min_length, from_end } => {
339                ProjectionElem::ConstantIndex { offset, min_length, from_end }
340            }
341            ProjectionElem::Subslice { from, to, from_end } => {
342                ProjectionElem::Subslice { from, to, from_end }
343            }
344            ProjectionElem::OpaqueCast(ty) => ProjectionElem::OpaqueCast(t(ty)),
345            ProjectionElem::UnwrapUnsafeBinder(ty) => ProjectionElem::UnwrapUnsafeBinder(t(ty)),
346            ProjectionElem::Index(val) => ProjectionElem::Index(v(val)?),
347        })
348    }
349}
350
351/// Alias for projections as they appear in `UserTypeProjection`, where we
352/// need neither the `V` parameter for `Index` nor the `T` for `Field`.
353pub type ProjectionKind = ProjectionElem<(), ()>;
354
355#[derive(#[automatically_derived]
impl<'tcx> ::core::clone::Clone for PlaceRef<'tcx> {
    #[inline]
    fn clone(&self) -> PlaceRef<'tcx> {
        let _: ::core::clone::AssertParamIsClone<Local>;
        let _: ::core::clone::AssertParamIsClone<&'tcx [PlaceElem<'tcx>]>;
        *self
    }
}Clone, #[automatically_derived]
impl<'tcx> ::core::marker::Copy for PlaceRef<'tcx> { }Copy, #[automatically_derived]
impl<'tcx> ::core::cmp::PartialEq for PlaceRef<'tcx> {
    #[inline]
    fn eq(&self, other: &PlaceRef<'tcx>) -> bool {
        self.local == other.local && self.projection == other.projection
    }
}PartialEq, #[automatically_derived]
impl<'tcx> ::core::cmp::Eq for PlaceRef<'tcx> {
    #[inline]
    #[doc(hidden)]
    #[coverage(off)]
    fn assert_fields_are_eq(&self) {
        let _: ::core::cmp::AssertParamIsEq<Local>;
        let _: ::core::cmp::AssertParamIsEq<&'tcx [PlaceElem<'tcx>]>;
    }
}Eq, #[automatically_derived]
impl<'tcx> ::core::hash::Hash for PlaceRef<'tcx> {
    #[inline]
    fn hash<__H: ::core::hash::Hasher>(&self, state: &mut __H) {
        ::core::hash::Hash::hash(&self.local, state);
        ::core::hash::Hash::hash(&self.projection, state)
    }
}Hash)]
356pub struct PlaceRef<'tcx> {
357    pub local: Local,
358    pub projection: &'tcx [PlaceElem<'tcx>],
359}
360
361// Once we stop implementing `Ord` for `DefId`,
362// this impl will be unnecessary. Until then, we'll
363// leave this impl in place to prevent re-adding a
364// dependency on the `Ord` impl for `DefId`
365impl<'tcx> !PartialOrd for PlaceRef<'tcx> {}
366
367impl<'tcx> Place<'tcx> {
368    // FIXME change this to a const fn by also making List::empty a const fn.
369    pub fn return_place() -> Place<'tcx> {
370        Place { local: RETURN_PLACE, projection: List::empty() }
371    }
372
373    /// Returns `true` if this `Place` contains a `Deref` projection.
374    ///
375    /// If `Place::is_indirect` returns false, the caller knows that the `Place` refers to the
376    /// same region of memory as its base.
377    pub fn is_indirect(&self) -> bool {
378        self.projection.iter().any(|elem| elem.is_indirect())
379    }
380
381    /// Returns `true` if the `Place` always refers to the same memory region
382    /// whatever the state of the program.
383    pub fn is_stable_offset(&self) -> bool {
384        self.projection.iter().all(|elem| elem.is_stable_offset())
385    }
386
387    /// Returns `true` if this `Place`'s first projection is `Deref`.
388    ///
389    /// This is useful because for MIR phases `AnalysisPhase::PostCleanup` and later,
390    /// `Deref` projections can only occur as the first projection. In that case this method
391    /// is equivalent to `is_indirect`, but faster.
392    pub fn is_indirect_first_projection(&self) -> bool {
393        self.as_ref().is_indirect_first_projection()
394    }
395
396    /// Finds the innermost `Local` from this `Place`, *if* it is either a local itself or
397    /// a single deref of a local.
398    #[inline(always)]
399    pub fn local_or_deref_local(&self) -> Option<Local> {
400        self.as_ref().local_or_deref_local()
401    }
402
403    /// If this place represents a local variable like `_X` with no
404    /// projections, return `Some(_X)`.
405    #[inline(always)]
406    pub fn as_local(&self) -> Option<Local> {
407        self.as_ref().as_local()
408    }
409
410    #[inline]
411    pub fn as_ref(&self) -> PlaceRef<'tcx> {
412        PlaceRef { local: self.local, projection: self.projection }
413    }
414
415    /// Iterate over the projections in evaluation order, i.e., the first element is the base with
416    /// its projection and then subsequently more projections are added.
417    /// As a concrete example, given the place a.b.c, this would yield:
418    /// - (a, .b)
419    /// - (a.b, .c)
420    ///
421    /// Given a place without projections, the iterator is empty.
422    #[inline]
423    pub fn iter_projections(
424        self,
425    ) -> impl Iterator<Item = (PlaceRef<'tcx>, PlaceElem<'tcx>)> + DoubleEndedIterator {
426        self.as_ref().iter_projections()
427    }
428
429    /// Generates a new place by appending `more_projections` to the existing ones
430    /// and interning the result.
431    pub fn project_deeper(self, more_projections: &[PlaceElem<'tcx>], tcx: TyCtxt<'tcx>) -> Self {
432        if more_projections.is_empty() {
433            return self;
434        }
435
436        self.as_ref().project_deeper(more_projections, tcx)
437    }
438
439    /// Return a place that projects to a field of the current place.
440    ///
441    /// The type of the current place must be an ADT.
442    pub fn project_to_field(
443        self,
444        idx: FieldIdx,
445        local_decls: &impl HasLocalDecls<'tcx>,
446        tcx: TyCtxt<'tcx>,
447    ) -> Self {
448        let ty = self.ty(local_decls, tcx).ty;
449        let ty::Adt(adt, args) = ty.kind() else { {
    ::core::panicking::panic_fmt(format_args!("projecting to field of non-ADT {0}",
            ty));
}panic!("projecting to field of non-ADT {ty}") };
450        let field = &adt.non_enum_variant().fields[idx];
451        let field_ty = field.ty(tcx, args).skip_norm_wip();
452        self.project_deeper(&[ProjectionElem::Field(idx, field_ty)], tcx)
453    }
454
455    pub fn ty_from<D>(
456        local: Local,
457        projection: &[PlaceElem<'tcx>],
458        local_decls: &D,
459        tcx: TyCtxt<'tcx>,
460    ) -> PlaceTy<'tcx>
461    where
462        D: ?Sized + HasLocalDecls<'tcx>,
463    {
464        // If there's a field projection element in `projection`, we *could* skip everything
465        // before that, but on 2026-01-31 a perf experiment showed no benefit from doing so.
466        PlaceTy::from_ty(local_decls.local_decls()[local].ty).multi_projection_ty(tcx, projection)
467    }
468
469    pub fn ty<D: ?Sized>(&self, local_decls: &D, tcx: TyCtxt<'tcx>) -> PlaceTy<'tcx>
470    where
471        D: HasLocalDecls<'tcx>,
472    {
473        Place::ty_from(self.local, self.projection, local_decls, tcx)
474    }
475}
476
477impl From<Local> for Place<'_> {
478    #[inline]
479    fn from(local: Local) -> Self {
480        Place { local, projection: List::empty() }
481    }
482}
483
484impl<'tcx> PlaceRef<'tcx> {
485    pub fn is_prefix_of(&self, other: PlaceRef<'tcx>) -> bool {
486        self.local == other.local
487            && self.projection.len() <= other.projection.len()
488            && self.projection == &other.projection[..self.projection.len()]
489    }
490
491    /// Finds the innermost `Local` from this `Place`, *if* it is either a local itself or
492    /// a single deref of a local.
493    pub fn local_or_deref_local(&self) -> Option<Local> {
494        match *self {
495            PlaceRef { local, projection: [] }
496            | PlaceRef { local, projection: [ProjectionElem::Deref] } => Some(local),
497            _ => None,
498        }
499    }
500
501    /// Returns `true` if this `Place` contains a `Deref` projection.
502    ///
503    /// If `Place::is_indirect` returns false, the caller knows that the `Place` refers to the
504    /// same region of memory as its base.
505    pub fn is_indirect(&self) -> bool {
506        self.projection.iter().any(|elem| elem.is_indirect())
507    }
508
509    /// Returns `true` if this `Place`'s first projection is `Deref`.
510    ///
511    /// This is useful because for MIR phases `AnalysisPhase::PostCleanup` and later,
512    /// `Deref` projections can only occur as the first projection. In that case this method
513    /// is equivalent to `is_indirect`, but faster.
514    pub fn is_indirect_first_projection(&self) -> bool {
515        // To make sure this is not accidentally used in wrong mir phase
516        if true {
    if !(self.projection.is_empty() ||
                !self.projection[1..].contains(&PlaceElem::Deref)) {
        ::core::panicking::panic("assertion failed: self.projection.is_empty() ||\n    !self.projection[1..].contains(&PlaceElem::Deref)")
    };
};debug_assert!(
517            self.projection.is_empty() || !self.projection[1..].contains(&PlaceElem::Deref)
518        );
519        self.projection.first() == Some(&PlaceElem::Deref)
520    }
521
522    /// If this place represents a local variable like `_X` with no
523    /// projections, return `Some(_X)`.
524    #[inline]
525    pub fn as_local(&self) -> Option<Local> {
526        match *self {
527            PlaceRef { local, projection: [] } => Some(local),
528            _ => None,
529        }
530    }
531
532    #[inline]
533    pub fn to_place(&self, tcx: TyCtxt<'tcx>) -> Place<'tcx> {
534        Place { local: self.local, projection: tcx.mk_place_elems(self.projection) }
535    }
536
537    #[inline]
538    pub fn last_projection(&self) -> Option<(PlaceRef<'tcx>, PlaceElem<'tcx>)> {
539        if let &[ref proj_base @ .., elem] = self.projection {
540            Some((PlaceRef { local: self.local, projection: proj_base }, elem))
541        } else {
542            None
543        }
544    }
545
546    /// Iterate over the projections in evaluation order, i.e., the first element is the base with
547    /// its projection and then subsequently more projections are added.
548    /// As a concrete example, given the place a.b.c, this would yield:
549    /// - (a, .b)
550    /// - (a.b, .c)
551    ///
552    /// Given a place without projections, the iterator is empty.
553    #[inline]
554    pub fn iter_projections(
555        self,
556    ) -> impl Iterator<Item = (PlaceRef<'tcx>, PlaceElem<'tcx>)> + DoubleEndedIterator {
557        self.projection.iter().enumerate().map(move |(i, proj)| {
558            let base = PlaceRef { local: self.local, projection: &self.projection[..i] };
559            (base, *proj)
560        })
561    }
562
563    /// Return the place accessed locals that include the base local.
564    pub fn accessed_locals(self) -> impl Iterator<Item = Local> {
565        std::iter::once(self.local).chain(self.projection.iter().filter_map(|proj| match proj {
566            ProjectionElem::Index(local) => Some(*local),
567            ProjectionElem::Deref
568            | ProjectionElem::Field(_, _)
569            | ProjectionElem::ConstantIndex { .. }
570            | ProjectionElem::Subslice { .. }
571            | ProjectionElem::Downcast(_, _)
572            | ProjectionElem::OpaqueCast(_)
573            | ProjectionElem::UnwrapUnsafeBinder(_) => None,
574        }))
575    }
576
577    /// Generates a new place by appending `more_projections` to the existing ones
578    /// and interning the result.
579    pub fn project_deeper(
580        self,
581        more_projections: &[PlaceElem<'tcx>],
582        tcx: TyCtxt<'tcx>,
583    ) -> Place<'tcx> {
584        let mut v: Vec<PlaceElem<'tcx>>;
585
586        let new_projections = if self.projection.is_empty() {
587            more_projections
588        } else {
589            v = Vec::with_capacity(self.projection.len() + more_projections.len());
590            v.extend(self.projection);
591            v.extend(more_projections);
592            &v
593        };
594
595        Place { local: self.local, projection: tcx.mk_place_elems(new_projections) }
596    }
597
598    pub fn ty<D>(&self, local_decls: &D, tcx: TyCtxt<'tcx>) -> PlaceTy<'tcx>
599    where
600        D: ?Sized + HasLocalDecls<'tcx>,
601    {
602        Place::ty_from(self.local, self.projection, local_decls, tcx)
603    }
604}
605
606impl From<Local> for PlaceRef<'_> {
607    #[inline]
608    fn from(local: Local) -> Self {
609        PlaceRef { local, projection: &[] }
610    }
611}
612
613///////////////////////////////////////////////////////////////////////////
614// Operands
615
616impl<'tcx> Operand<'tcx> {
617    /// Convenience helper to make a constant that refers to the fn
618    /// with given `DefId` and args. Since this is used to synthesize
619    /// MIR, assumes `user_ty` is None.
620    pub fn function_handle(
621        tcx: TyCtxt<'tcx>,
622        def_id: DefId,
623        args: impl IntoIterator<Item = GenericArg<'tcx>>,
624        span: Span,
625    ) -> Self {
626        let ty = Ty::new_fn_def(tcx, def_id, args);
627        Operand::Constant(Box::new(ConstOperand {
628            span,
629            user_ty: None,
630            const_: Const::Val(ConstValue::ZeroSized, ty),
631        }))
632    }
633
634    /// Convenience helper to make a constant that refers to the given `DefId` and args. Since this
635    /// is used to synthesize MIR, assumes `user_ty` is None.
636    pub fn unevaluated_constant(
637        tcx: TyCtxt<'tcx>,
638        def_id: DefId,
639        args: &[GenericArg<'tcx>],
640        span: Span,
641    ) -> Self {
642        let const_ = Const::from_unevaluated(tcx, def_id).instantiate(tcx, args).skip_norm_wip();
643        Operand::Constant(Box::new(ConstOperand { span, user_ty: None, const_ }))
644    }
645
646    pub fn is_move(&self) -> bool {
647        #[allow(non_exhaustive_omitted_patterns)] match self {
    Operand::Move(..) => true,
    _ => false,
}matches!(self, Operand::Move(..))
648    }
649
650    /// Convenience helper to make a literal-like constant from a given scalar value.
651    /// Since this is used to synthesize MIR, assumes `user_ty` is None.
652    pub fn const_from_scalar(
653        tcx: TyCtxt<'tcx>,
654        ty: Ty<'tcx>,
655        val: Scalar,
656        span: Span,
657    ) -> Operand<'tcx> {
658        if true {
    if !{
                let typing_env = ty::TypingEnv::fully_monomorphized();
                let type_size =
                    tcx.layout_of(typing_env.as_query_input(ty)).unwrap_or_else(|e|
                                {
                                    ::core::panicking::panic_fmt(format_args!("could not compute layout for {0:?}: {1:?}",
                                            ty, e));
                                }).size;
                let scalar_size =
                    match val {
                        Scalar::Int(int) => int.size(),
                        _ => {
                            ::core::panicking::panic_fmt(format_args!("Invalid scalar type {0:?}",
                                    val));
                        }
                    };
                scalar_size == type_size
            } {
        ::core::panicking::panic("assertion failed: {\n    let typing_env = ty::TypingEnv::fully_monomorphized();\n    let type_size =\n        tcx.layout_of(typing_env.as_query_input(ty)).unwrap_or_else(|e|\n                    panic!(\"could not compute layout for {ty:?}: {e:?}\")).size;\n    let scalar_size =\n        match val {\n            Scalar::Int(int) => int.size(),\n            _ => panic!(\"Invalid scalar type {val:?}\"),\n        };\n    scalar_size == type_size\n}")
    };
};debug_assert!({
659            let typing_env = ty::TypingEnv::fully_monomorphized();
660            let type_size = tcx
661                .layout_of(typing_env.as_query_input(ty))
662                .unwrap_or_else(|e| panic!("could not compute layout for {ty:?}: {e:?}"))
663                .size;
664            let scalar_size = match val {
665                Scalar::Int(int) => int.size(),
666                _ => panic!("Invalid scalar type {val:?}"),
667            };
668            scalar_size == type_size
669        });
670        Operand::Constant(Box::new(ConstOperand {
671            span,
672            user_ty: None,
673            const_: Const::Val(ConstValue::Scalar(val), ty),
674        }))
675    }
676
677    pub fn to_copy(&self) -> Self {
678        match *self {
679            Operand::Copy(_) | Operand::Constant(_) | Operand::RuntimeChecks(_) => self.clone(),
680            Operand::Move(place) => Operand::Copy(place),
681        }
682    }
683
684    /// Returns the `Place` that is the target of this `Operand`, or `None` if this `Operand` is a
685    /// constant.
686    pub fn place(&self) -> Option<Place<'tcx>> {
687        match self {
688            Operand::Copy(place) | Operand::Move(place) => Some(*place),
689            Operand::Constant(_) | Operand::RuntimeChecks(_) => None,
690        }
691    }
692
693    /// Returns the `ConstOperand` that is the target of this `Operand`, or `None` if this `Operand` is a
694    /// place.
695    pub fn constant(&self) -> Option<&ConstOperand<'tcx>> {
696        match self {
697            Operand::Constant(x) => Some(&**x),
698            Operand::Copy(_) | Operand::Move(_) | Operand::RuntimeChecks(_) => None,
699        }
700    }
701
702    /// Gets the `ty::FnDef` from an operand if it's a constant function item.
703    ///
704    /// While this is unlikely in general, it's the normal case of what you'll
705    /// find as the `func` in a [`TerminatorKind::Call`].
706    pub fn const_fn_def(&self) -> Option<(DefId, GenericArgsRef<'tcx>)> {
707        let const_ty = self.constant()?.const_.ty();
708        if let ty::FnDef(def_id, args) = *const_ty.kind() { Some((def_id, args)) } else { None }
709    }
710
711    pub fn ty<D>(&self, local_decls: &D, tcx: TyCtxt<'tcx>) -> Ty<'tcx>
712    where
713        D: ?Sized + HasLocalDecls<'tcx>,
714    {
715        match self {
716            &Operand::Copy(ref l) | &Operand::Move(ref l) => l.ty(local_decls, tcx).ty,
717            Operand::Constant(c) => c.const_.ty(),
718            Operand::RuntimeChecks(_) => tcx.types.bool,
719        }
720    }
721
722    pub fn span<D>(&self, local_decls: &D) -> Span
723    where
724        D: ?Sized + HasLocalDecls<'tcx>,
725    {
726        match self {
727            &Operand::Copy(ref l) | &Operand::Move(ref l) => {
728                local_decls.local_decls()[l.local].source_info.span
729            }
730            Operand::Constant(c) => c.span,
731            // User code should not contain this operand, so we should not need this span.
732            Operand::RuntimeChecks(_) => DUMMY_SP,
733        }
734    }
735}
736
737impl<'tcx> ConstOperand<'tcx> {
738    pub fn check_static_ptr(&self, tcx: TyCtxt<'_>) -> Option<DefId> {
739        match self.const_.try_to_scalar() {
740            Some(Scalar::Ptr(ptr, _size)) => match tcx.global_alloc(ptr.provenance.alloc_id()) {
741                GlobalAlloc::Static(def_id) => {
742                    if !!tcx.is_thread_local_static(def_id) {
    ::core::panicking::panic("assertion failed: !tcx.is_thread_local_static(def_id)")
};assert!(!tcx.is_thread_local_static(def_id));
743                    Some(def_id)
744                }
745                _ => None,
746            },
747            _ => None,
748        }
749    }
750
751    #[inline]
752    pub fn ty(&self) -> Ty<'tcx> {
753        self.const_.ty()
754    }
755}
756
757///////////////////////////////////////////////////////////////////////////
758// Rvalues
759
760impl<'tcx> Rvalue<'tcx> {
761    /// Returns true if rvalue can be safely removed when the result is unused.
762    #[inline]
763    pub fn is_safe_to_remove(&self) -> bool {
764        match self {
765            // Pointer to int casts may be side-effects due to exposing the provenance.
766            // While the model is undecided, we should be conservative. See
767            // <https://www.ralfj.de/blog/2022/04/11/provenance-exposed.html>
768            Rvalue::Cast(CastKind::PointerExposeProvenance, _, _) => false,
769
770            Rvalue::Use(_, _)
771            | Rvalue::CopyForDeref(_)
772            | Rvalue::Repeat(_, _)
773            | Rvalue::Ref(_, _, _)
774            | Rvalue::Reborrow(_, _, _)
775            | Rvalue::ThreadLocalRef(_)
776            | Rvalue::RawPtr(_, _)
777            | Rvalue::Cast(
778                CastKind::IntToInt
779                | CastKind::FloatToInt
780                | CastKind::FloatToFloat
781                | CastKind::IntToFloat
782                | CastKind::FnPtrToPtr
783                | CastKind::PtrToPtr
784                | CastKind::PointerCoercion(_, _)
785                | CastKind::PointerWithExposedProvenance
786                | CastKind::Transmute
787                | CastKind::Subtype,
788                _,
789                _,
790            )
791            | Rvalue::BinaryOp(_, _)
792            | Rvalue::UnaryOp(_, _)
793            | Rvalue::Discriminant(_)
794            | Rvalue::Aggregate(_, _)
795            | Rvalue::WrapUnsafeBinder(_, _) => true,
796        }
797    }
798
799    /// Returns true if rvalue is a generic Reborrow coercion (usage of Reborrow or CoerceShared
800    /// trait).
801    pub fn is_generic_reborrow(&self) -> bool {
802        #[allow(non_exhaustive_omitted_patterns)] match self {
    Self::Reborrow(..) => true,
    _ => false,
}matches!(self, Self::Reborrow(..))
803    }
804
805    pub fn ty<D>(&self, local_decls: &D, tcx: TyCtxt<'tcx>) -> Ty<'tcx>
806    where
807        D: ?Sized + HasLocalDecls<'tcx>,
808    {
809        match *self {
810            Rvalue::Use(ref operand, _) => operand.ty(local_decls, tcx),
811            Rvalue::Repeat(ref operand, count) => {
812                Ty::new_array_with_const_len(tcx, operand.ty(local_decls, tcx), count)
813            }
814            Rvalue::ThreadLocalRef(did) => tcx.thread_local_ptr_ty(did),
815            Rvalue::Ref(reg, bk, ref place) => {
816                let place_ty = place.ty(local_decls, tcx).ty;
817                Ty::new_ref(tcx, reg, place_ty, bk.to_mutbl_lossy())
818            }
819            Rvalue::Reborrow(target, _, _) => target,
820            Rvalue::RawPtr(kind, ref place) => {
821                let place_ty = place.ty(local_decls, tcx).ty;
822                Ty::new_ptr(tcx, place_ty, kind.to_mutbl_lossy())
823            }
824            Rvalue::Cast(.., ty) => ty,
825            Rvalue::BinaryOp(op, (ref lhs, ref rhs)) => {
826                let lhs_ty = lhs.ty(local_decls, tcx);
827                let rhs_ty = rhs.ty(local_decls, tcx);
828                op.ty(tcx, lhs_ty, rhs_ty)
829            }
830            Rvalue::UnaryOp(op, ref operand) => {
831                let arg_ty = operand.ty(local_decls, tcx);
832                op.ty(tcx, arg_ty)
833            }
834            Rvalue::Discriminant(ref place) => place.ty(local_decls, tcx).ty.discriminant_ty(tcx),
835            Rvalue::Aggregate(ref ak, ref ops) => match **ak {
836                AggregateKind::Array(ty) => Ty::new_array(tcx, ty, ops.len() as u64),
837                AggregateKind::Tuple => {
838                    Ty::new_tup_from_iter(tcx, ops.iter().map(|op| op.ty(local_decls, tcx)))
839                }
840                AggregateKind::Adt(did, _, args, _, _) => {
841                    tcx.type_of(did).instantiate(tcx, args).skip_norm_wip()
842                }
843                AggregateKind::Closure(did, args) => Ty::new_closure(tcx, did, args),
844                AggregateKind::Coroutine(did, args) => Ty::new_coroutine(tcx, did, args),
845                AggregateKind::CoroutineClosure(did, args) => {
846                    Ty::new_coroutine_closure(tcx, did, args)
847                }
848                AggregateKind::RawPtr(ty, mutability) => Ty::new_ptr(tcx, ty, mutability),
849            },
850            Rvalue::CopyForDeref(ref place) => place.ty(local_decls, tcx).ty,
851            Rvalue::WrapUnsafeBinder(_, ty) => ty,
852        }
853    }
854}
855
856impl BorrowKind {
857    pub fn mutability(&self) -> Mutability {
858        match *self {
859            BorrowKind::Shared | BorrowKind::Fake(_) => Mutability::Not,
860            BorrowKind::Mut { .. } => Mutability::Mut,
861        }
862    }
863
864    /// Returns whether borrows represented by this kind are allowed to be split into separate
865    /// Reservation and Activation phases.
866    pub fn is_two_phase_borrow(&self) -> bool {
867        match *self {
868            BorrowKind::Shared
869            | BorrowKind::Fake(_)
870            | BorrowKind::Mut { kind: MutBorrowKind::Default | MutBorrowKind::ClosureCapture } => {
871                false
872            }
873            BorrowKind::Mut { kind: MutBorrowKind::TwoPhaseBorrow } => true,
874        }
875    }
876
877    pub fn to_mutbl_lossy(self) -> hir::Mutability {
878        match self {
879            BorrowKind::Mut { .. } => hir::Mutability::Mut,
880            BorrowKind::Shared => hir::Mutability::Not,
881
882            // We have no type corresponding to a shallow borrow, so use
883            // `&` as an approximation.
884            BorrowKind::Fake(_) => hir::Mutability::Not,
885        }
886    }
887}
888
889impl<'tcx> UnOp {
890    pub fn ty(&self, tcx: TyCtxt<'tcx>, arg_ty: Ty<'tcx>) -> Ty<'tcx> {
891        match self {
892            UnOp::Not | UnOp::Neg => arg_ty,
893            UnOp::PtrMetadata => arg_ty.pointee_metadata_ty_or_projection(tcx),
894        }
895    }
896}
897
898impl<'tcx> BinOp {
899    pub fn ty(&self, tcx: TyCtxt<'tcx>, lhs_ty: Ty<'tcx>, rhs_ty: Ty<'tcx>) -> Ty<'tcx> {
900        // FIXME: handle SIMD correctly
901        match self {
902            &BinOp::Add
903            | &BinOp::AddUnchecked
904            | &BinOp::Sub
905            | &BinOp::SubUnchecked
906            | &BinOp::Mul
907            | &BinOp::MulUnchecked
908            | &BinOp::Div
909            | &BinOp::Rem
910            | &BinOp::BitXor
911            | &BinOp::BitAnd
912            | &BinOp::BitOr => {
913                // these should be integers or floats of the same size.
914                match (&lhs_ty, &rhs_ty) {
    (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::None);
        }
    }
};assert_eq!(lhs_ty, rhs_ty);
915                lhs_ty
916            }
917            &BinOp::AddWithOverflow | &BinOp::SubWithOverflow | &BinOp::MulWithOverflow => {
918                // these should be integers of the same size.
919                match (&lhs_ty, &rhs_ty) {
    (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::None);
        }
    }
};assert_eq!(lhs_ty, rhs_ty);
920                Ty::new_tup(tcx, &[lhs_ty, tcx.types.bool])
921            }
922            &BinOp::Shl
923            | &BinOp::ShlUnchecked
924            | &BinOp::Shr
925            | &BinOp::ShrUnchecked
926            | &BinOp::Offset => {
927                lhs_ty // lhs_ty can be != rhs_ty
928            }
929            &BinOp::Eq | &BinOp::Lt | &BinOp::Le | &BinOp::Ne | &BinOp::Ge | &BinOp::Gt => {
930                tcx.types.bool
931            }
932            &BinOp::Cmp => {
933                // these should be integer-like types of the same size.
934                match (&lhs_ty, &rhs_ty) {
    (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::None);
        }
    }
};assert_eq!(lhs_ty, rhs_ty);
935                tcx.ty_ordering_enum(DUMMY_SP)
936            }
937        }
938    }
939    pub(crate) fn to_hir_binop(self) -> hir::BinOpKind {
940        match self {
941            // HIR `+`/`-`/`*` can map to either of these MIR BinOp, depending
942            // on whether overflow checks are enabled or not.
943            BinOp::Add | BinOp::AddWithOverflow => hir::BinOpKind::Add,
944            BinOp::Sub | BinOp::SubWithOverflow => hir::BinOpKind::Sub,
945            BinOp::Mul | BinOp::MulWithOverflow => hir::BinOpKind::Mul,
946            BinOp::Div => hir::BinOpKind::Div,
947            BinOp::Rem => hir::BinOpKind::Rem,
948            BinOp::BitXor => hir::BinOpKind::BitXor,
949            BinOp::BitAnd => hir::BinOpKind::BitAnd,
950            BinOp::BitOr => hir::BinOpKind::BitOr,
951            BinOp::Shl => hir::BinOpKind::Shl,
952            BinOp::Shr => hir::BinOpKind::Shr,
953            BinOp::Eq => hir::BinOpKind::Eq,
954            BinOp::Ne => hir::BinOpKind::Ne,
955            BinOp::Lt => hir::BinOpKind::Lt,
956            BinOp::Gt => hir::BinOpKind::Gt,
957            BinOp::Le => hir::BinOpKind::Le,
958            BinOp::Ge => hir::BinOpKind::Ge,
959            // We don't have HIR syntax for these.
960            BinOp::Cmp
961            | BinOp::AddUnchecked
962            | BinOp::SubUnchecked
963            | BinOp::MulUnchecked
964            | BinOp::ShlUnchecked
965            | BinOp::ShrUnchecked
966            | BinOp::Offset => {
967                ::core::panicking::panic("internal error: entered unreachable code")unreachable!()
968            }
969        }
970    }
971
972    /// If this is a `FooWithOverflow`, return `Some(Foo)`.
973    pub fn overflowing_to_wrapping(self) -> Option<BinOp> {
974        Some(match self {
975            BinOp::AddWithOverflow => BinOp::Add,
976            BinOp::SubWithOverflow => BinOp::Sub,
977            BinOp::MulWithOverflow => BinOp::Mul,
978            _ => return None,
979        })
980    }
981
982    /// Returns whether this is a `FooWithOverflow`
983    pub fn is_overflowing(self) -> bool {
984        self.overflowing_to_wrapping().is_some()
985    }
986
987    /// If this is a `Foo`, return `Some(FooWithOverflow)`.
988    pub fn wrapping_to_overflowing(self) -> Option<BinOp> {
989        Some(match self {
990            BinOp::Add => BinOp::AddWithOverflow,
991            BinOp::Sub => BinOp::SubWithOverflow,
992            BinOp::Mul => BinOp::MulWithOverflow,
993            _ => return None,
994        })
995    }
996}
997
998impl From<Mutability> for RawPtrKind {
999    fn from(other: Mutability) -> Self {
1000        match other {
1001            Mutability::Mut => RawPtrKind::Mut,
1002            Mutability::Not => RawPtrKind::Const,
1003        }
1004    }
1005}
1006
1007impl RawPtrKind {
1008    pub fn is_fake(self) -> bool {
1009        match self {
1010            RawPtrKind::Mut | RawPtrKind::Const => false,
1011            RawPtrKind::FakeForPtrMetadata => true,
1012        }
1013    }
1014
1015    pub fn to_mutbl_lossy(self) -> Mutability {
1016        match self {
1017            RawPtrKind::Mut => Mutability::Mut,
1018            RawPtrKind::Const => Mutability::Not,
1019
1020            // We have no type corresponding to a fake borrow, so use
1021            // `*const` as an approximation.
1022            RawPtrKind::FakeForPtrMetadata => Mutability::Not,
1023        }
1024    }
1025
1026    pub fn ptr_str(self) -> &'static str {
1027        match self {
1028            RawPtrKind::Mut => "mut",
1029            RawPtrKind::Const => "const",
1030            RawPtrKind::FakeForPtrMetadata => "const (fake)",
1031        }
1032    }
1033}
1034
1035#[derive(#[automatically_derived]
impl<'tcx> ::core::default::Default for StmtDebugInfos<'tcx> {
    #[inline]
    fn default() -> StmtDebugInfos<'tcx> {
        StmtDebugInfos(::core::default::Default::default())
    }
}Default, #[automatically_derived]
impl<'tcx> ::core::fmt::Debug for StmtDebugInfos<'tcx> {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        ::core::fmt::Formatter::debug_tuple_field1_finish(f, "StmtDebugInfos",
            &&self.0)
    }
}Debug, #[automatically_derived]
impl<'tcx> ::core::clone::Clone for StmtDebugInfos<'tcx> {
    #[inline]
    fn clone(&self) -> StmtDebugInfos<'tcx> {
        StmtDebugInfos(::core::clone::Clone::clone(&self.0))
    }
}Clone, const _: () =
    {
        impl<'tcx, __E: ::rustc_middle::ty::codec::TyEncoder<'tcx>>
            ::rustc_serialize::Encodable<__E> for StmtDebugInfos<'tcx> {
            fn encode(&self, __encoder: &mut __E) {
                match *self {
                    StmtDebugInfos(ref __binding_0) => {
                        ::rustc_serialize::Encodable::<__E>::encode(__binding_0,
                            __encoder);
                    }
                }
            }
        }
    };TyEncodable, const _: () =
    {
        impl<'tcx, __D: ::rustc_middle::ty::codec::TyDecoder<'tcx>>
            ::rustc_serialize::Decodable<__D> for StmtDebugInfos<'tcx> {
            fn decode(__decoder: &mut __D) -> Self {
                StmtDebugInfos(::rustc_serialize::Decodable::decode(__decoder))
            }
        }
    };TyDecodable, const _: () =
    {
        impl<'tcx> ::rustc_data_structures::stable_hash::StableHash for
            StmtDebugInfos<'tcx> {
            #[inline]
            fn stable_hash<__Hcx: ::rustc_data_structures::stable_hash::StableHashCtxt>(&self,
                __hcx: &mut __Hcx,
                __hasher:
                    &mut ::rustc_data_structures::stable_hash::StableHasher) {
                match *self {
                    StmtDebugInfos(ref __binding_0) => {
                        { __binding_0.stable_hash(__hcx, __hasher); }
                    }
                }
            }
        }
    };StableHash, const _: () =
    {
        impl<'tcx>
            ::rustc_middle::ty::TypeFoldable<::rustc_middle::ty::TyCtxt<'tcx>>
            for StmtDebugInfos<'tcx> {
            fn try_fold_with<__F: ::rustc_middle::ty::FallibleTypeFolder<::rustc_middle::ty::TyCtxt<'tcx>>>(self,
                __folder: &mut __F) -> Result<Self, __F::Error> {
                Ok(match self {
                        StmtDebugInfos(__binding_0) => {
                            StmtDebugInfos(::rustc_middle::ty::TypeFoldable::try_fold_with(__binding_0,
                                        __folder)?)
                        }
                    })
            }
            fn fold_with<__F: ::rustc_middle::ty::TypeFolder<::rustc_middle::ty::TyCtxt<'tcx>>>(self,
                __folder: &mut __F) -> Self {
                match self {
                    StmtDebugInfos(__binding_0) => {
                        StmtDebugInfos(::rustc_middle::ty::TypeFoldable::fold_with(__binding_0,
                                __folder))
                    }
                }
            }
        }
    };TypeFoldable, const _: () =
    {
        impl<'tcx>
            ::rustc_middle::ty::TypeVisitable<::rustc_middle::ty::TyCtxt<'tcx>>
            for StmtDebugInfos<'tcx> {
            fn visit_with<__V: ::rustc_middle::ty::TypeVisitor<::rustc_middle::ty::TyCtxt<'tcx>>>(&self,
                __visitor: &mut __V) -> __V::Result {
                match *self {
                    StmtDebugInfos(ref __binding_0) => {
                        {
                            match ::rustc_middle::ty::VisitorResult::branch(::rustc_middle::ty::TypeVisitable::visit_with(__binding_0,
                                        __visitor)) {
                                ::core::ops::ControlFlow::Continue(()) => {}
                                ::core::ops::ControlFlow::Break(r) => {
                                    return ::rustc_middle::ty::VisitorResult::from_residual(r);
                                }
                            }
                        }
                    }
                }
                <__V::Result as ::rustc_middle::ty::VisitorResult>::output()
            }
        }
    };TypeVisitable)]
1036pub struct StmtDebugInfos<'tcx>(Vec<StmtDebugInfo<'tcx>>);
1037
1038impl<'tcx> StmtDebugInfos<'tcx> {
1039    pub fn push(&mut self, debuginfo: StmtDebugInfo<'tcx>) {
1040        self.0.push(debuginfo);
1041    }
1042
1043    pub fn drop_debuginfo(&mut self) {
1044        self.0.clear();
1045    }
1046
1047    pub fn is_empty(&self) -> bool {
1048        self.0.is_empty()
1049    }
1050
1051    pub fn prepend(&mut self, debuginfos: &mut Self) {
1052        if debuginfos.is_empty() {
1053            return;
1054        };
1055        debuginfos.0.append(self);
1056        std::mem::swap(debuginfos, self);
1057    }
1058
1059    pub fn append(&mut self, debuginfos: &mut Self) {
1060        if debuginfos.is_empty() {
1061            return;
1062        };
1063        self.0.append(debuginfos);
1064    }
1065
1066    pub fn extend(&mut self, debuginfos: &Self) {
1067        if debuginfos.is_empty() {
1068            return;
1069        };
1070        self.0.extend_from_slice(debuginfos);
1071    }
1072
1073    pub fn retain_locals(&mut self, locals: &DenseBitSet<Local>) {
1074        self.retain(|debuginfo| match debuginfo {
1075            StmtDebugInfo::AssignRef(local, _) | StmtDebugInfo::InvalidAssign(local) => {
1076                locals.contains(*local)
1077            }
1078        });
1079    }
1080}
1081
1082impl<'tcx> ops::Deref for StmtDebugInfos<'tcx> {
1083    type Target = Vec<StmtDebugInfo<'tcx>>;
1084
1085    #[inline]
1086    fn deref(&self) -> &Vec<StmtDebugInfo<'tcx>> {
1087        &self.0
1088    }
1089}
1090
1091impl<'tcx> ops::DerefMut for StmtDebugInfos<'tcx> {
1092    #[inline]
1093    fn deref_mut(&mut self) -> &mut Vec<StmtDebugInfo<'tcx>> {
1094        &mut self.0
1095    }
1096}
1097
1098#[derive(#[automatically_derived]
impl<'tcx> ::core::clone::Clone for StmtDebugInfo<'tcx> {
    #[inline]
    fn clone(&self) -> StmtDebugInfo<'tcx> {
        match self {
            StmtDebugInfo::AssignRef(__self_0, __self_1) =>
                StmtDebugInfo::AssignRef(::core::clone::Clone::clone(__self_0),
                    ::core::clone::Clone::clone(__self_1)),
            StmtDebugInfo::InvalidAssign(__self_0) =>
                StmtDebugInfo::InvalidAssign(::core::clone::Clone::clone(__self_0)),
        }
    }
}Clone, const _: () =
    {
        impl<'tcx, __E: ::rustc_middle::ty::codec::TyEncoder<'tcx>>
            ::rustc_serialize::Encodable<__E> for StmtDebugInfo<'tcx> {
            fn encode(&self, __encoder: &mut __E) {
                let disc =
                    match *self {
                        StmtDebugInfo::AssignRef(ref __binding_0, ref __binding_1)
                            => {
                            0usize
                        }
                        StmtDebugInfo::InvalidAssign(ref __binding_0) => { 1usize }
                    };
                ::rustc_serialize::Encoder::emit_u8(__encoder, disc as u8);
                match *self {
                    StmtDebugInfo::AssignRef(ref __binding_0, ref __binding_1)
                        => {
                        ::rustc_serialize::Encodable::<__E>::encode(__binding_0,
                            __encoder);
                        ::rustc_serialize::Encodable::<__E>::encode(__binding_1,
                            __encoder);
                    }
                    StmtDebugInfo::InvalidAssign(ref __binding_0) => {
                        ::rustc_serialize::Encodable::<__E>::encode(__binding_0,
                            __encoder);
                    }
                }
            }
        }
    };TyEncodable, const _: () =
    {
        impl<'tcx, __D: ::rustc_middle::ty::codec::TyDecoder<'tcx>>
            ::rustc_serialize::Decodable<__D> for StmtDebugInfo<'tcx> {
            fn decode(__decoder: &mut __D) -> Self {
                match ::rustc_serialize::Decoder::read_u8(__decoder) as usize
                    {
                    0usize => {
                        StmtDebugInfo::AssignRef(::rustc_serialize::Decodable::decode(__decoder),
                            ::rustc_serialize::Decodable::decode(__decoder))
                    }
                    1usize => {
                        StmtDebugInfo::InvalidAssign(::rustc_serialize::Decodable::decode(__decoder))
                    }
                    n => {
                        ::core::panicking::panic_fmt(format_args!("invalid enum variant tag while decoding `StmtDebugInfo`, expected 0..2, actual {0}",
                                n));
                    }
                }
            }
        }
    };TyDecodable, const _: () =
    {
        impl<'tcx> ::rustc_data_structures::stable_hash::StableHash for
            StmtDebugInfo<'tcx> {
            #[inline]
            fn stable_hash<__Hcx: ::rustc_data_structures::stable_hash::StableHashCtxt>(&self,
                __hcx: &mut __Hcx,
                __hasher:
                    &mut ::rustc_data_structures::stable_hash::StableHasher) {
                ::std::mem::discriminant(self).stable_hash(__hcx, __hasher);
                match *self {
                    StmtDebugInfo::AssignRef(ref __binding_0, ref __binding_1)
                        => {
                        { __binding_0.stable_hash(__hcx, __hasher); }
                        { __binding_1.stable_hash(__hcx, __hasher); }
                    }
                    StmtDebugInfo::InvalidAssign(ref __binding_0) => {
                        { __binding_0.stable_hash(__hcx, __hasher); }
                    }
                }
            }
        }
    };StableHash, const _: () =
    {
        impl<'tcx>
            ::rustc_middle::ty::TypeFoldable<::rustc_middle::ty::TyCtxt<'tcx>>
            for StmtDebugInfo<'tcx> {
            fn try_fold_with<__F: ::rustc_middle::ty::FallibleTypeFolder<::rustc_middle::ty::TyCtxt<'tcx>>>(self,
                __folder: &mut __F) -> Result<Self, __F::Error> {
                Ok(match self {
                        StmtDebugInfo::AssignRef(__binding_0, __binding_1) => {
                            StmtDebugInfo::AssignRef(::rustc_middle::ty::TypeFoldable::try_fold_with(__binding_0,
                                        __folder)?,
                                ::rustc_middle::ty::TypeFoldable::try_fold_with(__binding_1,
                                        __folder)?)
                        }
                        StmtDebugInfo::InvalidAssign(__binding_0) => {
                            StmtDebugInfo::InvalidAssign(::rustc_middle::ty::TypeFoldable::try_fold_with(__binding_0,
                                        __folder)?)
                        }
                    })
            }
            fn fold_with<__F: ::rustc_middle::ty::TypeFolder<::rustc_middle::ty::TyCtxt<'tcx>>>(self,
                __folder: &mut __F) -> Self {
                match self {
                    StmtDebugInfo::AssignRef(__binding_0, __binding_1) => {
                        StmtDebugInfo::AssignRef(::rustc_middle::ty::TypeFoldable::fold_with(__binding_0,
                                __folder),
                            ::rustc_middle::ty::TypeFoldable::fold_with(__binding_1,
                                __folder))
                    }
                    StmtDebugInfo::InvalidAssign(__binding_0) => {
                        StmtDebugInfo::InvalidAssign(::rustc_middle::ty::TypeFoldable::fold_with(__binding_0,
                                __folder))
                    }
                }
            }
        }
    };TypeFoldable, const _: () =
    {
        impl<'tcx>
            ::rustc_middle::ty::TypeVisitable<::rustc_middle::ty::TyCtxt<'tcx>>
            for StmtDebugInfo<'tcx> {
            fn visit_with<__V: ::rustc_middle::ty::TypeVisitor<::rustc_middle::ty::TyCtxt<'tcx>>>(&self,
                __visitor: &mut __V) -> __V::Result {
                match *self {
                    StmtDebugInfo::AssignRef(ref __binding_0, ref __binding_1)
                        => {
                        {
                            match ::rustc_middle::ty::VisitorResult::branch(::rustc_middle::ty::TypeVisitable::visit_with(__binding_0,
                                        __visitor)) {
                                ::core::ops::ControlFlow::Continue(()) => {}
                                ::core::ops::ControlFlow::Break(r) => {
                                    return ::rustc_middle::ty::VisitorResult::from_residual(r);
                                }
                            }
                        }
                        {
                            match ::rustc_middle::ty::VisitorResult::branch(::rustc_middle::ty::TypeVisitable::visit_with(__binding_1,
                                        __visitor)) {
                                ::core::ops::ControlFlow::Continue(()) => {}
                                ::core::ops::ControlFlow::Break(r) => {
                                    return ::rustc_middle::ty::VisitorResult::from_residual(r);
                                }
                            }
                        }
                    }
                    StmtDebugInfo::InvalidAssign(ref __binding_0) => {
                        {
                            match ::rustc_middle::ty::VisitorResult::branch(::rustc_middle::ty::TypeVisitable::visit_with(__binding_0,
                                        __visitor)) {
                                ::core::ops::ControlFlow::Continue(()) => {}
                                ::core::ops::ControlFlow::Break(r) => {
                                    return ::rustc_middle::ty::VisitorResult::from_residual(r);
                                }
                            }
                        }
                    }
                }
                <__V::Result as ::rustc_middle::ty::VisitorResult>::output()
            }
        }
    };TypeVisitable)]
1099pub enum StmtDebugInfo<'tcx> {
1100    AssignRef(Local, Place<'tcx>),
1101    InvalidAssign(Local),
1102}