@@ -11,6 +11,7 @@ use rustc_data_structures::base_n;
11
11
use rustc_data_structures:: fx:: FxHashMap ;
12
12
use rustc_hir as hir;
13
13
use rustc_middle:: ty:: layout:: IntegerExt ;
14
+ use rustc_middle:: ty:: TypeVisitableExt ;
14
15
use rustc_middle:: ty:: {
15
16
self , Const , ExistentialPredicate , FloatTy , FnSig , Instance , IntTy , List , Region , RegionKind ,
16
17
TermKind , Ty , TyCtxt , UintTy ,
@@ -21,7 +22,9 @@ use rustc_span::sym;
21
22
use rustc_target:: abi:: call:: { Conv , FnAbi , PassMode } ;
22
23
use rustc_target:: abi:: Integer ;
23
24
use rustc_target:: spec:: abi:: Abi ;
25
+ use rustc_trait_selection:: traits;
24
26
use std:: fmt:: Write as _;
27
+ use std:: iter;
25
28
26
29
use crate :: typeid:: TypeIdOptions ;
27
30
@@ -1110,51 +1113,45 @@ pub fn typeid_for_instance<'tcx>(
1110
1113
instance. args = strip_receiver_auto ( tcx, instance. args )
1111
1114
}
1112
1115
1116
+ if let Some ( impl_id) = tcx. impl_of_method ( instance. def_id ( ) )
1117
+ && let Some ( trait_ref) = tcx. impl_trait_ref ( impl_id)
1118
+ {
1119
+ let impl_method = tcx. associated_item ( instance. def_id ( ) ) ;
1120
+ let method_id = impl_method
1121
+ . trait_item_def_id
1122
+ . expect ( "Part of a trait implementation, but not linked to the def_id?" ) ;
1123
+ let trait_method = tcx. associated_item ( method_id) ;
1124
+ if traits:: is_vtable_safe_method ( tcx, trait_ref. skip_binder ( ) . def_id , trait_method) {
1125
+ // Trait methods will have a Self polymorphic parameter, where the concreteized
1126
+ // implementatation will not. We need to walk back to the more general trait method
1127
+ let trait_ref = tcx. normalize_erasing_regions (
1128
+ ty:: ParamEnv :: reveal_all ( ) ,
1129
+ trait_ref. instantiate ( tcx, instance. args ) ,
1130
+ ) ;
1131
+ let invoke_ty = trait_object_ty ( tcx, ty:: Binder :: dummy ( trait_ref) ) ;
1132
+
1133
+ // At the call site, any call to this concrete function through a vtable will be
1134
+ // `Virtual(method_id, idx)` with appropriate arguments for the method. Since we have the
1135
+ // original method id, and we've recovered the trait arguments, we can make the callee
1136
+ // instance we're computing the alias set for match the caller instance.
1137
+ //
1138
+ // Right now, our code ignores the vtable index everywhere, so we use 0 as a placeholder.
1139
+ // If we ever *do* start encoding the vtable index, we will need to generate an alias set
1140
+ // based on which vtables we are putting this method into, as there will be more than one
1141
+ // index value when supertraits are involved.
1142
+ instance. def = ty:: InstanceDef :: Virtual ( method_id, 0 ) ;
1143
+ let abstract_trait_args =
1144
+ tcx. mk_args_trait ( invoke_ty, trait_ref. args . into_iter ( ) . skip ( 1 ) ) ;
1145
+ instance. args = instance. args . rebase_onto ( tcx, impl_id, abstract_trait_args) ;
1146
+ }
1147
+ }
1148
+
1113
1149
let fn_abi = tcx
1114
1150
. fn_abi_of_instance ( tcx. param_env ( instance. def_id ( ) ) . and ( ( instance, ty:: List :: empty ( ) ) ) )
1115
1151
. unwrap_or_else ( |instance| {
1116
1152
bug ! ( "typeid_for_instance: couldn't get fn_abi of instance {:?}" , instance)
1117
1153
} ) ;
1118
1154
1119
- // If this instance is a method and self is a reference, get the impl it belongs to
1120
- let impl_def_id = tcx. impl_of_method ( instance. def_id ( ) ) ;
1121
- if impl_def_id. is_some ( ) && !fn_abi. args . is_empty ( ) && fn_abi. args [ 0 ] . layout . ty . is_ref ( ) {
1122
- // If this impl is not an inherent impl, get the trait it implements
1123
- if let Some ( trait_ref) = tcx. impl_trait_ref ( impl_def_id. unwrap ( ) ) {
1124
- // Transform the concrete self into a reference to a trait object
1125
- let existential_predicate = trait_ref. map_bound ( |trait_ref| {
1126
- ty:: ExistentialPredicate :: Trait ( ty:: ExistentialTraitRef :: erase_self_ty (
1127
- tcx, trait_ref,
1128
- ) )
1129
- } ) ;
1130
- let existential_predicates = tcx. mk_poly_existential_predicates ( & [ ty:: Binder :: dummy (
1131
- existential_predicate. skip_binder ( ) ,
1132
- ) ] ) ;
1133
- // Is the concrete self mutable?
1134
- let self_ty = if fn_abi. args [ 0 ] . layout . ty . is_mutable_ptr ( ) {
1135
- Ty :: new_mut_ref (
1136
- tcx,
1137
- tcx. lifetimes . re_erased ,
1138
- Ty :: new_dynamic ( tcx, existential_predicates, tcx. lifetimes . re_erased , ty:: Dyn ) ,
1139
- )
1140
- } else {
1141
- Ty :: new_imm_ref (
1142
- tcx,
1143
- tcx. lifetimes . re_erased ,
1144
- Ty :: new_dynamic ( tcx, existential_predicates, tcx. lifetimes . re_erased , ty:: Dyn ) ,
1145
- )
1146
- } ;
1147
-
1148
- // Replace the concrete self in an fn_abi clone by the reference to a trait object
1149
- let mut fn_abi = fn_abi. clone ( ) ;
1150
- // HACK(rcvalle): It is okay to not replace or update the entire ArgAbi here because the
1151
- // other fields are never used.
1152
- fn_abi. args [ 0 ] . layout . ty = self_ty;
1153
-
1154
- return typeid_for_fnabi ( tcx, & fn_abi, options) ;
1155
- }
1156
- }
1157
-
1158
1155
typeid_for_fnabi ( tcx, & fn_abi, options)
1159
1156
}
1160
1157
@@ -1180,3 +1177,36 @@ fn strip_receiver_auto<'tcx>(
1180
1177
} ;
1181
1178
tcx. mk_args_trait ( new_rcvr, args. into_iter ( ) . skip ( 1 ) )
1182
1179
}
1180
+
1181
+ fn trait_object_ty < ' tcx > ( tcx : TyCtxt < ' tcx > , poly_trait_ref : ty:: PolyTraitRef < ' tcx > ) -> Ty < ' tcx > {
1182
+ assert ! ( !poly_trait_ref. has_non_region_param( ) ) ;
1183
+ let principal_pred = poly_trait_ref. map_bound ( |trait_ref| {
1184
+ ty:: ExistentialPredicate :: Trait ( ty:: ExistentialTraitRef :: erase_self_ty ( tcx, trait_ref) )
1185
+ } ) ;
1186
+ let mut assoc_preds: Vec < _ > = traits:: supertraits ( tcx, poly_trait_ref)
1187
+ . flat_map ( |super_poly_trait_ref| {
1188
+ tcx. associated_items ( super_poly_trait_ref. def_id ( ) )
1189
+ . in_definition_order ( )
1190
+ . filter ( |item| item. kind == ty:: AssocKind :: Type )
1191
+ . map ( move |assoc_ty| {
1192
+ super_poly_trait_ref. map_bound ( |super_trait_ref| {
1193
+ let alias_ty = tcx. mk_alias_ty ( assoc_ty. def_id , super_trait_ref. args ) ;
1194
+ let resolved = tcx. normalize_erasing_regions (
1195
+ ty:: ParamEnv :: reveal_all ( ) ,
1196
+ alias_ty. to_ty ( tcx) ,
1197
+ ) ;
1198
+ ty:: ExistentialPredicate :: Projection ( ty:: ExistentialProjection {
1199
+ def_id : assoc_ty. def_id ,
1200
+ args : ty:: ExistentialTraitRef :: erase_self_ty ( tcx, super_trait_ref) . args ,
1201
+ term : resolved. into ( ) ,
1202
+ } )
1203
+ } )
1204
+ } )
1205
+ } )
1206
+ . collect ( ) ;
1207
+ assoc_preds. sort_by ( |a, b| a. skip_binder ( ) . stable_cmp ( tcx, & b. skip_binder ( ) ) ) ;
1208
+ let preds = tcx. mk_poly_existential_predicates_from_iter (
1209
+ iter:: once ( principal_pred) . chain ( assoc_preds. into_iter ( ) ) ,
1210
+ ) ;
1211
+ Ty :: new_dynamic ( tcx, preds, tcx. lifetimes . re_erased , ty:: Dyn )
1212
+ }
0 commit comments