Skip to content

Commit 7702ae1

Browse files
committed
Auto merge of #98221 - cjgillot:single-coh, r=lcnr
Perform coherence checking per impl. r? `@ghost`
2 parents 221bdb6 + 9388824 commit 7702ae1

18 files changed

+251
-284
lines changed

compiler/rustc_middle/src/query/mod.rs

+4-3
Original file line numberDiff line numberDiff line change
@@ -906,9 +906,10 @@ rustc_queries! {
906906

907907
/// Checks whether all impls in the crate pass the overlap check, returning
908908
/// which impls fail it. If all impls are correct, the returned slice is empty.
909-
query orphan_check_crate(_: ()) -> &'tcx [LocalDefId] {
910-
desc {
911-
"checking whether the immpl in the this crate follow the orphan rules",
909+
query orphan_check_impl(key: LocalDefId) -> Result<(), ErrorGuaranteed> {
910+
desc { |tcx|
911+
"checking whether impl `{}` follows the orphan rules",
912+
tcx.def_path_str(key.to_def_id()),
912913
}
913914
}
914915

compiler/rustc_trait_selection/src/traits/specialize/mod.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -452,7 +452,7 @@ fn report_conflicting_impls(
452452
match used_to_be_allowed {
453453
None => {
454454
let reported = if overlap.with_impl.is_local()
455-
|| !tcx.orphan_check_crate(()).contains(&impl_def_id)
455+
|| tcx.orphan_check_impl(impl_def_id).is_ok()
456456
{
457457
let err = struct_span_err!(tcx.sess, impl_span, E0119, "");
458458
Some(decorate(

compiler/rustc_typeck/src/coherence/mod.rs

+5-14
Original file line numberDiff line numberDiff line change
@@ -146,7 +146,7 @@ pub fn provide(providers: &mut Providers) {
146146
use self::builtin::coerce_unsized_info;
147147
use self::inherent_impls::{crate_incoherent_impls, crate_inherent_impls, inherent_impls};
148148
use self::inherent_impls_overlap::crate_inherent_impls_overlap_check;
149-
use self::orphan::orphan_check_crate;
149+
use self::orphan::orphan_check_impl;
150150

151151
*providers = Providers {
152152
coherent_trait,
@@ -155,7 +155,7 @@ pub fn provide(providers: &mut Providers) {
155155
inherent_impls,
156156
crate_inherent_impls_overlap_check,
157157
coerce_unsized_info,
158-
orphan_check_crate,
158+
orphan_check_impl,
159159
..*providers
160160
};
161161
}
@@ -171,21 +171,12 @@ fn coherent_trait(tcx: TyCtxt<'_>, def_id: DefId) {
171171

172172
check_impl(tcx, impl_def_id, trait_ref);
173173
check_object_overlap(tcx, impl_def_id, trait_ref);
174-
}
175-
builtin::check_trait(tcx, def_id);
176-
}
177174

178-
pub fn check_coherence(tcx: TyCtxt<'_>) {
179-
tcx.sess.time("unsafety_checking", || unsafety::check(tcx));
180-
tcx.ensure().orphan_check_crate(());
181-
182-
for &trait_def_id in tcx.all_local_trait_impls(()).keys() {
183-
tcx.ensure().coherent_trait(trait_def_id);
175+
tcx.sess.time("unsafety_checking", || unsafety::check_item(tcx, impl_def_id));
176+
tcx.sess.time("orphan_checking", || tcx.ensure().orphan_check_impl(impl_def_id));
184177
}
185178

186-
// these queries are executed for side-effects (error reporting):
187-
tcx.ensure().crate_inherent_impls(());
188-
tcx.ensure().crate_inherent_impls_overlap_check(());
179+
builtin::check_trait(tcx, def_id);
189180
}
190181

191182
/// Checks whether an impl overlaps with the automatic `impl Trait for dyn Trait`.

compiler/rustc_typeck/src/coherence/orphan.rs

+86-90
Original file line numberDiff line numberDiff line change
@@ -18,26 +18,29 @@ use rustc_span::Span;
1818
use rustc_trait_selection::traits;
1919
use std::ops::ControlFlow;
2020

21-
pub(super) fn orphan_check_crate(tcx: TyCtxt<'_>, (): ()) -> &[LocalDefId] {
22-
let mut errors = Vec::new();
23-
for (&trait_def_id, impls_of_trait) in tcx.all_local_trait_impls(()) {
24-
for &impl_of_trait in impls_of_trait {
25-
match orphan_check_impl(tcx, impl_of_trait) {
26-
Ok(()) => {}
27-
Err(_) => errors.push(impl_of_trait),
28-
}
29-
}
21+
#[instrument(skip(tcx), level = "debug")]
22+
pub(crate) fn orphan_check_impl(
23+
tcx: TyCtxt<'_>,
24+
impl_def_id: LocalDefId,
25+
) -> Result<(), ErrorGuaranteed> {
26+
let trait_ref = tcx.impl_trait_ref(impl_def_id).unwrap();
27+
if let Some(err) = trait_ref.error_reported() {
28+
return Err(err);
29+
}
3030

31-
if tcx.trait_is_auto(trait_def_id) {
32-
lint_auto_trait_impls(tcx, trait_def_id, impls_of_trait);
33-
}
31+
let ret = do_orphan_check_impl(tcx, trait_ref, impl_def_id);
32+
if tcx.trait_is_auto(trait_ref.def_id) {
33+
lint_auto_trait_impl(tcx, trait_ref, impl_def_id);
3434
}
35-
tcx.arena.alloc_slice(&errors)
35+
36+
ret
3637
}
3738

38-
#[instrument(skip(tcx), level = "debug")]
39-
fn orphan_check_impl(tcx: TyCtxt<'_>, def_id: LocalDefId) -> Result<(), ErrorGuaranteed> {
40-
let trait_ref = tcx.impl_trait_ref(def_id).unwrap();
39+
fn do_orphan_check_impl<'tcx>(
40+
tcx: TyCtxt<'tcx>,
41+
trait_ref: ty::TraitRef<'tcx>,
42+
def_id: LocalDefId,
43+
) -> Result<(), ErrorGuaranteed> {
4144
let trait_def_id = trait_ref.def_id;
4245

4346
let item = tcx.hir().item(hir::ItemId { def_id });
@@ -329,89 +332,82 @@ fn emit_orphan_check_error<'tcx>(
329332

330333
/// Lint impls of auto traits if they are likely to have
331334
/// unsound or surprising effects on auto impls.
332-
fn lint_auto_trait_impls(tcx: TyCtxt<'_>, trait_def_id: DefId, impls: &[LocalDefId]) {
333-
let mut non_covering_impls = Vec::new();
334-
for &impl_def_id in impls {
335-
let trait_ref = tcx.impl_trait_ref(impl_def_id).unwrap();
336-
if trait_ref.references_error() {
337-
return;
338-
}
335+
fn lint_auto_trait_impl<'tcx>(
336+
tcx: TyCtxt<'tcx>,
337+
trait_ref: ty::TraitRef<'tcx>,
338+
impl_def_id: LocalDefId,
339+
) {
340+
if tcx.impl_polarity(impl_def_id) != ImplPolarity::Positive {
341+
return;
342+
}
339343

340-
if tcx.impl_polarity(impl_def_id) != ImplPolarity::Positive {
344+
assert_eq!(trait_ref.substs.len(), 1);
345+
let self_ty = trait_ref.self_ty();
346+
let (self_type_did, substs) = match self_ty.kind() {
347+
ty::Adt(def, substs) => (def.did(), substs),
348+
_ => {
349+
// FIXME: should also lint for stuff like `&i32` but
350+
// considering that auto traits are unstable, that
351+
// isn't too important for now as this only affects
352+
// crates using `nightly`, and std.
341353
return;
342354
}
355+
};
343356

344-
assert_eq!(trait_ref.substs.len(), 1);
345-
let self_ty = trait_ref.self_ty();
346-
let (self_type_did, substs) = match self_ty.kind() {
347-
ty::Adt(def, substs) => (def.did(), substs),
348-
_ => {
349-
// FIXME: should also lint for stuff like `&i32` but
350-
// considering that auto traits are unstable, that
351-
// isn't too important for now as this only affects
352-
// crates using `nightly`, and std.
353-
continue;
354-
}
355-
};
357+
// Impls which completely cover a given root type are fine as they
358+
// disable auto impls entirely. So only lint if the substs
359+
// are not a permutation of the identity substs.
360+
let Err(arg) = tcx.uses_unique_generic_params(substs, IgnoreRegions::Yes) else {
361+
// ok
362+
return;
363+
};
356364

357-
// Impls which completely cover a given root type are fine as they
358-
// disable auto impls entirely. So only lint if the substs
359-
// are not a permutation of the identity substs.
360-
match tcx.uses_unique_generic_params(substs, IgnoreRegions::Yes) {
361-
Ok(()) => {} // ok
362-
Err(arg) => {
363-
// Ideally:
364-
//
365-
// - compute the requirements for the auto impl candidate
366-
// - check whether these are implied by the non covering impls
367-
// - if not, emit the lint
368-
//
369-
// What we do here is a bit simpler:
370-
//
371-
// - badly check if an auto impl candidate definitely does not apply
372-
// for the given simplified type
373-
// - if so, do not lint
374-
if fast_reject_auto_impl(tcx, trait_def_id, self_ty) {
375-
// ok
376-
} else {
377-
non_covering_impls.push((impl_def_id, self_type_did, arg));
378-
}
379-
}
380-
}
365+
// Ideally:
366+
//
367+
// - compute the requirements for the auto impl candidate
368+
// - check whether these are implied by the non covering impls
369+
// - if not, emit the lint
370+
//
371+
// What we do here is a bit simpler:
372+
//
373+
// - badly check if an auto impl candidate definitely does not apply
374+
// for the given simplified type
375+
// - if so, do not lint
376+
if fast_reject_auto_impl(tcx, trait_ref.def_id, self_ty) {
377+
// ok
378+
return;
381379
}
382380

383-
for &(impl_def_id, self_type_did, arg) in &non_covering_impls {
384-
tcx.struct_span_lint_hir(
385-
lint::builtin::SUSPICIOUS_AUTO_TRAIT_IMPLS,
386-
tcx.hir().local_def_id_to_hir_id(impl_def_id),
387-
tcx.def_span(impl_def_id),
388-
|err| {
389-
let item_span = tcx.def_span(self_type_did);
390-
let self_descr = tcx.def_kind(self_type_did).descr(self_type_did);
391-
let mut err = err.build(&format!(
392-
"cross-crate traits with a default impl, like `{}`, \
381+
tcx.struct_span_lint_hir(
382+
lint::builtin::SUSPICIOUS_AUTO_TRAIT_IMPLS,
383+
tcx.hir().local_def_id_to_hir_id(impl_def_id),
384+
tcx.def_span(impl_def_id),
385+
|err| {
386+
let item_span = tcx.def_span(self_type_did);
387+
let self_descr = tcx.def_kind(self_type_did).descr(self_type_did);
388+
let mut err = err.build(&format!(
389+
"cross-crate traits with a default impl, like `{}`, \
393390
should not be specialized",
394-
tcx.def_path_str(trait_def_id),
395-
));
396-
match arg {
397-
ty::util::NotUniqueParam::DuplicateParam(arg) => {
398-
err.note(&format!("`{}` is mentioned multiple times", arg));
399-
}
400-
ty::util::NotUniqueParam::NotParam(arg) => {
401-
err.note(&format!("`{}` is not a generic parameter", arg));
402-
}
391+
tcx.def_path_str(trait_ref.def_id),
392+
));
393+
match arg {
394+
ty::util::NotUniqueParam::DuplicateParam(arg) => {
395+
err.note(&format!("`{}` is mentioned multiple times", arg));
403396
}
404-
err.span_note(
405-
item_span,
406-
&format!(
407-
"try using the same sequence of generic parameters as the {} definition",
408-
self_descr,
409-
),
410-
);
411-
err.emit();
412-
},
413-
);
414-
}
397+
ty::util::NotUniqueParam::NotParam(arg) => {
398+
err.note(&format!("`{}` is not a generic parameter", arg));
399+
}
400+
}
401+
err.span_note(
402+
item_span,
403+
&format!(
404+
"try using the same sequence of generic parameters as the {} definition",
405+
self_descr,
406+
),
407+
);
408+
err.emit();
409+
},
410+
);
415411
}
416412

417413
fn fast_reject_auto_impl<'tcx>(tcx: TyCtxt<'tcx>, trait_def_id: DefId, self_ty: Ty<'tcx>) -> bool {

compiler/rustc_typeck/src/coherence/unsafety.rs

+8-27
Original file line numberDiff line numberDiff line change
@@ -6,37 +6,18 @@ use rustc_hir as hir;
66
use rustc_hir::def::DefKind;
77
use rustc_hir::Unsafety;
88
use rustc_middle::ty::TyCtxt;
9+
use rustc_span::def_id::LocalDefId;
910

10-
pub fn check(tcx: TyCtxt<'_>) {
11-
for id in tcx.hir().items() {
12-
if matches!(tcx.def_kind(id.def_id), DefKind::Impl) {
13-
let item = tcx.hir().item(id);
14-
if let hir::ItemKind::Impl(ref impl_) = item.kind {
15-
check_unsafety_coherence(
16-
tcx,
17-
item,
18-
Some(&impl_.generics),
19-
impl_.unsafety,
20-
impl_.polarity,
21-
);
22-
}
23-
}
24-
}
25-
}
11+
pub(super) fn check_item(tcx: TyCtxt<'_>, def_id: LocalDefId) {
12+
debug_assert!(matches!(tcx.def_kind(def_id), DefKind::Impl));
13+
let item = tcx.hir().expect_item(def_id);
14+
let hir::ItemKind::Impl(ref impl_) = item.kind else { bug!() };
2615

27-
fn check_unsafety_coherence<'tcx>(
28-
tcx: TyCtxt<'tcx>,
29-
item: &hir::Item<'_>,
30-
impl_generics: Option<&hir::Generics<'_>>,
31-
unsafety: hir::Unsafety,
32-
polarity: hir::ImplPolarity,
33-
) {
3416
if let Some(trait_ref) = tcx.impl_trait_ref(item.def_id) {
3517
let trait_def = tcx.trait_def(trait_ref.def_id);
36-
let unsafe_attr = impl_generics.and_then(|generics| {
37-
generics.params.iter().find(|p| p.pure_wrt_drop).map(|_| "may_dangle")
38-
});
39-
match (trait_def.unsafety, unsafe_attr, unsafety, polarity) {
18+
let unsafe_attr =
19+
impl_.generics.params.iter().find(|p| p.pure_wrt_drop).map(|_| "may_dangle");
20+
match (trait_def.unsafety, unsafe_attr, impl_.unsafety, impl_.polarity) {
4021
(Unsafety::Normal, None, Unsafety::Unsafe, hir::ImplPolarity::Positive) => {
4122
struct_span_err!(
4223
tcx.sess,

compiler/rustc_typeck/src/lib.rs

+9-1
Original file line numberDiff line numberDiff line change
@@ -513,7 +513,15 @@ pub fn check_crate(tcx: TyCtxt<'_>) -> Result<(), ErrorGuaranteed> {
513513
})?;
514514

515515
tcx.sess.track_errors(|| {
516-
tcx.sess.time("coherence_checking", || coherence::check_coherence(tcx));
516+
tcx.sess.time("coherence_checking", || {
517+
for &trait_def_id in tcx.all_local_trait_impls(()).keys() {
518+
tcx.ensure().coherent_trait(trait_def_id);
519+
}
520+
521+
// these queries are executed for side-effects (error reporting):
522+
tcx.ensure().crate_inherent_impls(());
523+
tcx.ensure().crate_inherent_impls_overlap_check(());
524+
});
517525
})?;
518526

519527
if tcx.features().rustc_attrs {

src/test/ui/coherence/coherence-impl-trait-for-marker-trait-negative.stderr

+12-12
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,15 @@
1+
error[E0371]: the object type `(dyn Object + Marker2 + 'static)` automatically implements the trait `Marker1`
2+
--> $DIR/coherence-impl-trait-for-marker-trait-negative.rs:15:1
3+
|
4+
LL | impl !Marker1 for dyn Object + Marker2 { }
5+
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ `(dyn Object + Marker2 + 'static)` automatically implements trait `Marker1`
6+
7+
error[E0371]: the object type `(dyn Object + Marker2 + 'static)` automatically implements the trait `Marker2`
8+
--> $DIR/coherence-impl-trait-for-marker-trait-negative.rs:17:1
9+
|
10+
LL | impl !Marker2 for dyn Object + Marker2 { }
11+
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ `(dyn Object + Marker2 + 'static)` automatically implements trait `Marker2`
12+
113
error[E0117]: only traits defined in the current crate can be implemented for arbitrary types
214
--> $DIR/coherence-impl-trait-for-marker-trait-negative.rs:23:1
315
|
@@ -21,18 +33,6 @@ error[E0321]: cross-crate traits with a default impl, like `Send`, can only be i
2133
LL | impl !Send for dyn Object + Marker2 {}
2234
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ can't implement cross-crate trait with a default impl for non-struct/enum type
2335

24-
error[E0371]: the object type `(dyn Object + Marker2 + 'static)` automatically implements the trait `Marker1`
25-
--> $DIR/coherence-impl-trait-for-marker-trait-negative.rs:15:1
26-
|
27-
LL | impl !Marker1 for dyn Object + Marker2 { }
28-
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ `(dyn Object + Marker2 + 'static)` automatically implements trait `Marker1`
29-
30-
error[E0371]: the object type `(dyn Object + Marker2 + 'static)` automatically implements the trait `Marker2`
31-
--> $DIR/coherence-impl-trait-for-marker-trait-negative.rs:17:1
32-
|
33-
LL | impl !Marker2 for dyn Object + Marker2 { }
34-
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ `(dyn Object + Marker2 + 'static)` automatically implements trait `Marker2`
35-
3636
error: aborting due to 5 previous errors
3737

3838
Some errors have detailed explanations: E0117, E0321, E0371.

0 commit comments

Comments
 (0)