Skip to main content

rustc_type_ir/
binder.rs

1use std::fmt;
2use std::hash::Hash;
3use std::marker::PhantomData;
4use std::ops::{ControlFlow, Deref};
5
6use derive_where::derive_where;
7#[cfg(feature = "nightly")]
8use rustc_macros::{Decodable_NoContext, Encodable_NoContext, StableHash, StableHash_NoContext};
9use rustc_type_ir_macros::{
10    GenericTypeVisitable, Lift_Generic, TypeFoldable_Generic, TypeVisitable_Generic,
11};
12use tracing::instrument;
13
14use crate::data_structures::SsoHashSet;
15use crate::fold::{FallibleTypeFolder, TypeFoldable, TypeFolder, TypeSuperFoldable};
16use crate::inherent::*;
17use crate::visit::{Flags, TypeSuperVisitable, TypeVisitable, TypeVisitableExt, TypeVisitor};
18use crate::{self as ty, DebruijnIndex, Interner, UniverseIndex, Unnormalized};
19
20/// `Binder` is a binder for higher-ranked lifetimes or types. It is part of the
21/// compiler's representation for things like `for<'a> Fn(&'a isize)`
22/// (which would be represented by the type `PolyTraitRef == Binder<I, TraitRef>`).
23///
24/// See <https://rustc-dev-guide.rust-lang.org/ty_module/instantiating_binders.html>
25/// for more details.
26///
27/// `Decodable` and `Encodable` are implemented for `Binder<T>` using the `impl_binder_encode_decode!` macro.
28#[automatically_derived]
impl<I: Interner, T> ::core::fmt::Debug for Binder<I, T> where I: Interner,
    T: ::core::fmt::Debug {
    fn fmt(&self, __f: &mut ::core::fmt::Formatter<'_>)
        -> ::core::fmt::Result {
        match self {
            Binder {
                value: ref __field_value, bound_vars: ref __field_bound_vars }
                => {
                let mut __builder =
                    ::core::fmt::Formatter::debug_struct(__f, "Binder");
                ::core::fmt::DebugStruct::field(&mut __builder, "value",
                    __field_value);
                ::core::fmt::DebugStruct::field(&mut __builder, "bound_vars",
                    __field_bound_vars);
                ::core::fmt::DebugStruct::finish(&mut __builder)
            }
        }
    }
}#[derive_where(Clone, Copy, Hash, PartialEq, Debug; I: Interner, T)]
29#[derive(GenericTypeVisitable, const _: () =
    {
        impl<I: Interner, T, J> ::rustc_type_ir::lift::Lift<J> for
            Binder<I, T> where J: Interner, I: ::rustc_type_ir::LiftInto<J>,
            T: ::rustc_type_ir::lift::Lift<J> {
            type Lifted =
                Binder<J, <T as ::rustc_type_ir::lift::Lift<J>>::Lifted>;
            fn lift_to_interner(self, interner: J) -> Self::Lifted {
                match self {
                    Binder { value: __binding_0, bound_vars: __binding_1 } => {
                        Binder {
                            value: __binding_0.lift_to_interner(interner),
                            bound_vars: __binding_1.lift_to_interner(interner),
                        }
                    }
                }
            }
        }
    };Lift_Generic)]
30#[cfg_attr(feature = "nightly", derive(const _: () =
    {
        impl<I: Interner, T> ::rustc_data_structures::stable_hash::StableHash
            for Binder<I, T> where
            T: ::rustc_data_structures::stable_hash::StableHash,
            I::BoundVarKinds: ::rustc_data_structures::stable_hash::StableHash
            {
            #[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 {
                    Binder { value: ref __binding_0, bound_vars: ref __binding_1
                        } => {
                        { __binding_0.stable_hash(__hcx, __hasher); }
                        { __binding_1.stable_hash(__hcx, __hasher); }
                    }
                }
            }
        }
    };StableHash_NoContext))]
