Skip to content

Commit ee25694

Browse files
committed
Auto merge of #121796 - oli-obk:eager_opaque_checks3, r=<try>
Make `DefiningAnchor::Bind` only store the opaque types that may be constrained, instead of the current infcx root item. This makes `Bind` almost always be empty, so we can start forwarding it to queries, allowing us to remove `Bubble` entirely (not done in this PR) The only behaviour change is in diagnostics. r? `@lcnr` `@compiler-errors`
2 parents c7beecf + 04461ee commit ee25694

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

51 files changed

+316
-280
lines changed

compiler/rustc_borrowck/src/consumers.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -106,7 +106,7 @@ pub fn get_body_with_borrowck_facts(
106106
options: ConsumerOptions,
107107
) -> BodyWithBorrowckFacts<'_> {
108108
let (input_body, promoted) = tcx.mir_promoted(def);
109-
let infcx = tcx.infer_ctxt().with_opaque_type_inference(DefiningAnchor::Bind(def)).build();
109+
let infcx = tcx.infer_ctxt().with_opaque_type_inference(DefiningAnchor::bind(tcx, def)).build();
110110
let input_body: &Body<'_> = &input_body.borrow();
111111
let promoted: &IndexSlice<_, _> = &promoted.borrow();
112112
*super::do_mir_borrowck(&infcx, input_body, promoted, Some(options)).1.unwrap()

compiler/rustc_borrowck/src/lib.rs

