Skip to content

Commit ad23942

Browse files
committed
Auto merge of #111680 - Dylan-DPC:rollup-1p45gxt, r=Dylan-DPC
Rollup of 6 pull requests Successful merges: - #110930 (Don't expect normalization to succeed in elaborate_drops) - #111557 (Revert "Validate resolution for SelfCtor too.") - #111565 (rustdoc-json: Add tests for visibility of impls) - #111588 (Emits E0599 when meeting `MyTrait::missing_method`) - #111625 (Move rustc_middle/src/ty/query.rs to rustc_middle/src/query/plumbing.rs) - #111674 (Add missing backslash in HTML string) Failed merges: r? `@ghost` `@rustbot` modify labels: rollup
2 parents c2ccc85 + b0d92c2 commit ad23942

File tree

40 files changed

+394
-137
lines changed

40 files changed

+394
-137
lines changed

compiler/rustc_const_eval/src/const_eval/error.rs

+2-1
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,8 @@ use std::fmt;
33

44
use rustc_errors::Diagnostic;
55
use rustc_middle::mir::AssertKind;
6-
use rustc_middle::ty::{layout::LayoutError, query::TyCtxtAt, ConstInt};
6+
use rustc_middle::query::TyCtxtAt;
7+
use rustc_middle::ty::{layout::LayoutError, ConstInt};
78
use rustc_span::{Span, Symbol};
89

910
use super::InterpCx;

compiler/rustc_const_eval/src/interpret/eval_context.rs

+2-3
Original file line numberDiff line numberDiff line change
@@ -8,13 +8,12 @@ use rustc_hir::{self as hir, def_id::DefId, definitions::DefPathData};
88
use rustc_index::IndexVec;
99
use rustc_middle::mir;
1010
use rustc_middle::mir::interpret::{ErrorHandled, InterpError, ReportedErrorInfo};
11+
use rustc_middle::query::TyCtxtAt;
1112
use rustc_middle::ty::layout::{
1213
self, FnAbiError, FnAbiOfHelpers, FnAbiRequest, LayoutError, LayoutOf, LayoutOfHelpers,
1314
TyAndLayout,
1415
};
15-
use rustc_middle::ty::{
16-
self, query::TyCtxtAt, subst::SubstsRef, ParamEnv, Ty, TyCtxt, TypeFoldable,
17-
};
16+
use rustc_middle::ty::{self, subst::SubstsRef, ParamEnv, Ty, TyCtxt, TypeFoldable};
1817
use rustc_mir_dataflow::storage::always_storage_live_locals;
1918
use rustc_session::Limit;
2019
use rustc_span::Span;

compiler/rustc_errors/src/lib.rs

+1
Original file line numberDiff line numberDiff line change
@@ -478,6 +478,7 @@ pub enum StashKey {
478478
/// FRU syntax
479479
MaybeFruTypo,
480480
CallAssocMethod,
481+
TraitMissingMethod,
481482
}
482483

483484
fn default_track_diagnostic(d: &mut Diagnostic, f: &mut dyn FnMut(&mut Diagnostic)) {

compiler/rustc_hir_analysis/src/astconv/mod.rs

+3-4
Original file line numberDiff line numberDiff line change
@@ -19,7 +19,7 @@ use rustc_ast::TraitObjectSyntax;
1919
use rustc_data_structures::fx::{FxHashMap, FxHashSet};
2020
use rustc_errors::{
2121
struct_span_err, Applicability, Diagnostic, DiagnosticBuilder, ErrorGuaranteed, FatalError,
22-
MultiSpan,
22+
MultiSpan, StashKey,
2323
};
2424
use rustc_hir as hir;
2525
use rustc_hir::def::{CtorOf, DefKind, Namespace, Res};
@@ -38,7 +38,6 @@ use rustc_middle::ty::{self, Const, IsSuggestable, Ty, TyCtxt, TypeVisitableExt}
3838
use rustc_middle::ty::{DynKind, ToPredicate};
3939
use rustc_session::lint::builtin::{AMBIGUOUS_ASSOCIATED_ITEMS, BARE_TRAIT_OBJECTS};
4040
use rustc_span::edit_distance::find_best_match_for_name;
41-
use rustc_span::edition::Edition;
4241
use rustc_span::symbol::{kw, Ident, Symbol};
4342
use rustc_span::{sym, Span, DUMMY_SP};
4443
use rustc_target::spec::abi;
@@ -3718,7 +3717,7 @@ impl<'o, 'tcx> dyn AstConv<'tcx> + 'o {
37183717
));
37193718
}
37203719