31pub struct Binder<I: Interner, T> {
32    value: T,
33    bound_vars: I::BoundVarKinds,
34}
35
36impl<I: Interner, T: Eq> Eq for Binder<I, T> {}
37
38#[cfg(feature = "nightly")]
39macro_rules! impl_binder_encode_decode {
40    ($($t:ty),+ $(,)?) => {
41        $(
42            impl<I: Interner, E: rustc_serialize::Encoder> rustc_serialize::Encodable<E> for ty::Binder<I, $t>
43            where
44                $t: rustc_serialize::Encodable<E>,
45                I::BoundVarKinds: rustc_serialize::Encodable<E>,
46            {
47                fn encode(&self, e: &mut E) {
48                    self.bound_vars().encode(e);
49                    self.as_ref().skip_binder().encode(e);
50                }
51            }
52            impl<I: Interner, D: rustc_serialize::Decoder> rustc_serialize::Decodable<D> for ty::Binder<I, $t>
53            where
54                $t: TypeVisitable<I> + rustc_serialize::Decodable<D>,
55                I::BoundVarKinds: rustc_serialize::Decodable<D>,
56            {
57                fn decode(decoder: &mut D) -> Self {
58                    let bound_vars = rustc_serialize::Decodable::decode(decoder);
59                    ty::Binder::bind_with_vars(rustc_serialize::Decodable::decode(decoder), bound_vars)
60                }
61            }
62        )*
63    }
64}
65
66#[cfg(feature = "nightly")]
67impl<I: Interner, E: rustc_serialize::Encoder> rustc_serialize::Encodable<E>
    for ty::Binder<I, ty::HostEffectPredicate<I>> where
    ty::HostEffectPredicate<I>: rustc_serialize::Encodable<E>,
    I::BoundVarKinds: rustc_serialize::Encodable<E> {
    fn encode(&self, e: &mut E) {
        self.bound_vars().encode(e);
        self.as_ref().skip_binder().encode(e);
    }
}
impl<I: Interner, D: rustc_serialize::Decoder> rustc_serialize::Decodable<D>
    for ty::Binder<I, ty::HostEffectPredicate<I>> where
    ty::HostEffectPredicate<I>: TypeVisitable<I> +
    rustc_serialize::Decodable<D>,
    I::BoundVarKinds: rustc_serialize::Decodable<D> {
    fn decode(decoder: &mut D) -> Self {
        let bound_vars = rustc_serialize::Decodable::decode(decoder);
        ty::Binder::bind_with_vars(rustc_serialize::Decodable::decode(decoder),
            bound_vars)
    }
}impl_binder_encode_decode! {
68    ty::FnSig<I>,
69    ty::FnSigTys<I>,
70    ty::TraitPredicate<I>,
71    ty::ExistentialPredicate<I>,
72    ty::TraitRef<I>,
73    ty::ExistentialTraitRef<I>,
74    ty::HostEffectPredicate<I>,
75}
76
77impl<I: Interner, T> Binder<I, T>
78where
79    T: TypeVisitable<I>,
80{
81    /// Wraps `value` in a binder, asserting that `value` does not
82    /// contain any bound vars that would be bound by the
83    /// binder. This is commonly used to 'inject' a value T into a
84    /// different binding level.
85    #[track_caller]
86    pub fn dummy(value: T) -> Binder<I, T> {
87        if !!value.has_escaping_bound_vars() {
    {
        ::core::panicking::panic_fmt(format_args!("`{0:?}` has escaping bound vars, so it cannot be wrapped in a dummy binder.",
                value));
    }
};assert!(
88            !value.has_escaping_bound_vars(),
89            "`{value:?}` has escaping bound vars, so it cannot be wrapped in a dummy binder."
90        );
91        Binder { value, bound_vars: Default::default() }
92    }
93
94    pub fn bind_with_vars(value: T, bound_vars: I::BoundVarKinds) -> Binder<I, T> {
95        if truecfg!(debug_assertions) {
96            let mut validator = ValidateBoundVars::new(bound_vars);
97            let _ = value.visit_with(&mut validator);
98        }
99        Binder { value, bound_vars }
100    }
101}
102
103impl<I: Interner, T: TypeFoldable<I>> TypeFoldable<I> for Binder<I, T> {
104    fn try_fold_with<F: FallibleTypeFolder<I>>(self, folder: &mut F) -> Result<Self, F::Error> {
105        folder.try_fold_binder(self)
106    }
107
108    fn fold_with<F: TypeFolder<I>>(self, folder: &mut F) -> Self {
109        folder.fold_binder(self)
110    }
111}
112
113impl<I: Interner, T: TypeVisitable<I>> TypeVisitable<I> for Binder<I, T> {
114    fn visit_with<V: TypeVisitor<I>>(&self, visitor: &mut V) -> V::Result {
115        visitor.visit_binder(self)
116    }
117}
118
119impl<I: Interner, T: TypeFoldable<I>> TypeSuperFoldable<I> for Binder<I, T> {
120    fn try_super_fold_with<F: FallibleTypeFolder<I>>(
121        self,
122        folder: &mut F,
123    ) -> Result<Self, F::Error> {
124        self.try_map_bound(|t| t.try_fold_with(folder))
125    }
126
127    fn super_fold_with<F: TypeFolder<I>>(self, folder: &mut F) -> Self {
128        self.map_bound(|t| t.fold_with(folder))
129    }
130}
131
132impl<I: Interner, T: TypeVisitable<I>> TypeSuperVisitable<I> for Binder<I, T> {
133    fn super_visit_with<V: TypeVisitor<I>>(&self, visitor: &mut V) -> V::Result {
134        self.as_ref().skip_binder().visit_with(visitor)
135    }
136}
137
138impl<I: Interner, T> Binder<I, T> {
139    /// Returns the value contained inside of this `for<'a>`. Accessing generic args
140    /// in the returned value is generally incorrect.
141    ///
142    /// Please read <https://rustc-dev-guide.rust-lang.org/ty_module/instantiating_binders.html>
143    /// before using this function. It is usually better to discharge the binder using
144    /// `no_bound_vars` or `instantiate_bound_regions` or something like that.
145    ///
146    /// `skip_binder` is only valid when you are either extracting data that does not reference
147    /// any generic arguments, e.g. a `DefId`, or when you're making sure you only pass the
148    /// value to things which can handle escaping bound vars.
149    ///
150    /// See existing uses of `.skip_binder()` in `rustc_trait_selection::traits::select`
151    /// or `rustc_next_trait_solver` for examples.
152    pub fn skip_binder(self) -> T {
153        self.value
154    }
155
156    pub fn bound_vars(&self) -> I::BoundVarKinds {
157        self.bound_vars
158    }
159
160    pub fn as_ref(&self) -> Binder<I, &T> {
161        Binder { value: &self.value, bound_vars: self.bound_vars }
162    }
163
164    pub fn as_deref(&self) -> Binder<I, &T::Target>
165    where
166        T: Deref,
167    {
168        Binder { value: &self.value, bound_vars: self.bound_vars }
169    }
170
171    pub fn map_bound_ref<F, U: TypeVisitable<I>>(&self, f: F) -> Binder<I, U>
172    where
173        F: FnOnce(&T) -> U,
174    {
175        self.as_ref().map_bound(f)
176    }
177
178    pub fn map_bound<F, U: TypeVisitable<I>>(self, f: F) -> Binder<I, U>
179    where
180        F: FnOnce(T) -> U,
181    {
182        let Binder { value, bound_vars } = self;
183        let value = f(value);
184        if truecfg!(debug_assertions) {
185            let mut validator = ValidateBoundVars::new(bound_vars);
186            let _ = value.visit_with(&mut validator);
187        }
188        Binder { value, bound_vars }
189    }
190
191    pub fn try_map_bound<F, U: TypeVisitable<I>, E>(self, f: F) -> Result<Binder<I, U>, E>
192    where
193        F: FnOnce(T) -> Result<U, E>,
194    {
195        let Binder { value, bound_vars } = self;
196        let value = f(value)?;
197        if truecfg!(debug_assertions) {
198            let mut validator = ValidateBoundVars::new(bound_vars);
199            let _ = value.visit_with(&mut validator);
200        }
201        Ok(Binder { value, bound_vars })
202    }
203
204    /// Wraps a `value` in a binder, using the same bound variables as the
205    /// current `Binder`. This should not be used if the new value *changes*
206    /// the bound variables. Note: the (old or new) value itself does not
207    /// necessarily need to *name* all the bound variables.
208    ///
209    /// This currently doesn't do anything different than `bind`, because we
210    /// don't actually track bound vars. However, semantically, it is different
211    /// because bound vars aren't allowed to change here, whereas they are
212    /// in `bind`. This may be (debug) asserted in the future.
213    pub fn rebind<U>(&self, value: U) -> Binder<I, U>
214    where
215        U: TypeVisitable<I>,
216    {
217        Binder::bind_with_vars(value, self.bound_vars)
218    }
219
220    /// Unwraps and returns the value within, but only if it contains
221    /// no bound vars at all. (In other words, if this binder --
222    /// and indeed any enclosing binder -- doesn't bind anything at
223    /// all.) Otherwise, returns `None`.
224    ///
225    /// (One could imagine having a method that just unwraps a single
226    /// binder, but permits late-bound vars bound by enclosing
227    /// binders, but that would require adjusting the debruijn
228    /// indices, and given the shallow binding structure we often use,
229    /// would not be that useful.)
230    pub fn no_bound_vars(self) -> Option<T>
231    where
232        T: TypeVisitable<I>,
233    {
234        // `self.value` is equivalent to `self.skip_binder()`
235        if self.value.has_escaping_bound_vars() { None } else { Some(self.skip_binder()) }
236    }
237}
238
239impl<I: Interner, T> Binder<I, Option<T>> {
240    pub fn transpose(self) -> Option<Binder<I, T>> {
241        let Binder { value, bound_vars } = self;
242        value.map(|value| Binder { value, bound_vars })
243    }
244}
245
246impl<I: Interner, T: IntoIterator> Binder<I, T> {
247    pub fn iter(self) -> impl Iterator<Item = Binder<I, T::Item>> {
248        let Binder { value, bound_vars } = self;
249        value.into_iter().map(move |value| Binder { value, bound_vars })
250    }
251}
252
253pub struct ValidateBoundVars<I: Interner> {
254    bound_vars: I::BoundVarKinds,
255    binder_index: ty::DebruijnIndex,
256    // We only cache types because any complex const will have to step through
257    // a type at some point anyways. We may encounter the same variable at
258    // different levels of binding, so this can't just be `Ty`.
259    visited: SsoHashSet<(ty::DebruijnIndex, I::Ty)>,
260}
261
262impl<I: Interner> ValidateBoundVars<I> {
263    pub fn new(bound_vars: I::BoundVarKinds) -> Self {
264        ValidateBoundVars {
265            bound_vars,
266            binder_index: ty::INNERMOST,
267            visited: SsoHashSet::default(),
268        }
269    }
270}
271
272impl<I: Interner> TypeVisitor<I> for ValidateBoundVars<I> {
273    type Result = ControlFlow<()>;
274
275    fn visit_binder<T: TypeVisitable<I>>(&mut self, t: &Binder<I, T>) -> Self::Result {
276        self.binder_index.shift_in(1);
277        let result = t.super_visit_with(self);
278        self.binder_index.shift_out(1);
279        result
280    }
281
282    fn visit_ty(&mut self, t: I::Ty) -> Self::Result {
283        if t.outer_exclusive_binder() < self.binder_index
284            || !self.visited.insert((self.binder_index, t))
285        {
286            return ControlFlow::Break(());
287        }
288        match t.kind() {
289            ty::Bound(ty::BoundVarIndexKind::Bound(debruijn), bound_ty)
290                if debruijn == self.binder_index =>
291            {
292                let idx = bound_ty.var().as_usize();
293                if self.bound_vars.len() <= idx {
294                    {
    ::core::panicking::panic_fmt(format_args!("Not enough bound vars: {0:?} not found in {1:?}",
            t, self.bound_vars));
};panic!("Not enough bound vars: {:?} not found in {:?}", t, self.bound_vars);
295                }
296                bound_ty.assert_eq(self.bound_vars.get(idx).unwrap());
297            }
298            _ => {}
299        };
300
301        t.super_visit_with(self)
302    }
303
304    fn visit_const(&mut self, c: I::Const) -> Self::Result {
305        if c.outer_exclusive_binder() < self.binder_index {
306            return ControlFlow::Break(());
307        }
308        match c.kind() {
309            ty::ConstKind::Bound(debruijn, bound_const)
310                if debruijn == ty::BoundVarIndexKind::Bound(self.binder_index) =>
311            {
312                let idx = bound_const.var().as_usize();
313                if self.bound_vars.len() <= idx {
314                    {
    ::core::panicking::panic_fmt(format_args!("Not enough bound vars: {0:?} not found in {1:?}",
            c, self.bound_vars));
};panic!("Not enough bound vars: {:?} not found in {:?}", c, self.bound_vars);
315                }
316                bound_const.assert_eq(self.bound_vars.get(idx).unwrap());
317            }
318            _ => {}
319        };
320
321        c.super_visit_with(self)
322    }
323
324    fn visit_region(&mut self, r: I::Region) -> Self::Result {
325        match r.kind() {
326            ty::ReBound(index, br) if index == ty::BoundVarIndexKind::Bound(self.binder_index) => {
327                let idx = br.var().as_usize();
328                if self.bound_vars.len() <= idx {
329                    {
    ::core::panicking::panic_fmt(format_args!("Not enough bound vars: {0:?} not found in {1:?}",
            r, self.bound_vars));
};panic!("Not enough bound vars: {:?} not found in {:?}", r, self.bound_vars);
330                }
331                br.assert_eq(self.bound_vars.get(idx).unwrap());
332            }
333
334            _ => (),
335        };
336
337        ControlFlow::Continue(())
338    }
339}
340
341/// Similar to [`Binder`] except that it tracks early bound generics, i.e. `struct Foo<T>(T)`
342/// needs `T` instantiated immediately. This type primarily exists to avoid forgetting to call
343/// `instantiate`.
344///
345/// See <https://rustc-dev-guide.rust-lang.org/ty_module/early_binder.html> for more details.
346#[automatically_derived]
impl<I: Interner, T> ::core::fmt::Debug for EarlyBinder<I, T> where
    I: Interner, T: ::core::fmt::Debug {
    fn fmt(&self, __f: &mut ::core::fmt::Formatter<'_>)
        -> ::core::fmt::Result {
        match self {
            EarlyBinder { value: ref __field_value, _tcx: ref __field__tcx }
                => {
                let mut __builder =
                    ::core::fmt::Formatter::debug_struct(__f, "EarlyBinder");
                ::core::fmt::DebugStruct::field(&mut __builder, "value",
                    __field_value);
                ::core::fmt::DebugStruct::finish_non_exhaustive(&mut __builder)
            }
        }
    }
}#[derive_where(Clone, Copy, PartialOrd, Ord, PartialEq, Hash, Debug; I: Interner, T)]
347#[derive(GenericTypeVisitable)]
348#[cfg_attr(
349    feature = "nightly",
350    derive(const _: () =
    {
        impl<I: Interner, T, __E: ::rustc_serialize::Encoder>
            ::rustc_serialize::Encodable<__E> for EarlyBinder<I, T> where
            T: ::rustc_serialize::Encodable<__E>,
            PhantomData<fn() -> I>: ::rustc_serialize::Encodable<__E> {
            fn encode(&self, __encoder: &mut __E) {
                match *self {
                    EarlyBinder { value: ref __binding_0, _tcx: ref __binding_1
                        } => {
                        ::rustc_serialize::Encodable::<__E>::encode(__binding_0,
                            __encoder);
                        ::rustc_serialize::Encodable::<__E>::encode(__binding_1,
                            __encoder);
                    }
                }
            }
        }
    };Encodable_NoContext, const _: () =
    {
        impl<I: Interner, T, __D: ::rustc_serialize::Decoder>
            ::rustc_serialize::Decodable<__D> for EarlyBinder<I, T> where
            T: ::rustc_serialize::Decodable<__D>,
            PhantomData<fn() -> I>: ::rustc_serialize::Decodable<__D> {
            fn decode(__decoder: &mut __D) -> Self {
                EarlyBinder {
                    value: ::rustc_serialize::Decodable::decode(__decoder),
                    _tcx: ::rustc_serialize::Decodable::decode(__decoder),
                }
            }
        }
    };Decodable_NoContext, const _: () =
    {
        impl<I: Interner, T> ::rustc_data_structures::stable_hash::StableHash
            for EarlyBinder<I, T> where
            T: ::rustc_data_structures::stable_hash::StableHash,
            PhantomData<fn()
                -> I>: ::rustc_data_structures::stable_hash::StableHash {
            #[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 {
                    EarlyBinder { value: ref __binding_0, _tcx: ref __binding_1
                        } => {
                        { __binding_0.stable_hash(__hcx, __hasher); }
                        { __binding_1.stable_hash(__hcx, __hasher); }
                    }
                }
            }
        }
    };StableHash_NoContext)
