Skip to content

Commit 697d953

Browse files
committed
Auto merge of rust-lang#129540 - tgross35:rollup-y6iaujw, r=tgross35
Rollup of 10 pull requests Successful merges: - rust-lang#128467 (Detect `*` operator on `!Sized` expression) - rust-lang#128524 (Don't suggest turning crate-level attributes into outer style) - rust-lang#128735 (Add a special case for `CStr`/`CString` in the `improper_ctypes` lint) - rust-lang#129429 (Print the generic parameter along with the variance in dumps.) - rust-lang#129430 (rustdoc: show exact case-sensitive matches first) - rust-lang#129449 (Put Pin::as_deref_mut in impl Pin<Ptr> / rearrange Pin methods) - rust-lang#129481 (Update `compiler_builtins` to `0.1.121`) - rust-lang#129482 (Add myself to the review rotation for libs) - rust-lang#129492 (Make wasm32 platform support docs easier to read) - rust-lang#129512 (update the doc comment on lintchecker b/c it parses html now) r? `@ghost` `@rustbot` modify labels: rollup
2 parents f167efa + 11ebb59 commit 697d953

File tree

71 files changed

+687
-303
lines changed

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

71 files changed

+687
-303
lines changed

compiler/rustc_ast/src/ast.rs

+11
Original file line numberDiff line numberDiff line change
@@ -2902,6 +2902,17 @@ pub struct AttrItem {
29022902
pub tokens: Option<LazyAttrTokenStream>,
29032903
}
29042904

2905+
impl AttrItem {
2906+
pub fn is_valid_for_outer_style(&self) -> bool {
2907+
self.path == sym::cfg_attr
2908+
|| self.path == sym::cfg
2909+
|| self.path == sym::forbid
2910+
|| self.path == sym::warn
2911+
|| self.path == sym::allow
2912+
|| self.path == sym::deny
2913+
}
2914+
}
2915+
29052916
/// `TraitRef`s appear in impls.
29062917
///
29072918
/// Resolution maps each `TraitRef`'s `ref_id` to its defining trait; that's all
Original file line numberDiff line numberDiff line change
@@ -1,18 +1,36 @@
1+
use std::fmt::Write;
2+
13
use rustc_hir::def::DefKind;
2-
use rustc_hir::def_id::CRATE_DEF_ID;
3-
use rustc_middle::ty::TyCtxt;
4+
use rustc_hir::def_id::{LocalDefId, CRATE_DEF_ID};
5+
use rustc_middle::ty::{GenericArgs, TyCtxt};
46
use rustc_span::symbol::sym;
57

8+
fn format_variances(tcx: TyCtxt<'_>, def_id: LocalDefId) -> String {
9+
let variances = tcx.variances_of(def_id);
10+
let generics = GenericArgs::identity_for_item(tcx, def_id);
11+
// 7 = 2-letter parameter + ": " + 1-letter variance + ", "
12+
let mut ret = String::with_capacity(2 + 7 * variances.len());
13+
ret.push('[');
14+
for (arg, variance) in generics.iter().zip(variances.iter()) {
15+
write!(ret, "{arg}: {variance:?}, ").unwrap();
16+
}
17+
// Remove trailing `, `.
18+
if !variances.is_empty() {
19+
ret.pop();
20+
ret.pop();
21+
}
22+
ret.push(']');
23+
ret
24+
}
25+
626
pub(crate) fn variances(tcx: TyCtxt<'_>) {
727
if tcx.has_attr(CRATE_DEF_ID, sym::rustc_variance_of_opaques) {
828
for id in tcx.hir().items() {
929
let DefKind::OpaqueTy = tcx.def_kind(id.owner_id) else { continue };
1030

11-
let variances = tcx.variances_of(id.owner_id);
12-
1331
tcx.dcx().emit_err(crate::errors::VariancesOf {
1432
span: tcx.def_span(id.owner_id),
15-
variances: format!("{variances:?}"),
33+
variances: format_variances(tcx, id.owner_id.def_id),
1634
});
1735
}
1836
}
@@ -22,11 +40,9 @@ pub(crate) fn variances(tcx: TyCtxt<'_>) {
2240
continue;
2341
}
2442

25-
let variances = tcx.variances_of(id.owner_id);
26-
2743
tcx.dcx().emit_err(crate::errors::VariancesOf {
2844
span: tcx.def_span(id.owner_id),
29-
variances: format!("{variances:?}"),
45+
variances: format_variances(tcx, id.owner_id.def_id),
3046
});
3147
}
3248
}

