Canonicalizer

Struct Canonicalizer 

Source
pub(super) struct Canonicalizer<'a, D: SolverDelegate<Interner = I>, I: Interner> {
    delegate: &'a D,
    canonicalize_mode: CanonicalizeMode,
    variables: &'a mut Vec<I::GenericArg>,
    var_kinds: Vec<CanonicalVarKind<I>>,
    variable_lookup_table: HashMap<I::GenericArg, usize>,
    sub_root_lookup_table: HashMap<TyVid, usize>,
    cache: HashMap<I::Ty, I::Ty>,
}

Fields§

§delegate: &'a D§canonicalize_mode: CanonicalizeMode§variables: &'a mut Vec<I::GenericArg>§var_kinds: Vec<CanonicalVarKind<I>>§variable_lookup_table: HashMap<I::GenericArg, usize>§sub_root_lookup_table: HashMap<TyVid, usize>

Maps each sub_unification_table_root_var to the index of the first variable which used it.

This means in case two type variables have the same sub relations root, we set the sub_root of the second variable to the position of the first. Otherwise the sub_root of each type variable is just its own position.

§cache: HashMap<I::Ty, I::Ty>

We can simply cache based on the ty itself, because we use ty::BoundVarIndexKind::Canonical.

Implementations§

Source§

impl<'a, D: SolverDelegate<Interner = I>, I: Interner> Canonicalizer<'a, D, I>

Source

pub(super) fn canonicalize_response<T: TypeFoldable<I>>( delegate: &'a D, max_input_universe: UniverseIndex, variables: &'a mut Vec<I::GenericArg>, value: T, ) -> Canonical<I, T>

Source

fn canonicalize_param_env( delegate: &'a D, variables: &'a mut Vec<I::GenericArg>, param_env: I::ParamEnv, ) -> (I::ParamEnv, HashMap<I::GenericArg, usize>, Vec<CanonicalVarKind<I>>)

Source

pub(super) fn canonicalize_input<P: TypeFoldable<I>>( delegate: &'a D, variables: &'a mut Vec<I::GenericArg>, input: QueryInput<I, P>, ) -> Canonical<I, QueryInput<I, P>>

When canonicalizing query inputs, we keep 'static in the param_env but erase it everywhere else. We generally don’t want to depend on region identity, so while it should not matter whether 'static is kept in the value or opaque type storage as well, this prevents us from accidentally relying on it in the future.

We want to keep the option of canonicalizing 'static to an existential variable in the future by changing the way we detect global where-bounds.

Source

fn get_or_insert_bound_var( &mut self, arg: impl Into<I::GenericArg>, kind: CanonicalVarKind<I>, ) -> BoundVar

Source

fn get_or_insert_sub_root(&mut self, vid: TyVid) -> BoundVar

Source

fn finalize(self) -> (UniverseIndex, I::CanonicalVarKinds)

Source

fn inner_fold_ty(&mut self, t: I::Ty) -> I::Ty

Trait Implementations§

Source§

impl<D: SolverDelegate<Interner = I>, I: Interner> TypeFolder<I> for Canonicalizer<'_, D, I>

Source§

fn cx(&self) -> I

Source§

fn fold_region(&mut self, r: I::Region) -> I::Region

Source§

fn fold_ty(&mut self, t: I::Ty) -> I::Ty

Source§

fn fold_const(&mut self, c: I::Const) -> I::Const

Source§

fn fold_predicate(&mut self, p: I::Predicate) -> I::Predicate

Source§

fn fold_clauses(&mut self, c: I::Clauses) -> I::Clauses

§

fn fold_binder<T>(&mut self, t: Binder<I, T>) -> Binder<I, T>
where T: TypeFoldable<I>,

Auto Trait Implementations§

§

impl<'a, D, I> DynSend for Canonicalizer<'a, D, I>

§

impl<'a, D, I> DynSync for Canonicalizer<'a, D, I>

§

impl<'a, D, I> Freeze for Canonicalizer<'a, D, I>

§

impl<'a, D, I> RefUnwindSafe for Canonicalizer<'a, D, I>

§

impl<'a, D, I> Send for Canonicalizer<'a, D, I>

§

impl<'a, D, I> Sync for Canonicalizer<'a, D, I>

§

impl<'a, D, I> Unpin for Canonicalizer<'a, D, I>