351)]
352pub struct EarlyBinder<I: Interner, T> {
353    value: T,
354    #[derive_where(skip(Debug))]
355    _tcx: PhantomData<fn() -> I>,
356}
357
358impl<I: Interner, T: Eq> Eq for EarlyBinder<I, T> {}
359
360// FIXME(154045): Recommended as per https://github.com/rust-lang/rust/issues/154045, this is so sad :((
361#[cfg(feature = "nightly")]
362macro_rules! generate { ($( $tt:tt )*) => { $( $tt )* } }
363
364#[cfg(feature = "nightly")]
365generate!(
366    /// For early binders, you should first call `instantiate` before using any visitors.
367    impl<I: Interner, T> !TypeFoldable<I> for ty::EarlyBinder<I, T> {}
368    /// For early binders, you should first call `instantiate` before using any visitors.
369    impl<I: Interner, T> !TypeVisitable<I> for ty::EarlyBinder<I, T> {}
370);
371
372impl<I: Interner, T> EarlyBinder<I, T> {
373    pub fn bind(value: T) -> EarlyBinder<I, T> {
374        EarlyBinder { value, _tcx: PhantomData }
375    }
376
377    pub fn as_ref(&self) -> EarlyBinder<I, &T> {
378        EarlyBinder { value: &self.value, _tcx: PhantomData }
379    }
380
381    pub fn map_bound_ref<F, U>(&self, f: F) -> EarlyBinder<I, U>
382    where
383        F: FnOnce(&T) -> U,
384    {
385        self.as_ref().map_bound(f)
386    }
387
388    pub fn map_bound<F, U>(self, f: F) -> EarlyBinder<I, U>
389    where
390        F: FnOnce(T) -> U,
391    {
392        let value = f(self.value);
393        EarlyBinder { value, _tcx: PhantomData }
394    }
395
396    pub fn try_map_bound<F, U, E>(self, f: F) -> Result<EarlyBinder<I, U>, E>
397    where
398        F: FnOnce(T) -> Result<U, E>,
399    {
400        let value = f(self.value)?;
401        Ok(EarlyBinder { value, _tcx: PhantomData })
402    }
403
404    pub fn rebind<U>(&self, value: U) -> EarlyBinder<I, U> {
405        EarlyBinder { value, _tcx: PhantomData }
406    }
407
408    /// Skips the binder and returns the "bound" value. Accessing generic args
409    /// in the returned value is generally incorrect.
410    ///
411    /// Please read <https://rustc-dev-guide.rust-lang.org/ty_module/early_binder.html>
412    /// before using this function.
413    ///
414    /// Only use this to extract data that does not depend on generic parameters, e.g.
415    /// to get the `DefId` of the inner value or the number of arguments ofan `FnSig`,
416    /// or while making sure to only pass the value to functions which are explicitly
417    /// set up to handle these uninstantiated generic parameters.
418    ///
419    /// To skip the binder on `x: &EarlyBinder<I, T>` to obtain `&T`, leverage
420    /// [`EarlyBinder::as_ref`](EarlyBinder::as_ref): `x.as_ref().skip_binder()`.
421    ///
422    /// See also [`Binder::skip_binder`](Binder::skip_binder), which is
423    /// the analogous operation on [`Binder`].
424    pub fn skip_binder(self) -> T {
425        self.value
426    }
427}
428
429impl<I: Interner, T> EarlyBinder<I, Option<T>> {
430    pub fn transpose(self) -> Option<EarlyBinder<I, T>> {
431        self.value.map(|value| EarlyBinder { value, _tcx: PhantomData })
432    }
433}
434
435impl<I: Interner, Iter: IntoIterator> EarlyBinder<I, Iter>
436where
437    Iter::Item: TypeFoldable<I>,
438{
439    pub fn iter_instantiated<A>(self, cx: I, args: A) -> IterInstantiated<I, Iter, A>
440    where
441        A: SliceLike<Item = I::GenericArg>,
442    {
443        IterInstantiated { it: self.value.into_iter(), cx, args }
444    }
445
446    /// Similar to [`instantiate_identity`](EarlyBinder::instantiate_identity),
447    /// but on an iterator of `TypeFoldable` values.
448    pub fn iter_identity(self) -> impl Iterator<Item = Unnormalized<I, Iter::Item>> {
449        self.value.into_iter().map(Unnormalized::new)
450    }
451}
452
453pub struct IterInstantiated<I: Interner, Iter: IntoIterator, A> {
454    it: Iter::IntoIter,
455    cx: I,
456    args: A,
457}
458
459impl<I: Interner, Iter: IntoIterator, A> Iterator for IterInstantiated<I, Iter, A>
460where
461    Iter::Item: TypeFoldable<I>,
462    A: SliceLike<Item = I::GenericArg>,
463{
464    type Item = Unnormalized<I, Iter::Item>;
465
466    fn next(&mut self) -> Option<Self::Item> {
467        Some(
468            EarlyBinder { value: self.it.next()?, _tcx: PhantomData }
469                .instantiate(self.cx, self.args),
470        )
471    }
472
473    fn size_hint(&self) -> (usize, Option<usize>) {
474        self.it.size_hint()
475    }
476}
477
478impl<I: Interner, Iter: IntoIterator, A> DoubleEndedIterator for IterInstantiated<I, Iter, A>
479where
480    Iter::IntoIter: DoubleEndedIterator,
481    Iter::Item: TypeFoldable<I>,
482    A: SliceLike<Item = I::GenericArg>,
483{
484    fn next_back(&mut self) -> Option<Self::Item> {
485        Some(
486            EarlyBinder { value: self.it.next_back()?, _tcx: PhantomData }
487                .instantiate(self.cx, self.args),
488        )
489    }
490}
491
492impl<I: Interner, Iter: IntoIterator, A> ExactSizeIterator for IterInstantiated<I, Iter, A>
493where
494    Iter::IntoIter: ExactSizeIterator,
495    Iter::Item: TypeFoldable<I>,
496    A: SliceLike<Item = I::GenericArg>,
497{
498}
499
500impl<'s, I: Interner, Iter: IntoIterator> EarlyBinder<I, Iter>
501where
502    Iter::Item: Deref,
503    <Iter::Item as Deref>::Target: Copy + TypeFoldable<I>,
504{
505    pub fn iter_instantiated_copied(
506        self,
507        cx: I,
508        args: &'s [I::GenericArg],
509    ) -> IterInstantiatedCopied<'s, I, Iter> {
510        IterInstantiatedCopied { it: self.value.into_iter(), cx, args }
511    }
512
513    /// Similar to [`instantiate_identity`](EarlyBinder::instantiate_identity),
514    /// but on an iterator of values that deref to a `TypeFoldable`.
515    pub fn iter_identity_copied(self) -> IterIdentityCopied<I, Iter> {
516        IterIdentityCopied { it: self.value.into_iter(), _tcx: PhantomData }
517    }
518}
519
520pub struct IterInstantiatedCopied<'a, I: Interner, Iter: IntoIterator> {
521    it: Iter::IntoIter,
522    cx: I,
523    args: &'a [I::GenericArg],
524}
525
526impl<I: Interner, Iter: IntoIterator> Iterator for IterInstantiatedCopied<'_, I, Iter>
527where
528    Iter::Item: Deref,
529    <Iter::Item as Deref>::Target: Copy + TypeFoldable<I>,
530{
531    type Item = Unnormalized<I, <Iter::Item as Deref>::Target>;
532
533    fn next(&mut self) -> Option<Self::Item> {
534        self.it.next().map(|value| {
535            EarlyBinder { value: *value, _tcx: PhantomData }.instantiate(self.cx, self.args)
536        })
537    }
538
539    fn size_hint(&self) -> (usize, Option<usize>) {
540        self.it.size_hint()
541    }
542}
543
544impl<I: Interner, Iter: IntoIterator> DoubleEndedIterator for IterInstantiatedCopied<'_, I, Iter>
545where
546    Iter::IntoIter: DoubleEndedIterator,
547    Iter::Item: Deref,
548    <Iter::Item as Deref>::Target: Copy + TypeFoldable<I>,
549{
550    fn next_back(&mut self) -> Option<Self::Item> {
551        self.it.next_back().map(|value| {
552            EarlyBinder { value: *value, _tcx: PhantomData }.instantiate(self.cx, self.args)
553        })
554    }
555}
556
557impl<I: Interner, Iter: IntoIterator> ExactSizeIterator for IterInstantiatedCopied<'_, I, Iter>
558where
559    Iter::IntoIter: ExactSizeIterator,
560    Iter::Item: Deref,
561    <Iter::Item as Deref>::Target: Copy + TypeFoldable<I>,
562{
563}
564
565pub struct IterIdentityCopied<I: Interner, Iter: IntoIterator> {
566    it: Iter::IntoIter,
567    _tcx: PhantomData<fn() -> I>,
568}
569
570impl<I: Interner, Iter: IntoIterator> Iterator for IterIdentityCopied<I, Iter>
571where
572    Iter::Item: Deref,
573    <Iter::Item as Deref>::Target: Copy,
574{
575    type Item = Unnormalized<I, <Iter::Item as Deref>::Target>;
576
577    fn next(&mut self) -> Option<Self::Item> {
578        self.it.next().map(|i| Unnormalized::new(*i))
579    }
580
581    fn size_hint(&self) -> (usize, Option<usize>) {
582        self.it.size_hint()
583    }
584}
585
586impl<I: Interner, Iter: IntoIterator> DoubleEndedIterator for IterIdentityCopied<I, Iter>
587where
588    Iter::IntoIter: DoubleEndedIterator,
589    Iter::Item: Deref,
590    <Iter::Item as Deref>::Target: Copy,
591{
592    fn next_back(&mut self) -> Option<Self::Item> {
593        self.it.next_back().map(|i| Unnormalized::new(*i))
594    }
595}
596
597impl<I: Interner, Iter: IntoIterator> ExactSizeIterator for IterIdentityCopied<I, Iter>
598where
599    Iter::IntoIter: ExactSizeIterator,
600    Iter::Item: Deref,
601    <Iter::Item as Deref>::Target: Copy,
602{
603}
604pub struct EarlyBinderIter<I, T> {
605    t: T,
606    _tcx: PhantomData<I>,
607}
608
609impl<I: Interner, T: IntoIterator> EarlyBinder<I, T> {
610    pub fn transpose_iter(self) -> EarlyBinderIter<I, T::IntoIter> {
611        EarlyBinderIter { t: self.value.into_iter(), _tcx: PhantomData }
612    }
613}
614
615impl<I: Interner, T: Iterator> Iterator for EarlyBinderIter<I, T> {
616    type Item = EarlyBinder<I, T::Item>;
617
618    fn next(&mut self) -> Option<Self::Item> {
619        self.t.next().map(|value| EarlyBinder { value, _tcx: PhantomData })
620    }
621
622    fn size_hint(&self) -> (usize, Option<usize>) {
623        self.t.size_hint()
624    }
625}
626
627impl<I: Interner, T: TypeFoldable<I>> ty::EarlyBinder<I, T> {
628    pub fn instantiate<A>(self, cx: I, args: A) -> Unnormalized<I, T>
629    where
630        A: SliceLike<Item = I::GenericArg>,
631    {
632        // Nothing to fold, so let's avoid visiting things and possibly re-hashing/equating
633        // them when interning. Perf testing found this to be a modest improvement.
634        // See: <https://github.com/rust-lang/rust/pull/142317>
635        if args.is_empty() {
636            if !!self.value.has_param() {
    {
        ::core::panicking::panic_fmt(format_args!("{0:?} has parameters, but no args were provided in instantiate",
                self.value));
    }
};assert!(
637                !self.value.has_param(),
638                "{:?} has parameters, but no args were provided in instantiate",
639                self.value,
640            );
641            return Unnormalized::new(self.value);
642        }
643        let mut folder = ArgFolder { cx, args: args.as_slice(), binders_passed: 0 };
644        Unnormalized::new(self.value.fold_with(&mut folder))
645    }
646
647    /// Makes the identity replacement `T0 => T0, ..., TN => TN`.
648    /// Conceptually, this converts universally bound variables into placeholders
649    /// when inside of a given item.
650    ///
651    /// For example, consider `for<T> fn foo<T>(){ .. }`:
652    /// - Outside of `foo`, `T` is bound (represented by the presence of `EarlyBinder`).
653    /// - Inside of the body of `foo`, we treat `T` as a placeholder by calling
654    /// `instantiate_identity` to discharge the `EarlyBinder`.
655    pub fn instantiate_identity(self) -> Unnormalized<I, T> {
656        // FIXME(#155345): In case the bound value was already normalized, this
657        // is unnecessary. We may want to track explicitly whether `EarlyBinder`
658        // contains something that has been normalized already.
659        // Also do that for other types who have `instantiate_identity` method,
660        // e.g., `GenericPredicates` and `ConstConditions`.
661        //
662        // This is annoying, as e.g. `type_of` for opaque types is normalized,
663        // while `type_of` for free type aliases is not.
664        Unnormalized::new(self.value)
665    }
666
667    /// Returns the inner value, but only if it contains no bound vars.
668    pub fn no_bound_vars(self) -> Option<T> {
669        if !self.value.has_param() { Some(self.value) } else { None }
670    }
671}
672
673///////////////////////////////////////////////////////////////////////////
674// The actual instantiation engine itself is a type folder.
675
676struct ArgFolder<'a, I: Interner> {
677    cx: I,
678    args: &'a [I::GenericArg],
679
680    /// Number of region binders we have passed through while doing the instantiation
681    binders_passed: u32,
682}
683
684impl<'a, I: Interner> TypeFolder<I> for ArgFolder<'a, I> {
685    #[inline]
686    fn cx(&self) -> I {
687        self.cx
688    }
689
690    fn fold_binder<T: TypeFoldable<I>>(&mut self, t: ty::Binder<I, T>) -> ty::Binder<I, T> {
691        self.binders_passed += 1;
692        let t = t.super_fold_with(self);
693        self.binders_passed -= 1;
694        t
695    }
696
697    fn fold_region(&mut self, r: I::Region) -> I::Region {
698        // Note: This routine only handles regions that are bound on
699        // type declarations and other outer declarations, not those
700        // bound in *fn types*. Region instantiation of the bound
701        // regions that appear in a function signature is done using
702        // the specialized routine `ty::replace_late_regions()`.
703        match r.kind() {
704            ty::ReEarlyParam(data) => {
705                let rk = self.args.get(data.index() as usize).map(|arg| arg.kind());
706                match rk {
707                    Some(ty::GenericArgKind::Lifetime(lt)) => self.shift_region_through_binders(lt),
708                    Some(other) => self.region_param_expected(data, r, other),
709                    None => self.region_param_out_of_range(data, r),
710                }
711            }
712            ty::ReBound(..)
713            | ty::ReLateParam(_)
714            | ty::ReStatic
715            | ty::RePlaceholder(_)
716            | ty::ReErased
717            | ty::ReError(_) => r,
718            ty::ReVar(_) => { ::core::panicking::panic_fmt(format_args!("unexpected region: {0:?}", r)); }panic!("unexpected region: {r:?}"),
719        }
720    }
721
722    fn fold_ty(&mut self, t: I::Ty) -> I::Ty {
723        if !t.has_param() {
724            return t;
725        }
726
727        match t.kind() {
728            ty::Param(p) => self.ty_for_param(p, t),
729            _ => t.super_fold_with(self),
730        }
731    }
732
733    fn fold_const(&mut self, c: I::Const) -> I::Const {
734        if let ty::ConstKind::Param(p) = c.kind() {
735            self.const_for_param(p, c)
736        } else {
737            c.super_fold_with(self)
738        }
739    }
740
741    fn fold_predicate(&mut self, p: I::Predicate) -> I::Predicate {
742        if p.has_param() { p.super_fold_with(self) } else { p }
743    }
744
745    fn fold_clauses(&mut self, c: I::Clauses) -> I::Clauses {
746        if c.has_param() { c.super_fold_with(self) } else { c }
747    }
748}
749
750impl<'a, I: Interner> ArgFolder<'a, I> {
751    fn ty_for_param(&self, p: I::ParamTy, source_ty: I::Ty) -> I::Ty {
752        // Look up the type in the args. It really should be in there.
753        let opt_ty = self.args.get(p.index() as usize).map(|arg| arg.kind());
754        let ty = match opt_ty {
755            Some(ty::GenericArgKind::Type(ty)) => ty,
756            Some(kind) => self.type_param_expected(p, source_ty, kind),
757            None => self.type_param_out_of_range(p, source_ty),
758        };
759
760        self.shift_vars_through_binders(ty)
761    }
762
763    #[cold]
764    #[inline(never)]
765    fn type_param_expected(&self, p: I::ParamTy, ty: I::Ty, kind: ty::GenericArgKind<I>) -> ! {
766        {
    ::core::panicking::panic_fmt(format_args!("expected type for `{0:?}` ({1:?}/{2}) but found {3:?} when instantiating, args={4:?}",
            p, ty, p.index(), kind, self.args));
}panic!(
767            "expected type for `{:?}` ({:?}/{}) but found {:?} when instantiating, args={:?}",
768            p,
769            ty,
770            p.index(),
771            kind,
772            self.args,
773        )
774    }
775
776    #[cold]
777    #[inline(never)]
778    fn type_param_out_of_range(&self, p: I::ParamTy, ty: I::Ty) -> ! {
779        {
    ::core::panicking::panic_fmt(format_args!("type parameter `{0:?}` ({1:?}/{2}) out of range when instantiating, args={3:?}",
            p, ty, p.index(), self.args));
}panic!(
780            "type parameter `{:?}` ({:?}/{}) out of range when instantiating, args={:?}",
781            p,
782            ty,
783            p.index(),
784            self.args,
785        )
786    }
787
788    fn const_for_param(&self, p: I::ParamConst, source_ct: I::Const) -> I::Const {
789        // Look up the const in the args. It really should be in there.
790        let opt_ct = self.args.get(p.index() as usize).map(|arg| arg.kind());
791        let ct = match opt_ct {
792            Some(ty::GenericArgKind::Const(ct)) => ct,
793            Some(kind) => self.const_param_expected(p, source_ct, kind),
794            None => self.const_param_out_of_range(p, source_ct),
795        };
796
797        self.shift_vars_through_binders(ct)
798    }
799
800    #[cold]
801    #[inline(never)]
802    fn const_param_expected(
803        &self,
804        p: I::ParamConst,
805        ct: I::Const,
806        kind: ty::GenericArgKind<I>,
807    ) -> ! {
808        {
    ::core::panicking::panic_fmt(format_args!("expected const for `{0:?}` ({1:?}/{2}) but found {3:?} when instantiating args={4:?}",
            p, ct, p.index(), kind, self.args));
}panic!(
809            "expected const for `{:?}` ({:?}/{}) but found {:?} when instantiating args={:?}",
810            p,
811            ct,
812            p.index(),
813            kind,
814            self.args,
815        )
816    }
817
818    #[cold]
819    #[inline(never)]
820    fn const_param_out_of_range(&self, p: I::ParamConst, ct: I::Const) -> ! {
821        {
    ::core::panicking::panic_fmt(format_args!("const parameter `{0:?}` ({1:?}/{2}) out of range when instantiating args={3:?}",
            p, ct, p.index(), self.args));
}panic!(
822            "const parameter `{:?}` ({:?}/{}) out of range when instantiating args={:?}",
823            p,
824            ct,
825            p.index(),
826            self.args,
827        )
828    }
829
830    #[cold]
831    #[inline(never)]
832    fn region_param_expected(
833        &self,
834        ebr: I::EarlyParamRegion,
835        r: I::Region,
836        kind: ty::GenericArgKind<I>,
837    ) -> ! {
838        {
    ::core::panicking::panic_fmt(format_args!("expected region for `{0:?}` ({1:?}/{2}) but found {3:?} when instantiating args={4:?}",
            ebr, r, ebr.index(), kind, self.args));
}panic!(
839            "expected region for `{:?}` ({:?}/{}) but found {:?} when instantiating args={:?}",
840            ebr,
841            r,
842            ebr.index(),
843            kind,
844            self.args,
845        )
846    }
847
848    #[cold]
849    #[inline(never)]
850    fn region_param_out_of_range(&self, ebr: I::EarlyParamRegion, r: I::Region) -> ! {
851        {
    ::core::panicking::panic_fmt(format_args!("region parameter `{0:?}` ({1:?}/{2}) out of range when instantiating args={3:?}",
            ebr, r, ebr.index(), self.args));
}panic!(
852            "region parameter `{:?}` ({:?}/{}) out of range when instantiating args={:?}",
853            ebr,
854            r,
855            ebr.index(),
856            self.args,
857        )
858    }
859
860    /// It is sometimes necessary to adjust the De Bruijn indices during instantiation. This occurs
861    /// when we are instantiating a type with escaping bound vars into a context where we have
862    /// passed through binders. That's quite a mouthful. Let's see an example:
863    ///
864    /// ```
865    /// type Func<A> = fn(A);
866    /// type MetaFunc = for<'a> fn(Func<&'a i32>);
867    /// ```
868    ///
869    /// The type `MetaFunc`, when fully expanded, will be
870    /// ```ignore (illustrative)
871    /// for<'a> fn(fn(&'a i32))
872    /// //      ^~ ^~ ^~~
873    /// //      |  |  |
874    /// //      |  |  DebruijnIndex of 2
875    /// //      Binders
876    /// ```
877    /// Here the `'a` lifetime is bound in the outer function, but appears as an argument of the
878    /// inner one. Therefore, that appearance will have a DebruijnIndex of 2, because we must skip
879    /// over the inner binder (remember that we count De Bruijn indices from 1). However, in the
880    /// definition of `MetaFunc`, the binder is not visible, so the type `&'a i32` will have a
881    /// De Bruijn index of 1. It's only during the instantiation that we can see we must increase the
882    /// depth by 1 to account for the binder that we passed through.
883    ///
884    /// As a second example, consider this twist:
885    ///
886    /// ```
887    /// type FuncTuple<A> = (A,fn(A));
888    /// type MetaFuncTuple = for<'a> fn(FuncTuple<&'a i32>);
889    /// ```
890    ///
891    /// Here the final type will be:
892    /// ```ignore (illustrative)
893    /// for<'a> fn((&'a i32, fn(&'a i32)))
894    /// //          ^~~         ^~~
895    /// //          |           |
896    /// //   DebruijnIndex of 1 |
897    /// //               DebruijnIndex of 2
898    /// ```
899    /// As indicated in the diagram, here the same type `&'a i32` is instantiated once, but in the
900    /// first case we do not increase the De Bruijn index and in the second case we do. The reason
901    /// is that only in the second case have we passed through a fn binder.
902    x;#[instrument(level = "trace", skip(self), fields(binders_passed = self.binders_passed), ret)]
903    fn shift_vars_through_binders<T: TypeFoldable<I>>(&self, val: T) -> T {
904        if self.binders_passed == 0 || !val.has_escaping_bound_vars() {
905            val
906        } else {
907            ty::shift_vars(self.cx, val, self.binders_passed)
908        }
909    }
910
911    fn shift_region_through_binders(&self, region: I::Region) -> I::Region {
912        if self.binders_passed == 0 || !region.has_escaping_bound_vars() {
913            region
914        } else {
915            ty::shift_region(self.cx, region, self.binders_passed)
916        }
917    }
918}
919
920/// Okay, we do something fun for `Bound` types/regions/consts:
921/// Specifically, we distinguish between *canonically* bound things and
922/// `for<>` bound things. And, really, it comes down to caching during
923/// canonicalization and instantiation.
924///
925/// To understand why we do this, imagine we have a type `(T, for<> fn(T))`.
926/// If we just tracked canonically bound types with a `DebruijnIndex` (as we
927/// used to), then the canonicalized type would be something like
928/// `for<0> (^0.0, for<> fn(^1.0))` and so we can't cache `T -> ^0.0`,
929/// we have to also factor in binder level. (Of course, we don't cache that
930/// exactly, but rather the entire enclosing type, but the point stands.)
931///
932/// Of course, this is okay because we don't ever nest canonicalization, so
933/// `BoundVarIndexKind::Canonical` is unambiguous. We, alternatively, could
934/// have some sentinel `DebruijinIndex`, but that just seems too scary.
935///
936/// This doesn't seem to have a huge perf swing either way, but in the next
937/// solver, canonicalization is hot and there are some pathological cases where
938/// this is needed (`post-mono-higher-ranked-hang`).
939#[derive(#[automatically_derived]
impl ::core::clone::Clone for BoundVarIndexKind {
    #[inline]
    fn clone(&self) -> BoundVarIndexKind {
        let _: ::core::clone::AssertParamIsClone<DebruijnIndex>;
        *self
    }
}Clone, #[automatically_derived]
impl ::core::marker::Copy for BoundVarIndexKind { }Copy, #[automatically_derived]
impl ::core::fmt::Debug for BoundVarIndexKind {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        match self {
            BoundVarIndexKind::Bound(__self_0) =>
                ::core::fmt::Formatter::debug_tuple_field1_finish(f, "Bound",
                    &__self_0),
            BoundVarIndexKind::Canonical =>
                ::core::fmt::Formatter::write_str(f, "Canonical"),
        }
    }
}Debug, #[automatically_derived]
impl ::core::cmp::PartialEq for BoundVarIndexKind {
    #[inline]
    fn eq(&self, other: &BoundVarIndexKind) -> bool {
        let __self_discr = ::core::intrinsics::discriminant_value(self);
        let __arg1_discr = ::core::intrinsics::discriminant_value(other);
        __self_discr == __arg1_discr &&
            match (self, other) {
                (BoundVarIndexKind::Bound(__self_0),
                    BoundVarIndexKind::Bound(__arg1_0)) => __self_0 == __arg1_0,
                _ => true,
            }
    }
}PartialEq, #[automatically_derived]
impl ::core::cmp::Eq for BoundVarIndexKind {
    #[inline]
    #[doc(hidden)]
    #[coverage(off)]
    fn assert_fields_are_eq(&self) {
        let _: ::core::cmp::AssertParamIsEq<DebruijnIndex>;
    }
}Eq, #[automatically_derived]
impl ::core::hash::Hash for BoundVarIndexKind {
    #[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 {
            BoundVarIndexKind::Bound(__self_0) =>
                ::core::hash::Hash::hash(__self_0, state),
            _ => {}
        }
    }
}Hash)]
940#[cfg_attr(feature = "nightly", derive(const _: () =
    {
        impl<__E: ::rustc_serialize::Encoder>
            ::rustc_serialize::Encodable<__E> for BoundVarIndexKind {
            fn encode(&self, __encoder: &mut __E) {
                let disc =
                    match *self {
                        BoundVarIndexKind::Bound(ref __binding_0) => { 0usize }
                        BoundVarIndexKind::Canonical => { 1usize }
                    };
                ::rustc_serialize::Encoder::emit_u8(__encoder, disc as u8);
                match *self {
                    BoundVarIndexKind::Bound(ref __binding_0) => {
                        ::rustc_serialize::Encodable::<__E>::encode(__binding_0,
                            __encoder);
                    }
                    BoundVarIndexKind::Canonical => {}
                }
            }
        }
    };Encodable_NoContext, const _: () =
    {
        impl<__D: ::rustc_serialize::Decoder>
            ::rustc_serialize::Decodable<__D> for BoundVarIndexKind {
            fn decode(__decoder: &mut __D) -> Self {
                match ::rustc_serialize::Decoder::read_u8(__decoder) as usize
                    {
                    0usize => {
                        BoundVarIndexKind::Bound(::rustc_serialize::Decodable::decode(__decoder))
                    }
                    1usize => { BoundVarIndexKind::Canonical }
                    n => {
                        ::core::panicking::panic_fmt(format_args!("invalid enum variant tag while decoding `BoundVarIndexKind`, expected 0..2, actual {0}",
                                n));
                    }
                }
            }
        }
    };Decodable_NoContext, const _: () =
    {
        impl ::rustc_data_structures::stable_hash::StableHash for
            BoundVarIndexKind {
            #[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 {
                    BoundVarIndexKind::Bound(ref __binding_0) => {
                        { __binding_0.stable_hash(__hcx, __hasher); }
                    }
                    BoundVarIndexKind::Canonical => {}
                }
            }
        }
    };StableHash))]