compiler/rustc_hir_typeck/src/expr.rs

+5-1
Original file line numberDiff line numberDiff line change
@@ -841,13 +841,15 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
841841
let ret_ty = ret_coercion.borrow().expected_ty();
842842
let return_expr_ty = self.check_expr_with_hint(return_expr, ret_ty);
843843
let mut span = return_expr.span;
844+
let mut hir_id = return_expr.hir_id;
844845
// Use the span of the trailing expression for our cause,
845846
// not the span of the entire function
846847
if !explicit_return
847848
&& let ExprKind::Block(body, _) = return_expr.kind
848849
&& let Some(last_expr) = body.expr
849850
{
850851
span = last_expr.span;
852+
hir_id = last_expr.hir_id;
851853
}
852854
ret_coercion.borrow_mut().coerce(
853855
self,
@@ -864,6 +866,7 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
864866
self.select_obligations_where_possible(|errors| {
865867
self.point_at_return_for_opaque_ty_error(
866868
errors,
869+
hir_id,
867870
span,
868871
return_expr_ty,
869872
return_expr.span,
@@ -921,6 +924,7 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
921924
fn point_at_return_for_opaque_ty_error(
922925
&self,
923926
errors: &mut Vec<traits::FulfillmentError<'tcx>>,
927+
hir_id: HirId,
924928
span: Span,
925929
return_expr_ty: Ty<'tcx>,
926930
return_span: Span,
@@ -935,7 +939,7 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
935939
let new_cause = ObligationCause::new(
936940
cause.span,
937941
cause.body_id,
938-
ObligationCauseCode::OpaqueReturnType(Some((return_expr_ty, span))),
942+
ObligationCauseCode::OpaqueReturnType(Some((return_expr_ty, hir_id))),
939943
);
940944
*cause = new_cause;
941945
}

compiler/rustc_lint/messages.ftl

+5
Original file line numberDiff line numberDiff line change
@@ -361,6 +361,11 @@ lint_improper_ctypes_box = box cannot be represented as a single pointer
361361
lint_improper_ctypes_char_help = consider using `u32` or `libc::wchar_t` instead
362362
363363
lint_improper_ctypes_char_reason = the `char` type has no C equivalent
364+
365+
lint_improper_ctypes_cstr_help =
366+
consider passing a `*const std::ffi::c_char` instead, and use `CStr::as_ptr()`
367+
lint_improper_ctypes_cstr_reason = `CStr`/`CString` do not have a guaranteed layout
368+
364369
lint_improper_ctypes_dyn = trait objects have no C equivalent
365370
366371
lint_improper_ctypes_enum_repr_help =

compiler/rustc_lint/src/types.rs

+39-15
Original file line numberDiff line numberDiff line change
@@ -985,6 +985,14 @@ struct ImproperCTypesVisitor<'a, 'tcx> {
985985
mode: CItemKind,
986986
}
987987

988+
/// Accumulator for recursive ffi type checking
989+
struct CTypesVisitorState<'tcx> {
990+
cache: FxHashSet<Ty<'tcx>>,
991+
/// The original type being checked, before we recursed
992+
/// to any other types it contains.
993+
base_ty: Ty<'tcx>,
994+
}
995+
988996
enum FfiResult<'tcx> {
989997
FfiSafe,
990998
FfiPhantom(Ty<'tcx>),
@@ -1213,7 +1221,7 @@ impl<'a, 'tcx> ImproperCTypesVisitor<'a, 'tcx> {
12131221
/// Checks if the given field's type is "ffi-safe".
12141222
fn check_field_type_for_ffi(
12151223
&self,
1216-
cache: &mut FxHashSet<Ty<'tcx>>,
1224+
acc: &mut CTypesVisitorState<'tcx>,
12171225
field: &ty::FieldDef,
12181226
args: GenericArgsRef<'tcx>,
12191227
) -> FfiResult<'tcx> {
@@ -1223,13 +1231,13 @@ impl<'a, 'tcx> ImproperCTypesVisitor<'a, 'tcx> {
12231231
.tcx
12241232
.try_normalize_erasing_regions(self.cx.param_env, field_ty)
12251233
.unwrap_or(field_ty);
1226-
self.check_type_for_ffi(cache, field_ty)
1234+
self.check_type_for_ffi(acc, field_ty)
12271235
}
12281236

12291237
/// Checks if the given `VariantDef`'s field types are "ffi-safe".
12301238
fn check_variant_for_ffi(
12311239
&self,
1232-
cache: &mut FxHashSet<Ty<'tcx>>,
1240+
acc: &mut CTypesVisitorState<'tcx>,
12331241
ty: Ty<'tcx>,
12341242
def: ty::AdtDef<'tcx>,
12351243
variant: &ty::VariantDef,
@@ -1239,7 +1247,7 @@ impl<'a, 'tcx> ImproperCTypesVisitor<'a, 'tcx> {
12391247
let transparent_with_all_zst_fields = if def.repr().transparent() {
12401248
if let Some(field) = transparent_newtype_field(self.cx.tcx, variant) {
12411249
// Transparent newtypes have at most one non-ZST field which needs to be checked..
1242-
match self.check_field_type_for_ffi(cache, field, args) {
1250+
match self.check_field_type_for_ffi(acc, field, args) {
12431251
FfiUnsafe { ty, .. } if ty.is_unit() => (),
12441252
r => return r,
12451253
}
@@ -1257,7 +1265,7 @@ impl<'a, 'tcx> ImproperCTypesVisitor<'a, 'tcx> {
12571265
// We can't completely trust `repr(C)` markings, so make sure the fields are actually safe.
12581266
let mut all_phantom = !variant.fields.is_empty();
12591267
for field in &variant.fields {
1260-
all_phantom &= match self.check_field_type_for_ffi(cache, field, args) {
1268+
all_phantom &= match self.check_field_type_for_ffi(acc, field, args) {
12611269
FfiSafe => false,
12621270
// `()` fields are FFI-safe!
12631271
FfiUnsafe { ty, .. } if ty.is_unit() => false,
@@ -1277,7 +1285,11 @@ impl<'a, 'tcx> ImproperCTypesVisitor<'a, 'tcx> {
12771285

12781286
/// Checks if the given type is "ffi-safe" (has a stable, well-defined
12791287
/// representation which can be exported to C code).
1280-
fn check_type_for_ffi(&self, cache: &mut FxHashSet<Ty<'tcx>>, ty: Ty<'tcx>) -> FfiResult<'tcx> {
1288+
fn check_type_for_ffi(
1289+
&self,
1290+
acc: &mut CTypesVisitorState<'tcx>,
1291+
ty: Ty<'tcx>,
1292+
) -> FfiResult<'tcx> {
12811293
use FfiResult::*;
12821294

12831295
let tcx = self.cx.tcx;
@@ -1286,7 +1298,7 @@ impl<'a, 'tcx> ImproperCTypesVisitor<'a, 'tcx> {
12861298
// `struct S(*mut S);`.
12871299
// FIXME: A recursion limit is necessary as well, for irregular
12881300
// recursive types.
1289-
if !cache.insert(ty) {
1301+
if !acc.cache.insert(ty) {
12901302
return FfiSafe;
12911303
}
12921304

@@ -1308,6 +1320,17 @@ impl<'a, 'tcx> ImproperCTypesVisitor<'a, 'tcx> {
13081320
}
13091321
match def.adt_kind() {
13101322
AdtKind::Struct | AdtKind::Union => {
1323+
if let Some(sym::cstring_type | sym::cstr_type) =
1324+
tcx.get_diagnostic_name(def.did())
1325+
&& !acc.base_ty.is_mutable_ptr()
1326+
{
1327+
return FfiUnsafe {
1328+
ty,
1329+
reason: fluent::lint_improper_ctypes_cstr_reason,
1330+
help: Some(fluent::lint_improper_ctypes_cstr_help),
1331+
};
1332+
}
1333+
13111334
if !def.repr().c() && !def.repr().transparent() {
13121335
return FfiUnsafe {
13131336
ty,
@@ -1354,7 +1377,7 @@ impl<'a, 'tcx> ImproperCTypesVisitor<'a, 'tcx> {
13541377
};
13551378
}
13561379

1357-
self.check_variant_for_ffi(cache, ty, def, def.non_enum_variant(), args)
1380+
self.check_variant_for_ffi(acc, ty, def, def.non_enum_variant(), args)
13581381
}
13591382
AdtKind::Enum => {
13601383
if def.variants().is_empty() {
@@ -1378,7 +1401,7 @@ impl<'a, 'tcx> ImproperCTypesVisitor<'a, 'tcx> {
13781401
if let Some(ty) =
13791402
repr_nullable_ptr(self.cx.tcx, self.cx.param_env, ty, self.mode)
13801403
{
1381-
return self.check_type_for_ffi(cache, ty);
1404+
return self.check_type_for_ffi(acc, ty);
13821405
}
13831406

13841407
return FfiUnsafe {
@@ -1399,7 +1422,7 @@ impl<'a, 'tcx> ImproperCTypesVisitor<'a, 'tcx> {
13991422
};
14001423
}
14011424

1402-
match self.check_variant_for_ffi(cache, ty, def, variant, args) {
1425+
match self.check_variant_for_ffi(acc, ty, def, variant, args) {
14031426
FfiSafe => (),
14041427
r => return r,
14051428
}
@@ -1469,9 +1492,9 @@ impl<'a, 'tcx> ImproperCTypesVisitor<'a, 'tcx> {
14691492
FfiSafe
14701493
}
14711494

1472-
ty::RawPtr(ty, _) | ty::Ref(_, ty, _) => self.check_type_for_ffi(cache, ty),
1495+
ty::RawPtr(ty, _) | ty::Ref(_, ty, _) => self.check_type_for_ffi(acc, ty),
14731496

1474-
ty::Array(inner_ty, _) => self.check_type_for_ffi(cache, inner_ty),
1497+
ty::Array(inner_ty, _) => self.check_type_for_ffi(acc, inner_ty),
14751498

14761499
ty::FnPtr(sig_tys, hdr) => {
14771500
let sig = sig_tys.with(hdr);
@@ -1485,7 +1508,7 @@ impl<'a, 'tcx> ImproperCTypesVisitor<'a, 'tcx> {
14851508

14861509
let sig = tcx.instantiate_bound_regions_with_erased(sig);
14871510
for arg in sig.inputs() {
1488-
match self.check_type_for_ffi(cache, *arg) {
1511+
match self.check_type_for_ffi(acc, *arg) {
14891512
FfiSafe => {}
14901513
r => return r,
14911514
}
@@ -1496,7 +1519,7 @@ impl<'a, 'tcx> ImproperCTypesVisitor<'a, 'tcx> {
14961519
return FfiSafe;
14971520
}
14981521

1499-
self.check_type_for_ffi(cache, ret_ty)
1522+
self.check_type_for_ffi(acc, ret_ty)
15001523
}
15011524

15021525
ty::Foreign(..) => FfiSafe,
@@ -1619,7 +1642,8 @@ impl<'a, 'tcx> ImproperCTypesVisitor<'a, 'tcx> {
16191642
return;
16201643
}
16211644

1622-
match self.check_type_for_ffi(&mut FxHashSet::default(), ty) {
1645+
let mut acc = CTypesVisitorState { cache: FxHashSet::default(), base_ty: ty };
1646+
match self.check_type_for_ffi(&mut acc, ty) {
16231647
FfiResult::FfiSafe => {}
16241648
FfiResult::FfiPhantom(ty) => {
16251649
self.emit_ffi_unsafe_type_lint(

compiler/rustc_middle/src/traits/mod.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -353,7 +353,7 @@ pub enum ObligationCauseCode<'tcx> {
353353
ReturnValue(HirId),
354354

355355
/// Opaque return type of this function
356-
OpaqueReturnType(Option<(Ty<'tcx>, Span)>),
356+
OpaqueReturnType(Option<(Ty<'tcx>, HirId)>),
357357

358358
/// Block implicit return
359359
BlockTailExpression(HirId, hir::MatchSource),

compiler/rustc_parse/src/parser/attr.rs

+26-12
Original file line numberDiff line numberDiff line change
@@ -76,6 +76,7 @@ impl<'a> Parser<'a> {
7676
token::CommentKind::Line => OuterAttributeType::DocComment,
7777
token::CommentKind::Block => OuterAttributeType::DocBlockComment,
7878
},
79+
true,
7980
) {
8081
err.note(fluent::parse_note);
8182
err.span_suggestion_verbose(
@@ -139,7 +140,11 @@ impl<'a> Parser<'a> {
139140

140141
// Emit error if inner attribute is encountered and forbidden.
141142
if style == ast::AttrStyle::Inner {
142-
this.error_on_forbidden_inner_attr(attr_sp, inner_parse_policy);
143+
this.error_on_forbidden_inner_attr(
144+
attr_sp,
145+
inner_parse_policy,
146+
item.is_valid_for_outer_style(),
147+
);
143148
}
144149

145150
Ok(attr::mk_attr_from_item(&self.psess.attr_id_generator, item, None, style, attr_sp))
@@ -151,6 +156,7 @@ impl<'a> Parser<'a> {
151156
err: &mut Diag<'_>,
152157
span: Span,
153158
attr_type: OuterAttributeType,
159+
suggest_to_outer: bool,
154160
) -> Option<Span> {
155161
let mut snapshot = self.create_snapshot_for_diagnostic();
156162
let lo = span.lo()
@@ -185,16 +191,18 @@ impl<'a> Parser<'a> {
185191
// FIXME(#100717)
186192
err.arg("item", item.kind.descr());
187193
err.span_label(item.span, fluent::parse_label_does_not_annotate_this);
188-
err.span_suggestion_verbose(
189-
replacement_span,
190-
fluent::parse_sugg_change_inner_to_outer,
191-
match attr_type {
192-
OuterAttributeType::Attribute => "",
193-
OuterAttributeType::DocBlockComment => "*",
194-
OuterAttributeType::DocComment => "/",
195-
},
196-
rustc_errors::Applicability::MachineApplicable,
197-
);
194+
if suggest_to_outer {
195+
err.span_suggestion_verbose(
196+
replacement_span,
197+
fluent::parse_sugg_change_inner_to_outer,
198+
match attr_type {
199+
OuterAttributeType::Attribute => "",
200+
OuterAttributeType::DocBlockComment => "*",
201+
OuterAttributeType::DocComment => "/",
202+
},
203+
rustc_errors::Applicability::MachineApplicable,
204+
);
205+
}
198206
return None;
199207
}
200208
Err(item_err) => {
@@ -205,7 +213,12 @@ impl<'a> Parser<'a> {
205213
Some(replacement_span)
206214
}
207215

208-
pub(super) fn error_on_forbidden_inner_attr(&self, attr_sp: Span, policy: InnerAttrPolicy) {
216+
pub(super) fn error_on_forbidden_inner_attr(
217+
&self,
218+
attr_sp: Span,
219+
policy: InnerAttrPolicy,
220+
suggest_to_outer: bool,
221+
) {
209222
if let InnerAttrPolicy::Forbidden(reason) = policy {
210223
let mut diag = match reason.as_ref().copied() {
211224
Some(InnerAttrForbiddenReason::AfterOuterDocComment { prev_doc_comment_span }) => {
@@ -239,6 +252,7 @@ impl<'a> Parser<'a> {
239252
&mut diag,
240253
attr_sp,
241254
OuterAttributeType::Attribute,
255+
suggest_to_outer,
242256
)
243257
.is_some()
244258
{

compiler/rustc_parse/src/parser/stmt.rs

+5
Original file line numberDiff line numberDiff line change
@@ -459,11 +459,16 @@ impl<'a> Parser<'a> {
459459
pub fn parse_block(&mut self) -> PResult<'a, P<Block>> {
460460
let (attrs, block) = self.parse_inner_attrs_and_block()?;
461461
if let [.., last] = &*attrs {
462+
let suggest_to_outer = match &last.kind {
463+
ast::AttrKind::Normal(attr) => attr.item.is_valid_for_outer_style(),
464+
_ => false,
465+
};
462466
self.error_on_forbidden_inner_attr(
463467
last.span,
464468
super::attr::InnerAttrPolicy::Forbidden(Some(
465469
InnerAttrForbiddenReason::InCodeBlock,
466470
)),
471+
suggest_to_outer,
467472
);
468473
}
469474
Ok(block)

0 commit comments

Comments
 (0)