Skip to content

Commit b0dadff

Browse files
committed
error/E0691: include alignment in error message
Include the computed alignment of the violating field when rejecting transparent types with non-trivially aligned ZSTs. ZST member fields in transparent types must have an alignment of 1 (to ensure it does not raise the layout requirements of the transparent field). The current error message looks like this: LL | struct Foobar(u32, [u32; 0]); | ^^^^^^^^ has alignment larger than 1 This patch changes the report to include the alignment of the violating field: LL | struct Foobar(u32, [u32; 0]); | ^^^^^^^^ has alignment of 4, which is larger than 1 In case of unknown alignments, it will yield: LL | struct Foobar<T>(u32, [T; 0]); | ^^^^^^ may have alignment larger than 1 This allows developers to get a better grasp why a specific field is rejected. Knowing the alignment of the violating field makes it easier to judge where that alignment-requirement originates, and thus hopefully provide better hints on how to mitigate the problem. This idea was proposed in 2022 in #98071 as part of a bigger change. This commit simply extracts this error-message change, to decouple it from the other diagnostic improvements.
1 parent 1a44b45 commit b0dadff

File tree

3 files changed

+25
-15
lines changed

3 files changed

+25
-15
lines changed

compiler/rustc_error_codes/src/error_codes/E0691.md

+2-1
Original file line numberDiff line numberDiff line change
@@ -11,7 +11,8 @@ struct ForceAlign32;
1111
1212
#[repr(transparent)]
1313
struct Wrapper(f32, ForceAlign32); // error: zero-sized field in transparent
14-
// struct has alignment larger than 1
14+
// struct has alignment of 32, which
15+
// is larger than 1
1516
```
1617

1718
A transparent struct, enum, or union is supposed to be represented exactly like

compiler/rustc_hir_analysis/src/check/check.rs

+19-10
Original file line numberDiff line numberDiff line change
@@ -1078,9 +1078,9 @@ pub(super) fn check_transparent<'tcx>(tcx: TyCtxt<'tcx>, adt: ty::AdtDef<'tcx>)
10781078
// We are currently checking the type this field came from, so it must be local
10791079
let span = tcx.hir().span_if_local(field.did).unwrap();
10801080
let zst = layout.is_ok_and(|layout| layout.is_zst());
1081-
let align1 = layout.is_ok_and(|layout| layout.align.abi.bytes() == 1);
1081+
let align = layout.ok().map(|layout| layout.align.abi.bytes());
10821082
if !zst {
1083-
return (span, zst, align1, None);
1083+
return (span, zst, align, None);
10841084
}
10851085

10861086
fn check_non_exhaustive<'tcx>(
@@ -1115,30 +1115,39 @@ pub(super) fn check_transparent<'tcx>(tcx: TyCtxt<'tcx>, adt: ty::AdtDef<'tcx>)
11151115
}
11161116
}
11171117

1118-
(span, zst, align1, check_non_exhaustive(tcx, ty).break_value())
1118+
(span, zst, align, check_non_exhaustive(tcx, ty).break_value())
11191119
});
11201120

11211121
let non_zst_fields = field_infos
11221122
.clone()
1123-
.filter_map(|(span, zst, _align1, _non_exhaustive)| if !zst { Some(span) } else { None });
1123+
.filter_map(|(span, zst, _align, _non_exhaustive)| if !zst { Some(span) } else { None });
11241124
let non_zst_count = non_zst_fields.clone().count();
11251125
if non_zst_count >= 2 {
11261126
bad_non_zero_sized_fields(tcx, adt, non_zst_count, non_zst_fields, tcx.def_span(adt.did()));
11271127
}
11281128
let incompatible_zst_fields =
11291129
field_infos.clone().filter(|(_, _, _, opt)| opt.is_some()).count();
11301130
let incompat = incompatible_zst_fields + non_zst_count >= 2 && non_zst_count < 2;
1131-
for (span, zst, align1, non_exhaustive) in field_infos {
1132-
if zst && !align1 {
1133-
struct_span_err!(
1131+
for (span, zst, align, non_exhaustive) in field_infos {
1132+
if zst && align != Some(1) {
1133+
let mut err = struct_span_err!(
11341134
tcx.sess,
11351135
span,
11361136
E0691,
11371137
"zero-sized field in transparent {} has alignment larger than 1",
11381138
adt.descr(),
1139-
)
1140-
.span_label(span, "has alignment larger than 1")
1141-
.emit();
1139+
);
1140+
1141+
if let Some(align_bytes) = align {
1142+
err.span_label(
1143+
span,
1144+
format!("has alignment of {align_bytes}, which is larger than 1"),
1145+
);
1146+
} else {
1147+
err.span_label(span, "may have alignment larger than 1");
1148+
}
1149+
1150+
err.emit();
11421151
}
11431152
if incompat && let Some((descr, def_id, args, non_exhaustive)) = non_exhaustive {
11441153
tcx.struct_span_lint_hir(

tests/ui/repr/repr-transparent.stderr

+4-4
Original file line numberDiff line numberDiff line change
@@ -20,13 +20,13 @@ error[E0691]: zero-sized field in transparent struct has alignment larger than 1
2020
--> $DIR/repr-transparent.rs:36:32
2121
|
2222
LL | struct NontrivialAlignZst(u32, [u16; 0]);
23-
| ^^^^^^^^ has alignment larger than 1
23+
| ^^^^^^^^ has alignment of 2, which is larger than 1
2424

2525
error[E0691]: zero-sized field in transparent struct has alignment larger than 1
2626
--> $DIR/repr-transparent.rs:42:24
2727
|
2828
LL | struct GenericAlign<T>(ZstAlign32<T>, u32);
29-
| ^^^^^^^^^^^^^ has alignment larger than 1
29+
| ^^^^^^^^^^^^^ has alignment of 32, which is larger than 1
3030

3131
error[E0084]: unsupported representation for zero-variant enum
3232
--> $DIR/repr-transparent.rs:44:1
@@ -66,13 +66,13 @@ error[E0691]: zero-sized field in transparent enum has alignment larger than 1
6666
--> $DIR/repr-transparent.rs:71:14
6767
|
6868
LL | Foo(u32, [u16; 0]),
69-
| ^^^^^^^^ has alignment larger than 1
69+
| ^^^^^^^^ has alignment of 2, which is larger than 1
7070

7171
error[E0691]: zero-sized field in transparent enum has alignment larger than 1
7272
--> $DIR/repr-transparent.rs:76:11
7373
|
7474
LL | Foo { bar: ZstAlign32<T>, baz: u32 }
75-
| ^^^^^^^^^^^^^^^^^^ has alignment larger than 1
75+
| ^^^^^^^^^^^^^^^^^^ has alignment of 32, which is larger than 1
7676

7777
error[E0690]: transparent union needs at most one non-zero-sized field, but has 2
7878
--> $DIR/repr-transparent.rs:85:1

0 commit comments

Comments
 (0)