941#[derive(const _: () =
    {
        impl<I> ::rustc_type_ir::TypeVisitable<I> for BoundVarIndexKind where
            I: Interner {
            fn visit_with<__V: ::rustc_type_ir::TypeVisitor<I>>(&self,
                __visitor: &mut __V) -> __V::Result {
                match *self {
                    BoundVarIndexKind::Bound(ref __binding_0) => {
                        {
                            match ::rustc_type_ir::VisitorResult::branch(::rustc_type_ir::TypeVisitable::visit_with(__binding_0,
                                        __visitor)) {
                                ::core::ops::ControlFlow::Continue(()) => {}
                                ::core::ops::ControlFlow::Break(r) => {
                                    return ::rustc_type_ir::VisitorResult::from_residual(r);
                                }
                            }
                        }
                    }
                    BoundVarIndexKind::Canonical => {}
                }
                <__V::Result as ::rustc_type_ir::VisitorResult>::output()
            }
        }
    };TypeVisitable_Generic, GenericTypeVisitable, const _: () =
    {
        impl<I> ::rustc_type_ir::TypeFoldable<I> for BoundVarIndexKind where
            I: Interner {
            fn try_fold_with<__F: ::rustc_type_ir::FallibleTypeFolder<I>>(self,
                __folder: &mut __F) -> Result<Self, __F::Error> {
                Ok(match self {
                        BoundVarIndexKind::Bound(__binding_0) => {
                            BoundVarIndexKind::Bound(::rustc_type_ir::TypeFoldable::try_fold_with(__binding_0,
                                        __folder)?)
                        }
                        BoundVarIndexKind::Canonical => {
                            BoundVarIndexKind::Canonical
                        }
                    })
            }
            fn fold_with<__F: ::rustc_type_ir::TypeFolder<I>>(self,
                __folder: &mut __F) -> Self {
                match self {
                    BoundVarIndexKind::Bound(__binding_0) => {
                        BoundVarIndexKind::Bound(::rustc_type_ir::TypeFoldable::fold_with(__binding_0,
                                __folder))
                    }
                    BoundVarIndexKind::Canonical => {
                        BoundVarIndexKind::Canonical
                    }
                }
            }
        }
    };TypeFoldable_Generic)]
