Skip to content

Commit 1fd3ee0

Browse files
authored
Rollup merge of #123066 - maurer:cfi-erased-lifetime-ice, r=compiler-errors
CFI: (actually) check that methods are object-safe before projecting their receivers to `dyn Trait` in CFI `trait_object_ty` assumed that associated types would be fully determined by the trait. This is *almost* true - const parameters and type parameters are no longer allowed, but lifetime parameters are. Since we erase all lifetime parameters anyways, instantiate it with as many erased regions as it needs. Fixes: #123053 r? `@compiler-errors`
2 parents 6ecbc34 + 70e1d23 commit 1fd3ee0

File tree

2 files changed

+46
-3
lines changed

2 files changed

+46
-3
lines changed

compiler/rustc_symbol_mangling/src/typeid/typeid_itanium_cxx_abi.rs

+8-3
Original file line numberDiff line numberDiff line change
@@ -1124,7 +1124,10 @@ pub fn typeid_for_instance<'tcx>(
11241124
.trait_item_def_id
11251125
.expect("Part of a trait implementation, but not linked to the def_id?");
11261126
let trait_method = tcx.associated_item(method_id);
1127-
if traits::is_vtable_safe_method(tcx, trait_ref.skip_binder().def_id, trait_method) {
1127+
let trait_id = trait_ref.skip_binder().def_id;
1128+
if traits::is_vtable_safe_method(tcx, trait_id, trait_method)
1129+
&& tcx.object_safety_violations(trait_id).is_empty()
1130+
{
11281131
// Trait methods will have a Self polymorphic parameter, where the concreteized
11291132
// implementatation will not. We need to walk back to the more general trait method
11301133
let trait_ref = tcx.instantiate_and_normalize_erasing_regions(
@@ -1152,8 +1155,8 @@ pub fn typeid_for_instance<'tcx>(
11521155

11531156
let fn_abi = tcx
11541157
.fn_abi_of_instance(tcx.param_env(instance.def_id()).and((instance, ty::List::empty())))
1155-
.unwrap_or_else(|instance| {
1156-
bug!("typeid_for_instance: couldn't get fn_abi of instance {:?}", instance)
1158+
.unwrap_or_else(|error| {
1159+
bug!("typeid_for_instance: couldn't get fn_abi of instance {instance:?}: {error:?}")
11571160
});
11581161

11591162
typeid_for_fnabi(tcx, fn_abi, options)
@@ -1182,6 +1185,7 @@ fn strip_receiver_auto<'tcx>(
11821185
tcx.mk_args_trait(new_rcvr, args.into_iter().skip(1))
11831186
}
11841187

1188+
#[instrument(skip(tcx), ret)]
11851189
fn trait_object_ty<'tcx>(tcx: TyCtxt<'tcx>, poly_trait_ref: ty::PolyTraitRef<'tcx>) -> Ty<'tcx> {
11861190
assert!(!poly_trait_ref.has_non_region_param());
11871191
let principal_pred = poly_trait_ref.map_bound(|trait_ref| {
@@ -1199,6 +1203,7 @@ fn trait_object_ty<'tcx>(tcx: TyCtxt<'tcx>, poly_trait_ref: ty::PolyTraitRef<'tc
11991203
ty::ParamEnv::reveal_all(),
12001204
alias_ty.to_ty(tcx),
12011205
);
1206+
debug!("Resolved {:?} -> {resolved}", alias_ty.to_ty(tcx));
12021207
ty::ExistentialPredicate::Projection(ty::ExistentialProjection {
12031208
def_id: assoc_ty.def_id,
12041209
args: ty::ExistentialTraitRef::erase_self_ty(tcx, super_trait_ref).args,
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,38 @@
1+
// Regression test for issue 123053, where associated types with lifetimes caused generation of the
2+
// trait object type to fail, causing an ICE.
3+
//
4+
//@ needs-sanitizer-cfi
5+
//@ compile-flags: -Ccodegen-units=1 -Clto -Ctarget-feature=-crt-static -Zsanitizer=cfi --edition=2021
6+
//@ no-prefer-dynamic
7+
//@ only-x86_64-unknown-linux-gnu
8+
//@ build-pass
9+
10+
trait Iterable {
11+
type Item<'a>
12+
where
13+
Self: 'a;
14+
type Iter<'a>: Iterator<Item = Self::Item<'a>>
15+
where
16+
Self: 'a;
17+
18+
fn iter<'a>(&'a self) -> Self::Iter<'a>;
19+
}
20+
21+
impl<T> Iterable for [T] {
22+
type Item<'a> = <std::slice::Iter<'a, T> as Iterator>::Item where T: 'a;
23+
type Iter<'a> = std::slice::Iter<'a, T> where T: 'a;
24+
25+
fn iter<'a>(&'a self) -> Self::Iter<'a> {
26+
self.iter()
27+
}
28+
}
29+
30+
fn get_first<'a, I: Iterable + ?Sized>(it: &'a I) -> Option<I::Item<'a>> {
31+
it.iter().next()
32+
}
33+
34+
fn main() {
35+
let v = vec![1, 2, 3];
36+
37+
assert_eq!(Some(&1), get_first(&*v));
38+
}

0 commit comments

Comments
 (0)