Skip to main content

rustc_hir_typeck/
typeck_root_ctxt.rs

1use std::cell::{Cell, RefCell};
2use std::ops::Deref;
3
4use rustc_hir::def_id::LocalDefId;
5use rustc_hir::{self as hir, HirId, HirIdMap};
6use rustc_infer::infer::{InferCtxt, InferOk, OpaqueTypeStorageEntries, TyCtxtInferExt};
7use rustc_middle::span_bug;
8use rustc_middle::ty::{self, Ty, TyCtxt, TyVid, TypeVisitableExt, TypingMode};
9use rustc_span::Span;
10use rustc_span::def_id::LocalDefIdMap;
11use rustc_trait_selection::traits::{self, FulfillmentError, TraitEngine, TraitEngineExt as _};
12use tracing::instrument;
13
14use super::callee::DeferredCallResolution;
15
16/// Data shared between a "typeck root" and its nested bodies,
17/// e.g. closures defined within the function. For example:
18/// ```ignore (illustrative)
19/// fn foo() {
20///     bar(move || { ... })
21/// }
22/// ```
23/// Here, the function `foo()` and the closure passed to
24/// `bar()` will each have their own `FnCtxt`, but they will
25/// share the inference context, will process obligations together,
26/// can access each other's local types (scoping permitted), etc.
27pub(crate) struct TypeckRootCtxt<'tcx> {
28    pub(super) infcx: InferCtxt<'tcx>,
29
30    pub(super) typeck_results: RefCell<ty::TypeckResults<'tcx>>,
31
32    pub(super) locals: RefCell<HirIdMap<Ty<'tcx>>>,
33
34    pub(super) fulfillment_cx: RefCell<Box<dyn TraitEngine<'tcx, FulfillmentError<'tcx>>>>,
35
36    // Used to detect opaque types uses added after we've already checked them.
37    //
38    // See [FnCtxt::detect_opaque_types_added_during_writeback] for more details.
39    pub(super) checked_opaque_types_storage_entries: Cell<Option<OpaqueTypeStorageEntries>>,
40
41    /// Some additional `Sized` obligations badly affect type inference.
42    /// These obligations are added in a later stage of typeck.
43    /// Removing these may also cause additional complications, see #101066.
44    pub(super) deferred_sized_obligations:
45        RefCell<Vec<(Ty<'tcx>, Span, traits::ObligationCauseCode<'tcx>)>>,
46
47    /// When we process a call like `c()` where `c` is a closure type,
48    /// we may not have decided yet whether `c` is a `Fn`, `FnMut`, or
49    /// `FnOnce` closure. In that case, we defer full resolution of the
50    /// call until upvar inference can kick in and make the
51    /// decision. We keep these deferred resolutions grouped by the
52    /// def-id of the closure, so that once we decide, we can easily go
53    /// back and process them.
54    pub(super) deferred_call_resolutions: RefCell<LocalDefIdMap<Vec<DeferredCallResolution<'tcx>>>>,
55
56    pub(super) deferred_cast_checks: RefCell<Vec<super::cast::CastCheck<'tcx>>>,
57
58    pub(super) deferred_transmute_checks: RefCell<Vec<(Ty<'tcx>, Ty<'tcx>, HirId)>>,
59
60    pub(super) deferred_asm_checks: RefCell<Vec<(&'tcx hir::InlineAsm<'tcx>, HirId)>>,
61
62    pub(super) deferred_repeat_expr_checks:
63        RefCell<Vec<(&'tcx hir::Expr<'tcx>, Ty<'tcx>, ty::Const<'tcx>)>>,
64
65    /// Whenever we introduce an adjustment from `!` into a type variable,
66    /// we record that type variable here. This is later used to inform
67    /// fallback. See the `fallback` module for details.
68    pub(super) diverging_type_vars: RefCell<Vec<TyVid>>,
69}
70
71impl<'tcx> Deref for TypeckRootCtxt<'tcx> {
72    type Target = InferCtxt<'tcx>;
73    fn deref(&self) -> &Self::Target {
74        &self.infcx
75    }
76}
77
78impl<'tcx> TypeckRootCtxt<'tcx> {
79    pub(crate) fn new(tcx: TyCtxt<'tcx>, def_id: LocalDefId) -> Self {
80        let hir_owner = tcx.local_def_id_to_hir_id(def_id).owner;
81
82        let infcx = tcx
83            .infer_ctxt()
84            .ignoring_regions()
85            .in_hir_typeck()
86            .build(TypingMode::typeck_for_body(tcx, def_id));
87        let typeck_results = RefCell::new(ty::TypeckResults::new(hir_owner));
88        let fulfillment_cx = RefCell::new(<dyn TraitEngine<'_, _>>::new(&infcx));
89
90        TypeckRootCtxt {
91            infcx,
92            typeck_results,
93            locals: RefCell::new(Default::default()),
94            fulfillment_cx,
95            checked_opaque_types_storage_entries: Cell::new(None),
96            deferred_sized_obligations: RefCell::new(Vec::new()),
97            deferred_call_resolutions: RefCell::new(Default::default()),
98            deferred_cast_checks: RefCell::new(Vec::new()),
99            deferred_transmute_checks: RefCell::new(Vec::new()),
100            deferred_asm_checks: RefCell::new(Vec::new()),
101            deferred_repeat_expr_checks: RefCell::new(Vec::new()),
102            diverging_type_vars: RefCell::new(Default::default()),
103        }
104    }
105
106    #[allow(clippy :: suspicious_else_formatting)]
{
    let __tracing_attr_span;
    let __tracing_attr_guard;
    if ::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
                &&
                ::tracing::Level::DEBUG <=
                    ::tracing::level_filters::LevelFilter::current() ||
            { false } {
        __tracing_attr_span =
            {
                use ::tracing::__macro_support::Callsite as _;
                static __CALLSITE: ::tracing::callsite::DefaultCallsite =
                    {
                        static META: ::tracing::Metadata<'static> =
                            {
                                ::tracing_core::metadata::Metadata::new("register_predicate",
                                    "rustc_hir_typeck::typeck_root_ctxt",
                                    ::tracing::Level::DEBUG,
                                    ::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/typeck_root_ctxt.rs"),
                                    ::tracing_core::__macro_support::Option::Some(106u32),
                                    ::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::typeck_root_ctxt"),
                                    ::tracing_core::field::FieldSet::new(&["obligation"],
                                        ::tracing_core::callsite::Identifier(&__CALLSITE)),
                                    ::tracing::metadata::Kind::SPAN)
                            };
                        ::tracing::callsite::DefaultCallsite::new(&META)
                    };
                let mut interest = ::tracing::subscriber::Interest::never();
                if ::tracing::Level::DEBUG <=
                                    ::tracing::level_filters::STATIC_MAX_LEVEL &&
                                ::tracing::Level::DEBUG <=
                                    ::tracing::level_filters::LevelFilter::current() &&
                            { interest = __CALLSITE.interest(); !interest.is_never() }
                        &&
                        ::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
                            interest) {
                    let meta = __CALLSITE.metadata();
                    ::tracing::Span::new(meta,
                        &{
                                #[allow(unused_imports)]
                                use ::tracing::field::{debug, display, Value};
                                let mut iter = meta.fields().iter();
                                meta.fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
                                                    ::tracing::__macro_support::Option::Some(&::tracing::field::debug(&obligation)
                                                            as &dyn Value))])
                            })
                } else {
                    let span =
                        ::tracing::__macro_support::__disabled_span(__CALLSITE.metadata());
                    {};
                    span
                }
            };
        __tracing_attr_guard = __tracing_attr_span.enter();
    }

    #[warn(clippy :: suspicious_else_formatting)]
    {

        #[allow(unknown_lints, unreachable_code, clippy ::
        diverging_sub_expression, clippy :: empty_loop, clippy ::
        let_unit_value, clippy :: let_with_type_underscore, clippy ::
        needless_return, clippy :: unreachable)]
        if false {
            let __tracing_attr_fake_return: () = loop {};
            return __tracing_attr_fake_return;
        }
        {
            if obligation.has_escaping_bound_vars() {
                ::rustc_middle::util::bug::span_bug_fmt(obligation.cause.span,
                    format_args!("escaping bound vars in predicate {0:?}",
                        obligation));
            }
            self.fulfillment_cx.borrow_mut().register_predicate_obligation(self,
                obligation);
        }
    }
}#[instrument(level = "debug", skip(self))]
107    pub(super) fn register_predicate(&self, obligation: traits::PredicateObligation<'tcx>) {
108        if obligation.has_escaping_bound_vars() {
109            span_bug!(obligation.cause.span, "escaping bound vars in predicate {:?}", obligation);
110        }
111
112        self.fulfillment_cx.borrow_mut().register_predicate_obligation(self, obligation);
113    }
114
115    pub(super) fn register_predicates<I>(&self, obligations: I)
116    where
117        I: IntoIterator<Item = traits::PredicateObligation<'tcx>>,
118    {
119        for obligation in obligations {
120            self.register_predicate(obligation);
121        }
122    }
123
124    pub(super) fn register_infer_ok_obligations<T>(&self, infer_ok: InferOk<'tcx, T>) -> T {
125        self.register_predicates(infer_ok.obligations);
126        infer_ok.value
127    }
128}