Skip to main content

rustc_type_ir/relate/
combine.rs

1use std::iter;
2
3use tracing::debug;
4
5use super::{
6    ExpectedFound, RelateResult, StructurallyRelateAliases, TypeRelation,
7    structurally_relate_consts, structurally_relate_tys,
8};
9use crate::error::TypeError;
10use crate::inherent::*;
11use crate::relate::VarianceDiagInfo;
12use crate::solve::Goal;
13use crate::visit::TypeVisitableExt as _;
14use crate::{self as ty, InferCtxtLike, Interner, TypingMode, Upcast};
15
16pub trait PredicateEmittingRelation<Infcx, I = <Infcx as InferCtxtLike>::Interner>:
17    TypeRelation<I>
18where
19    Infcx: InferCtxtLike<Interner = I>,
20    I: Interner,
21{
22    fn span(&self) -> I::Span;
23
24    fn param_env(&self) -> I::ParamEnv;
25
26    /// Whether aliases should be related structurally. This is pretty much
27    /// always `No` unless you're equating in some specific locations of the
28    /// new solver. See the comments in these use-cases for more details.
29    fn structurally_relate_aliases(&self) -> StructurallyRelateAliases;
30
31    /// Register obligations that must hold in order for this relation to hold
32    fn register_goals(&mut self, obligations: impl IntoIterator<Item = Goal<I, I::Predicate>>);
33
34    /// Register predicates that must hold in order for this relation to hold.
35    /// This uses the default `param_env` of the obligation.
36    fn register_predicates(
37        &mut self,
38        obligations: impl IntoIterator<Item: Upcast<I, I::Predicate>>,
39    );
40
41    /// Register `AliasRelate` obligation(s) that both types must be related to each other.
42    fn register_alias_relate_predicate(&mut self, a: I::Ty, b: I::Ty);
43}
44
45pub fn super_combine_tys<Infcx, I, R>(
46    infcx: &Infcx,
47    relation: &mut R,
48    a: I::Ty,
49    b: I::Ty,
50) -> RelateResult<I, I::Ty>
51where
52    Infcx: InferCtxtLike<Interner = I>,
53    I: Interner,
54    R: PredicateEmittingRelation<Infcx>,
55{
56    {
    use ::tracing::__macro_support::Callsite as _;
    static __CALLSITE: ::tracing::callsite::DefaultCallsite =
        {
            static META: ::tracing::Metadata<'static> =
                {
                    ::tracing_core::metadata::Metadata::new("event compiler/rustc_type_ir/src/relate/combine.rs:56",
                        "rustc_type_ir::relate::combine", ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_type_ir/src/relate/combine.rs"),
                        ::tracing_core::__macro_support::Option::Some(56u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_type_ir::relate::combine"),
                        ::tracing_core::field::FieldSet::new(&["message"],
                            ::tracing_core::callsite::Identifier(&__CALLSITE)),
                        ::tracing::metadata::Kind::EVENT)
                };
            ::tracing::callsite::DefaultCallsite::new(&META)
        };
    let enabled =
        ::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
                &&
                ::tracing::Level::DEBUG <=
                    ::tracing::level_filters::LevelFilter::current() &&
            {
                let interest = __CALLSITE.interest();
                !interest.is_never() &&
                    ::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
                        interest)
            };
    if enabled {
        (|value_set: ::tracing::field::ValueSet|
                    {
                        let meta = __CALLSITE.metadata();
                        ::tracing::Event::dispatch(meta, &value_set);
                        ;
                    })({
                #[allow(unused_imports)]
                use ::tracing::field::{debug, display, Value};
                let mut iter = __CALLSITE.metadata().fields().iter();
                __CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
                                    ::tracing::__macro_support::Option::Some(&format_args!("super_combine_tys::<{0}>({1:?}, {2:?})",
                                                    std::any::type_name::<R>(), a, b) as &dyn Value))])
            });
    } else { ; }
};debug!("super_combine_tys::<{}>({:?}, {:?})", std::any::type_name::<R>(), a, b);
57    if true {
    if !!a.has_escaping_bound_vars() {
        ::core::panicking::panic("assertion failed: !a.has_escaping_bound_vars()")
    };
};debug_assert!(!a.has_escaping_bound_vars());
58    if true {
    if !!b.has_escaping_bound_vars() {
        ::core::panicking::panic("assertion failed: !b.has_escaping_bound_vars()")
    };
};debug_assert!(!b.has_escaping_bound_vars());
59
60    match (a.kind(), b.kind()) {
61        (ty::Error(e), _) | (_, ty::Error(e)) => {
62            infcx.set_tainted_by_errors(e);
63            return Ok(Ty::new_error(infcx.cx(), e));
64        }
65
66        // Relate integral variables to other types
67        (ty::Infer(ty::IntVar(a_id)), ty::Infer(ty::IntVar(b_id))) => {
68            infcx.equate_int_vids_raw(a_id, b_id);
69            Ok(a)
70        }
71        (ty::Infer(ty::IntVar(v_id)), ty::Int(v)) => {
72            infcx.instantiate_int_var_raw(v_id, ty::IntVarValue::IntType(v));
73            Ok(b)
74        }
75        (ty::Int(v), ty::Infer(ty::IntVar(v_id))) => {
76            infcx.instantiate_int_var_raw(v_id, ty::IntVarValue::IntType(v));
77            Ok(a)
78        }
79        (ty::Infer(ty::IntVar(v_id)), ty::Uint(v)) => {
80            infcx.instantiate_int_var_raw(v_id, ty::IntVarValue::UintType(v));
81            Ok(b)
82        }
83        (ty::Uint(v), ty::Infer(ty::IntVar(v_id))) => {
84            infcx.instantiate_int_var_raw(v_id, ty::IntVarValue::UintType(v));
85            Ok(a)
86        }
87
88        // Relate floating-point variables to other types
89        (ty::Infer(ty::FloatVar(a_id)), ty::Infer(ty::FloatVar(b_id))) => {
90            infcx.equate_float_vids_raw(a_id, b_id);
91            Ok(a)
92        }
93        (ty::Infer(ty::FloatVar(v_id)), ty::Float(v)) => {
94            infcx.instantiate_float_var_raw(v_id, ty::FloatVarValue::Known(v));
95            Ok(b)
96        }
97        (ty::Float(v), ty::Infer(ty::FloatVar(v_id))) => {
98            infcx.instantiate_float_var_raw(v_id, ty::FloatVarValue::Known(v));
99            Ok(a)
100        }
101
102        // We don't expect `TyVar` or `Fresh*` vars at this point with lazy norm.
103        (ty::Alias(..), ty::Infer(ty::TyVar(_))) | (ty::Infer(ty::TyVar(_)), ty::Alias(..))
104            if infcx.next_trait_solver() =>
105        {
106            {
    ::core::panicking::panic_fmt(format_args!("We do not expect to encounter `TyVar` this late in combine -- they should have been handled earlier"));
}panic!(
107                "We do not expect to encounter `TyVar` this late in combine \
108                    -- they should have been handled earlier"
109            )
110        }
111        (_, ty::Infer(ty::FreshTy(_) | ty::FreshIntTy(_) | ty::FreshFloatTy(_)))
112        | (ty::Infer(ty::FreshTy(_) | ty::FreshIntTy(_) | ty::FreshFloatTy(_)), _)
113            if infcx.next_trait_solver() =>
114        {
115            {
    ::core::panicking::panic_fmt(format_args!("We do not expect to encounter `Fresh` variables in the new solver"));
}panic!("We do not expect to encounter `Fresh` variables in the new solver")
116        }
117
118        (_, ty::Alias(..)) | (ty::Alias(..), _) if infcx.next_trait_solver() => {
119            match relation.structurally_relate_aliases() {
120                StructurallyRelateAliases::Yes => structurally_relate_tys(relation, a, b),
121                StructurallyRelateAliases::No => {
122                    relation.register_alias_relate_predicate(a, b);
123                    Ok(a)
124                }
125            }
126        }
127
128        // All other cases of inference are errors
129        (ty::Infer(_), _) | (_, ty::Infer(_)) => Err(TypeError::Sorts(ExpectedFound::new(a, b))),
130
131        (ty::Alias(ty::AliasTy { kind: ty::Opaque { .. }, .. }), _)
132        | (_, ty::Alias(ty::AliasTy { kind: ty::Opaque { .. }, .. })) => {
133            if !!infcx.next_trait_solver() {
    ::core::panicking::panic("assertion failed: !infcx.next_trait_solver()")
};assert!(!infcx.next_trait_solver());
134            match infcx.typing_mode_raw().assert_not_erased() {
135                // During coherence, opaque types should be treated as *possibly*
136                // equal to any other type. This is an
137                // extremely heavy hammer, but can be relaxed in a forwards-compatible
138                // way later.
139                TypingMode::Coherence => {
140                    relation.register_predicates([ty::Binder::dummy(ty::PredicateKind::Ambiguous)]);
141                    Ok(a)
142                }
143                TypingMode::Analysis { .. }
144                | TypingMode::Borrowck { .. }
145                | TypingMode::PostBorrowckAnalysis { .. }
146                | TypingMode::PostAnalysis
147                | TypingMode::Codegen => structurally_relate_tys(relation, a, b),
148            }
149        }
150
151        _ => structurally_relate_tys(relation, a, b),
152    }
153}
154
155pub fn super_combine_consts<Infcx, I, R>(
156    infcx: &Infcx,
157    relation: &mut R,
158    a: I::Const,
159    b: I::Const,
160) -> RelateResult<I, I::Const>
161where
162    Infcx: InferCtxtLike<Interner = I>,
163    I: Interner,
164    R: PredicateEmittingRelation<Infcx>,
165{
166    {
    use ::tracing::__macro_support::Callsite as _;
    static __CALLSITE: ::tracing::callsite::DefaultCallsite =
        {
            static META: ::tracing::Metadata<'static> =
                {
                    ::tracing_core::metadata::Metadata::new("event compiler/rustc_type_ir/src/relate/combine.rs:166",
                        "rustc_type_ir::relate::combine", ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_type_ir/src/relate/combine.rs"),
                        ::tracing_core::__macro_support::Option::Some(166u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_type_ir::relate::combine"),
                        ::tracing_core::field::FieldSet::new(&["message"],
                            ::tracing_core::callsite::Identifier(&__CALLSITE)),
                        ::tracing::metadata::Kind::EVENT)
                };
            ::tracing::callsite::DefaultCallsite::new(&META)
        };
    let enabled =
        ::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
                &&
                ::tracing::Level::DEBUG <=
                    ::tracing::level_filters::LevelFilter::current() &&
            {
                let interest = __CALLSITE.interest();
                !interest.is_never() &&
                    ::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
                        interest)
            };
    if enabled {
        (|value_set: ::tracing::field::ValueSet|
                    {
                        let meta = __CALLSITE.metadata();
                        ::tracing::Event::dispatch(meta, &value_set);
                        ;
                    })({
                #[allow(unused_imports)]
                use ::tracing::field::{debug, display, Value};
                let mut iter = __CALLSITE.metadata().fields().iter();
                __CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
                                    ::tracing::__macro_support::Option::Some(&format_args!("super_combine_consts::<{0}>({1:?}, {2:?})",
                                                    std::any::type_name::<R>(), a, b) as &dyn Value))])
            });
    } else { ; }
};debug!("super_combine_consts::<{}>({:?}, {:?})", std::any::type_name::<R>(), a, b);
167    if true {
    if !!a.has_escaping_bound_vars() {
        ::core::panicking::panic("assertion failed: !a.has_escaping_bound_vars()")
    };
};debug_assert!(!a.has_escaping_bound_vars());
168    if true {
    if !!b.has_escaping_bound_vars() {
        ::core::panicking::panic("assertion failed: !b.has_escaping_bound_vars()")
    };
};debug_assert!(!b.has_escaping_bound_vars());
169
170    if a == b {
171        return Ok(a);
172    }
173
174    let a = infcx.shallow_resolve_const(a);
175    let b = infcx.shallow_resolve_const(b);
176
177    match (a.kind(), b.kind()) {
178        (
179            ty::ConstKind::Infer(ty::InferConst::Var(a_vid)),
180            ty::ConstKind::Infer(ty::InferConst::Var(b_vid)),
181        ) => {
182            infcx.equate_const_vids_raw(a_vid, b_vid);
183            Ok(a)
184        }
185
186        // All other cases of inference with other variables are errors.
187        (ty::ConstKind::Infer(ty::InferConst::Var(_)), ty::ConstKind::Infer(_))
188        | (ty::ConstKind::Infer(_), ty::ConstKind::Infer(ty::InferConst::Var(_))) => {
189            {
    ::core::panicking::panic_fmt(format_args!("tried to combine ConstKind::Infer/ConstKind::Infer(InferConst::Var): {0:?} and {1:?}",
            a, b));
}panic!(
190                "tried to combine ConstKind::Infer/ConstKind::Infer(InferConst::Var): {a:?} and {b:?}"
191            )
192        }
193
194        (ty::ConstKind::Infer(ty::InferConst::Var(vid)), _) => {
195            infcx.instantiate_const_var_raw(relation, true, vid, b)?;
196            Ok(b)
197        }
198
199        (_, ty::ConstKind::Infer(ty::InferConst::Var(vid))) => {
200            infcx.instantiate_const_var_raw(relation, false, vid, a)?;
201            Ok(a)
202        }
203
204        (ty::ConstKind::Unevaluated(..), _) | (_, ty::ConstKind::Unevaluated(..))
205            if infcx.cx().features().generic_const_exprs() || infcx.next_trait_solver() =>
206        {
207            match relation.structurally_relate_aliases() {
208                StructurallyRelateAliases::No => {
209                    relation.register_predicates([if infcx.next_trait_solver() {
210                        ty::PredicateKind::AliasRelate(
211                            a.into(),
212                            b.into(),
213                            ty::AliasRelationDirection::Equate,
214                        )
215                    } else {
216                        ty::PredicateKind::ConstEquate(a, b)
217                    }]);
218
219                    Ok(b)
220                }
221                StructurallyRelateAliases::Yes => structurally_relate_consts(relation, a, b),
222            }
223        }
224        _ => structurally_relate_consts(relation, a, b),
225    }
226}
227
228pub fn combine_ty_args<Infcx, I, R>(
229    infcx: &Infcx,
230    relation: &mut R,
231    a_ty: I::Ty,
232    b_ty: I::Ty,
233    variances: I::VariancesOf,
234    a_args: I::GenericArgs,
235    b_args: I::GenericArgs,
236    mk: impl FnOnce(I::GenericArgs) -> I::Ty,
237) -> RelateResult<I, I::Ty>
238where
239    Infcx: InferCtxtLike<Interner = I>,
240    I: Interner,
241    R: PredicateEmittingRelation<Infcx>,
242{
243    let cx = infcx.cx();
244    let mut has_unconstrained_bivariant_arg = false;
245    let args = iter::zip(a_args.iter(), b_args.iter()).enumerate().map(|(i, (a, b))| {
246        let variance = variances.get(i).unwrap();
247        let variance_info = match variance {
248            ty::Invariant => {
249                VarianceDiagInfo::Invariant { ty: a_ty, param_index: i.try_into().unwrap() }
250            }
251            ty::Covariant | ty::Contravariant => VarianceDiagInfo::default(),
252            ty::Bivariant => {
253                let has_non_region_infer = |arg: I::GenericArg| {
254                    arg.has_non_region_infer()
255                        && infcx.resolve_vars_if_possible(arg).has_non_region_infer()
256                };
257                if has_non_region_infer(a) || has_non_region_infer(b) {
258                    has_unconstrained_bivariant_arg = true;
259                }
260                VarianceDiagInfo::default()
261            }
262        };
263        relation.relate_with_variance(variance, variance_info, a, b)
264    });
265    let args = cx.mk_args_from_iter(args)?;
266
267    // In general, we do not check whether all types which occur during
268    // type checking are well-formed. We only check wf of user-provided types
269    // and when actually using a type, e.g. for method calls.
270    //
271    // This means that when subtyping, we may end up with unconstrained
272    // inference variables if a generalized type has bivariant parameters.
273    // A parameter may only be bivariant if it is constrained by a projection
274    // bound in a where-clause. As an example, imagine a type:
275    //
276    //     struct Foo<A, B> where A: Iterator<Item = B> {
277    //         data: A
278    //     }
279    //
280    // here, `A` will be covariant, but `B` is unconstrained. However, whatever it is,
281    // for `Foo` to be WF, it must be equal to `A::Item`.
282    //
283    // If we have an input `Foo<?A, ?B>`, then after generalization we will wind
284    // up with a type like `Foo<?C, ?D>`. When we enforce `Foo<?A, ?B> <: Foo<?C, ?D>`,
285    // we will wind up with the requirement that `?A <: ?C`, but no particular
286    // relationship between `?B` and `?D` (after all, these types may be completely
287    // different). If we do nothing else, this may mean that `?D` goes unconstrained
288    // (as in #41677). To avoid this we emit a `WellFormed` when relating types with
289    // bivariant arguments.
290    if has_unconstrained_bivariant_arg {
291        relation.register_predicates([
292            ty::ClauseKind::WellFormed(a_ty.into()),
293            ty::ClauseKind::WellFormed(b_ty.into()),
294        ]);
295    }
296
297    if a_args == args { Ok(a_ty) } else { Ok(mk(args)) }
298}