942pub enum BoundVarIndexKind {
943    Bound(DebruijnIndex),
944    Canonical,
945}
946
947/// The "placeholder index" fully defines a placeholder region, type, or const. Placeholders are
948/// identified by both a universe, as well as a name residing within that universe. Distinct bound
949/// regions/types/consts within the same universe simply have an unknown relationship to one
950#[automatically_derived]
impl<I: Interner, T> ::core::hash::Hash for Placeholder<I, T> where
    I: Interner, T: ::core::hash::Hash {
    fn hash<__H: ::core::hash::Hasher>(&self, __state: &mut __H) {
        match self {
            Placeholder {
                universe: ref __field_universe,
                bound: ref __field_bound,
                _tcx: ref __field__tcx } => {
                ::core::hash::Hash::hash(__field_universe, __state);
                ::core::hash::Hash::hash(__field_bound, __state);
                ::core::hash::Hash::hash(__field__tcx, __state);
            }
        }
    }
}#[derive_where(Clone, Copy, PartialOrd, Ord, PartialEq, Eq, Hash; I: Interner, T)]
951#[derive(const _: () =
    {
        impl<I: Interner, T> ::rustc_type_ir::TypeVisitable<I> for
            Placeholder<I, T> where I: Interner,
            T: ::rustc_type_ir::TypeVisitable<I> {
            fn visit_with<__V: ::rustc_type_ir::TypeVisitor<I>>(&self,
                __visitor: &mut __V) -> __V::Result {
                match *self {
                    Placeholder {
                        universe: ref __binding_0, bound: ref __binding_1, .. } => {
                        {
                            match ::rustc_type_ir::VisitorResult::branch(::rustc_type_ir::TypeVisitable::visit_with(__binding_0,
                                        __visitor)) {
                                ::core::ops::ControlFlow::Continue(()) => {}
                                ::core::ops::ControlFlow::Break(r) => {
                                    return ::rustc_type_ir::VisitorResult::from_residual(r);
                                }
                            }
                        }
                        {
                            match ::rustc_type_ir::VisitorResult::branch(::rustc_type_ir::TypeVisitable::visit_with(__binding_1,
                                        __visitor)) {
                                ::core::ops::ControlFlow::Continue(()) => {}
                                ::core::ops::ControlFlow::Break(r) => {
                                    return ::rustc_type_ir::VisitorResult::from_residual(r);
                                }
                            }
                        }
                    }
                }
                <__V::Result as ::rustc_type_ir::VisitorResult>::output()
            }
        }
    };TypeVisitable_Generic, const _: () =
    {
        impl<I: Interner, T> ::rustc_type_ir::TypeFoldable<I> for
            Placeholder<I, T> where I: Interner,
            T: ::rustc_type_ir::TypeFoldable<I>,
            T: ::rustc_type_ir::TypeFoldable<I> {
            fn try_fold_with<__F: ::rustc_type_ir::FallibleTypeFolder<I>>(self,
                __folder: &mut __F) -> Result<Self, __F::Error> {
                Ok(match self {
                        Placeholder {
                            universe: __binding_0, bound: __binding_1, _tcx: __binding_2
                            } => {
                            Placeholder {
                                universe: ::rustc_type_ir::TypeFoldable::try_fold_with(__binding_0,
                                        __folder)?,
                                bound: ::rustc_type_ir::TypeFoldable::try_fold_with(__binding_1,
                                        __folder)?,
                                _tcx: __binding_2,
                            }
                        }
                    })
            }
            fn fold_with<__F: ::rustc_type_ir::TypeFolder<I>>(self,
                __folder: &mut __F) -> Self {
                match self {
                    Placeholder {
                        universe: __binding_0, bound: __binding_1, _tcx: __binding_2
                        } => {
                        Placeholder {
                            universe: ::rustc_type_ir::TypeFoldable::fold_with(__binding_0,
                                __folder),
                            bound: ::rustc_type_ir::TypeFoldable::fold_with(__binding_1,
                                __folder),
                            _tcx: __binding_2,
                        }
                    }
                }
            }
        }
    };TypeFoldable_Generic, GenericTypeVisitable, const _: () =
    {
        impl<I: Interner, T, J> ::rustc_type_ir::lift::Lift<J> for
            Placeholder<I, T> where J: Interner,
            I: ::rustc_type_ir::LiftInto<J>, T: ::rustc_type_ir::lift::Lift<J>
            {
            type Lifted =
                Placeholder<J, <T as ::rustc_type_ir::lift::Lift<J>>::Lifted>;
            fn lift_to_interner(self, interner: J) -> Self::Lifted {
                match self {
                    Placeholder {
                        universe: __binding_0, bound: __binding_1, _tcx: __binding_2
                        } => {
                        Placeholder {
                            universe: __binding_0,
                            bound: __binding_1.lift_to_interner(interner),
                            _tcx: PhantomData,
                        }
                    }
                }
            }
        }
    };Lift_Generic)]
