Skip to content

Commit af98101

Browse files
committed
Auto merge of #123029 - matthiaskrgr:rollup-6qsevhx, r=matthiaskrgr
Rollup of 7 pull requests Successful merges: - #122858 (Tweak `parse_dot_suffix_expr`) - #122982 (Add more comments to the bootstrap code that handles `tests/coverage`) - #122990 (Clarify transmute example) - #122995 (Clean up unnecessary headers/flags in coverage mir-opt tests) - #123003 (CFI: Handle dyn with no principal) - #123005 (CFI: Support complex receivers) - #123020 (Temporarily remove nnethercote from the review rotation.) r? `@ghost` `@rustbot` modify labels: rollup
2 parents dda2372 + ab8084a commit af98101

16 files changed

+278
-160
lines changed

Cargo.lock

+1
Original file line numberDiff line numberDiff line change
@@ -4634,6 +4634,7 @@ dependencies = [
46344634
"rustc_session",
46354635
"rustc_span",
46364636
"rustc_target",
4637+
"rustc_trait_selection",
46374638
"tracing",
46384639
"twox-hash",
46394640
]

compiler/rustc_parse/src/parser/expr.rs

+50-48
Original file line numberDiff line numberDiff line change
@@ -16,7 +16,6 @@ use core::mem;
1616
use core::ops::ControlFlow;
1717
use rustc_ast::ptr::P;
1818
use rustc_ast::token::{self, Delimiter, Token, TokenKind};
19-
use rustc_ast::tokenstream::Spacing;
2019
use rustc_ast::util::case::Case;
2120
use rustc_ast::util::classify;
2221
use rustc_ast::util::parser::{prec_let_scrutinee_needs_par, AssocOp, Fixity};
@@ -999,13 +998,57 @@ impl<'a> Parser<'a> {
999998
}
1000999

