@@ -11,7 +11,7 @@ use crate::traits::{self, Normalized, Obligation, ObligationCause, SelectionCont
11
11
use rustc_hir:: def_id:: { DefId , LOCAL_CRATE } ;
12
12
use rustc_middle:: ty:: fold:: TypeFoldable ;
13
13
use rustc_middle:: ty:: subst:: Subst ;
14
- use rustc_middle:: ty:: { self , Ty , TyCtxt } ;
14
+ use rustc_middle:: ty:: { self , fast_reject , Ty , TyCtxt } ;
15
15
use rustc_span:: symbol:: sym;
16
16
use rustc_span:: DUMMY_SP ;
17
17
use std:: iter;
67
67
impl2_def_id={:?})",
68
68
impl1_def_id, impl2_def_id,
69
69
) ;
70
+ // Before doing expensive operations like entering an inference context, do
71
+ // a quick check via fast_reject to tell if the impl headers could possibly
72
+ // unify.
73
+ let impl1_ref = tcx. impl_trait_ref ( impl1_def_id) ;
74
+ let impl2_ref = tcx. impl_trait_ref ( impl2_def_id) ;
75
+
76
+ // Check if any of the input types definitely do not unify.
77
+ if impl1_ref
78
+ . iter ( )
79
+ . flat_map ( |tref| tref. substs . types ( ) )
80
+ . zip ( impl2_ref. iter ( ) . flat_map ( |tref| tref. substs . types ( ) ) )
81
+ . any ( |( ty1, ty2) | {
82
+ let t1 = fast_reject:: simplify_type ( tcx, ty1, false ) ;
83
+ let t2 = fast_reject:: simplify_type ( tcx, ty2, false ) ;
84
+ if let ( Some ( t1) , Some ( t2) ) = ( t1, t2) {
85
+ // Simplified successfully
86
+ // Types cannot unify if they differ in their reference mutability or simplify to different types
87
+ t1 != t2 || ty1. ref_mutability ( ) != ty2. ref_mutability ( )
88
+ } else {
89
+ // Types might unify
90
+ false
91
+ }
92
+ } )
93
+ {
94
+ // Some types involved are definitely different, so the impls couldn't possibly overlap.
95
+ debug ! ( "overlapping_impls: fast_reject early-exit" ) ;
96
+ return no_overlap ( ) ;
97
+ }
70
98
71
99
let overlaps = tcx. infer_ctxt ( ) . enter ( |infcx| {
72
100
let selcx = & mut SelectionContext :: intercrate ( & infcx) ;
0 commit comments