952#[cfg_attr(
953    feature = "nightly",
954    derive(const _: () =
    {
        impl<I: Interner, T, __E: ::rustc_serialize::Encoder>
            ::rustc_serialize::Encodable<__E> for Placeholder<I, T> where
            T: ::rustc_serialize::Encodable<__E>,
            PhantomData<fn() -> I>: ::rustc_serialize::Encodable<__E> {
            fn encode(&self, __encoder: &mut __E) {
                match *self {
                    Placeholder {
                        universe: ref __binding_0,
                        bound: ref __binding_1,
                        _tcx: 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);
                    }
                }
            }
        }
    };Encodable_NoContext, const _: () =
    {
        impl<I: Interner, T, __D: ::rustc_serialize::Decoder>
            ::rustc_serialize::Decodable<__D> for Placeholder<I, T> where
            T: ::rustc_serialize::Decodable<__D>,
            PhantomData<fn() -> I>: ::rustc_serialize::Decodable<__D> {
            fn decode(__decoder: &mut __D) -> Self {
                Placeholder {
                    universe: ::rustc_serialize::Decodable::decode(__decoder),
                    bound: ::rustc_serialize::Decodable::decode(__decoder),
                    _tcx: ::rustc_serialize::Decodable::decode(__decoder),
                }
            }
        }
    };Decodable_NoContext, const _: () =
    {
        impl<I: Interner, T> ::rustc_data_structures::stable_hash::StableHash
            for Placeholder<I, T> where
            T: ::rustc_data_structures::stable_hash::StableHash,
            PhantomData<fn()
                -> I>: ::rustc_data_structures::stable_hash::StableHash {
            #[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 {
                    Placeholder {
                        universe: ref __binding_0,
                        bound: ref __binding_1,
                        _tcx: ref __binding_2 } => {
                        { __binding_0.stable_hash(__hcx, __hasher); }
                        { __binding_1.stable_hash(__hcx, __hasher); }
                        { __binding_2.stable_hash(__hcx, __hasher); }
                    }
                }
            }
        }
    };StableHash_NoContext)
955)]
956pub struct Placeholder<I: Interner, T> {
957    #[lift(identity)]
958    pub universe: UniverseIndex,
959    pub bound: T,
960    #[type_foldable(identity)]
961    #[type_visitable(ignore)]
962    _tcx: PhantomData<fn() -> I>,
963}
964
965impl<I: Interner, T: fmt::Debug> fmt::Debug for ty::Placeholder<I, T> {
966    fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
967        if self.universe == ty::UniverseIndex::ROOT {
968            f.write_fmt(format_args!("!{0:?}", self.bound))write!(f, "!{:?}", self.bound)
969        } else {
970            f.write_fmt(format_args!("!{0}_{1:?}", self.universe.index(), self.bound))write!(f, "!{}_{:?}", self.universe.index(), self.bound)
971        }
972    }
973}
974
975#[automatically_derived]
impl<I: Interner> ::core::hash::Hash for BoundRegionKind<I> where I: Interner
    {
    fn hash<__H: ::core::hash::Hasher>(&self, __state: &mut __H) {
        match self {
            BoundRegionKind::Anon => {
                ::core::hash::Hash::hash(&::core::mem::discriminant(self),
                    __state);
            }
            BoundRegionKind::NamedForPrinting(ref __field_0) => {
                ::core::hash::Hash::hash(&::core::mem::discriminant(self),
                    __state);
                ::core::hash::Hash::hash(__field_0, __state);
            }
            BoundRegionKind::Named(ref __field_0) => {
                ::core::hash::Hash::hash(&::core::mem::discriminant(self),
                    __state);
                ::core::hash::Hash::hash(__field_0, __state);
            }
            BoundRegionKind::ClosureEnv => {
                ::core::hash::Hash::hash(&::core::mem::discriminant(self),
                    __state);
            }
        }
    }
}#[derive_where(Clone, Copy, PartialEq, Eq, Hash; I: Interner)]
976#[derive(const _: () =
    {
        impl<I: Interner, J> ::rustc_type_ir::lift::Lift<J> for
            BoundRegionKind<I> where J: Interner,
            I: ::rustc_type_ir::LiftInto<J> {
            type Lifted = BoundRegionKind<J>;
            fn lift_to_interner(self, interner: J) -> Self::Lifted {
                match self {
                    BoundRegionKind::Anon => { BoundRegionKind::Anon }
                    BoundRegionKind::NamedForPrinting(__binding_0) => {
                        BoundRegionKind::NamedForPrinting(__binding_0.lift_to_interner(interner))
                    }
                    BoundRegionKind::Named(__binding_0) => {
                        BoundRegionKind::Named(__binding_0.lift_to_interner(interner))
                    }
                    BoundRegionKind::ClosureEnv => {
                        BoundRegionKind::ClosureEnv
                    }
                }
            }
        }
    };Lift_Generic, GenericTypeVisitable)]
977#[cfg_attr(
978    feature = "nightly",
979    derive(const _: () =
    {
        impl<I: Interner, __E: ::rustc_serialize::Encoder>
            ::rustc_serialize::Encodable<__E> for BoundRegionKind<I> where
            I::Symbol: ::rustc_serialize::Encodable<__E>,
            I::DefId: ::rustc_serialize::Encodable<__E> {
            fn encode(&self, __encoder: &mut __E) {
                let disc =
                    match *self {
                        BoundRegionKind::Anon => { 0usize }
                        BoundRegionKind::NamedForPrinting(ref __binding_0) => {
                            1usize
                        }
                        BoundRegionKind::Named(ref __binding_0) => { 2usize }
                        BoundRegionKind::ClosureEnv => { 3usize }
                    };
                ::rustc_serialize::Encoder::emit_u8(__encoder, disc as u8);
                match *self {
                    BoundRegionKind::Anon => {}
                    BoundRegionKind::NamedForPrinting(ref __binding_0) => {
                        ::rustc_serialize::Encodable::<__E>::encode(__binding_0,
                            __encoder);
                    }
                    BoundRegionKind::Named(ref __binding_0) => {
                        ::rustc_serialize::Encodable::<__E>::encode(__binding_0,
                            __encoder);
                    }
                    BoundRegionKind::ClosureEnv => {}
                }
            }
        }
    };Encodable_NoContext, const _: () =
    {
        impl<I: Interner, __D: ::rustc_serialize::Decoder>
            ::rustc_serialize::Decodable<__D> for BoundRegionKind<I> where
            I::Symbol: ::rustc_serialize::Decodable<__D>,
            I::DefId: ::rustc_serialize::Decodable<__D> {
            fn decode(__decoder: &mut __D) -> Self {
                match ::rustc_serialize::Decoder::read_u8(__decoder) as usize
                    {
                    0usize => { BoundRegionKind::Anon }
                    1usize => {
                        BoundRegionKind::NamedForPrinting(::rustc_serialize::Decodable::decode(__decoder))
                    }
                    2usize => {
                        BoundRegionKind::Named(::rustc_serialize::Decodable::decode(__decoder))
                    }
                    3usize => { BoundRegionKind::ClosureEnv }
                    n => {
                        ::core::panicking::panic_fmt(format_args!("invalid enum variant tag while decoding `BoundRegionKind`, expected 0..4, actual {0}",
                                n));
                    }
                }
            }
        }
    };Decodable_NoContext, const _: () =
    {
        impl<I: Interner> ::rustc_data_structures::stable_hash::StableHash for
            BoundRegionKind<I> where
            I::Symbol: ::rustc_data_structures::stable_hash::StableHash,
            I::DefId: ::rustc_data_structures::stable_hash::StableHash {
            #[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 {
                    BoundRegionKind::Anon => {}
                    BoundRegionKind::NamedForPrinting(ref __binding_0) => {
                        { __binding_0.stable_hash(__hcx, __hasher); }
                    }
                    BoundRegionKind::Named(ref __binding_0) => {
                        { __binding_0.stable_hash(__hcx, __hasher); }
                    }
                    BoundRegionKind::ClosureEnv => {}
                }
            }
        }
    };StableHash_NoContext)
980)]
981pub enum BoundRegionKind<I: Interner> {
982    /// An anonymous region parameter for a given fn (&T)
983    Anon,
984
985    /// An anonymous region parameter with a `Symbol` name.
986    ///
987    /// Used to give late-bound regions names for things like pretty printing.
988    NamedForPrinting(I::Symbol),
989
990    /// Late-bound regions that appear in the AST.
991    Named(I::DefId),
992
993    /// Anonymous region for the implicit env pointer parameter
994    /// to a closure
995    ClosureEnv,
996}
997
998impl<I: Interner> fmt::Debug for ty::BoundRegionKind<I> {
999    fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
1000        match *self {
1001            ty::BoundRegionKind::Anon => f.write_fmt(format_args!("BrAnon"))write!(f, "BrAnon"),
1002            ty::BoundRegionKind::NamedForPrinting(name) => {
1003                f.write_fmt(format_args!("BrNamedForPrinting({0:?})", name))write!(f, "BrNamedForPrinting({:?})", name)
1004            }
1005            ty::BoundRegionKind::Named(did) => {
1006                f.write_fmt(format_args!("BrNamed({0:?})", did))write!(f, "BrNamed({did:?})")
1007            }
1008            ty::BoundRegionKind::ClosureEnv => f.write_fmt(format_args!("BrEnv"))write!(f, "BrEnv"),
1009        }
1010    }
1011}
1012
1013impl<I: Interner> BoundRegionKind<I> {
1014    pub fn is_named(&self, tcx: I) -> bool {
1015        self.get_name(tcx).is_some()
1016    }
1017
1018    pub fn get_name(&self, tcx: I) -> Option<I::Symbol> {
1019        match *self {
1020            ty::BoundRegionKind::Named(def_id) => {
1021                let name = tcx.item_name(def_id);
1022                if name.is_kw_underscore_lifetime() { None } else { Some(name) }
1023            }
1024            ty::BoundRegionKind::NamedForPrinting(name) => Some(name),
1025            _ => None,
1026        }
1027    }
1028
1029    pub fn get_id(&self) -> Option<I::DefId> {
1030        match *self {
1031            ty::BoundRegionKind::Named(id) => Some(id),
1032            _ => None,
1033        }
1034    }
1035}
1036
1037#[automatically_derived]
impl<I: Interner> ::core::hash::Hash for BoundTyKind<I> where I: Interner {
    fn hash<__H: ::core::hash::Hasher>(&self, __state: &mut __H) {
        match self {
            BoundTyKind::Anon => {
                ::core::hash::Hash::hash(&::core::mem::discriminant(self),
                    __state);
            }
            BoundTyKind::Param(ref __field_0) => {
                ::core::hash::Hash::hash(&::core::mem::discriminant(self),
                    __state);
                ::core::hash::Hash::hash(__field_0, __state);
            }
        }
    }
}#[derive_where(Clone, Copy, PartialEq, Eq, Debug, Hash; I: Interner)]
1038#[derive(const _: () =
    {
        impl<I: Interner, J> ::rustc_type_ir::lift::Lift<J> for BoundTyKind<I>
            where J: Interner, I: ::rustc_type_ir::LiftInto<J> {
            type Lifted = BoundTyKind<J>;
            fn lift_to_interner(self, interner: J) -> Self::Lifted {
                match self {
                    BoundTyKind::Anon => { BoundTyKind::Anon }
                    BoundTyKind::Param(__binding_0) => {
                        BoundTyKind::Param(__binding_0.lift_to_interner(interner))
                    }
                }
            }
        }
    };Lift_Generic, GenericTypeVisitable)]
