Skip to content

Commit f748bb1

Browse files
authored
Rollup merge of #111252 - matthewjasper:min-spec-improvements, r=compiler-errors
Min specialization improvements - Don't allow specialization impls with no items, such implementations are probably not correct and only occur as mistakes in the compiler and standard library - Fix a missing normalization call - Adds spans for lifetime errors from overly general specializations Closes #79457 Closes #109815
2 parents 8c51701 + f46eabb commit f748bb1

20 files changed

+346
-48
lines changed

compiler/rustc_data_structures/src/owned_slice.rs

+2
Original file line numberDiff line numberDiff line change
@@ -109,9 +109,11 @@ impl Borrow<[u8]> for OwnedSlice {
109109
}
110110

111111
// Safety: `OwnedSlice` is conceptually `(&'self.1 [u8], Box<dyn Send + Sync>)`, which is `Send`
112+
#[cfg(parallel_compiler)]
112113
unsafe impl Send for OwnedSlice {}
113114

114115
// Safety: `OwnedSlice` is conceptually `(&'self.1 [u8], Box<dyn Send + Sync>)`, which is `Sync`
116+
#[cfg(parallel_compiler)]
115117
unsafe impl Sync for OwnedSlice {}
116118

117119
#[cfg(test)]

compiler/rustc_hir_analysis/messages.ftl

+3
Original file line numberDiff line numberDiff line change
@@ -279,6 +279,9 @@ hir_analysis_specialization_trait = implementing `rustc_specialization_trait` tr
279279
hir_analysis_closure_implicit_hrtb = implicit types in closure signatures are forbidden when `for<...>` is present
280280
.label = `for<...>` is here
281281
282+
hir_analysis_empty_specialization = specialization impl does not specialize any associated items
283+
.note = impl is a specialization of this impl
284+
282285
hir_analysis_const_specialize = cannot specialize on const impl with non-const impl
283286
284287
hir_analysis_static_specialize = cannot specialize on `'static` lifetime

compiler/rustc_hir_analysis/src/errors.rs

+9
Original file line numberDiff line numberDiff line change
@@ -814,6 +814,15 @@ pub(crate) struct ClosureImplicitHrtb {
814814
pub for_sp: Span,
815815
}
816816

