1
1
//! Check properties that are required by built-in traits and set
2
2
//! up data structures required by type-checking/codegen.
3
3
4
+ mod diagnostics;
5
+
4
6
use std:: assert_matches:: assert_matches;
5
7
use std:: collections:: BTreeMap ;
6
8
9
+ use diagnostics:: { extract_coerce_pointee_data, redact_fulfillment_err_for_coerce_pointee} ;
7
10
use rustc_data_structures:: fx:: FxHashSet ;
8
11
use rustc_errors:: { ErrorGuaranteed , MultiSpan } ;
9
12
use rustc_hir as hir;
@@ -12,6 +15,7 @@ use rustc_hir::def_id::{DefId, LocalDefId};
12
15
use rustc_hir:: lang_items:: LangItem ;
13
16
use rustc_infer:: infer:: { self , RegionResolutionError , TyCtxtInferExt } ;
14
17
use rustc_infer:: traits:: Obligation ;
18
+ use rustc_middle:: bug;
15
19
use rustc_middle:: ty:: adjustment:: CoerceUnsizedInfo ;
16
20
use rustc_middle:: ty:: print:: PrintTraitRefExt as _;
17
21
use rustc_middle:: ty:: {
@@ -307,29 +311,45 @@ fn visit_implementation_of_dispatch_from_dyn(checker: &Checker<'_>) -> Result<()
307
311
. collect :: < Vec < _ > > ( ) ;
308
312
309
313
if coerced_fields. is_empty ( ) {
310
- res = Err ( tcx. dcx ( ) . emit_err ( errors:: DispatchFromDynSingle {
311
- span,
312
- trait_name : "DispatchFromDyn" ,
313
- note : true ,
314
- } ) ) ;
314
+ if extract_coerce_pointee_data ( tcx, def_a. did ( ) ) . is_some ( ) {
315
+ res = Err ( tcx. dcx ( ) . span_delayed_bug (
316
+ span,
317
+ "a specialised message for CoercePointee is expected" ,
318
+ ) ) ;
319
+ } else {
320
+ res = Err ( tcx. dcx ( ) . emit_err ( errors:: DispatchFromDynSingle {
321
+ span,
322
+ trait_name : "DispatchFromDyn" ,
323
+ note : true ,
324
+ } ) ) ;
325
+ }
315
326
} else if coerced_fields. len ( ) > 1 {
316
- res = Err ( tcx. dcx ( ) . emit_err ( errors:: DispatchFromDynMulti {
317
- span,
318
- coercions_note : true ,
319
- number : coerced_fields. len ( ) ,
320
- coercions : coerced_fields
321
- . iter ( )
322
- . map ( |field| {
323
- format ! (
324
- "`{}` (`{}` to `{}`)" ,
325
- field. name,
326
- field. ty( tcx, args_a) ,
327
- field. ty( tcx, args_b) ,
328
- )
329
- } )
330
- . collect :: < Vec < _ > > ( )
331
- . join ( ", " ) ,
332
- } ) ) ;
327
+ if extract_coerce_pointee_data ( tcx, def_a. did ( ) ) . is_some ( ) {
328
+ let spans =
329
+ coerced_fields. iter ( ) . map ( |field| tcx. def_span ( field. did ) ) . collect ( ) ;
330
+ res = Err ( tcx. dcx ( ) . emit_err ( errors:: CoercePointeeMultipleTargets {
331
+ spans,
332
+ diag_trait : "DispatchFromDyn" ,
333
+ } ) ) ;
334
+ } else {
335
+ res = Err ( tcx. dcx ( ) . emit_err ( errors:: DispatchFromDynMulti {
336
+ span,
337
+ coercions_note : true ,
338
+ number : coerced_fields. len ( ) ,
339
+ coercions : coerced_fields
340
+ . iter ( )
341
+ . map ( |field| {
342
+ format ! (
343
+ "`{}` (`{}` to `{}`)" ,
344
+ field. name,
345
+ field. ty( tcx, args_a) ,
346
+ field. ty( tcx, args_b) ,
347
+ )
348
+ } )
349
+ . collect :: < Vec < _ > > ( )
350
+ . join ( ", " ) ,
351
+ } ) ) ;
352
+ }
333
353
} else {
334
354
let ocx = ObligationCtxt :: new_with_diagnostics ( & infcx) ;
335
355
for field in coerced_fields {
@@ -344,9 +364,31 @@ fn visit_implementation_of_dispatch_from_dyn(checker: &Checker<'_>) -> Result<()
344
364
) ,
345
365
) ) ;
346
366
}
347
- let errors = ocx. select_all_or_error ( ) ;
367
+ let mut errors = ocx. select_all_or_error ( ) ;
348
368
if !errors. is_empty ( ) {
349
- res = Err ( infcx. err_ctxt ( ) . report_fulfillment_errors ( errors) ) ;
369
+ if let Some ( ( pointee_idx, _) ) = extract_coerce_pointee_data ( tcx, def_a. did ( ) ) {
370
+ let target_pointee = args_b. type_at ( pointee_idx) ;
371
+
372
+ errors = errors
373
+ . into_iter ( )
374
+ . filter_map ( |err| {
375
+ redact_fulfillment_err_for_coerce_pointee (
376
+ tcx,
377
+ err,
378
+ target_pointee,
379
+ span,
380
+ )
381
+ } )
382
+ . collect ( ) ;
383
+ }
384
+ if errors. is_empty ( ) {
385
+ res = Err ( tcx. dcx ( ) . span_delayed_bug (
386
+ span,
387
+ "a specialised CoercePointee error is expected" ,
388
+ ) ) ;
389
+ } else {
390
+ res = Err ( infcx. err_ctxt ( ) . report_fulfillment_errors ( errors) ) ;
391
+ }
350
392
}
351
393
352
394
// Finally, resolve all regions.
@@ -372,9 +414,11 @@ pub(crate) fn coerce_unsized_info<'tcx>(
372
414
let unsize_trait = tcx. require_lang_item ( LangItem :: Unsize , Some ( span) ) ;
373
415
374
416
let source = tcx. type_of ( impl_did) . instantiate_identity ( ) ;
417
+ let self_ty = source;
375
418
let trait_ref = tcx. impl_trait_ref ( impl_did) . unwrap ( ) . instantiate_identity ( ) ;
376
419
assert_eq ! ( trait_ref. def_id, coerce_unsized_trait) ;
377
420
let target = trait_ref. args . type_at ( 1 ) ;
421
+ let coerced_ty = target;
378
422
debug ! ( "visit_implementation_of_coerce_unsized: {:?} -> {:?} (bound)" , source, target) ;
379
423
380
424
let param_env = tcx. param_env ( impl_did) ;
@@ -509,12 +553,28 @@ pub(crate) fn coerce_unsized_info<'tcx>(
509
553
. collect :: < Vec < _ > > ( ) ;
510
554
511
555
if diff_fields. is_empty ( ) {
556
+ if extract_coerce_pointee_data ( tcx, def_a. did ( ) ) . is_some ( ) {
557
+ return Err ( tcx. dcx ( ) . span_delayed_bug (
558
+ span,
559
+ "a specialised message for CoercePointee is expected" ,
560
+ ) ) ;
561
+ }
512
562
return Err ( tcx. dcx ( ) . emit_err ( errors:: CoerceUnsizedOneField {
513
563
span,
514
564
trait_name : "CoerceUnsized" ,
515
565
note : true ,
516
566
} ) ) ;
517
567
} else if diff_fields. len ( ) > 1 {
568
+ if extract_coerce_pointee_data ( tcx, def_a. did ( ) ) . is_some ( ) {
569
+ let spans = diff_fields
570
+ . iter ( )
571
+ . map ( |& ( idx, _, _) | tcx. def_span ( fields[ idx] . did ) )
572
+ . collect ( ) ;
573
+ return Err ( tcx. dcx ( ) . emit_err ( errors:: CoercePointeeMultipleTargets {
574
+ spans,
575
+ diag_trait : "CoerceUnsized" ,
576
+ } ) ) ;
577
+ }
518
578
let item = tcx. hir ( ) . expect_item ( impl_did) ;
519
579
let span = if let ItemKind :: Impl ( hir:: Impl { of_trait : Some ( t) , .. } ) = & item. kind {
520
580
t. path . span
@@ -547,18 +607,41 @@ pub(crate) fn coerce_unsized_info<'tcx>(
547
607
} ;
548
608
549
609
// Register an obligation for `A: Trait<B>`.
610
+ let coerce_pointee_data = if let ty:: Adt ( def, _) = self_ty. kind ( ) {
611
+ extract_coerce_pointee_data ( tcx, def. did ( ) )
612
+ } else {
613
+ None
614
+ } ;
550
615
let ocx = ObligationCtxt :: new_with_diagnostics ( & infcx) ;
551
- let cause = traits:: ObligationCause :: misc ( span, impl_did) ;
616
+ let cause = traits:: ObligationCause :: misc (
617
+ span,
618
+ coerce_pointee_data. map_or ( impl_did, |( _, impl_did) | impl_did. expect_local ( ) ) ,
619
+ ) ;
552
620
let obligation = Obligation :: new (
553
621
tcx,
554
622
cause,
555
623
param_env,
556
624
ty:: TraitRef :: new ( tcx, trait_def_id, [ source, target] ) ,
557
625
) ;
558
626
ocx. register_obligation ( obligation) ;
559
- let errors = ocx. select_all_or_error ( ) ;
627
+ let mut errors = ocx. select_all_or_error ( ) ;
560
628
if !errors. is_empty ( ) {
561
- infcx. err_ctxt ( ) . report_fulfillment_errors ( errors) ;
629
+ if let Some ( ( pointee, _) ) = coerce_pointee_data {
630
+ let ty:: Adt ( _def, args) = coerced_ty. kind ( ) else { bug ! ( ) } ;
631
+ let target_pointee = args. type_at ( pointee) ;
632
+ let ty:: Adt ( _def, _) = self_ty. kind ( ) else { bug ! ( ) } ;
633
+ errors = errors
634
+ . into_iter ( )
635
+ . filter_map ( |err| {
636
+ redact_fulfillment_err_for_coerce_pointee ( tcx, err, target_pointee, span)
637
+ } )
638
+ . collect ( ) ;
639
+ }
640
+ if errors. is_empty ( ) {
641
+ tcx. dcx ( ) . span_delayed_bug ( span, "a specialised CoercePointee error is expected" ) ;
642
+ } else {
643
+ infcx. err_ctxt ( ) . report_fulfillment_errors ( errors) ;
644
+ }
562
645
}
563
646
564
647
// Finally, resolve all regions.
0 commit comments