mirror of
https://github.com/rust-lang/rust.git
synced 2025-10-02 18:27:37 +00:00
325 lines
10 KiB
Rust
325 lines
10 KiB
Rust
///! Definition of `InferCtxtLike` from the librarified type layer.
|
|
use rustc_hir::def_id::DefId;
|
|
use rustc_middle::traits::ObligationCause;
|
|
use rustc_middle::ty::relate::RelateResult;
|
|
use rustc_middle::ty::relate::combine::PredicateEmittingRelation;
|
|
use rustc_middle::ty::{self, Ty, TyCtxt, TypeFoldable};
|
|
use rustc_span::{DUMMY_SP, ErrorGuaranteed, Span};
|
|
|
|
use super::{
|
|
BoundRegionConversionTime, InferCtxt, OpaqueTypeStorageEntries, RegionVariableOrigin,
|
|
SubregionOrigin,
|
|
};
|
|
|
|
impl<'tcx> rustc_type_ir::InferCtxtLike for InferCtxt<'tcx> {
|
|
type Interner = TyCtxt<'tcx>;
|
|
|
|
fn cx(&self) -> TyCtxt<'tcx> {
|
|
self.tcx
|
|
}
|
|
|
|
fn next_trait_solver(&self) -> bool {
|
|
self.next_trait_solver
|
|
}
|
|
|
|
fn typing_mode(&self) -> ty::TypingMode<'tcx> {
|
|
self.typing_mode()
|
|
}
|
|
|
|
fn universe(&self) -> ty::UniverseIndex {
|
|
self.universe()
|
|
}
|
|
|
|
fn create_next_universe(&self) -> ty::UniverseIndex {
|
|
self.create_next_universe()
|
|
}
|
|
|
|
fn universe_of_ty(&self, vid: ty::TyVid) -> Option<ty::UniverseIndex> {
|
|
match self.probe_ty_var(vid) {
|
|
Err(universe) => Some(universe),
|
|
Ok(_) => None,
|
|
}
|
|
}
|
|
|
|
fn universe_of_lt(&self, lt: ty::RegionVid) -> Option<ty::UniverseIndex> {
|
|
match self.inner.borrow_mut().unwrap_region_constraints().probe_value(lt) {
|
|
Err(universe) => Some(universe),
|
|
Ok(_) => None,
|
|
}
|
|
}
|
|
|
|
fn universe_of_ct(&self, ct: ty::ConstVid) -> Option<ty::UniverseIndex> {
|
|
match self.probe_const_var(ct) {
|
|
Err(universe) => Some(universe),
|
|
Ok(_) => None,
|
|
}
|
|
}
|
|
|
|
fn root_ty_var(&self, var: ty::TyVid) -> ty::TyVid {
|
|
self.root_var(var)
|
|
}
|
|
|
|
fn root_const_var(&self, var: ty::ConstVid) -> ty::ConstVid {
|
|
self.root_const_var(var)
|
|
}
|
|
|
|
fn opportunistic_resolve_ty_var(&self, vid: ty::TyVid) -> Ty<'tcx> {
|
|
match self.probe_ty_var(vid) {
|
|
Ok(ty) => ty,
|
|
Err(_) => Ty::new_var(self.tcx, self.root_var(vid)),
|
|
}
|
|
}
|
|
|
|
fn opportunistic_resolve_int_var(&self, vid: ty::IntVid) -> Ty<'tcx> {
|
|
self.opportunistic_resolve_int_var(vid)
|
|
}
|
|
|
|
fn opportunistic_resolve_float_var(&self, vid: ty::FloatVid) -> Ty<'tcx> {
|
|
self.opportunistic_resolve_float_var(vid)
|
|
}
|
|
|
|
fn opportunistic_resolve_ct_var(&self, vid: ty::ConstVid) -> ty::Const<'tcx> {
|
|
match self.probe_const_var(vid) {
|
|
Ok(ct) => ct,
|
|
Err(_) => ty::Const::new_var(self.tcx, self.root_const_var(vid)),
|
|
}
|
|
}
|
|
|
|
fn opportunistic_resolve_lt_var(&self, vid: ty::RegionVid) -> ty::Region<'tcx> {
|
|
self.inner.borrow_mut().unwrap_region_constraints().opportunistic_resolve_var(self.tcx, vid)
|
|
}
|
|
|
|
fn is_changed_arg(&self, arg: ty::GenericArg<'tcx>) -> bool {
|
|
match arg.kind() {
|
|
ty::GenericArgKind::Lifetime(_) => {
|
|
// Lifetimes should not change affect trait selection.
|
|
false
|
|
}
|
|
ty::GenericArgKind::Type(ty) => {
|
|
if let ty::Infer(infer_ty) = *ty.kind() {
|
|
match infer_ty {
|
|
ty::InferTy::TyVar(vid) => {
|
|
!self.probe_ty_var(vid).is_err_and(|_| self.root_var(vid) == vid)
|
|
}
|
|
ty::InferTy::IntVar(vid) => {
|
|
let mut inner = self.inner.borrow_mut();
|
|
!matches!(
|
|
inner.int_unification_table().probe_value(vid),
|
|
ty::IntVarValue::Unknown
|
|
if inner.int_unification_table().find(vid) == vid
|
|
)
|
|
}
|
|
ty::InferTy::FloatVar(vid) => {
|
|
let mut inner = self.inner.borrow_mut();
|
|
!matches!(
|
|
inner.float_unification_table().probe_value(vid),
|
|
ty::FloatVarValue::Unknown
|
|
if inner.float_unification_table().find(vid) == vid
|
|
)
|
|
}
|
|
ty::InferTy::FreshTy(_)
|
|
| ty::InferTy::FreshIntTy(_)
|
|
| ty::InferTy::FreshFloatTy(_) => true,
|
|
}
|
|
} else {
|
|
true
|
|
}
|
|
}
|
|
ty::GenericArgKind::Const(ct) => {
|
|
if let ty::ConstKind::Infer(infer_ct) = ct.kind() {
|
|
match infer_ct {
|
|
ty::InferConst::Var(vid) => !self
|
|
.probe_const_var(vid)
|
|
.is_err_and(|_| self.root_const_var(vid) == vid),
|
|
ty::InferConst::Fresh(_) => true,
|
|
}
|
|
} else {
|
|
true
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
fn next_region_infer(&self) -> ty::Region<'tcx> {
|
|
self.next_region_var(RegionVariableOrigin::Misc(DUMMY_SP))
|
|
}
|
|
|
|
fn next_ty_infer(&self) -> Ty<'tcx> {
|
|
self.next_ty_var(DUMMY_SP)
|
|
}
|
|
|
|
fn next_const_infer(&self) -> ty::Const<'tcx> {
|
|
self.next_const_var(DUMMY_SP)
|
|
}
|
|
|
|
fn fresh_args_for_item(&self, def_id: DefId) -> ty::GenericArgsRef<'tcx> {
|
|
self.fresh_args_for_item(DUMMY_SP, def_id)
|
|
}
|
|
|
|
fn instantiate_binder_with_infer<T: TypeFoldable<TyCtxt<'tcx>> + Copy>(
|
|
&self,
|
|
value: ty::Binder<'tcx, T>,
|
|
) -> T {
|
|
self.instantiate_binder_with_fresh_vars(
|
|
DUMMY_SP,
|
|
BoundRegionConversionTime::HigherRankedType,
|
|
value,
|
|
)
|
|
}
|
|
|
|
fn enter_forall<T: TypeFoldable<TyCtxt<'tcx>>, U>(
|
|
&self,
|
|
value: ty::Binder<'tcx, T>,
|
|
f: impl FnOnce(T) -> U,
|
|
) -> U {
|
|
self.enter_forall(value, f)
|
|
}
|
|
|
|
fn equate_ty_vids_raw(&self, a: ty::TyVid, b: ty::TyVid) {
|
|
self.inner.borrow_mut().type_variables().equate(a, b);
|
|
}
|
|
|
|
fn equate_int_vids_raw(&self, a: ty::IntVid, b: ty::IntVid) {
|
|
self.inner.borrow_mut().int_unification_table().union(a, b);
|
|
}
|
|
|
|
fn equate_float_vids_raw(&self, a: ty::FloatVid, b: ty::FloatVid) {
|
|
self.inner.borrow_mut().float_unification_table().union(a, b);
|
|
}
|
|
|
|
fn equate_const_vids_raw(&self, a: ty::ConstVid, b: ty::ConstVid) {
|
|
self.inner.borrow_mut().const_unification_table().union(a, b);
|
|
}
|
|
|
|
fn instantiate_ty_var_raw<R: PredicateEmittingRelation<Self>>(
|
|
&self,
|
|
relation: &mut R,
|
|
target_is_expected: bool,
|
|
target_vid: ty::TyVid,
|
|
instantiation_variance: ty::Variance,
|
|
source_ty: Ty<'tcx>,
|
|
) -> RelateResult<'tcx, ()> {
|
|
self.instantiate_ty_var(
|
|
relation,
|
|
target_is_expected,
|
|
target_vid,
|
|
instantiation_variance,
|
|
source_ty,
|
|
)
|
|
}
|
|
|
|
fn instantiate_int_var_raw(&self, vid: ty::IntVid, value: ty::IntVarValue) {
|
|
self.inner.borrow_mut().int_unification_table().union_value(vid, value);
|
|
}
|
|
|
|
fn instantiate_float_var_raw(&self, vid: ty::FloatVid, value: ty::FloatVarValue) {
|
|
self.inner.borrow_mut().float_unification_table().union_value(vid, value);
|
|
}
|
|
|
|
fn instantiate_const_var_raw<R: PredicateEmittingRelation<Self>>(
|
|
&self,
|
|
relation: &mut R,
|
|
target_is_expected: bool,
|
|
target_vid: ty::ConstVid,
|
|
source_ct: ty::Const<'tcx>,
|
|
) -> RelateResult<'tcx, ()> {
|
|
self.instantiate_const_var(relation, target_is_expected, target_vid, source_ct)
|
|
}
|
|
|
|
fn set_tainted_by_errors(&self, e: ErrorGuaranteed) {
|
|
self.set_tainted_by_errors(e)
|
|
}
|
|
|
|
fn shallow_resolve(&self, ty: Ty<'tcx>) -> Ty<'tcx> {
|
|
self.shallow_resolve(ty)
|
|
}
|
|
fn shallow_resolve_const(&self, ct: ty::Const<'tcx>) -> ty::Const<'tcx> {
|
|
self.shallow_resolve_const(ct)
|
|
}
|
|
|
|
fn resolve_vars_if_possible<T>(&self, value: T) -> T
|
|
where
|
|
T: TypeFoldable<TyCtxt<'tcx>>,
|
|
{
|
|
self.resolve_vars_if_possible(value)
|
|
}
|
|
|
|
fn probe<T>(&self, probe: impl FnOnce() -> T) -> T {
|
|
self.probe(|_| probe())
|
|
}
|
|
|
|
fn sub_regions(&self, sub: ty::Region<'tcx>, sup: ty::Region<'tcx>, span: Span) {
|
|
self.inner.borrow_mut().unwrap_region_constraints().make_subregion(
|
|
SubregionOrigin::RelateRegionParamBound(span, None),
|
|
sub,
|
|
sup,
|
|
);
|
|
}
|
|
|
|
fn equate_regions(&self, a: ty::Region<'tcx>, b: ty::Region<'tcx>, span: Span) {
|
|
self.inner.borrow_mut().unwrap_region_constraints().make_eqregion(
|
|
SubregionOrigin::RelateRegionParamBound(span, None),
|
|
a,
|
|
b,
|
|
);
|
|
}
|
|
|
|
fn register_ty_outlives(&self, ty: Ty<'tcx>, r: ty::Region<'tcx>, span: Span) {
|
|
self.register_type_outlives_constraint(ty, r, &ObligationCause::dummy_with_span(span));
|
|
}
|
|
|
|
type OpaqueTypeStorageEntries = OpaqueTypeStorageEntries;
|
|
fn opaque_types_storage_num_entries(&self) -> OpaqueTypeStorageEntries {
|
|
self.inner.borrow_mut().opaque_types().num_entries()
|
|
}
|
|
fn clone_opaque_types_lookup_table(&self) -> Vec<(ty::OpaqueTypeKey<'tcx>, Ty<'tcx>)> {
|
|
self.inner.borrow_mut().opaque_types().iter_lookup_table().map(|(k, h)| (k, h.ty)).collect()
|
|
}
|
|
fn clone_duplicate_opaque_types(&self) -> Vec<(ty::OpaqueTypeKey<'tcx>, Ty<'tcx>)> {
|
|
self.inner
|
|
.borrow_mut()
|
|
.opaque_types()
|
|
.iter_duplicate_entries()
|
|
.map(|(k, h)| (k, h.ty))
|
|
.collect()
|
|
}
|
|
fn clone_opaque_types_added_since(
|
|
&self,
|
|
prev_entries: OpaqueTypeStorageEntries,
|
|
) -> Vec<(ty::OpaqueTypeKey<'tcx>, Ty<'tcx>)> {
|
|
self.inner
|
|
.borrow_mut()
|
|
.opaque_types()
|
|
.opaque_types_added_since(prev_entries)
|
|
.map(|(k, h)| (k, h.ty))
|
|
.collect()
|
|
}
|
|
|
|
fn register_hidden_type_in_storage(
|
|
&self,
|
|
opaque_type_key: ty::OpaqueTypeKey<'tcx>,
|
|
hidden_ty: Ty<'tcx>,
|
|
span: Span,
|
|
) -> Option<Ty<'tcx>> {
|
|
self.register_hidden_type_in_storage(
|
|
opaque_type_key,
|
|
ty::OpaqueHiddenType { span, ty: hidden_ty },
|
|
)
|
|
}
|
|
fn add_duplicate_opaque_type(
|
|
&self,
|
|
opaque_type_key: ty::OpaqueTypeKey<'tcx>,
|
|
hidden_ty: Ty<'tcx>,
|
|
span: Span,
|
|
) {
|
|
self.inner
|
|
.borrow_mut()
|
|
.opaque_types()
|
|
.add_duplicate(opaque_type_key, ty::OpaqueHiddenType { span, ty: hidden_ty })
|
|
}
|
|
|
|
fn reset_opaque_types(&self) {
|
|
let _ = self.take_opaque_types();
|
|
}
|
|
}
|