@@ -16,6 +16,7 @@ use super::translate_substs;
16
16
use super :: Obligation ;
17
17
use super :: ObligationCause ;
18
18
use super :: PredicateObligation ;
19
+ use super :: Selection ;
19
20
use super :: SelectionContext ;
20
21
use super :: SelectionError ;
21
22
use super :: VtableClosureData ;
@@ -101,7 +102,7 @@ pub struct MismatchedProjectionTypes<'tcx> {
101
102
pub err : ty:: error:: TypeError < ' tcx >
102
103
}
103
104
104
- #[ derive( PartialEq , Eq , PartialOrd , Ord , Debug ) ]
105
+ #[ derive( PartialEq , Eq , Debug ) ]
105
106
enum ProjectionTyCandidate < ' tcx > {
106
107
// from a where-clause in the env or object type
107
108
ParamEnv ( ty:: PolyProjectionPredicate < ' tcx > ) ,
@@ -110,7 +111,7 @@ enum ProjectionTyCandidate<'tcx> {
110
111
TraitDef ( ty:: PolyProjectionPredicate < ' tcx > ) ,
111
112
112
113
// from a "impl" (or a "pseudo-impl" returned by select)
113
- Select ,
114
+ Select ( Selection < ' tcx > ) ,
114
115
}
115
116
116
117
struct ProjectionTyCandidateSet < ' tcx > {
@@ -818,55 +819,57 @@ fn project_type<'cx, 'gcx, 'tcx>(
818
819
& obligation_trait_ref,
819
820
& mut candidates) ;
820
821
822
+ let decide_commit = |candidates : & mut ProjectionTyCandidateSet < ' tcx > | {
823
+ debug ! ( "{} candidates, ambiguous={}" ,
824
+ candidates. vec. len( ) ,
825
+ candidates. ambiguous) ;
826
+
827
+ // Prefer where-clauses. As in select, if there are multiple
828
+ // candidates, we prefer where-clause candidates over impls. This
829
+ // may seem a bit surprising, since impls are the source of
830
+ // "truth" in some sense, but in fact some of the impls that SEEM
831
+ // applicable are not, because of nested obligations. Where
832
+ // clauses are the safer choice. See the comment on
833
+ // `select::SelectionCandidate` and #21974 for more details.
834
+ if candidates. vec . len ( ) > 1 {
835
+ debug ! ( "retaining param-env candidates only from {:?}" , candidates. vec) ;
836
+ candidates. vec . retain ( |c| match * c {
837
+ ProjectionTyCandidate :: ParamEnv ( ..) => true ,
838
+ ProjectionTyCandidate :: TraitDef ( ..) |
839
+ ProjectionTyCandidate :: Select ( ..) => false ,
840
+ } ) ;
841
+ debug ! ( "resulting candidate set: {:?}" , candidates. vec) ;
842
+ if candidates. vec . len ( ) != 1 {
843
+ candidates. ambiguous = true ;
844
+ return false ;
845
+ }
846
+ }
847
+
848
+ assert ! ( candidates. vec. len( ) <= 1 ) ;
849
+ match candidates. vec . get ( 0 ) {
850
+ Some ( & ProjectionTyCandidate :: Select ( ..) ) => true ,
851
+ _ => false ,
852
+ }
853
+ } ;
854
+
855
+ // Note that at here, if `ProjectionTyCandidate::Select` is not going to be
856
+ // a valid candidate, the closure is not executed at all. There are two cases,
857
+ // one being trait selection error, and another being ambiguous candidates.
858
+ // We handle both by the early return below.
821
859
if let Err ( e) = assemble_candidates_from_impls ( selcx,
822
860
obligation,
823
861
& obligation_trait_ref,
824
- & mut candidates) {
862
+ & mut candidates,
863
+ decide_commit) {
825
864
return Err ( ProjectionTyError :: TraitSelectionError ( e) ) ;
826
865
}
827
866
828
- debug ! ( "{} candidates, ambiguous={}" ,
829
- candidates. vec. len( ) ,
830
- candidates. ambiguous) ;
831
-
832
867
// Inherent ambiguity that prevents us from even enumerating the
833
868
// candidates.
834
869
if candidates. ambiguous {
835
870
return Err ( ProjectionTyError :: TooManyCandidates ) ;
836
871
}
837
872
838
- // Drop duplicates.
839
- //
840
- // Note: `candidates.vec` seems to be on the critical path of the
841
- // compiler. Replacing it with an HashSet was also tried, which would
842
- // render the following dedup unnecessary. The original comment indicated
843
- // that it was 9% slower, but that data is now obsolete and a new
844
- // benchmark should be performed.
845
- candidates. vec . sort_unstable ( ) ;
846
- candidates. vec . dedup ( ) ;
847
-
848
- // Prefer where-clauses. As in select, if there are multiple
849
- // candidates, we prefer where-clause candidates over impls. This
850
- // may seem a bit surprising, since impls are the source of
851
- // "truth" in some sense, but in fact some of the impls that SEEM
852
- // applicable are not, because of nested obligations. Where
853
- // clauses are the safer choice. See the comment on
854
- // `select::SelectionCandidate` and #21974 for more details.
855
- if candidates. vec . len ( ) > 1 {
856
- debug ! ( "retaining param-env candidates only from {:?}" , candidates. vec) ;
857
- candidates. vec . retain ( |c| match * c {
858
- ProjectionTyCandidate :: ParamEnv ( ..) => true ,
859
- ProjectionTyCandidate :: TraitDef ( ..) |
860
- ProjectionTyCandidate :: Select => false ,
861
- } ) ;
862
- debug ! ( "resulting candidate set: {:?}" , candidates. vec) ;
863
- if candidates. vec . len ( ) != 1 {
864
- return Err ( ProjectionTyError :: TooManyCandidates ) ;
865
- }
866
- }
867
-
868
- assert ! ( candidates. vec. len( ) <= 1 ) ;
869
-
870
873
match candidates. vec . pop ( ) {
871
874
Some ( candidate) => {
872
875
Ok ( ProjectedTy :: Progress (
@@ -962,7 +965,7 @@ fn assemble_candidates_from_predicates<'cx, 'gcx, 'tcx, I>(
962
965
debug ! ( "assemble_candidates_from_predicates: predicate={:?}" ,
963
966
predicate) ;
964
967
match predicate {
965
- ty:: Predicate :: Projection ( ref data) => {
968
+ ty:: Predicate :: Projection ( data) => {
966
969
let same_def_id =
967
970
data. 0 . projection_ty . item_def_id == obligation. predicate . item_def_id ;
968
971
@@ -985,26 +988,33 @@ fn assemble_candidates_from_predicates<'cx, 'gcx, 'tcx, I>(
985
988
data, is_match, same_def_id) ;
986
989
987
990
if is_match {
988
- candidate_set. vec . push ( ctor ( data. clone ( ) ) ) ;
991
+ candidate_set. vec . push ( ctor ( data) ) ;
989
992
}
990
993
}
991
- _ => { }
994
+ _ => { }
992
995
}
993
996
}
994
997
}
995
998
996
- fn assemble_candidates_from_impls < ' cx , ' gcx , ' tcx > (
999
+ enum NoImplCandidateError < ' tcx > {
1000
+ CanceledCommit ,
1001
+ SelectionError ( SelectionError < ' tcx > ) ,
1002
+ }
1003
+
1004
+ fn assemble_candidates_from_impls < ' cx , ' gcx , ' tcx ,
1005
+ F : FnOnce ( & mut ProjectionTyCandidateSet < ' tcx > ) -> bool > (
997
1006
selcx : & mut SelectionContext < ' cx , ' gcx , ' tcx > ,
998
1007
obligation : & ProjectionTyObligation < ' tcx > ,
999
1008
obligation_trait_ref : & ty:: TraitRef < ' tcx > ,
1000
- candidate_set : & mut ProjectionTyCandidateSet < ' tcx > )
1009
+ candidate_set : & mut ProjectionTyCandidateSet < ' tcx > ,
1010
+ decide_commit : F )
1001
1011
-> Result < ( ) , SelectionError < ' tcx > >
1002
1012
{
1003
1013
// If we are resolving `<T as TraitRef<...>>::Item == Type`,
1004
1014
// start out by selecting the predicate `T as TraitRef<...>`:
1005
1015
let poly_trait_ref = obligation_trait_ref. to_poly_trait_ref ( ) ;
1006
1016
let trait_obligation = obligation. with ( poly_trait_ref. to_poly_trait_predicate ( ) ) ;
1007
- selcx. infcx ( ) . probe ( |_| {
1017
+ match selcx. infcx ( ) . commit_if_ok ( |_| {
1008
1018
let vtable = match selcx. select ( & trait_obligation) {
1009
1019
Ok ( Some ( vtable) ) => vtable,
1010
1020
Ok ( None ) => {
@@ -1014,21 +1024,20 @@ fn assemble_candidates_from_impls<'cx, 'gcx, 'tcx>(
1014
1024
Err ( e) => {
1015
1025
debug ! ( "assemble_candidates_from_impls: selection error {:?}" ,
1016
1026
e) ;
1017
- return Err ( e ) ;
1027
+ return Err ( NoImplCandidateError :: SelectionError ( e ) ) ;
1018
1028
}
1019
1029
} ;
1020
1030
1021
- match vtable {
1022
- super :: VtableClosure ( _) |
1023
- super :: VtableGenerator ( _) |
1024
- super :: VtableFnPointer ( _) |
1025
- super :: VtableObject ( _) => {
1031
+ let eligible = match & vtable {
1032
+ & super :: VtableClosure ( _) |
1033
+ & super :: VtableGenerator ( _) |
1034
+ & super :: VtableFnPointer ( _) |
1035
+ & super :: VtableObject ( _) => {
1026
1036
debug ! ( "assemble_candidates_from_impls: vtable={:?}" ,
1027
1037
vtable) ;
1028
-
1029
- candidate_set. vec . push ( ProjectionTyCandidate :: Select ) ;
1038
+ true
1030
1039
}
1031
- super :: VtableImpl ( ref impl_data) => {
1040
+ & super :: VtableImpl ( ref impl_data) => {
1032
1041
// We have to be careful when projecting out of an
1033
1042
// impl because of specialization. If we are not in
1034
1043
// trans (i.e., projection mode is not "any"), and the
@@ -1072,29 +1081,27 @@ fn assemble_candidates_from_impls<'cx, 'gcx, 'tcx>(
1072
1081
node_item. item . defaultness . has_value ( )
1073
1082
} else {
1074
1083
node_item. item . defaultness . is_default ( ) ||
1075
- selcx. tcx ( ) . impl_is_default ( node_item. node . def_id ( ) )
1084
+ selcx. tcx ( ) . impl_is_default ( node_item. node . def_id ( ) )
1076
1085
} ;
1077
1086
1078
1087
// Only reveal a specializable default if we're past type-checking
1079
1088
// and the obligations is monomorphic, otherwise passes such as
1080
1089
// transmute checking and polymorphic MIR optimizations could
1081
1090
// get a result which isn't correct for all monomorphizations.
1082
- let new_candidate = if !is_default {
1083
- Some ( ProjectionTyCandidate :: Select )
1091
+ if !is_default {
1092
+ true
1084
1093
} else if obligation. param_env . reveal == Reveal :: All {
1085
1094
assert ! ( !poly_trait_ref. needs_infer( ) ) ;
1086
1095
if !poly_trait_ref. needs_subst ( ) {
1087
- Some ( ProjectionTyCandidate :: Select )
1096
+ true
1088
1097
} else {
1089
- None
1098
+ false
1090
1099
}
1091
1100
} else {
1092
- None
1093
- } ;
1094
-
1095
- candidate_set. vec . extend ( new_candidate) ;
1101
+ false
1102
+ }
1096
1103
}
1097
- super :: VtableParam ( ..) => {
1104
+ & super :: VtableParam ( ..) => {
1098
1105
// This case tell us nothing about the value of an
1099
1106
// associated type. Consider:
1100
1107
//
@@ -1120,19 +1127,32 @@ fn assemble_candidates_from_impls<'cx, 'gcx, 'tcx>(
1120
1127
// in the compiler: a trait predicate (`T : SomeTrait`) and a
1121
1128
// projection. And the projection where clause is handled
1122
1129
// in `assemble_candidates_from_param_env`.
1130
+ false
1123
1131
}
1124
- super :: VtableAutoImpl ( ..) |
1125
- super :: VtableBuiltin ( ..) => {
1132
+ & super :: VtableAutoImpl ( ..) |
1133
+ & super :: VtableBuiltin ( ..) => {
1126
1134
// These traits have no associated types.
1127
1135
span_bug ! (
1128
1136
obligation. cause. span,
1129
1137
"Cannot project an associated type from `{:?}`" ,
1130
1138
vtable) ;
1131
1139
}
1140
+ } ;
1141
+
1142
+ if eligible {
1143
+ candidate_set. vec . push ( ProjectionTyCandidate :: Select ( vtable) ) ;
1132
1144
}
1133
1145
1134
- Ok ( ( ) )
1135
- } )
1146
+ if decide_commit ( candidate_set) {
1147
+ Ok ( ( ) )
1148
+ } else {
1149
+ Err ( NoImplCandidateError :: CanceledCommit )
1150
+ }
1151
+ } ) {
1152
+ Ok ( ( ) ) => Ok ( ( ) ) ,
1153
+ Err ( NoImplCandidateError :: CanceledCommit ) => Ok ( ( ) ) ,
1154
+ Err ( NoImplCandidateError :: SelectionError ( e) ) => Err ( e) ,
1155
+ }
1136
1156
}
1137
1157
1138
1158
fn confirm_candidate < ' cx , ' gcx , ' tcx > (
@@ -1152,30 +1172,19 @@ fn confirm_candidate<'cx, 'gcx, 'tcx>(
1152
1172
confirm_param_env_candidate ( selcx, obligation, poly_projection)
1153
1173
}
1154
1174
1155
- ProjectionTyCandidate :: Select => {
1156
- confirm_select_candidate ( selcx, obligation, obligation_trait_ref)
1175
+ ProjectionTyCandidate :: Select ( vtable ) => {
1176
+ confirm_select_candidate ( selcx, obligation, obligation_trait_ref, vtable )
1157
1177
}
1158
1178
}
1159
1179
}
1160
1180
1161
1181
fn confirm_select_candidate < ' cx , ' gcx , ' tcx > (
1162
1182
selcx : & mut SelectionContext < ' cx , ' gcx , ' tcx > ,
1163
1183
obligation : & ProjectionTyObligation < ' tcx > ,
1164
- obligation_trait_ref : & ty:: TraitRef < ' tcx > )
1184
+ obligation_trait_ref : & ty:: TraitRef < ' tcx > ,
1185
+ vtable : Selection < ' tcx > )
1165
1186
-> Progress < ' tcx >
1166
1187
{
1167
- let poly_trait_ref = obligation_trait_ref. to_poly_trait_ref ( ) ;
1168
- let trait_obligation = obligation. with ( poly_trait_ref. to_poly_trait_predicate ( ) ) ;
1169
- let vtable = match selcx. select ( & trait_obligation) {
1170
- Ok ( Some ( vtable) ) => vtable,
1171
- _ => {
1172
- span_bug ! (
1173
- obligation. cause. span,
1174
- "Failed to select `{:?}`" ,
1175
- trait_obligation) ;
1176
- }
1177
- } ;
1178
-
1179
1188
match vtable {
1180
1189
super :: VtableImpl ( data) =>
1181
1190
confirm_impl_candidate ( selcx, obligation, data) ,
0 commit comments