1use std::cell::{Cell, RefCell};
2use std::ops::Deref;
34use rustc_hir::def_id::LocalDefId;
5use rustc_hir::{selfas 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;
1314use super::callee::DeferredCallResolution;
1516/// 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> {
28pub(super) infcx: InferCtxt<'tcx>,
2930pub(super) typeck_results: RefCell<ty::TypeckResults<'tcx>>,
3132pub(super) locals: RefCell<HirIdMap<Ty<'tcx>>>,
3334pub(super) fulfillment_cx: RefCell<Box<dyn TraitEngine<'tcx, FulfillmentError<'tcx>>>>,
3536// 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.
39pub(super) checked_opaque_types_storage_entries: Cell<Option<OpaqueTypeStorageEntries>>,
4041/// 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.
44pub(super) deferred_sized_obligations:
45RefCell<Vec<(Ty<'tcx>, Span, traits::ObligationCauseCode<'tcx>)>>,
4647/// 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.
54pub(super) deferred_call_resolutions: RefCell<LocalDefIdMap<Vec<DeferredCallResolution<'tcx>>>>,
5556pub(super) deferred_cast_checks: RefCell<Vec<super::cast::CastCheck<'tcx>>>,
5758pub(super) deferred_transmute_checks: RefCell<Vec<(Ty<'tcx>, Ty<'tcx>, HirId)>>,
5960pub(super) deferred_asm_checks: RefCell<Vec<(&'tcx hir::InlineAsm<'tcx>, HirId)>>,
6162pub(super) deferred_repeat_expr_checks:
63RefCell<Vec<(&'tcx hir::Expr<'tcx>, Ty<'tcx>, ty::Const<'tcx>)>>,
6465/// 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.
68pub(super) diverging_type_vars: RefCell<Vec<TyVid>>,
69}
7071impl<'tcx> Dereffor TypeckRootCtxt<'tcx> {
72type Target = InferCtxt<'tcx>;
73fn deref(&self) -> &Self::Target {
74&self.infcx
75 }
76}
7778impl<'tcx> TypeckRootCtxt<'tcx> {
79pub(crate) fn new(tcx: TyCtxt<'tcx>, def_id: LocalDefId) -> Self {
80let hir_owner = tcx.local_def_id_to_hir_id(def_id).owner;
8182let infcx = tcx83 .infer_ctxt()
84 .ignoring_regions()
85 .in_hir_typeck()
86 .build(TypingMode::typeck_for_body(tcx, def_id));
87let typeck_results = RefCell::new(ty::TypeckResults::new(hir_owner));
88let fulfillment_cx = RefCell::new(<dyn TraitEngine<'_, _>>::new(&infcx));
8990TypeckRootCtxt {
91infcx,
92typeck_results,
93 locals: RefCell::new(Default::default()),
94fulfillment_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 }
105106#[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))]107pub(super) fn register_predicate(&self, obligation: traits::PredicateObligation<'tcx>) {
108if obligation.has_escaping_bound_vars() {
109span_bug!(obligation.cause.span, "escaping bound vars in predicate {:?}", obligation);
110 }
111112self.fulfillment_cx.borrow_mut().register_predicate_obligation(self, obligation);
113 }
114115pub(super) fn register_predicates<I>(&self, obligations: I)
116where
117I: IntoIterator<Item = traits::PredicateObligation<'tcx>>,
118 {
119for obligation in obligations {
120self.register_predicate(obligation);
121 }
122 }
123124pub(super) fn register_infer_ok_obligations<T>(&self, infer_ok: InferOk<'tcx, T>) -> T {
125self.register_predicates(infer_ok.obligations);
126infer_ok.value
127 }
128}