10011000
pub fn parse_dot_suffix_expr(&mut self, lo: Span, base: P<Expr>) -> PResult<'a, P<Expr>> {
1001+
// At this point we've consumed something like `expr.` and `self.token` holds the token
1002+
// after the dot.
10021003
match self.token.uninterpolate().kind {
10031004
token::Ident(..) => self.parse_dot_suffix(base, lo),
10041005
token::Literal(token::Lit { kind: token::Integer, symbol, suffix }) => {
1005-
Ok(self.parse_expr_tuple_field_access(lo, base, symbol, suffix, None))
1006+
let ident_span = self.token.span;
1007+
self.bump();
1008+
Ok(self.mk_expr_tuple_field_access(lo, ident_span, base, symbol, suffix))
10061009
}
10071010
token::Literal(token::Lit { kind: token::Float, symbol, suffix }) => {
1008-
Ok(self.parse_expr_tuple_field_access_float(lo, base, symbol, suffix))
1011+
Ok(match self.break_up_float(symbol, self.token.span) {
1012+
// 1e2
1013+
DestructuredFloat::Single(sym, _sp) => {
1014+
// `foo.1e2`: a single complete dot access, fully consumed. We end up with
1015+
// the `1e2` token in `self.prev_token` and the following token in
1016+
// `self.token`.
1017+
let ident_span = self.token.span;
1018+
self.bump();
1019+
self.mk_expr_tuple_field_access(lo, ident_span, base, sym, suffix)
1020+
}
1021+
// 1.
1022+
DestructuredFloat::TrailingDot(sym, ident_span, dot_span) => {
1023+
// `foo.1.`: a single complete dot access and the start of another.
1024+
// We end up with the `sym` (`1`) token in `self.prev_token` and a dot in
1025+
// `self.token`.
1026+
assert!(suffix.is_none());
1027+
self.token = Token::new(token::Ident(sym, IdentIsRaw::No), ident_span);
1028+
self.bump_with((Token::new(token::Dot, dot_span), self.token_spacing));
1029+
self.mk_expr_tuple_field_access(lo, ident_span, base, sym, None)
1030+
}
1031+
// 1.2 | 1.2e3
1032+
DestructuredFloat::MiddleDot(
1033+
sym1,
1034+
ident1_span,
1035+
_dot_span,
1036+
sym2,
1037+
ident2_span,
1038+
) => {
1039+
// `foo.1.2` (or `foo.1.2e3`): two complete dot accesses. We end up with
1040+
// the `sym2` (`2` or `2e3`) token in `self.prev_token` and the following
1041+
// token in `self.token`.
1042+
let next_token2 =
1043+
Token::new(token::Ident(sym2, IdentIsRaw::No), ident2_span);
1044+
self.bump_with((next_token2, self.token_spacing));
1045+
self.bump();
1046+
let base1 =
1047+
self.mk_expr_tuple_field_access(lo, ident1_span, base, sym1, None);
1048+
self.mk_expr_tuple_field_access(lo, ident2_span, base1, sym2, suffix)
1049+
}
1050+
DestructuredFloat::Error => base,
1051+
})
10091052
}
10101053
_ => {
10111054
self.error_unexpected_after_dot();
@@ -1119,41 +1162,6 @@ impl<'a> Parser<'a> {
11191162
}
11201163
}
11211164

1122-
fn parse_expr_tuple_field_access_float(
1123-
&mut self,
1124-
lo: Span,
1125-
base: P<Expr>,
1126-
float: Symbol,
1127-
suffix: Option<Symbol>,
1128-
) -> P<Expr> {
1129-
match self.break_up_float(float, self.token.span) {
1130-
// 1e2
1131-
DestructuredFloat::Single(sym, _sp) => {
1132-
self.parse_expr_tuple_field_access(lo, base, sym, suffix, None)
1133-
}
1134-
// 1.
1135-
DestructuredFloat::TrailingDot(sym, ident_span, dot_span) => {
1136-
assert!(suffix.is_none());
1137-
self.token = Token::new(token::Ident(sym, IdentIsRaw::No), ident_span);
1138-
let next_token = (Token::new(token::Dot, dot_span), self.token_spacing);
1139-
self.parse_expr_tuple_field_access(lo, base, sym, None, Some(next_token))
1140-
}
1141-
// 1.2 | 1.2e3
1142-
DestructuredFloat::MiddleDot(symbol1, ident1_span, dot_span, symbol2, ident2_span) => {
1143-
self.token = Token::new(token::Ident(symbol1, IdentIsRaw::No), ident1_span);
1144-
// This needs to be `Spacing::Alone` to prevent regressions.
1145-
// See issue #76399 and PR #76285 for more details
1146-
let next_token1 = (Token::new(token::Dot, dot_span), Spacing::Alone);
1147-
let base1 =
1148-
self.parse_expr_tuple_field_access(lo, base, symbol1, None, Some(next_token1));
1149-
let next_token2 = Token::new(token::Ident(symbol2, IdentIsRaw::No), ident2_span);
1150-
self.bump_with((next_token2, self.token_spacing)); // `.`
1151-
self.parse_expr_tuple_field_access(lo, base1, symbol2, suffix, None)
1152-
}
1153-
DestructuredFloat::Error => base,
1154-
}
1155-
}
1156-
11571165
/// Parse the field access used in offset_of, matched by `$(e:expr)+`.
11581166
/// Currently returns a list of idents. However, it should be possible in
11591167
/// future to also do array indices, which might be arbitrary expressions.
@@ -1255,24 +1263,18 @@ impl<'a> Parser<'a> {
12551263
Ok(fields.into_iter().collect())
12561264
}
12571265

1258-
fn parse_expr_tuple_field_access(
1266+
fn mk_expr_tuple_field_access(
12591267
&mut self,
12601268
lo: Span,
1269+
ident_span: Span,
12611270
base: P<Expr>,
12621271
field: Symbol,
12631272
suffix: Option<Symbol>,
1264-
next_token: Option<(Token, Spacing)>,
12651273
) -> P<Expr> {
1266-
match next_token {
1267-
Some(next_token) => self.bump_with(next_token),
1268-
None => self.bump(),
1269-
}
1270-
let span = self.prev_token.span;
1271-
let field = ExprKind::Field(base, Ident::new(field, span));
12721274
if let Some(suffix) = suffix {
1273-
self.expect_no_tuple_index_suffix(span, suffix);
1275+
self.expect_no_tuple_index_suffix(ident_span, suffix);
12741276
}
1275-
self.mk_expr(lo.to(span), field)
1277+
self.mk_expr(lo.to(ident_span), ExprKind::Field(base, Ident::new(field, ident_span)))
12761278
}
12771279

12781280
/// Parse a function call expression, `expr(...)`.

compiler/rustc_symbol_mangling/Cargo.toml

+1
Original file line numberDiff line numberDiff line change
@@ -15,6 +15,7 @@ rustc_middle = { path = "../rustc_middle" }
1515
rustc_session = { path = "../rustc_session" }
1616
rustc_span = { path = "../rustc_span" }
1717
rustc_target = { path = "../rustc_target" }
18+
rustc_trait_selection = { path = "../rustc_trait_selection" }
1819
tracing = "0.1"
1920
twox-hash = "1.6.3"
2021
# tidy-alphabetical-end

compiler/rustc_symbol_mangling/src/lib.rs

+1
Original file line numberDiff line numberDiff line change
@@ -90,6 +90,7 @@
9090
#![doc(html_root_url = "https://doc.rust-lang.org/nightly/nightly-rustc/")]
9191
#![doc(rust_logo)]
9292
#![feature(rustdoc_internals)]
93+
#![feature(let_chains)]
9394
#![allow(internal_features)]
9495

9596
#[macro_use]

compiler/rustc_symbol_mangling/src/typeid/typeid_itanium_cxx_abi.rs

+84-52
Original file line numberDiff line numberDiff line change
@@ -11,6 +11,7 @@ use rustc_data_structures::base_n;
1111
use rustc_data_structures::fx::FxHashMap;
1212
use rustc_hir as hir;
1313
use rustc_middle::ty::layout::IntegerExt;
14+
use rustc_middle::ty::TypeVisitableExt;
1415
use rustc_middle::ty::{
1516
self, Const, ExistentialPredicate, FloatTy, FnSig, Instance, IntTy, List, Region, RegionKind,
1617
TermKind, Ty, TyCtxt, UintTy,
@@ -21,7 +22,9 @@ use rustc_span::sym;
2122
use rustc_target::abi::call::{Conv, FnAbi, PassMode};
2223
use rustc_target::abi::Integer;
2324
use rustc_target::spec::abi::Abi;
25+
use rustc_trait_selection::traits;
2426
use std::fmt::Write as _;
27+
use std::iter;
2528

2629
use crate::typeid::TypeIdOptions;
2730

@@ -747,9 +750,8 @@ fn transform_predicates<'tcx>(
747750
tcx: TyCtxt<'tcx>,
748751
predicates: &List<ty::PolyExistentialPredicate<'tcx>>,
749752
) -> &'tcx List<ty::PolyExistentialPredicate<'tcx>> {
750-
let predicates: Vec<ty::PolyExistentialPredicate<'tcx>> = predicates
751-
.iter()
752-
.filter_map(|predicate| match predicate.skip_binder() {
753+
tcx.mk_poly_existential_predicates_from_iter(predicates.iter().filter_map(|predicate| {
754+
match predicate.skip_binder() {
753755
ty::ExistentialPredicate::Trait(trait_ref) => {
754756
let trait_ref = ty::TraitRef::identity(tcx, trait_ref.def_id);
755757
Some(ty::Binder::dummy(ty::ExistentialPredicate::Trait(
@@ -758,9 +760,8 @@ fn transform_predicates<'tcx>(
758760
}
759761
ty::ExistentialPredicate::Projection(..) => None,
760762
ty::ExistentialPredicate::AutoTrait(..) => Some(predicate),
761-
})
762-
.collect();
763-
tcx.mk_poly_existential_predicates(&predicates)
763+
}
764+
}))
764765
}
765766

766767
/// Transforms args for being encoded and used in the substitution dictionary.
@@ -1115,51 +1116,46 @@ pub fn typeid_for_instance<'tcx>(
11151116
instance.args = strip_receiver_auto(tcx, instance.args)
11161117
}
11171118

1119+
if let Some(impl_id) = tcx.impl_of_method(instance.def_id())
1120+
&& let Some(trait_ref) = tcx.impl_trait_ref(impl_id)
1121+
{
1122+
let impl_method = tcx.associated_item(instance.def_id());
1123+
let method_id = impl_method
1124+
.trait_item_def_id
1125+
.expect("Part of a trait implementation, but not linked to the def_id?");
1126+
let trait_method = tcx.associated_item(method_id);
1127+
if traits::is_vtable_safe_method(tcx, trait_ref.skip_binder().def_id, trait_method) {
1128+
// Trait methods will have a Self polymorphic parameter, where the concreteized
1129+
// implementatation will not. We need to walk back to the more general trait method
1130+
let trait_ref = tcx.instantiate_and_normalize_erasing_regions(
1131+
instance.args,
1132+
ty::ParamEnv::reveal_all(),
1133+
trait_ref,
1134+
);
1135+
let invoke_ty = trait_object_ty(tcx, ty::Binder::dummy(trait_ref));
1136+
1137+
// At the call site, any call to this concrete function through a vtable will be
1138+
// `Virtual(method_id, idx)` with appropriate arguments for the method. Since we have the
1139+
// original method id, and we've recovered the trait arguments, we can make the callee
1140+
// instance we're computing the alias set for match the caller instance.
1141+
//
1142+
// Right now, our code ignores the vtable index everywhere, so we use 0 as a placeholder.
1143+
// If we ever *do* start encoding the vtable index, we will need to generate an alias set
1144+
// based on which vtables we are putting this method into, as there will be more than one
1145+
// index value when supertraits are involved.
1146+
instance.def = ty::InstanceDef::Virtual(method_id, 0);
1147+
let abstract_trait_args =
1148+
tcx.mk_args_trait(invoke_ty, trait_ref.args.into_iter().skip(1));
1149+
instance.args = instance.args.rebase_onto(tcx, impl_id, abstract_trait_args);
1150+
}
1151+
}
1152+
11181153
let fn_abi = tcx
11191154
.fn_abi_of_instance(tcx.param_env(instance.def_id()).and((instance, ty::List::empty())))
11201155
.unwrap_or_else(|instance| {
11211156
bug!("typeid_for_instance: couldn't get fn_abi of instance {:?}", instance)
11221157
});
11231158

1124-
// If this instance is a method and self is a reference, get the impl it belongs to
1125-
let impl_def_id = tcx.impl_of_method(instance.def_id());
1126-
if impl_def_id.is_some() && !fn_abi.args.is_empty() && fn_abi.args[0].layout.ty.is_ref() {
1127-
// If this impl is not an inherent impl, get the trait it implements
1128-
if let Some(trait_ref) = tcx.impl_trait_ref(impl_def_id.unwrap()) {
1129-
// Transform the concrete self into a reference to a trait object
1130-
let existential_predicate = trait_ref.map_bound(|trait_ref| {
1131-
ty::ExistentialPredicate::Trait(ty::ExistentialTraitRef::erase_self_ty(
1132-
tcx, trait_ref,
1133-
))
1134-
});
1135-
let existential_predicates = tcx.mk_poly_existential_predicates(&[ty::Binder::dummy(
1136-
existential_predicate.skip_binder(),
1137-
)]);
1138-
// Is the concrete self mutable?
1139-
let self_ty = if fn_abi.args[0].layout.ty.is_mutable_ptr() {
1140-
Ty::new_mut_ref(
1141-
tcx,
1142-
tcx.lifetimes.re_erased,
1143-
Ty::new_dynamic(tcx, existential_predicates, tcx.lifetimes.re_erased, ty::Dyn),
1144-
)
1145-
} else {
1146-
Ty::new_imm_ref(
1147-
tcx,
1148-
tcx.lifetimes.re_erased,
1149-
Ty::new_dynamic(tcx, existential_predicates, tcx.lifetimes.re_erased, ty::Dyn),
1150-
)
1151-
};
1152-
1153-
// Replace the concrete self in an fn_abi clone by the reference to a trait object
1154-
let mut fn_abi = fn_abi.clone();
1155-
// HACK(rcvalle): It is okay to not replace or update the entire ArgAbi here because the
1156-
// other fields are never used.
1157-
fn_abi.args[0].layout.ty = self_ty;
1158-
1159-
return typeid_for_fnabi(tcx, &fn_abi, options);
1160-
}
1161-
}
1162-
11631159
typeid_for_fnabi(tcx, fn_abi, options)
11641160
}
11651161

@@ -1171,14 +1167,50 @@ fn strip_receiver_auto<'tcx>(
11711167
let ty::Dynamic(preds, lifetime, kind) = ty.kind() else {
11721168
bug!("Tried to strip auto traits from non-dynamic type {ty}");
11731169
};
1174-
let filtered_preds =
1175-
if preds.principal().is_some() {
1170+
let new_rcvr = if preds.principal().is_some() {
1171+
let filtered_preds =
11761172
tcx.mk_poly_existential_predicates_from_iter(preds.into_iter().filter(|pred| {
11771173
!matches!(pred.skip_binder(), ty::ExistentialPredicate::AutoTrait(..))
1178-
}))
1179-
} else {
1180-
ty::List::empty()
1181-
};
1182-
let new_rcvr = Ty::new_dynamic(tcx, filtered_preds, *lifetime, *kind);
1174+
}));
1175+
Ty::new_dynamic(tcx, filtered_preds, *lifetime, *kind)
1176+
} else {
1177+
// If there's no principal type, re-encode it as a unit, since we don't know anything
1178+
// about it. This technically discards the knowledge that it was a type that was made
1179+
// into a trait object at some point, but that's not a lot.
1180+
tcx.types.unit
1181+
};
11831182
tcx.mk_args_trait(new_rcvr, args.into_iter().skip(1))
11841183
}
1184+
1185+
fn trait_object_ty<'tcx>(tcx: TyCtxt<'tcx>, poly_trait_ref: ty::PolyTraitRef<'tcx>) -> Ty<'tcx> {
1186+
assert!(!poly_trait_ref.has_non_region_param());
1187+
let principal_pred = poly_trait_ref.map_bound(|trait_ref| {
1188+
ty::ExistentialPredicate::Trait(ty::ExistentialTraitRef::erase_self_ty(tcx, trait_ref))
1189+
});
1190+
let mut assoc_preds: Vec<_> = traits::supertraits(tcx, poly_trait_ref)
1191+
.flat_map(|super_poly_trait_ref| {
1192+
tcx.associated_items(super_poly_trait_ref.def_id())
1193+
.in_definition_order()
1194+
.filter(|item| item.kind == ty::AssocKind::Type)
1195+
.map(move |assoc_ty| {
1196+
super_poly_trait_ref.map_bound(|super_trait_ref| {
1197+
let alias_ty = ty::AliasTy::new(tcx, assoc_ty.def_id, super_trait_ref.args);
1198+
let resolved = tcx.normalize_erasing_regions(
1199+
ty::ParamEnv::reveal_all(),
1200+
alias_ty.to_ty(tcx),
1201+
);
1202+
ty::ExistentialPredicate::Projection(ty::ExistentialProjection {
1203+
def_id: assoc_ty.def_id,
1204+
args: ty::ExistentialTraitRef::erase_self_ty(tcx, super_trait_ref).args,
1205+
term: resolved.into(),
1206+
})
1207+
})
1208+
})
1209+
})
1210+
.collect();
1211+
assoc_preds.sort_by(|a, b| a.skip_binder().stable_cmp(tcx, &b.skip_binder()));
1212+
let preds = tcx.mk_poly_existential_predicates_from_iter(
1213+
iter::once(principal_pred).chain(assoc_preds.into_iter()),
1214+
);
1215+
Ty::new_dynamic(tcx, preds, tcx.lifetimes.re_erased, ty::Dyn)
1216+
}

library/core/src/intrinsics.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -1356,7 +1356,7 @@ extern "rust-intrinsic" {
13561356
/// let v_clone = v_orig.clone();
13571357
///
13581358
/// // This is the suggested, safe way.
1359-
/// // It does copy the entire vector, though, into a new array.
1359+
/// // It may copy the entire vector into a new one though, but also may not.
13601360
/// let v_collected = v_clone.into_iter()
13611361
/// .map(Some)
13621362
/// .collect::<Vec<Option<&i32>>>();

0 commit comments

Comments
 (0)