817+
#[derive(Diagnostic)]
818+
#[diag(hir_analysis_empty_specialization)]
819+
pub(crate) struct EmptySpecialization {
820+
#[primary_span]
821+
pub span: Span,
822+
#[note]
823+
pub base_impl_span: Span,
824+
}
825+
817826
#[derive(Diagnostic)]
818827
#[diag(hir_analysis_const_specialize)]
819828
pub(crate) struct ConstSpecialize {

compiler/rustc_hir_analysis/src/impl_wf_check/min_specialization.rs

+30-4
Original file line numberDiff line numberDiff line change
@@ -80,7 +80,7 @@ use rustc_middle::ty::{self, TyCtxt, TypeVisitableExt};
8080
use rustc_span::Span;
8181
use rustc_trait_selection::traits::error_reporting::TypeErrCtxtExt;
8282
use rustc_trait_selection::traits::outlives_bounds::InferCtxtExt as _;
83-
use rustc_trait_selection::traits::{self, translate_substs, wf, ObligationCtxt};
83+
use rustc_trait_selection::traits::{self, translate_substs_with_cause, wf, ObligationCtxt};
8484

8585
pub(super) fn check_min_specialization(tcx: TyCtxt<'_>, impl_def_id: LocalDefId) {
8686
if let Some(node) = parent_specialization_node(tcx, impl_def_id) {
@@ -100,12 +100,19 @@ fn parent_specialization_node(tcx: TyCtxt<'_>, impl1_def_id: LocalDefId) -> Opti
100100
// Implementing a normal trait isn't a specialization.
101101
return None;
102102
}
103+
if trait_def.is_marker {
104+
// Overlapping marker implementations are not really specializations.
105+
return None;
106+
}
103107
Some(impl2_node)
104108
}
105109

106110
/// Check that `impl1` is a sound specialization
107111
#[instrument(level = "debug", skip(tcx))]
108112
fn check_always_applicable(tcx: TyCtxt<'_>, impl1_def_id: LocalDefId, impl2_node: Node) {
113+
let span = tcx.def_span(impl1_def_id);
114+
check_has_items(tcx, impl1_def_id, impl2_node, span);
115+
109116
if let Some((impl1_substs, impl2_substs)) = get_impl_substs(tcx, impl1_def_id, impl2_node) {
110117
let impl2_def_id = impl2_node.def_id();
111118
debug!(?impl2_def_id, ?impl2_substs);
@@ -116,14 +123,20 @@ fn check_always_applicable(tcx: TyCtxt<'_>, impl1_def_id: LocalDefId, impl2_node
116123
unconstrained_parent_impl_substs(tcx, impl2_def_id, impl2_substs)
117124
};
118125

119-
let span = tcx.def_span(impl1_def_id);
120126
check_constness(tcx, impl1_def_id, impl2_node, span);
121127
check_static_lifetimes(tcx, &parent_substs, span);
122128
check_duplicate_params(tcx, impl1_substs, &parent_substs, span);
123129
check_predicates(tcx, impl1_def_id, impl1_substs, impl2_node, impl2_substs, span);
124130
}
125131
}
126132

133+
fn check_has_items(tcx: TyCtxt<'_>, impl1_def_id: LocalDefId, impl2_node: Node, span: Span) {
134+
if let Node::Impl(impl2_id) = impl2_node && tcx.associated_item_def_ids(impl1_def_id).is_empty() {
135+
let base_impl_span = tcx.def_span(impl2_id);
136+
tcx.sess.emit_err(errors::EmptySpecialization { span, base_impl_span });
137+
}
138+
}
139+
127140
/// Check that the specializing impl `impl1` is at least as const as the base
128141
/// impl `impl2`
129142
fn check_constness(tcx: TyCtxt<'_>, impl1_def_id: LocalDefId, impl2_node: Node, span: Span) {
@@ -167,8 +180,21 @@ fn get_impl_substs(
167180
ocx.assumed_wf_types(param_env, tcx.def_span(impl1_def_id), impl1_def_id);
168181

169182
let impl1_substs = InternalSubsts::identity_for_item(tcx, impl1_def_id);
170-
let impl2_substs =
171-
translate_substs(infcx, param_env, impl1_def_id.to_def_id(), impl1_substs, impl2_node);
183+
let impl1_span = tcx.def_span(impl1_def_id);
184+
let impl2_substs = translate_substs_with_cause(
185+
infcx,
186+
param_env,
187+
impl1_def_id.to_def_id(),
188+
impl1_substs,
189+
impl2_node,
190+
|_, span| {
191+
traits::ObligationCause::new(
192+
impl1_span,
193+
impl1_def_id,
194+
traits::ObligationCauseCode::BindingObligation(impl2_node.def_id(), span),
195+
)
196+
},
197+
);
172198

173199
let errors = ocx.select_all_or_error();
174200
if !errors.is_empty() {

compiler/rustc_middle/src/mir/mod.rs

-2
Original file line numberDiff line numberDiff line change
@@ -2728,8 +2728,6 @@ pub struct UserTypeProjection {
27282728
pub projs: Vec<ProjectionKind>,
27292729
}
27302730

2731-
impl Copy for ProjectionKind {}
2732-
27332731
impl UserTypeProjection {
27342732
pub(crate) fn index(mut self) -> Self {
27352733
self.projs.push(ProjectionElem::Index(()));

compiler/rustc_trait_selection/src/traits/coherence.rs

+3-1
Original file line numberDiff line numberDiff line change
@@ -322,7 +322,9 @@ fn negative_impl(tcx: TyCtxt<'_>, impl1_def_id: DefId, impl2_def_id: DefId) -> b
322322
let selcx = &mut SelectionContext::new(&infcx);
323323
let impl2_substs = infcx.fresh_substs_for_item(DUMMY_SP, impl2_def_id);
324324
let (subject2, obligations) =
325-
impl_subject_and_oblig(selcx, impl_env, impl2_def_id, impl2_substs);
325+
impl_subject_and_oblig(selcx, impl_env, impl2_def_id, impl2_substs, |_, _| {
326+
ObligationCause::dummy()
327+
});
326328

327329
!equate(&infcx, impl_env, subject1, subject2, obligations, impl1_def_id)
328330
}

compiler/rustc_trait_selection/src/traits/mod.rs

+3-1
Original file line numberDiff line numberDiff line change
@@ -55,7 +55,9 @@ pub use self::select::{EvaluationCache, SelectionCache, SelectionContext};
5555
pub use self::select::{EvaluationResult, IntercrateAmbiguityCause, OverflowError};
5656
pub use self::specialize::specialization_graph::FutureCompatOverlapError;
5757
pub use self::specialize::specialization_graph::FutureCompatOverlapErrorKind;
58-
pub use self::specialize::{specialization_graph, translate_substs, OverlapError};
58+
pub use self::specialize::{
59+
specialization_graph, translate_substs, translate_substs_with_cause, OverlapError,
60+
};
5961
pub use self::structural_match::{
6062
search_for_adt_const_param_violation, search_for_structural_match_violation,
6163
};

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

+51-18
Original file line numberDiff line numberDiff line change
@@ -82,6 +82,30 @@ pub fn translate_substs<'tcx>(
8282
source_impl: DefId,
8383
source_substs: SubstsRef<'tcx>,
8484
target_node: specialization_graph::Node,
85+
) -> SubstsRef<'tcx> {
86+
translate_substs_with_cause(
87+
infcx,
88+
param_env,
89+
source_impl,
90+
source_substs,
91+
target_node,
92+
|_, _| ObligationCause::dummy(),
93+
)
94+
}
95+
96+
/// Like [translate_substs], but obligations from the parent implementation
97+
/// are registered with the provided `ObligationCause`.
98+
///
99+
/// This is for reporting *region* errors from those bounds. Type errors should
100+
/// not happen because the specialization graph already checks for those, and
101+
/// will result in an ICE.
102+
pub fn translate_substs_with_cause<'tcx>(
103+
infcx: &InferCtxt<'tcx>,
104+
param_env: ty::ParamEnv<'tcx>,
105+
source_impl: DefId,
106+
source_substs: SubstsRef<'tcx>,
107+
target_node: specialization_graph::Node,
108+
cause: impl Fn(usize, Span) -> ObligationCause<'tcx>,
85109
) -> SubstsRef<'tcx> {
86110
debug!(
87111
"translate_substs({:?}, {:?}, {:?}, {:?})",
@@ -99,14 +123,13 @@ pub fn translate_substs<'tcx>(
99123
return source_substs;
100124
}
101125

102-
fulfill_implication(infcx, param_env, source_trait_ref, target_impl).unwrap_or_else(
103-
|()| {
126+
fulfill_implication(infcx, param_env, source_trait_ref, source_impl, target_impl, cause)
127+
.unwrap_or_else(|()| {
104128
bug!(
105129
"When translating substitutions from {source_impl:?} to {target_impl:?}, \
106130
the expected specialization failed to hold"
107131
)
108-
},
109-
)
132+
})
110133
}
111134
specialization_graph::Node::Trait(..) => source_trait_ref.substs,
112135
};
@@ -153,20 +176,12 @@ pub(super) fn specializes(tcx: TyCtxt<'_>, (impl1_def_id, impl2_def_id): (DefId,
153176

154177
// Create an infcx, taking the predicates of impl1 as assumptions:
155178
let infcx = tcx.infer_ctxt().build();
156-
let impl1_trait_ref =
157-
match traits::fully_normalize(&infcx, ObligationCause::dummy(), penv, impl1_trait_ref) {
158-
Ok(impl1_trait_ref) => impl1_trait_ref,
159-
Err(_errors) => {
160-
tcx.sess.delay_span_bug(
161-
tcx.def_span(impl1_def_id),
162-
format!("failed to fully normalize {impl1_trait_ref}"),
163-
);
164-
impl1_trait_ref
165-
}
166-
};
167179

168180
// Attempt to prove that impl2 applies, given all of the above.
169-
fulfill_implication(&infcx, penv, impl1_trait_ref, impl2_def_id).is_ok()
181+
fulfill_implication(&infcx, penv, impl1_trait_ref, impl1_def_id, impl2_def_id, |_, _| {
182+
ObligationCause::dummy()
183+
})
184+
.is_ok()
170185
}
171186

172187
/// Attempt to fulfill all obligations of `target_impl` after unification with
@@ -178,23 +193,41 @@ fn fulfill_implication<'tcx>(
178193
infcx: &InferCtxt<'tcx>,
179194
param_env: ty::ParamEnv<'tcx>,
180195
source_trait_ref: ty::TraitRef<'tcx>,
196+
source_impl: DefId,
181197
target_impl: DefId,
198+
error_cause: impl Fn(usize, Span) -> ObligationCause<'tcx>,
182199
) -> Result<SubstsRef<'tcx>, ()> {
183200
debug!(
184201
"fulfill_implication({:?}, trait_ref={:?} |- {:?} applies)",
185202
param_env, source_trait_ref, target_impl
186203
);
187204

205+
let source_trait_ref = match traits::fully_normalize(
206+
&infcx,
207+
ObligationCause::dummy(),
208+
param_env,
209+
source_trait_ref,
210+
) {
211+
Ok(source_trait_ref) => source_trait_ref,
212+
Err(_errors) => {
213+
infcx.tcx.sess.delay_span_bug(
214+
infcx.tcx.def_span(source_impl),
215+
format!("failed to fully normalize {source_trait_ref}"),
216+
);
217+
source_trait_ref
218+
}
219+
};
220+
188221
let source_trait = ImplSubject::Trait(source_trait_ref);
189222

190223
let selcx = &mut SelectionContext::new(&infcx);
191224
let target_substs = infcx.fresh_substs_for_item(DUMMY_SP, target_impl);
192225
let (target_trait, obligations) =
193-
util::impl_subject_and_oblig(selcx, param_env, target_impl, target_substs);
226+
util::impl_subject_and_oblig(selcx, param_env, target_impl, target_substs, error_cause);
194227

195228
// do the impls unify? If not, no specialization.
196229
let Ok(InferOk { obligations: more_obligations, .. }) =
197-
infcx.at(&ObligationCause::dummy(), param_env, ).eq(DefineOpaqueTypes::No,source_trait, target_trait)
230+
infcx.at(&ObligationCause::dummy(), param_env).eq(DefineOpaqueTypes::No, source_trait, target_trait)
198231
else {
199232
debug!(
200233
"fulfill_implication: {:?} does not unify with {:?}",

compiler/rustc_trait_selection/src/traits/util.rs

+2-2
Original file line numberDiff line numberDiff line change
@@ -197,6 +197,7 @@ pub fn impl_subject_and_oblig<'a, 'tcx>(
197197
param_env: ty::ParamEnv<'tcx>,
198198
impl_def_id: DefId,
199199
impl_substs: SubstsRef<'tcx>,
200+
cause: impl Fn(usize, Span) -> ObligationCause<'tcx>,
200201
) -> (ImplSubject<'tcx>, impl Iterator<Item = PredicateObligation<'tcx>>) {
201202
let subject = selcx.tcx().impl_subject(impl_def_id);
202203
let subject = subject.subst(selcx.tcx(), impl_substs);
@@ -208,8 +209,7 @@ pub fn impl_subject_and_oblig<'a, 'tcx>(
208209
let predicates = predicates.instantiate(selcx.tcx(), impl_substs);
209210
let InferOk { value: predicates, obligations: normalization_obligations2 } =
210211
selcx.infcx.at(&ObligationCause::dummy(), param_env).normalize(predicates);
211-
let impl_obligations =
212-
super::predicates_for_generics(|_, _| ObligationCause::dummy(), param_env, predicates);
212+
let impl_obligations = super::predicates_for_generics(cause, param_env, predicates);
213213

214214
let impl_obligations = impl_obligations
215215
.chain(normalization_obligations1.into_iter())

tests/ui/rfc-2632-const-trait-impl/specialization/const-default-bound-non-const-specialized-bound.rs

+18-6
Original file line numberDiff line numberDiff line change
@@ -12,7 +12,9 @@ trait Specialize {}
1212
trait Foo {}
1313

1414
#[const_trait]
15-
trait Bar {}
15+
trait Bar {
16+
fn bar();
17+
}
1618

1719
// bgr360: I was only able to exercise the code path that raises the
1820
// "missing ~const qualifier" error by making this base impl non-const, even
@@ -21,26 +23,36 @@ trait Bar {}
2123
impl<T> Bar for T
2224
where
2325
T: ~const Foo,
24-
{}
26+
{
27+
default fn bar() {}
28+
}
2529

2630
impl<T> Bar for T
2731
where
2832
T: Foo, //~ ERROR missing `~const` qualifier
2933
T: Specialize,
30-
{}
34+
{
35+
fn bar() {}
36+
}
3137

3238
#[const_trait]
33-
trait Baz {}
39+
trait Baz {
40+
fn baz();
41+
}
3442

3543
impl<T> const Baz for T
3644
where
3745
T: ~const Foo,
38-
{}
46+
{
47+
default fn baz() {}
48+
}
3949

4050
impl<T> const Baz for T //~ ERROR conflicting implementations of trait `Baz`
4151
where
4252
T: Foo,
4353
T: Specialize,
44-
{}
54+
{
55+
fn baz() {}
56+
}
4557

4658
fn main() {}

tests/ui/rfc-2632-const-trait-impl/specialization/const-default-bound-non-const-specialized-bound.stderr

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,11 +1,11 @@
11
error: missing `~const` qualifier for specialization
2-
--> $DIR/const-default-bound-non-const-specialized-bound.rs:28:8
2+
--> $DIR/const-default-bound-non-const-specialized-bound.rs:32:8
33
|
44
LL | T: Foo,
55
| ^^^
66

77
error[E0119]: conflicting implementations of trait `Baz`
8-
--> $DIR/const-default-bound-non-const-specialized-bound.rs:40:1
8+
--> $DIR/const-default-bound-non-const-specialized-bound.rs:50:1
99
|
1010
LL | impl<T> const Baz for T
1111
| ----------------------- first implementation here

tests/ui/rfc-2632-const-trait-impl/specialization/issue-95186-specialize-on-tilde-const.rs

+18-6
Original file line numberDiff line numberDiff line change
@@ -11,27 +11,39 @@
1111
trait Specialize {}
1212

1313
#[const_trait]
14-
trait Foo {}
14+
trait Foo {
15+
fn foo();
16+
}
1517

16-
impl<T> const Foo for T {}
18+
impl<T> const Foo for T {
19+
default fn foo() {}
20+
}
1721

1822
impl<T> const Foo for T
1923
where
2024
T: ~const Specialize,
21-
{}
25+
{
26+
fn foo() {}
27+
}
2228

2329
#[const_trait]
24-
trait Bar {}
30+
trait Bar {
31+
fn bar() {}
32+
}
2533

2634
impl<T> const Bar for T
2735
where
2836
T: ~const Foo,
29-
{}
37+
{
38+
default fn bar() {}
39+
}
3040

3141
impl<T> const Bar for T
3242
where
3343
T: ~const Foo,
3444
T: ~const Specialize,
35-
{}
45+
{
46+
fn bar() {}
47+
}
3648

3749
fn main() {}

0 commit comments

Comments
 (0)