3721-
if self_ty.span.edition() >= Edition::Edition2021 {
3720+
if self_ty.span.edition().rust_2021() {
37223721
let msg = "trait objects must include the `dyn` keyword";
37233722
let label = "add `dyn` keyword before this trait";
37243723
let mut diag =
@@ -3732,7 +3731,7 @@ impl<'o, 'tcx> dyn AstConv<'tcx> + 'o {
37323731
}
37333732
// check if the impl trait that we are considering is a impl of a local trait
37343733
self.maybe_lint_blanket_trait_impl(&self_ty, &mut diag);
3735-
diag.emit();
3734+
diag.stash(self_ty.span, StashKey::TraitMissingMethod);
37363735
} else {
37373736
let msg = "trait objects without an explicit `dyn` are deprecated";
37383737
tcx.struct_span_lint_hir(

compiler/rustc_hir_typeck/messages.ftl

+11
Original file line numberDiff line numberDiff line change
@@ -79,3 +79,14 @@ hir_typeck_arg_mismatch_indeterminate = argument type mismatch was detected, but
7979
hir_typeck_suggest_boxing_note = for more on the distinction between the stack and the heap, read https://doc.rust-lang.org/book/ch15-01-box.html, https://doc.rust-lang.org/rust-by-example/std/box.html, and https://doc.rust-lang.org/std/boxed/index.html
8080
8181
hir_typeck_suggest_boxing_when_appropriate = store this in the heap by calling `Box::new`
82+
83+
hir_typeck_no_associated_item = no {$item_kind} named `{$item_name}` found for {$ty_prefix} `{$ty_str}`{$trait_missing_method ->
84+
[true] {""}
85+
*[other] {" "}in the current scope
86+
}
87+
88+
hir_typeck_candidate_trait_note = `{$trait_name}` defines an item `{$item_name}`{$action_or_ty ->
89+
[NONE] {""}
90+
[implement] , perhaps you need to implement it
91+
*[other] , perhaps you need to restrict type parameter `{$action_or_ty}` with it
92+
}

compiler/rustc_hir_typeck/src/errors.rs

+24
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,6 @@
11
//! Errors emitted by `rustc_hir_typeck`.
2+
use std::borrow::Cow;
3+
24
use crate::fluent_generated as fluent;
35
use rustc_errors::{AddToDiagnostic, Applicability, Diagnostic, MultiSpan, SubdiagnosticMessage};
46
use rustc_macros::{Diagnostic, Subdiagnostic};
@@ -295,3 +297,25 @@ pub enum SuggestBoxing {
295297
end: Span,
296298
},
297299
}
300+
301+
#[derive(Diagnostic)]
302+
#[diag(hir_typeck_no_associated_item, code = "E0599")]
303+
pub struct NoAssociatedItem {
304+
#[primary_span]
305+
pub span: Span,
306+
pub item_kind: &'static str,
307+
pub item_name: Ident,
308+
pub ty_prefix: Cow<'static, str>,
309+
pub ty_str: String,
310+
pub trait_missing_method: bool,
311+
}
312+
313+
#[derive(Subdiagnostic)]
314+
#[note(hir_typeck_candidate_trait_note)]
315+
pub struct CandidateTraitNote {
316+
#[primary_span]
317+
pub span: Span,
318+
pub trait_name: String,
319+
pub item_name: Ident,
320+
pub action_or_ty: String,
321+
}

compiler/rustc_hir_typeck/src/expr.rs

+1
Original file line numberDiff line numberDiff line change
@@ -1245,6 +1245,7 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
12451245
error,
12461246
Some((rcvr, args)),
12471247
expected,
1248+
false,
12481249
) {
12491250
err.emit();
12501251
}

compiler/rustc_hir_typeck/src/fn_ctxt/_impl.rs

+24-2
Original file line numberDiff line numberDiff line change
@@ -4,7 +4,7 @@ use crate::rvalue_scopes;
44
use crate::{BreakableCtxt, Diverges, Expectation, FnCtxt, LocalTy, RawTy};
55
use rustc_data_structures::captures::Captures;
66
use rustc_data_structures::fx::FxHashSet;
7-
use rustc_errors::{Applicability, Diagnostic, ErrorGuaranteed, MultiSpan};
7+
use rustc_errors::{Applicability, Diagnostic, ErrorGuaranteed, MultiSpan, StashKey};
88
use rustc_hir as hir;
99
use rustc_hir::def::{CtorOf, DefKind, Res};
1010
use rustc_hir::def_id::DefId;
@@ -853,6 +853,13 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
853853
let item_name = item_segment.ident;
854854
let result = self
855855
.resolve_fully_qualified_call(span, item_name, ty.normalized, qself.span, hir_id)
856+
.and_then(|r| {
857+
// lint bare trait if the method is found in the trait
858+
if span.edition().rust_2021() && let Some(mut diag) = self.tcx.sess.diagnostic().steal_diagnostic(qself.span, StashKey::TraitMissingMethod) {
859+
diag.emit();
860+
}
861+
Ok(r)
862+
})
856863
.or_else(|error| {
857864
let guar = self
858865
.tcx
@@ -863,17 +870,30 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
863870
_ => Err(guar),
864871
};
865872

873+
let trait_missing_method =
874+
matches!(error, method::MethodError::NoMatch(_)) && ty.normalized.is_trait();
866875
// If we have a path like `MyTrait::missing_method`, then don't register
867876
// a WF obligation for `dyn MyTrait` when method lookup fails. Otherwise,
868877
// register a WF obligation so that we can detect any additional
869878
// errors in the self type.
870-
if !(matches!(error, method::MethodError::NoMatch(_)) && ty.normalized.is_trait()) {
879+
if !trait_missing_method {
871880
self.register_wf_obligation(
872881
ty.raw.into(),
873882
qself.span,
874883
traits::WellFormed(None),
875884
);
876885
}
886+
887+
// emit or cancel the diagnostic for bare traits
888+
if span.edition().rust_2021() && let Some(mut diag) = self.tcx.sess.diagnostic().steal_diagnostic(qself.span, StashKey::TraitMissingMethod) {
889+
if trait_missing_method {
890+
// cancel the diag for bare traits when meeting `MyTrait::missing_method`
891+
diag.cancel();
892+
} else {
893+
diag.emit();
894+
}
895+
}
896+
877897
if item_name.name != kw::Empty {
878898
if let Some(mut e) = self.report_method_error(
879899
span,
@@ -883,10 +903,12 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
883903
error,
884904
None,
885905
Expectation::NoExpectation,
906+
trait_missing_method && span.edition().rust_2021(), // emits missing method for trait only after edition 2021
886907
) {
887908
e.emit();
888909
}
889910
}
911+
890912
result
891913
});
892914

compiler/rustc_hir_typeck/src/method/suggest.rs

+55-30
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,8 @@
22
//! found or is otherwise invalid.
33
44
use crate::errors;
5+
use crate::errors::CandidateTraitNote;
6+
use crate::errors::NoAssociatedItem;
57
use crate::Expectation;
68
use crate::FnCtxt;
79
use rustc_ast::ast::Mutability;
@@ -38,6 +40,7 @@ use rustc_trait_selection::traits::query::evaluate_obligation::InferCtxtExt as _
3840
use rustc_trait_selection::traits::{
3941
FulfillmentError, Obligation, ObligationCause, ObligationCauseCode,
4042
};
43+
use std::borrow::Cow;
4144

4245
use super::probe::{AutorefOrPtrAdjustment, IsSuggestion, Mode, ProbeScope};
4346
use super::{CandidateSource, MethodError, NoMatchData};
@@ -112,6 +115,7 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
112115
error: MethodError<'tcx>,
113116
args: Option<(&'tcx hir::Expr<'tcx>, &'tcx [hir::Expr<'tcx>])>,
114117
expected: Expectation<'tcx>,
118+
trait_missing_method: bool,
115119
) -> Option<DiagnosticBuilder<'_, ErrorGuaranteed>> {
116120
// Avoid suggestions when we don't know what's going on.
117121
if rcvr_ty.references_error() {
@@ -136,6 +140,7 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
136140
sugg_span,
137141
&mut no_match_data,
138142
expected,
143+
trait_missing_method,
139144
);
140145
}
141146