1039#[cfg_attr(
1040    feature = "nightly",
1041    derive(const _: () =
    {
        impl<I: Interner, __E: ::rustc_serialize::Encoder>
            ::rustc_serialize::Encodable<__E> for BoundTyKind<I> where
            I::DefId: ::rustc_serialize::Encodable<__E> {
            fn encode(&self, __encoder: &mut __E) {
                let disc =
                    match *self {
                        BoundTyKind::Anon => { 0usize }
                        BoundTyKind::Param(ref __binding_0) => { 1usize }
                    };
                ::rustc_serialize::Encoder::emit_u8(__encoder, disc as u8);
                match *self {
                    BoundTyKind::Anon => {}
                    BoundTyKind::Param(ref __binding_0) => {
                        ::rustc_serialize::Encodable::<__E>::encode(__binding_0,
                            __encoder);
                    }
                }
            }
        }
    };Encodable_NoContext, const _: () =
    {
        impl<I: Interner, __D: ::rustc_serialize::Decoder>
            ::rustc_serialize::Decodable<__D> for BoundTyKind<I> where
            I::DefId: ::rustc_serialize::Decodable<__D> {
            fn decode(__decoder: &mut __D) -> Self {
                match ::rustc_serialize::Decoder::read_u8(__decoder) as usize
                    {
                    0usize => { BoundTyKind::Anon }
                    1usize => {
                        BoundTyKind::Param(::rustc_serialize::Decodable::decode(__decoder))
                    }
                    n => {
                        ::core::panicking::panic_fmt(format_args!("invalid enum variant tag while decoding `BoundTyKind`, expected 0..2, actual {0}",
                                n));
                    }
                }
            }
        }
    };Decodable_NoContext, const _: () =
    {
        impl<I: Interner> ::rustc_data_structures::stable_hash::StableHash for
            BoundTyKind<I> where
            I::DefId: ::rustc_data_structures::stable_hash::StableHash {
            #[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 {
                    BoundTyKind::Anon => {}
                    BoundTyKind::Param(ref __binding_0) => {
                        { __binding_0.stable_hash(__hcx, __hasher); }
                    }
                }
            }
        }
    };StableHash_NoContext)
1042)]
1043pub enum BoundTyKind<I: Interner> {
1044    Anon,
1045    Param(I::DefId),
1046}
1047
1048#[automatically_derived]
impl<I: Interner> ::core::hash::Hash for BoundVariableKind<I> where
    I: Interner {
    fn hash<__H: ::core::hash::Hasher>(&self, __state: &mut __H) {
        match self {
            BoundVariableKind::Ty(ref __field_0) => {
                ::core::hash::Hash::hash(&::core::mem::discriminant(self),
                    __state);
                ::core::hash::Hash::hash(__field_0, __state);
            }
            BoundVariableKind::Region(ref __field_0) => {
                ::core::hash::Hash::hash(&::core::mem::discriminant(self),
                    __state);
                ::core::hash::Hash::hash(__field_0, __state);
            }
            BoundVariableKind::Const => {
                ::core::hash::Hash::hash(&::core::mem::discriminant(self),
                    __state);
            }
        }
    }
}#[derive_where(Clone, Copy, PartialEq, Eq, Debug, Hash; I: Interner)]
1049#[derive(const _: () =
    {
        impl<I: Interner, J> ::rustc_type_ir::lift::Lift<J> for
            BoundVariableKind<I> where J: Interner,
            I: ::rustc_type_ir::LiftInto<J> {
            type Lifted = BoundVariableKind<J>;
            fn lift_to_interner(self, interner: J) -> Self::Lifted {
                match self {
                    BoundVariableKind::Ty(__binding_0) => {
                        BoundVariableKind::Ty(__binding_0.lift_to_interner(interner))
                    }
                    BoundVariableKind::Region(__binding_0) => {
                        BoundVariableKind::Region(__binding_0.lift_to_interner(interner))
                    }
                    BoundVariableKind::Const => { BoundVariableKind::Const }
                }
            }
        }
    };Lift_Generic, GenericTypeVisitable)]
1050#[cfg_attr(
1051    feature = "nightly",
1052    derive(const _: () =
    {
        impl<I: Interner, __E: ::rustc_serialize::Encoder>
            ::rustc_serialize::Encodable<__E> for BoundVariableKind<I> where
            BoundTyKind<I>: ::rustc_serialize::Encodable<__E>,
            BoundRegionKind<I>: ::rustc_serialize::Encodable<__E> {
            fn encode(&self, __encoder: &mut __E) {
                let disc =
                    match *self {
                        BoundVariableKind::Ty(ref __binding_0) => { 0usize }
                        BoundVariableKind::Region(ref __binding_0) => { 1usize }
                        BoundVariableKind::Const => { 2usize }
                    };
                ::rustc_serialize::Encoder::emit_u8(__encoder, disc as u8);
                match *self {
                    BoundVariableKind::Ty(ref __binding_0) => {
                        ::rustc_serialize::Encodable::<__E>::encode(__binding_0,
                            __encoder);
                    }
                    BoundVariableKind::Region(ref __binding_0) => {
                        ::rustc_serialize::Encodable::<__E>::encode(__binding_0,
                            __encoder);
                    }
                    BoundVariableKind::Const => {}
                }
            }
        }
    };Encodable_NoContext, const _: () =
    {
        impl<I: Interner, __D: ::rustc_serialize::Decoder>
            ::rustc_serialize::Decodable<__D> for BoundVariableKind<I> where
            BoundTyKind<I>: ::rustc_serialize::Decodable<__D>,
            BoundRegionKind<I>: ::rustc_serialize::Decodable<__D> {
            fn decode(__decoder: &mut __D) -> Self {
                match ::rustc_serialize::Decoder::read_u8(__decoder) as usize
                    {
                    0usize => {
                        BoundVariableKind::Ty(::rustc_serialize::Decodable::decode(__decoder))
                    }
                    1usize => {
                        BoundVariableKind::Region(::rustc_serialize::Decodable::decode(__decoder))
                    }
                    2usize => { BoundVariableKind::Const }
                    n => {
                        ::core::panicking::panic_fmt(format_args!("invalid enum variant tag while decoding `BoundVariableKind`, expected 0..3, actual {0}",
                                n));
                    }
                }
            }
        }
    };Decodable_NoContext, const _: () =
    {
        impl<I: Interner> ::rustc_data_structures::stable_hash::StableHash for
            BoundVariableKind<I> where
            BoundTyKind<I>: ::rustc_data_structures::stable_hash::StableHash,
            BoundRegionKind<I>: ::rustc_data_structures::stable_hash::StableHash
            {
            #[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 {
                    BoundVariableKind::Ty(ref __binding_0) => {
                        { __binding_0.stable_hash(__hcx, __hasher); }
                    }
                    BoundVariableKind::Region(ref __binding_0) => {
                        { __binding_0.stable_hash(__hcx, __hasher); }
                    }
                    BoundVariableKind::Const => {}
                }
            }
        }
    };StableHash_NoContext)
1053)]
1054pub enum BoundVariableKind<I: Interner> {
1055    Ty(BoundTyKind<I>),
1056    Region(BoundRegionKind<I>),
1057    Const,
1058}
1059
1060impl<I: Interner> BoundVariableKind<I> {
1061    pub fn expect_region(self) -> BoundRegionKind<I> {
1062        match self {
1063            BoundVariableKind::Region(lt) => lt,
1064            _ => {
    ::core::panicking::panic_fmt(format_args!("expected a region, but found another kind"));
}panic!("expected a region, but found another kind"),
1065        }
1066    }
1067
1068    pub fn expect_ty(self) -> BoundTyKind<I> {
1069        match self {
1070            BoundVariableKind::Ty(ty) => ty,
1071            _ => {
    ::core::panicking::panic_fmt(format_args!("expected a type, but found another kind"));
}panic!("expected a type, but found another kind"),
1072        }
1073    }
1074
1075    pub fn expect_const(self) {
1076        match self {
1077            BoundVariableKind::Const => (),
1078            _ => {
    ::core::panicking::panic_fmt(format_args!("expected a const, but found another kind"));
}panic!("expected a const, but found another kind"),
1079        }
1080    }
1081}
1082
1083#[automatically_derived]
impl<I: Interner> ::core::hash::Hash for BoundRegion<I> where I: Interner {
    fn hash<__H: ::core::hash::Hasher>(&self, __state: &mut __H) {
        match self {
            BoundRegion { var: ref __field_var, kind: ref __field_kind } => {
                ::core::hash::Hash::hash(__field_var, __state);
                ::core::hash::Hash::hash(__field_kind, __state);
            }
        }
    }
}#[derive_where(Clone, Copy, PartialEq, Eq, Hash; I: Interner)]
1084#[derive(GenericTypeVisitable)]
1085#[cfg_attr(
1086    feature = "nightly",
1087    derive(const _: () =
    {
        impl<I: Interner, __E: ::rustc_serialize::Encoder>
            ::rustc_serialize::Encodable<__E> for BoundRegion<I> where
            BoundRegionKind<I>: ::rustc_serialize::Encodable<__E> {
            fn encode(&self, __encoder: &mut __E) {
                match *self {
                    BoundRegion { var: ref __binding_0, kind: ref __binding_1 }
                        => {
                        ::rustc_serialize::Encodable::<__E>::encode(__binding_0,
                            __encoder);
                        ::rustc_serialize::Encodable::<__E>::encode(__binding_1,
                            __encoder);
                    }
                }
            }
        }
    };Encodable_NoContext, const _: () =
    {
        impl<I: Interner> ::rustc_data_structures::stable_hash::StableHash for
            BoundRegion<I> where
            BoundRegionKind<I>: ::rustc_data_structures::stable_hash::StableHash
            {
            #[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 {
                    BoundRegion { var: ref __binding_0, kind: ref __binding_1 }
                        => {
                        { __binding_0.stable_hash(__hcx, __hasher); }
                        { __binding_1.stable_hash(__hcx, __hasher); }
                    }
                }
            }
        }
    };StableHash_NoContext, const _: () =
    {
        impl<I: Interner, __D: ::rustc_serialize::Decoder>
            ::rustc_serialize::Decodable<__D> for BoundRegion<I> where
            BoundRegionKind<I>: ::rustc_serialize::Decodable<__D> {
            fn decode(__decoder: &mut __D) -> Self {
                BoundRegion {
                    var: ::rustc_serialize::Decodable::decode(__decoder),
                    kind: ::rustc_serialize::Decodable::decode(__decoder),
                }
            }
        }
    };Decodable_NoContext)
1088)]
1089pub struct BoundRegion<I: Interner> {
1090    pub var: ty::BoundVar,
1091    pub kind: BoundRegionKind<I>,
1092}
1093
1094impl<I: Interner> core::fmt::Debug for BoundRegion<I> {
1095    fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
1096        match self.kind {
1097            BoundRegionKind::Anon => f.write_fmt(format_args!("{0:?}", self.var))write!(f, "{:?}", self.var),
1098            BoundRegionKind::ClosureEnv => f.write_fmt(format_args!("{0:?}.Env", self.var))write!(f, "{:?}.Env", self.var),
1099            BoundRegionKind::Named(def) => {
1100                f.write_fmt(format_args!("{0:?}.Named({1:?})", self.var, def))write!(f, "{:?}.Named({:?})", self.var, def)
1101            }
1102            BoundRegionKind::NamedForPrinting(symbol) => {
1103                f.write_fmt(format_args!("{0:?}.NamedAnon({1:?})", self.var, symbol))write!(f, "{:?}.NamedAnon({:?})", self.var, symbol)
1104            }
1105        }
1106    }
1107}
1108
1109impl<I: Interner> BoundRegion<I> {
1110    pub fn var(self) -> ty::BoundVar {
1111        self.var
1112    }
1113
1114    pub fn assert_eq(self, var: BoundVariableKind<I>) {
1115        match (&self.kind, &var.expect_region()) {
    (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!(self.kind, var.expect_region())
1116    }
1117}
1118
1119pub type PlaceholderRegion<I> = ty::Placeholder<I, BoundRegion<I>>;
1120
1121impl<I: Interner> PlaceholderRegion<I> {
1122    pub fn universe(self) -> UniverseIndex {
1123        self.universe
1124    }
1125
1126    pub fn var(self) -> ty::BoundVar {
1127        self.bound.var()
1128    }
1129
1130    pub fn with_updated_universe(self, ui: UniverseIndex) -> Self {
1131        Self { universe: ui, bound: self.bound, _tcx: PhantomData }
1132    }
1133
1134    pub fn new(ui: UniverseIndex, bound: BoundRegion<I>) -> Self {
1135        Self { universe: ui, bound, _tcx: PhantomData }
1136    }
1137
1138    pub fn new_anon(ui: UniverseIndex, var: ty::BoundVar) -> Self {
1139        let bound = BoundRegion { var, kind: BoundRegionKind::Anon };
1140        Self { universe: ui, bound, _tcx: PhantomData }
1141    }
1142}
1143
1144#[automatically_derived]
impl<I: Interner> ::core::hash::Hash for BoundTy<I> where I: Interner {
    fn hash<__H: ::core::hash::Hasher>(&self, __state: &mut __H) {
        match self {
            BoundTy { var: ref __field_var, kind: ref __field_kind } => {
                ::core::hash::Hash::hash(__field_var, __state);
                ::core::hash::Hash::hash(__field_kind, __state);
            }
        }
    }
}#[derive_where(Clone, Copy, PartialEq, Eq, Hash; I: Interner)]
1145#[derive(GenericTypeVisitable, const _: () =
    {
        impl<I: Interner, J> ::rustc_type_ir::lift::Lift<J> for BoundTy<I>
            where J: Interner, I: ::rustc_type_ir::LiftInto<J> {
            type Lifted = BoundTy<J>;
            fn lift_to_interner(self, interner: J) -> Self::Lifted {
                match self {
                    BoundTy { var: __binding_0, kind: __binding_1 } => {
                        BoundTy {
                            var: __binding_0,
                            kind: __binding_1.lift_to_interner(interner),
                        }
                    }
                }
            }
        }
    };Lift_Generic)]
