@@ -9,7 +9,7 @@ use rustc_hir as hir;
9
9
use rustc_hir:: def_id:: { DefId , LocalDefId , LOCAL_CRATE } ;
10
10
use rustc_hir:: intravisit:: Visitor ;
11
11
use rustc_hir:: lang_items:: LangItem ;
12
- use rustc_hir:: { ItemKind , Node } ;
12
+ use rustc_hir:: { def :: Res , ItemKind , Node , PathSegment } ;
13
13
use rustc_infer:: infer:: type_variable:: { TypeVariableOrigin , TypeVariableOriginKind } ;
14
14
use rustc_infer:: infer:: { RegionVariableOrigin , TyCtxtInferExt } ;
15
15
use rustc_middle:: ty:: fold:: TypeFoldable ;
@@ -516,10 +516,11 @@ pub(super) fn check_opaque_for_inheriting_lifetimes(
516
516
}
517
517
}
518
518
519
- #[ derive( Debug ) ]
520
519
struct ProhibitOpaqueVisitor < ' tcx > {
521
520
opaque_identity_ty : Ty < ' tcx > ,
522
521
generics : & ' tcx ty:: Generics ,
522
+ tcx : TyCtxt < ' tcx > ,
523
+ selftys : Vec < ( Span , Option < String > ) > ,
523
524
}
524
525
525
526
impl < ' tcx > ty:: fold:: TypeVisitor < ' tcx > for ProhibitOpaqueVisitor < ' tcx > {
@@ -536,6 +537,29 @@ pub(super) fn check_opaque_for_inheriting_lifetimes(
536
537
}
537
538
}
538
539
540
+ impl Visitor < ' tcx > for ProhibitOpaqueVisitor < ' tcx > {
541
+ type Map = rustc_middle:: hir:: map:: Map < ' tcx > ;
542
+
543
+ fn nested_visit_map ( & mut self ) -> hir:: intravisit:: NestedVisitorMap < Self :: Map > {
544
+ hir:: intravisit:: NestedVisitorMap :: OnlyBodies ( self . tcx . hir ( ) )
545
+ }
546
+
547
+ fn visit_ty ( & mut self , arg : & ' tcx hir:: Ty < ' tcx > ) {
548
+ match arg. kind {
549
+ hir:: TyKind :: Path ( hir:: QPath :: Resolved ( None , path) ) => match & path. segments {
550
+ [ PathSegment { res : Some ( Res :: SelfTy ( _, impl_ref) ) , .. } ] => {
551
+ let impl_ty_name =
552
+ impl_ref. map ( |( def_id, _) | self . tcx . def_path_str ( def_id) ) ;
553
+ self . selftys . push ( ( path. span , impl_ty_name) ) ;
554
+ }
555
+ _ => { }
556
+ } ,
557
+ _ => { }
558
+ }
559
+ hir:: intravisit:: walk_ty ( self , arg) ;
560
+ }
561
+ }
562
+
539
563
if let ItemKind :: OpaqueTy ( hir:: OpaqueTy {
540
564
origin : hir:: OpaqueTyOrigin :: AsyncFn | hir:: OpaqueTyOrigin :: FnReturn ,
541
565
..
@@ -547,18 +571,19 @@ pub(super) fn check_opaque_for_inheriting_lifetimes(
547
571
InternalSubsts :: identity_for_item ( tcx, def_id. to_def_id ( ) ) ,
548
572
) ,
549
573
generics : tcx. generics_of ( def_id) ,
574
+ tcx,
575
+ selftys : vec ! [ ] ,
550
576
} ;
551
577
let prohibit_opaque = tcx
552
578
. explicit_item_bounds ( def_id)
553
579
. iter ( )
554
580
. try_for_each ( |( predicate, _) | predicate. visit_with ( & mut visitor) ) ;
555
581
debug ! (
556
- "check_opaque_for_inheriting_lifetimes: prohibit_opaque={:?}, visitor={:?}" ,
557
- prohibit_opaque, visitor
582
+ "check_opaque_for_inheriting_lifetimes: prohibit_opaque={:?}, visitor.opaque_identity_ty={:?}, visitor.generics ={:?}" ,
583
+ prohibit_opaque, visitor. opaque_identity_ty , visitor . generics
558
584
) ;
559
585
560
586
if let Some ( ty) = prohibit_opaque. break_value ( ) {
561
- let mut visitor = SelfTySpanVisitor { tcx, selfty_spans : vec ! [ ] } ;
562
587
visitor. visit_item ( & item) ;
563
588
let is_async = match item. kind {
564
589
ItemKind :: OpaqueTy ( hir:: OpaqueTy { origin, .. } ) => {
@@ -576,11 +601,11 @@ pub(super) fn check_opaque_for_inheriting_lifetimes(
576
601
if is_async { "async fn" } else { "impl Trait" } ,
577
602
) ;
578
603
579
- for span in visitor. selfty_spans {
604
+ for ( span, name ) in visitor. selftys {
580
605
err. span_suggestion (
581
606
span,
582
607
"consider spelling out the type instead" ,
583
- format ! ( "{:?}" , ty) ,
608
+ name . unwrap_or_else ( || format ! ( "{:?}" , ty) ) ,
584
609
Applicability :: MaybeIncorrect ,
585
610
) ;
586
611
}
@@ -1591,31 +1616,3 @@ fn opaque_type_cycle_error(tcx: TyCtxt<'tcx>, def_id: LocalDefId, span: Span) {
1591
1616
}
1592
1617
err. emit ( ) ;
1593
1618
}
1594
-
1595
- struct SelfTySpanVisitor < ' tcx > {
1596
- tcx : TyCtxt < ' tcx > ,
1597
- selfty_spans : Vec < Span > ,
1598
- }
1599
-
1600
- impl Visitor < ' tcx > for SelfTySpanVisitor < ' tcx > {
1601
- type Map = rustc_middle:: hir:: map:: Map < ' tcx > ;
1602
-
1603
- fn nested_visit_map ( & mut self ) -> hir:: intravisit:: NestedVisitorMap < Self :: Map > {
1604
- hir:: intravisit:: NestedVisitorMap :: OnlyBodies ( self . tcx . hir ( ) )
1605
- }
1606
-
1607
- fn visit_ty ( & mut self , arg : & ' tcx hir:: Ty < ' tcx > ) {
1608
- match arg. kind {
1609
- hir:: TyKind :: Path ( hir:: QPath :: Resolved ( None , path) ) => match & path. segments {
1610
- [ segment]
1611
- if segment. res . map ( |res| matches ! ( res, Res :: SelfTy ( _, _) ) ) . unwrap_or ( false ) =>
1612
- {
1613
- self . selfty_spans . push ( path. span ) ;
1614
- }
1615
- _ => { }
1616
- } ,
1617
- _ => { }
1618
- }
1619
- hir:: intravisit:: walk_ty ( self , arg) ;
1620
- }
1621
- }
0 commit comments