1
1
use crate :: errors:: * ;
2
+ use crate :: fluent_generated as fluent;
2
3
3
4
use rustc_arena:: { DroplessArena , TypedArena } ;
4
5
use rustc_ast:: Mutability ;
@@ -16,8 +17,8 @@ use rustc_middle::ty::print::with_no_trimmed_paths;
16
17
use rustc_middle:: ty:: { self , AdtDef , Ty , TyCtxt } ;
17
18
use rustc_pattern_analysis:: errors:: Uncovered ;
18
19
use rustc_pattern_analysis:: rustc:: {
19
- Constructor , DeconstructedPat , MatchArm , RevealedTy , RustcPatCtxt as PatCtxt , Usefulness ,
20
- UsefulnessReport , WitnessPat ,
20
+ Constructor , DeconstructedPat , MatchArm , RedundancyExplanation , RevealedTy ,
21
+ RustcPatCtxt as PatCtxt , Usefulness , UsefulnessReport , WitnessPat ,
21
22
} ;
22
23
use rustc_session:: lint:: builtin:: {
23
24
BINDINGS_WITH_VARIANT_NAME , IRREFUTABLE_LET_PATTERNS , UNREACHABLE_PATTERNS ,
@@ -391,12 +392,16 @@ impl<'p, 'tcx> MatchVisitor<'p, 'tcx> {
391
392
) -> Result < UsefulnessReport < ' p , ' tcx > , ErrorGuaranteed > {
392
393
let pattern_complexity_limit =
393
394
get_limit_size ( cx. tcx . hir ( ) . krate_attrs ( ) , cx. tcx . sess , sym:: pattern_complexity) ;
394
- let report =
395
- rustc_pattern_analysis:: analyze_match ( & cx, & arms, scrut_ty, pattern_complexity_limit)
396
- . map_err ( |err| {
397
- self . error = Err ( err) ;
398
- err
399
- } ) ?;
395
+ let report = rustc_pattern_analysis:: rustc:: analyze_match (
396
+ & cx,
397
+ & arms,
398
+ scrut_ty,
399
+ pattern_complexity_limit,
400
+ )
401
+ . map_err ( |err| {
402
+ self . error = Err ( err) ;
403
+ err
404
+ } ) ?;
400
405
401
406
// Warn unreachable subpatterns.
402
407
for ( arm, is_useful) in report. arm_usefulness . iter ( ) {
@@ -405,9 +410,9 @@ impl<'p, 'tcx> MatchVisitor<'p, 'tcx> {
405
410
{
406
411
let mut redundant_subpats = redundant_subpats. clone ( ) ;
407
412
// Emit lints in the order in which they occur in the file.
408
- redundant_subpats. sort_unstable_by_key ( |pat| pat. data ( ) . span ) ;
409
- for pat in redundant_subpats {
410
- report_unreachable_pattern ( cx, arm. arm_data , pat. data ( ) . span , None )
413
+ redundant_subpats. sort_unstable_by_key ( |( pat, _ ) | pat. data ( ) . span ) ;
414
+ for ( pat, explanation ) in redundant_subpats {
415
+ report_unreachable_pattern ( cx, arm. arm_data , pat, & explanation )
411
416
}
412
417
}
413
418
}
@@ -906,26 +911,60 @@ fn report_irrefutable_let_patterns(
906
911
fn report_unreachable_pattern < ' p , ' tcx > (
907
912
cx : & PatCtxt < ' p , ' tcx > ,
908
913
hir_id : HirId ,
909
- span : Span ,
910
- catchall : Option < Span > ,
914
+ pat : & DeconstructedPat < ' p , ' tcx > ,
915
+ explanation : & RedundancyExplanation < ' p , ' tcx > ,
911
916
) {
912
- cx. tcx . emit_node_span_lint (
913
- UNREACHABLE_PATTERNS ,
914
- hir_id,
915
- span,
916
- UnreachablePattern { span : if catchall. is_some ( ) { Some ( span) } else { None } , catchall } ,
917
- ) ;
917
+ let pat_span = pat. data ( ) . span ;
918
+ let mut lint = UnreachablePattern {
919
+ span : Some ( pat_span) ,
920
+ matches_no_values : None ,
921
+ covered_by_catchall : None ,
922
+ covered_by_one : None ,
923
+ covered_by_many : None ,
924
+ } ;
925
+ match explanation. covered_by . as_slice ( ) {
926
+ [ ] => {
927
+ // Empty pattern; we report the uninhabited type that caused the emptiness.
928
+ lint. span = None ; // Don't label the pattern itself
929
+ pat. walk ( & mut |subpat| {
930
+ let ty = * * subpat. ty ( ) ;
931
+ if cx. is_uninhabited ( ty) {
932
+ lint. matches_no_values = Some ( UnreachableMatchesNoValues { ty } ) ;
933
+ false // No need to dig further.
934
+ } else if matches ! ( subpat. ctor( ) , Constructor :: Ref | Constructor :: UnionField ) {
935
+ false // Don't explore further since they are not by-value.
936
+ } else {
937
+ true
938
+ }
939
+ } ) ;
940
+ }
941
+ [ covering_pat] if pat_is_catchall ( covering_pat) => {
942
+ lint. covered_by_catchall = Some ( covering_pat. data ( ) . span ) ;
943
+ }
944
+ [ covering_pat] => {
945
+ lint. covered_by_one = Some ( covering_pat. data ( ) . span ) ;
946
+ }
947
+ covering_pats => {
948
+ let mut multispan = MultiSpan :: from_span ( pat_span) ;
949
+ for p in covering_pats {
950
+ multispan. push_span_label (
951
+ p. data ( ) . span ,
952
+ fluent:: mir_build_unreachable_matches_same_values,
953
+ ) ;
954
+ }
955
+ multispan
956
+ . push_span_label ( pat_span, fluent:: mir_build_unreachable_making_this_unreachable) ;
957
+ lint. covered_by_many = Some ( multispan) ;
958
+ }
959
+ }
960
+ cx. tcx . emit_node_span_lint ( UNREACHABLE_PATTERNS , hir_id, pat_span, lint) ;
918
961
}
919
962
920
963
/// Report unreachable arms, if any.
921
964
fn report_arm_reachability < ' p , ' tcx > ( cx : & PatCtxt < ' p , ' tcx > , report : & UsefulnessReport < ' p , ' tcx > ) {
922
- let mut catchall = None ;
923
965
for ( arm, is_useful) in report. arm_usefulness . iter ( ) {
924
- if matches ! ( is_useful, Usefulness :: Redundant ) {
925
- report_unreachable_pattern ( cx, arm. arm_data , arm. pat . data ( ) . span , catchall)
926
- }
927
- if !arm. has_guard && catchall. is_none ( ) && pat_is_catchall ( arm. pat ) {
928
- catchall = Some ( arm. pat . data ( ) . span ) ;
966
+ if let Usefulness :: Redundant ( explanation) = is_useful {
967
+ report_unreachable_pattern ( cx, arm. arm_data , arm. pat , explanation)
929
968
}
930
969
}
931
970
}
0 commit comments