1146#[cfg_attr(
1147    feature = "nightly",
1148    derive(const _: () =
    {
        impl<I: Interner, __E: ::rustc_serialize::Encoder>
            ::rustc_serialize::Encodable<__E> for BoundTy<I> where
            BoundTyKind<I>: ::rustc_serialize::Encodable<__E> {
            fn encode(&self, __encoder: &mut __E) {
                match *self {
                    BoundTy { var: ref __binding_0, kind: ref __binding_1 } => {
                        ::rustc_serialize::Encodable::<__E>::encode(__binding_0,
                            __encoder);
                        ::rustc_serialize::Encodable::<__E>::encode(__binding_1,
                            __encoder);
                    }
                }
            }
        }
    };Encodable_NoContext, const _: () =
    {
        impl<I: Interner, __D: ::rustc_serialize::Decoder>
            ::rustc_serialize::Decodable<__D> for BoundTy<I> where
            BoundTyKind<I>: ::rustc_serialize::Decodable<__D> {
            fn decode(__decoder: &mut __D) -> Self {
                BoundTy {
                    var: ::rustc_serialize::Decodable::decode(__decoder),
                    kind: ::rustc_serialize::Decodable::decode(__decoder),
                }
            }
        }
    };Decodable_NoContext, const _: () =
    {
        impl<I: Interner> ::rustc_data_structures::stable_hash::StableHash for
            BoundTy<I> where
            BoundTyKind<I>: ::rustc_data_structures::stable_hash::StableHash {
            #[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 {
                    BoundTy { var: ref __binding_0, kind: ref __binding_1 } => {
                        { __binding_0.stable_hash(__hcx, __hasher); }
                        { __binding_1.stable_hash(__hcx, __hasher); }
                    }
                }
            }
        }
    };StableHash_NoContext)
1149)]
1150pub struct BoundTy<I: Interner> {
1151    #[lift(identity)]
1152    pub var: ty::BoundVar,
1153    pub kind: BoundTyKind<I>,
1154}
1155
1156impl<I: Interner> fmt::Debug for ty::BoundTy<I> {
1157    fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
1158        match self.kind {
1159            ty::BoundTyKind::Anon => f.write_fmt(format_args!("{0:?}", self.var))write!(f, "{:?}", self.var),
1160            ty::BoundTyKind::Param(def_id) => f.write_fmt(format_args!("{0:?}", def_id))write!(f, "{def_id:?}"),
1161        }
1162    }
1163}
1164
1165impl<I: Interner> BoundTy<I> {
1166    pub fn var(self) -> ty::BoundVar {
1167        self.var
1168    }
1169
1170    pub fn assert_eq(self, var: BoundVariableKind<I>) {
1171        match (&self.kind, &var.expect_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!(self.kind, var.expect_ty())
1172    }
1173}
1174
1175pub type PlaceholderType<I> = ty::Placeholder<I, BoundTy<I>>;
1176
1177impl<I: Interner> PlaceholderType<I> {
1178    pub fn universe(self) -> UniverseIndex {
1179        self.universe
1180    }
1181
1182    pub fn var(self) -> ty::BoundVar {
1183        self.bound.var
1184    }
1185
1186    pub fn with_updated_universe(self, ui: UniverseIndex) -> Self {
1187        Self { universe: ui, bound: self.bound, _tcx: PhantomData }
1188    }
1189
1190    pub fn new(ui: UniverseIndex, bound: BoundTy<I>) -> Self {
1191        Self { universe: ui, bound, _tcx: PhantomData }
1192    }
1193
1194    pub fn new_anon(ui: UniverseIndex, var: ty::BoundVar) -> Self {
1195        let bound = BoundTy { var, kind: BoundTyKind::Anon };
1196        Self { universe: ui, bound, _tcx: PhantomData }
1197    }
1198}
1199
1200#[automatically_derived]
impl<I: Interner> ::core::hash::Hash for BoundConst<I> where I: Interner {
    fn hash<__H: ::core::hash::Hasher>(&self, __state: &mut __H) {
        match self {
            BoundConst { var: ref __field_var, _tcx: ref __field__tcx } => {
                ::core::hash::Hash::hash(__field_var, __state);
                ::core::hash::Hash::hash(__field__tcx, __state);
            }
        }
    }
}#[derive_where(Clone, Copy, PartialEq, Debug, Eq, Hash; I: Interner)]
1201#[derive(GenericTypeVisitable)]
1202#[cfg_attr(
1203    feature = "nightly",
1204    derive(const _: () =
    {
        impl<I: Interner, __E: ::rustc_serialize::Encoder>
            ::rustc_serialize::Encodable<__E> for BoundConst<I> where
            PhantomData<fn() -> I>: ::rustc_serialize::Encodable<__E> {
            fn encode(&self, __encoder: &mut __E) {
                match *self {
                    BoundConst { var: ref __binding_0, _tcx: ref __binding_1 }
                        => {
                        ::rustc_serialize::Encodable::<__E>::encode(__binding_0,
                            __encoder);
                        ::rustc_serialize::Encodable::<__E>::encode(__binding_1,
                            __encoder);
                    }
                }
            }
        }
    };Encodable_NoContext, const _: () =
    {
        impl<I: Interner, __D: ::rustc_serialize::Decoder>
            ::rustc_serialize::Decodable<__D> for BoundConst<I> where
            PhantomData<fn() -> I>: ::rustc_serialize::Decodable<__D> {
            fn decode(__decoder: &mut __D) -> Self {
                BoundConst {
                    var: ::rustc_serialize::Decodable::decode(__decoder),
                    _tcx: ::rustc_serialize::Decodable::decode(__decoder),
                }
            }
        }
    };Decodable_NoContext, const _: () =
    {
        impl<I: Interner> ::rustc_data_structures::stable_hash::StableHash for
            BoundConst<I> where
            PhantomData<fn()
                -> I>: ::rustc_data_structures::stable_hash::StableHash {
            #[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 {
                    BoundConst { var: ref __binding_0, _tcx: ref __binding_1 }
                        => {
                        { __binding_0.stable_hash(__hcx, __hasher); }
                        { __binding_1.stable_hash(__hcx, __hasher); }
                    }
                }
            }
        }
    };StableHash_NoContext)
1205)]
1206pub struct BoundConst<I: Interner> {
1207    pub var: ty::BoundVar,
1208    #[derive_where(skip(Debug))]
1209    pub _tcx: PhantomData<fn() -> I>,
1210}
1211
1212impl<I: Interner> BoundConst<I> {
1213    pub fn var(self) -> ty::BoundVar {
1214        self.var
1215    }
1216
1217    pub fn assert_eq(self, var: BoundVariableKind<I>) {
1218        var.expect_const()
1219    }
1220
1221    pub fn new(var: ty::BoundVar) -> Self {
1222        Self { var, _tcx: PhantomData }
1223    }
1224}
1225
1226pub type PlaceholderConst<I> = ty::Placeholder<I, BoundConst<I>>;
1227
1228impl<I: Interner> PlaceholderConst<I> {
1229    pub fn universe(self) -> UniverseIndex {
1230        self.universe
1231    }
1232
1233    pub fn var(self) -> ty::BoundVar {
1234        self.bound.var
1235    }
1236
1237    pub fn with_updated_universe(self, ui: UniverseIndex) -> Self {
1238        Self { universe: ui, bound: self.bound, _tcx: PhantomData }
1239    }
1240
1241    pub fn new(ui: UniverseIndex, bound: BoundConst<I>) -> Self {
1242        Self { universe: ui, bound, _tcx: PhantomData }
1243    }
1244
1245    pub fn new_anon(ui: UniverseIndex, var: ty::BoundVar) -> Self {
1246        let bound = BoundConst::new(var);
1247        Self { universe: ui, bound, _tcx: PhantomData }
1248    }
1249
1250    pub fn find_const_ty_from_env(self, env: I::ParamEnv) -> I::Ty {
1251        let mut candidates = env.caller_bounds().iter().filter_map(|clause| {
1252            // `ConstArgHasType` are never desugared to be higher ranked.
1253            match clause.kind().skip_binder() {
1254                ty::ClauseKind::ConstArgHasType(placeholder_ct, ty) => {
1255                    if !!(placeholder_ct, ty).has_escaping_bound_vars() {
    ::core::panicking::panic("assertion failed: !(placeholder_ct, ty).has_escaping_bound_vars()")
};assert!(!(placeholder_ct, ty).has_escaping_bound_vars());
1256
1257                    match placeholder_ct.kind() {
1258                        ty::ConstKind::Placeholder(placeholder_ct) if placeholder_ct == self => {
1259                            Some(ty)
1260                        }
1261                        _ => None,
1262                    }
1263                }
1264                _ => None,
1265            }
1266        });
1267
1268        // N.B. it may be tempting to fix ICEs by making this function return
1269        // `Option<Ty<'tcx>>` instead of `Ty<'tcx>`; however, this is generally
1270        // considered to be a bandaid solution, since it hides more important
1271        // underlying issues with how we construct generics and predicates of
1272        // items. It's advised to fix the underlying issue rather than trying
1273        // to modify this function.
1274        let ty = candidates.next().unwrap_or_else(|| {
1275            {
    ::core::panicking::panic_fmt(format_args!("cannot find `{0:?}` in param-env: {1:#?}",
            self, env));
};panic!("cannot find `{self:?}` in param-env: {env:#?}");
1276        });
1277        if !candidates.next().is_none() {
    {
        ::core::panicking::panic_fmt(format_args!("did not expect duplicate `ConstParamHasTy` for `{0:?}` in param-env: {1:#?}",
                self, env));
    }
};assert!(
1278            candidates.next().is_none(),
1279            "did not expect duplicate `ConstParamHasTy` for `{self:?}` in param-env: {env:#?}"
1280        );
1281        ty
1282    }
1283}