§

impl<'a, D, I> !UnwindSafe for Canonicalizer<'a, D, I>

Blanket Implementations§

§

impl<T> Aligned for T

§

const ALIGN: Alignment

Alignment of Self.
§

impl<T> Any for T
where T: 'static + ?Sized,

§

fn type_id(&self) -> TypeId

Gets the TypeId of self. Read more
§

impl<T> Borrow<T> for T
where T: ?Sized,

§

fn borrow(&self) -> &T

Immutably borrows from an owned value. Read more
§

impl<T> BorrowMut<T> for T
where T: ?Sized,

§

fn borrow_mut(&mut self) -> &mut T

Mutably borrows from an owned value. Read more
§

impl<T, R> CollectAndApply<T, R> for T

§

fn collect_and_apply<I, F>(iter: I, f: F) -> R
where I: Iterator<Item = T>, F: FnOnce(&[T]) -> R,

Equivalent to f(&iter.collect::<Vec<_>>()).

§

type Output = R

§

impl<T> From<T> for T

§

fn from(t: T) -> T

Returns the argument unchanged.

§

impl<T> Instrument for T

§

fn instrument(self, span: Span) -> Instrumented<Self>

Instruments this type with the provided Span, returning an Instrumented wrapper. Read more
§

fn in_current_span(self) -> Instrumented<Self>

Instruments this type with the current Span, returning an Instrumented wrapper. Read more
§

impl<T, U> Into<U> for T
where U: From<T>,

§

fn into(self) -> U

Calls U::from(self).

That is, this conversion is whatever the implementation of From<T> for U chooses to do.

§

impl<T> IntoEither for T

§

fn into_either(self, into_left: bool) -> Either<Self, Self>

Converts self into a Left variant of Either<Self, Self> if into_left is true. Converts self into a Right variant of Either<Self, Self> otherwise. Read more
§

fn into_either_with<F>(self, into_left: F) -> Either<Self, Self>
where F: FnOnce(&Self) -> bool,

Converts self into a Left variant of Either<Self, Self> if into_left(&self) returns true. Converts self into a Right variant of Either<Self, Self> otherwise. Read more
§

impl<T> Pointable for T

§

const ALIGN: usize

The alignment of pointer.
§

type Init = T

The type for initializers.
§

unsafe fn init(init: <T as Pointable>::Init) -> usize

Initializes a with the given initializer. Read more
§

unsafe fn deref<'a>(ptr: usize) -> &'a T

Dereferences the given pointer. Read more
§

unsafe fn deref_mut<'a>(ptr: usize) -> &'a mut T

Mutably dereferences the given pointer. Read more
§

unsafe fn drop(ptr: usize)

Drops the object pointed to by the given pointer. Read more
§

impl<T> Same for T

§

type Output = T

Should always be Self
§

impl<T, U> TryFrom<U> for T
where U: Into<T>,

§

type Error = Infallible

The type returned in the event of a conversion error.
§

fn try_from(value: U) -> Result<T, <T as TryFrom<U>>::Error>

Performs the conversion.
§

impl<T, U> TryInto<U> for T
where U: TryFrom<T>,

§

type Error = <U as TryFrom<T>>::Error

The type returned in the event of a conversion error.
§

fn try_into(self) -> Result<U, <U as TryFrom<T>>::Error>

Performs the conversion.
§

impl<I, T, U> Upcast<I, U> for T
where U: UpcastFrom<I, T>,

§

fn upcast(self, interner: I) -> U

§

impl<I, T> UpcastFrom<I, T> for T

§

fn upcast_from(from: T, _tcx: I) -> T

§

impl<T> WithSubscriber for T

§

fn with_subscriber<S>(self, subscriber: S) -> WithDispatch<Self>
where S: Into<Dispatch>,

Attaches the provided Subscriber to this type, returning a WithDispatch wrapper. Read more
§

fn with_current_subscriber(self) -> WithDispatch<Self>

Attaches the current default Subscriber to this type, returning a WithDispatch wrapper. Read more
§

impl<T> ErasedDestructor for T
where T: 'static,

Layout§

Note: Most layout information is completely unstable and may even differ between compilations. The only exception is types with certain repr(...) attributes. Please see the Rust Reference's “Type Layout” chapter for details on type layout guarantees.

Size: 144 bytes