+4-2
Original file line numberDiff line numberDiff line change
@@ -127,8 +127,10 @@ fn mir_borrowck(tcx: TyCtxt<'_>, def: LocalDefId) -> &BorrowCheckResult<'_> {
127127

128128
let hir_owner = tcx.local_def_id_to_hir_id(def).owner;
129129

130-
let infcx =
131-
tcx.infer_ctxt().with_opaque_type_inference(DefiningAnchor::Bind(hir_owner.def_id)).build();
130+
let infcx = tcx
131+
.infer_ctxt()
132+
.with_opaque_type_inference(DefiningAnchor::bind(tcx, hir_owner.def_id))
133+
.build();
132134
let promoted: &IndexSlice<_, _> = &promoted.borrow();
133135
let opt_closure_req = do_mir_borrowck(&infcx, input_body, promoted, None).0;
134136
debug!("mir_borrowck done");

compiler/rustc_borrowck/src/region_infer/opaque_types.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -317,7 +317,7 @@ fn check_opaque_type_well_formed<'tcx>(
317317
let infcx = tcx
318318
.infer_ctxt()
319319
.with_next_trait_solver(next_trait_solver)
320-
.with_opaque_type_inference(DefiningAnchor::Bind(parent_def_id))
320+
.with_opaque_type_inference(DefiningAnchor::bind(tcx, parent_def_id))
321321
.build();
322322
let ocx = ObligationCtxt::new(&infcx);
323323
let identity_args = GenericArgs::identity_for_item(tcx, def_id);

compiler/rustc_hir_analysis/src/check/check.rs

+2-2
Original file line numberDiff line numberDiff line change
@@ -346,7 +346,7 @@ fn check_opaque_meets_bounds<'tcx>(
346346

347347
let infcx = tcx
348348
.infer_ctxt()
349-
.with_opaque_type_inference(DefiningAnchor::Bind(defining_use_anchor))
349+
.with_opaque_type_inference(DefiningAnchor::bind(tcx, defining_use_anchor))
350350
.build();
351351
let ocx = ObligationCtxt::new(&infcx);
352352

@@ -1569,7 +1569,7 @@ pub(super) fn check_coroutine_obligations(
15691569
.ignoring_regions()
15701570
// Bind opaque types to type checking root, as they should have been checked by borrowck,
15711571
// but may show up in some cases, like when (root) obligations are stalled in the new solver.
1572-
.with_opaque_type_inference(DefiningAnchor::Bind(typeck.hir_owner.def_id))
1572+
.with_opaque_type_inference(DefiningAnchor::bind(tcx, typeck.hir_owner.def_id))
15731573
.build();
15741574

15751575
let mut fulfillment_cx = <dyn TraitEngine<'_>>::new(&infcx);

compiler/rustc_hir_analysis/src/collect/type_of/opaque.rs

+8-20
Original file line numberDiff line numberDiff line change
@@ -46,9 +46,7 @@ pub(super) fn find_opaque_ty_constraints_for_impl_trait_in_assoc_type(
4646
for &assoc_id in tcx.associated_item_def_ids(impl_def_id) {
4747
let assoc = tcx.associated_item(assoc_id);
4848
match assoc.kind {
49-
ty::AssocKind::Const | ty::AssocKind::Fn => {
50-
locator.check(assoc_id.expect_local(), ImplTraitSource::AssocTy)
51-
}
49+
ty::AssocKind::Const | ty::AssocKind::Fn => locator.check(assoc_id.expect_local()),
5250
// Associated types don't have bodies, so they can't constrain hidden types
5351
ty::AssocKind::Type => {}
5452
}
@@ -182,15 +180,9 @@ struct TaitConstraintLocator<'tcx> {
182180
typeck_types: Vec<ty::OpaqueHiddenType<'tcx>>,
183181
}
184182

185-
#[derive(Debug)]
186-
enum ImplTraitSource {
187-
AssocTy,
188-
TyAlias,
189-
}
190-
191183
impl TaitConstraintLocator<'_> {
192184
#[instrument(skip(self), level = "debug")]
193-
fn check(&mut self, item_def_id: LocalDefId, source: ImplTraitSource) {
185+
fn check(&mut self, item_def_id: LocalDefId) {
194186
// Don't try to check items that cannot possibly constrain the type.
195187
if !self.tcx.has_typeck_results(item_def_id) {
196188
debug!("no constraint: no typeck results");
@@ -242,12 +234,8 @@ impl TaitConstraintLocator<'_> {
242234
continue;
243235
}
244236
constrained = true;
245-
let opaque_types_defined_by = match source {
246-
ImplTraitSource::AssocTy => {
247-
self.tcx.impl_trait_in_assoc_types_defined_by(item_def_id)
248-
}
249-
ImplTraitSource::TyAlias => self.tcx.opaque_types_defined_by(item_def_id),
250-
};
237+
let opaque_types_defined_by = self.tcx.opaque_types_defined_by(item_def_id);
238+
251239
if !opaque_types_defined_by.contains(&self.def_id) {
252240
self.tcx.dcx().emit_err(TaitForwardCompat {
253241
span: hidden_type.span,
@@ -308,29 +296,29 @@ impl<'tcx> intravisit::Visitor<'tcx> for TaitConstraintLocator<'tcx> {
308296
}
309297
fn visit_expr(&mut self, ex: &'tcx Expr<'tcx>) {
310298
if let hir::ExprKind::Closure(closure) = ex.kind {
311-
self.check(closure.def_id, ImplTraitSource::TyAlias);
299+
self.check(closure.def_id);
312300
}
313301
intravisit::walk_expr(self, ex);
314302
}
315303
fn visit_item(&mut self, it: &'tcx Item<'tcx>) {
316304
trace!(?it.owner_id);
317305
// The opaque type itself or its children are not within its reveal scope.
318306
if it.owner_id.def_id != self.def_id {
319-
self.check(it.owner_id.def_id, ImplTraitSource::TyAlias);
307+
self.check(it.owner_id.def_id);
320308
intravisit::walk_item(self, it);
321309
}
322310
}
323311
fn visit_impl_item(&mut self, it: &'tcx ImplItem<'tcx>) {
324312
trace!(?it.owner_id);
325313
// The opaque type itself or its children are not within its reveal scope.
326314
if it.owner_id.def_id != self.def_id {
327-
self.check(it.owner_id.def_id, ImplTraitSource::TyAlias);
315+
self.check(it.owner_id.def_id);
328316
intravisit::walk_impl_item(self, it);
329317
}
330318
}
331319
fn visit_trait_item(&mut self, it: &'tcx TraitItem<'tcx>) {
332320
trace!(?it.owner_id);
333-
self.check(it.owner_id.def_id, ImplTraitSource::TyAlias);
321+
self.check(it.owner_id.def_id);
334322
intravisit::walk_trait_item(self, it);
335323
}
336324
fn visit_foreign_item(&mut self, it: &'tcx hir::ForeignItem<'tcx>) {

compiler/rustc_hir_typeck/src/inherited.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -79,7 +79,7 @@ impl<'tcx> Inherited<'tcx> {
7979
let infcx = tcx
8080
.infer_ctxt()
8181
.ignoring_regions()
82-
.with_opaque_type_inference(DefiningAnchor::Bind(def_id))
82+
.with_opaque_type_inference(DefiningAnchor::bind(tcx, def_id))
8383
.build();
8484
let typeck_results = RefCell::new(ty::TypeckResults::new(hir_owner));
8585

compiler/rustc_infer/src/infer/mod.rs

+3-3
Original file line numberDiff line numberDiff line change
@@ -252,7 +252,7 @@ pub struct InferCtxt<'tcx> {
252252
///
253253
/// Its default value is `DefiningAnchor::Error`, this way it is easier to catch errors that
254254
/// might come up during inference or typeck.
255-
pub defining_use_anchor: DefiningAnchor,
255+
pub defining_use_anchor: DefiningAnchor<'tcx>,
256256

257257
/// Whether this inference context should care about region obligations in
258258
/// the root universe. Most notably, this is used during hir typeck as region
@@ -613,7 +613,7 @@ impl fmt::Display for FixupError {
613613
/// Used to configure inference contexts before their creation.
614614
pub struct InferCtxtBuilder<'tcx> {
615615
tcx: TyCtxt<'tcx>,
616-
defining_use_anchor: DefiningAnchor,
616+
defining_use_anchor: DefiningAnchor<'tcx>,
617617
considering_regions: bool,
618618
skip_leak_check: bool,
619619
/// Whether we are in coherence mode.
@@ -644,7 +644,7 @@ impl<'tcx> InferCtxtBuilder<'tcx> {
644644
/// It is only meant to be called in two places, for typeck
645645
/// (via `Inherited::build`) and for the inference context used
646646
/// in mir borrowck.
647-
pub fn with_opaque_type_inference(mut self, defining_use_anchor: DefiningAnchor) -> Self {
647+
pub fn with_opaque_type_inference(mut self, defining_use_anchor: DefiningAnchor<'tcx>) -> Self {
648648
self.defining_use_anchor = defining_use_anchor;
649649
self
650650
}

compiler/rustc_infer/src/infer/opaque_types.rs

+3-57
Original file line numberDiff line numberDiff line change
@@ -368,28 +368,14 @@ impl<'tcx> InferCtxt<'tcx> {
368368
/// in its defining scope.
369369
#[instrument(skip(self), level = "trace", ret)]
370370
pub fn opaque_type_origin(&self, def_id: LocalDefId) -> Option<OpaqueTyOrigin> {
371-
let opaque_hir_id = self.tcx.local_def_id_to_hir_id(def_id);
372-
let parent_def_id = match self.defining_use_anchor {
371+
let defined_opaque_types = match self.defining_use_anchor {
373372
DefiningAnchor::Bubble | DefiningAnchor::Error => return None,
374373
DefiningAnchor::Bind(bind) => bind,
375374
};
376375

377376
let origin = self.tcx.opaque_type_origin(def_id);
378-
let in_definition_scope = match origin {
379-
// Async `impl Trait`
380-
hir::OpaqueTyOrigin::AsyncFn(parent) => parent == parent_def_id,
381-
// Anonymous `impl Trait`
382-
hir::OpaqueTyOrigin::FnReturn(parent) => parent == parent_def_id,
383-
// Named `type Foo = impl Bar;`
384-
hir::OpaqueTyOrigin::TyAlias { in_assoc_ty } => {
385-
if in_assoc_ty {
386-
self.tcx.opaque_types_defined_by(parent_def_id).contains(&def_id)
387-
} else {
388-
may_define_opaque_type(self.tcx, parent_def_id, opaque_hir_id)
389-
}
390-
}
391-
};
392-
in_definition_scope.then_some(origin)
377+
378+
defined_opaque_types.contains(&def_id).then_some(origin)
393379
}
394380
}
395381

@@ -638,43 +624,3 @@ impl<'tcx> InferCtxt<'tcx> {
638624
}
639625
}
640626
}
641-
642-
/// Returns `true` if `opaque_hir_id` is a sibling or a child of a sibling of `def_id`.
643-
///
644-
/// Example:
645-
/// ```ignore UNSOLVED (is this a bug?)
646-
/// # #![feature(type_alias_impl_trait)]
647-
/// pub mod foo {
648-
/// pub mod bar {
649-
/// pub trait Bar { /* ... */ }
650-
/// pub type Baz = impl Bar;
651-
///
652-
/// # impl Bar for () {}
653-
/// fn f1() -> Baz { /* ... */ }
654-
/// }
655-
/// fn f2() -> bar::Baz { /* ... */ }
656-
/// }
657-
/// ```
658-
///
659-
/// Here, `def_id` is the `LocalDefId` of the defining use of the opaque type (e.g., `f1` or `f2`),
660-
/// and `opaque_hir_id` is the `HirId` of the definition of the opaque type `Baz`.
661-
/// For the above example, this function returns `true` for `f1` and `false` for `f2`.
662-
fn may_define_opaque_type(tcx: TyCtxt<'_>, def_id: LocalDefId, opaque_hir_id: hir::HirId) -> bool {
663-
let mut hir_id = tcx.local_def_id_to_hir_id(def_id);
664-
665-
// Named opaque types can be defined by any siblings or children of siblings.
666-
let scope = tcx.hir().get_defining_scope(opaque_hir_id);
667-
// We walk up the node tree until we hit the root or the scope of the opaque type.
668-
while hir_id != scope && hir_id != hir::CRATE_HIR_ID {
669-
hir_id = tcx.hir().get_parent_item(hir_id).into();
670-
}
671-
// Syntactically, we are allowed to define the concrete type if:
672-
let res = hir_id == scope;
673-
trace!(
674-
"may_define_opaque_type(def={:?}, opaque_node={:?}) = {}",
675-
tcx.hir_node(hir_id),
676-
tcx.hir_node(opaque_hir_id),
677-
res
678-
);
679-
res
680-
}

compiler/rustc_middle/src/mir/type_foldable.rs

+10
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
//! `TypeFoldable` implementations for MIR types
22
33
use rustc_ast::InlineAsmTemplatePiece;
4+
use rustc_hir::def_id::LocalDefId;
45

56
use super::*;
67

@@ -44,6 +45,15 @@ impl<'tcx> TypeFoldable<TyCtxt<'tcx>> for &'tcx [Span] {
4445
}
4546
}
4647

48+
impl<'tcx> TypeFoldable<TyCtxt<'tcx>> for &'tcx [LocalDefId] {
49+
fn try_fold_with<F: FallibleTypeFolder<TyCtxt<'tcx>>>(
50+
self,
51+
_folder: &mut F,
52+
) -> Result<Self, F::Error> {
53+
Ok(self)
54+
}
55+
}
56+
4757
impl<'tcx> TypeFoldable<TyCtxt<'tcx>> for &'tcx ty::List<PlaceElem<'tcx>> {
4858
fn try_fold_with<F: FallibleTypeFolder<TyCtxt<'tcx>>>(
4959
self,

compiler/rustc_middle/src/query/mod.rs

-9
Original file line numberDiff line numberDiff line change
@@ -344,15 +344,6 @@ rustc_queries! {
344344
}
345345
}
346346

347-
query impl_trait_in_assoc_types_defined_by(
348-
key: LocalDefId
349-
) -> &'tcx ty::List<LocalDefId> {
350-
desc {
351-
|tcx| "computing the opaque types defined by `{}`",
352-
tcx.def_path_str(key.to_def_id())
353-
}
354-
}
355-
356347
/// Returns the list of bounds that can be used for
357348
/// `SelectionCandidate::ProjectionCandidate(_)` and
358349
/// `ProjectionTyCandidate::TraitDef`.

compiler/rustc_middle/src/traits/mod.rs

+11-5
Original file line numberDiff line numberDiff line change
@@ -12,8 +12,8 @@ pub mod util;
1212
use crate::infer::canonical::Canonical;
1313
use crate::mir::ConstraintCategory;
1414
use crate::ty::abstract_const::NotConstEvaluatable;
15-
use crate::ty::GenericArgsRef;
1615
use crate::ty::{self, AdtKind, Ty};
16+
use crate::ty::{GenericArgsRef, TyCtxt};
1717

1818
use rustc_data_structures::sync::Lrc;
1919
use rustc_errors::{Applicability, Diag, EmissionGuarantee};
@@ -1001,10 +1001,10 @@ pub enum CodegenObligationError {
10011001
/// opaques are replaced with inference vars eagerly in the old solver (e.g.
10021002
/// in projection, and in the signature during function type-checking).
10031003
#[derive(Debug, PartialEq, Eq, Clone, Copy, Hash, HashStable, TypeFoldable, TypeVisitable)]
1004-
pub enum DefiningAnchor {
1005-
/// Define opaques which are in-scope of the `LocalDefId`. Also, eagerly
1006-
/// replace opaque types in `replace_opaque_types_with_inference_vars`.
1007-
Bind(LocalDefId),
1004+
pub enum DefiningAnchor<'tcx> {
1005+
/// Define opaques which are in-scope of the current item being analyzed.
1006+
/// Also, eagerly replace these opaque types in `replace_opaque_types_with_inference_vars`.
1007+
Bind(&'tcx [LocalDefId]),
10081008
/// In contexts where we don't currently know what opaques are allowed to be
10091009
/// defined, such as (old solver) canonical queries, we will simply allow
10101010
/// opaques to be defined, but "bubble" them up in the canonical response or
@@ -1018,3 +1018,9 @@ pub enum DefiningAnchor {
10181018
/// otherwise reveal opaques (such as [`Reveal::All`] reveal mode).
10191019
Error,
10201020
}
1021+
1022+
impl<'tcx> DefiningAnchor<'tcx> {
1023+
pub fn bind(tcx: TyCtxt<'tcx>, item: LocalDefId) -> Self {
1024+
Self::Bind(tcx.opaque_types_defined_by(item))
1025+
}
1026+
}

compiler/rustc_middle/src/traits/solve.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -115,7 +115,7 @@ impl MaybeCause {
115115
#[derive(Debug, PartialEq, Eq, Clone, Copy, Hash, HashStable, TypeFoldable, TypeVisitable)]
116116
pub struct QueryInput<'tcx, T> {
117117
pub goal: Goal<'tcx, T>,
118-
pub anchor: DefiningAnchor,
118+
pub anchor: DefiningAnchor<'tcx>,
119119
pub predefined_opaques_in_body: PredefinedOpaques<'tcx>,
120120
}
121121

compiler/rustc_trait_selection/src/traits/error_reporting/type_err_ctxt_ext.rs

+6-13
Original file line numberDiff line numberDiff line change
@@ -31,7 +31,7 @@ use rustc_hir::{GenericParam, Item, Node};
3131
use rustc_infer::infer::error_reporting::TypeErrCtxt;
3232
use rustc_infer::infer::{InferOk, TypeTrace};
3333
use rustc_middle::traits::select::OverflowError;
34-
use rustc_middle::traits::{DefiningAnchor, SignatureMismatchData};
34+
use rustc_middle::traits::SignatureMismatchData;
3535
use rustc_middle::ty::abstract_const::NotConstEvaluatable;
3636
use rustc_middle::ty::error::{ExpectedFound, TypeError};
3737
use rustc_middle::ty::fold::{BottomUpFolder, TypeFolder, TypeSuperFoldable};
@@ -3373,19 +3373,12 @@ impl<'tcx> TypeErrCtxt<'_, 'tcx> {
33733373
obligation.cause.span,
33743374
format!("cannot check whether the hidden type of {name} satisfies auto traits"),
33753375
);
3376+
3377+
err.note(
3378+
"fetching the hidden types of an opaque inside of the defining scope is not supported. \
3379+
You can try moving the opaque type and the item that actually registers a hidden type into a new submodule",
3380+
);
33763381
err.span_note(self.tcx.def_span(def_id), "opaque type is declared here");
3377-
match self.defining_use_anchor {
3378-
DefiningAnchor::Bubble | DefiningAnchor::Error => {}
3379-
DefiningAnchor::Bind(bind) => {
3380-
err.span_note(
3381-
self.tcx.def_ident_span(bind).unwrap_or_else(|| self.tcx.def_span(bind)),
3382-
"this item depends on auto traits of the hidden type, \
3383-
but may also be registering the hidden type. \
3384-
This is not supported right now. \
3385-
You can try moving the opaque type and the item that actually registers a hidden type into a new submodule".to_string(),
3386-
);
3387-
}
3388-
};
33893382

33903383
self.note_obligation_cause(&mut err, &obligation);
33913384
self.point_at_returns_when_relevant(&mut err, &obligation);

0 commit comments

Comments
 (0)