@@ -278,6 +283,7 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
278283
sugg_span: Span,
279284
no_match_data: &mut NoMatchData<'tcx>,
280285
expected: Expectation<'tcx>,
286+
trait_missing_method: bool,
281287
) -> Option<DiagnosticBuilder<'_, ErrorGuaranteed>> {
282288
let mode = no_match_data.mode;
283289
let tcx = self.tcx;
@@ -323,7 +329,12 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
323329
span = item_name.span;
324330

325331
// Don't show generic arguments when the method can't be found in any implementation (#81576).
326-
let mut ty_str_reported = ty_str.clone();
332+
let mut ty_str_reported = if trait_missing_method {
333+
ty_str.strip_prefix("dyn ").expect("Failed to remove the prefix dyn").to_owned()
334+
} else {
335+
ty_str.clone()
336+
};
337+
327338
if let ty::Adt(_, generics) = rcvr_ty.kind() {
328339
if generics.len() > 0 {
329340
let mut autoderef = self.autoderef(span, rcvr_ty);
@@ -355,25 +366,33 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
355366
{
356367
self.suggest_missing_writer(rcvr_ty, args)
357368
} else {
358-
struct_span_err!(
359-
tcx.sess,
369+
tcx.sess.create_err(NoAssociatedItem {
360370
span,
361-
E0599,
362-
"no {} named `{}` found for {} `{}` in the current scope",
363371
item_kind,
364372
item_name,
365-
rcvr_ty.prefix_string(self.tcx),
366-
ty_str_reported,
367-
)
373+
ty_prefix: if trait_missing_method {
374+
// FIXME(mu001999) E0599 maybe not suitable here because it is for types
375+
Cow::from("trait")
376+
} else {
377+
rcvr_ty.prefix_string(self.tcx)
378+
},
379+
ty_str: ty_str_reported,
380+
trait_missing_method,
381+
})
368382
};
369383
if tcx.sess.source_map().is_multiline(sugg_span) {
370384
err.span_label(sugg_span.with_hi(span.lo()), "");
371385
}
372-
let ty_str = if short_ty_str.len() < ty_str.len() && ty_str.len() > 10 {
386+
let mut ty_str = if short_ty_str.len() < ty_str.len() && ty_str.len() > 10 {
373387
short_ty_str
374388
} else {
375389
ty_str
376390
};
391+
if trait_missing_method {
392+
ty_str =
393+
ty_str.strip_prefix("dyn ").expect("Failed to remove the prefix dyn").to_owned();
394+
}
395+
377396
if let Some(file) = ty_file {
378397
err.note(format!("the full type name has been written to '{}'", file.display(),));
379398
}
@@ -1067,6 +1086,7 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
10671086
&static_candidates,
10681087
unsatisfied_bounds,
10691088
expected.only_has_type(self),
1089+
trait_missing_method,
10701090
);
10711091
}
10721092

@@ -2375,6 +2395,7 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
23752395
static_candidates: &[CandidateSource],
23762396
unsatisfied_bounds: bool,
23772397
return_type: Option<Ty<'tcx>>,
2398+
trait_missing_method: bool,
23782399
) {
23792400
let mut alt_rcvr_sugg = false;
23802401
if let (SelfSource::MethodCall(rcvr), false) = (source, unsatisfied_bounds) {
@@ -2598,11 +2619,14 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
25982619
},
25992620
_ => None,
26002621
};
2601-
err.help(if param_type.is_some() {
2602-
"items from traits can only be used if the type parameter is bounded by the trait"
2603-
} else {
2604-
"items from traits can only be used if the trait is implemented and in scope"
2605-
});
2622+
if !trait_missing_method {
2623+
err.help(if param_type.is_some() {
2624+
"items from traits can only be used if the type parameter is bounded by the trait"
2625+
} else {
2626+
"items from traits can only be used if the trait is implemented and in scope"
2627+
});
2628+
}
2629+
26062630
let candidates_len = candidates.len();
26072631
let message = |action| {
26082632
format!(
@@ -2751,27 +2775,28 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
27512775
(candidates, Vec::new())
27522776
};
27532777

2754-
let action = if let Some(param) = param_type {
2755-
format!("restrict type parameter `{}` with", param)
2756-
} else {
2757-
// FIXME: it might only need to be imported into scope, not implemented.
2758-
"implement".to_string()
2759-
};
27602778
match &potential_candidates[..] {
27612779
[] => {}
27622780
[trait_info] if trait_info.def_id.is_local() => {
2763-
err.span_note(
2764-
self.tcx.def_span(trait_info.def_id),
2765-
format!(
2766-
"`{}` defines an item `{}`, perhaps you need to {} it",
2767-
self.tcx.def_path_str(trait_info.def_id),
2768-
item_name,
2769-
action
2770-
),
2771-
);
2781+
err.subdiagnostic(CandidateTraitNote {
2782+
span: self.tcx.def_span(trait_info.def_id),
2783+
trait_name: self.tcx.def_path_str(trait_info.def_id),
2784+
item_name,
2785+
action_or_ty: if trait_missing_method {
2786+
"NONE".to_string()
2787+
} else {
2788+
param_type.map_or_else(
2789+
|| "implement".to_string(), // FIXME: it might only need to be imported into scope, not implemented.
2790+
ToString::to_string,
2791+
)
2792+
},
2793+
});
27722794
}
27732795
trait_infos => {
2774-
let mut msg = message(action);
2796+
let mut msg = message(param_type.map_or_else(
2797+
|| "implement".to_string(), // FIXME: it might only need to be imported into scope, not implemented.
2798+
|param| format!("restrict type parameter `{}` with", param),
2799+
));
27752800
for (i, trait_info) in trait_infos.iter().enumerate() {
27762801
msg.push_str(&format!(
27772802
"\ncandidate #{}: `{}`",

compiler/rustc_middle/src/lib.rs

-1
Original file line numberDiff line numberDiff line change
@@ -95,7 +95,6 @@ pub mod middle;
9595
pub mod mir;
9696
pub mod thir;
9797
pub mod traits;
98-
#[macro_use]
9998
pub mod ty;
10099
pub mod util;
101100
mod values;

compiler/rustc_middle/src/mir/interpret/error.rs

+2-1
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,8 @@
11
use super::{AllocId, AllocRange, ConstAlloc, Pointer, Scalar};
22

33
use crate::mir::interpret::ConstValue;
4-
use crate::ty::{layout, query::TyCtxtAt, tls, Ty, ValTree};
4+
use crate::query::TyCtxtAt;
5+
use crate::ty::{layout, tls, Ty, ValTree};
56

67
use rustc_data_structures::sync::Lock;
78
use rustc_errors::{pluralize, struct_span_err, DiagnosticBuilder, ErrorGuaranteed};

0 commit comments

Comments
 (0)