2
2
//! found or is otherwise invalid.
3
3
4
4
use crate :: errors;
5
+ use crate :: errors:: CandidateTraitNote ;
6
+ use crate :: errors:: NoAssociatedItem ;
5
7
use crate :: Expectation ;
6
8
use crate :: FnCtxt ;
7
9
use rustc_ast:: ast:: Mutability ;
@@ -38,6 +40,7 @@ use rustc_trait_selection::traits::query::evaluate_obligation::InferCtxtExt as _
38
40
use rustc_trait_selection:: traits:: {
39
41
FulfillmentError , Obligation , ObligationCause , ObligationCauseCode ,
40
42
} ;
43
+ use std:: borrow:: Cow ;
41
44
42
45
use super :: probe:: { AutorefOrPtrAdjustment , IsSuggestion , Mode , ProbeScope } ;
43
46
use super :: { CandidateSource , MethodError , NoMatchData } ;
@@ -112,6 +115,7 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
112
115
error : MethodError < ' tcx > ,
113
116
args : Option < ( & ' tcx hir:: Expr < ' tcx > , & ' tcx [ hir:: Expr < ' tcx > ] ) > ,
114
117
expected : Expectation < ' tcx > ,
118
+ trait_missing_method : bool ,
115
119
) -> Option < DiagnosticBuilder < ' _ , ErrorGuaranteed > > {
116
120
// Avoid suggestions when we don't know what's going on.
117
121
if rcvr_ty. references_error ( ) {
@@ -136,6 +140,7 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
136
140
sugg_span,
137
141
& mut no_match_data,
138
142
expected,
143
+ trait_missing_method,
139
144
) ;
140
145
}
141
146
@@ -278,6 +283,7 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
278
283
sugg_span : Span ,
279
284
no_match_data : & mut NoMatchData < ' tcx > ,
280
285
expected : Expectation < ' tcx > ,
286
+ trait_missing_method : bool ,
281
287
) -> Option < DiagnosticBuilder < ' _ , ErrorGuaranteed > > {
282
288
let mode = no_match_data. mode ;
283
289
let tcx = self . tcx ;
@@ -323,7 +329,12 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
323
329
span = item_name. span ;
324
330
325
331
// 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
+
327
338
if let ty:: Adt ( _, generics) = rcvr_ty. kind ( ) {
328
339
if generics. len ( ) > 0 {
329
340
let mut autoderef = self . autoderef ( span, rcvr_ty) ;
@@ -355,25 +366,33 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
355
366
{
356
367
self . suggest_missing_writer ( rcvr_ty, args)
357
368
} else {
358
- struct_span_err ! (
359
- tcx. sess,
369
+ tcx. sess . create_err ( NoAssociatedItem {
360
370
span,
361
- E0599 ,
362
- "no {} named `{}` found for {} `{}` in the current scope" ,
363
371
item_kind,
364
372
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
+ } )
368
382
} ;
369
383
if tcx. sess . source_map ( ) . is_multiline ( sugg_span) {
370
384
err. span_label ( sugg_span. with_hi ( span. lo ( ) ) , "" ) ;
371
385
}
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 {
373
387
short_ty_str
374
388
} else {
375
389
ty_str
376
390
} ;
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
+
377
396
if let Some ( file) = ty_file {
378
397
err. note ( format ! ( "the full type name has been written to '{}'" , file. display( ) , ) ) ;
379
398
}
@@ -1067,6 +1086,7 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
1067
1086
& static_candidates,
1068
1087
unsatisfied_bounds,
1069
1088
expected. only_has_type ( self ) ,
1089
+ trait_missing_method,
1070
1090
) ;
1071
1091
}
1072
1092
@@ -2375,6 +2395,7 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
2375
2395
static_candidates : & [ CandidateSource ] ,
2376
2396
unsatisfied_bounds : bool ,
2377
2397
return_type : Option < Ty < ' tcx > > ,
2398
+ trait_missing_method : bool ,
2378
2399
) {
2379
2400
let mut alt_rcvr_sugg = false ;
2380
2401
if let ( SelfSource :: MethodCall ( rcvr) , false ) = ( source, unsatisfied_bounds) {
@@ -2598,11 +2619,14 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
2598
2619
} ,
2599
2620
_ => None ,
2600
2621
} ;
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
+
2606
2630
let candidates_len = candidates. len ( ) ;
2607
2631
let message = |action| {
2608
2632
format ! (
@@ -2736,27 +2760,28 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
2736
2760
( candidates, Vec :: new ( ) )
2737
2761
} ;
2738
2762
2739
- let action = if let Some ( param) = param_type {
2740
- format ! ( "restrict type parameter `{}` with" , param)
2741
- } else {
2742
- // FIXME: it might only need to be imported into scope, not implemented.
2743
- "implement" . to_string ( )
2744
- } ;
2745
2763
match & potential_candidates[ ..] {
2746
2764
[ ] => { }
2747
2765
[ trait_info] if trait_info. def_id . is_local ( ) => {
2748
- err. span_note (
2749
- self . tcx . def_span ( trait_info. def_id ) ,
2750
- format ! (
2751
- "`{}` defines an item `{}`, perhaps you need to {} it" ,
2752
- self . tcx. def_path_str( trait_info. def_id) ,
2753
- item_name,
2754
- action
2755
- ) ,
2756
- ) ;
2766
+ err. subdiagnostic ( CandidateTraitNote {
2767
+ span : self . tcx . def_span ( trait_info. def_id ) ,
2768
+ trait_name : self . tcx . def_path_str ( trait_info. def_id ) ,
2769
+ item_name,
2770
+ action_or_ty : if trait_missing_method {
2771
+ "NONE" . to_string ( )
2772
+ } else {
2773
+ param_type. map_or_else (
2774
+ || "implement" . to_string ( ) , // FIXME: it might only need to be imported into scope, not implemented.
2775
+ ToString :: to_string,
2776
+ )
2777
+ } ,
2778
+ } ) ;
2757
2779
}
2758
2780
trait_infos => {
2759
- let mut msg = message ( action) ;
2781
+ let mut msg = message ( param_type. map_or_else (
2782
+ || "implement" . to_string ( ) , // FIXME: it might only need to be imported into scope, not implemented.
2783
+ |param| format ! ( "restrict type parameter `{}` with" , param) ,
2784
+ ) ) ;
2760
2785
for ( i, trait_info) in trait_infos. iter ( ) . enumerate ( ) {
2761
2786
msg. push_str ( & format ! (
2762
2787
"\n candidate #{}: `{}`" ,
0 commit comments