Skip to content

Commit d0b99e3

Browse files
committed
Make #[repr(Rust)] and #[repr(C)] incompatible with one another
1 parent 1bc0463 commit d0b99e3

File tree

4 files changed

+80
-4
lines changed

4 files changed

+80
-4
lines changed

compiler/rustc_passes/src/check_attr.rs

+10-3
Original file line numberDiff line numberDiff line change
@@ -1767,6 +1767,7 @@ impl CheckAttrVisitor<'_> {
17671767
.collect();
17681768

17691769
let mut int_reprs = 0;
1770+
let mut is_explicit_rust = false;
17701771
let mut is_c = false;
17711772
let mut is_simd = false;
17721773
let mut is_transparent = false;
@@ -1778,7 +1779,9 @@ impl CheckAttrVisitor<'_> {
17781779
}
17791780

17801781
match hint.name_or_empty() {
1781-
sym::Rust => {}
1782+
sym::Rust => {
1783+
is_explicit_rust = true;
1784+
}
17821785
sym::C => {
17831786
is_c = true;
17841787
match target {
@@ -1888,12 +1891,16 @@ impl CheckAttrVisitor<'_> {
18881891

18891892
// Error on repr(transparent, <anything else>).
18901893
if is_transparent && hints.len() > 1 {
1891-
let hint_spans: Vec<_> = hint_spans.clone().collect();
1894+
let hint_spans = hint_spans.clone().collect();
18921895
self.tcx.sess.emit_err(errors::TransparentIncompatible {
18931896
hint_spans,
18941897
target: target.to_string(),
18951898
});
18961899
}
1900+
if is_explicit_rust && (int_reprs > 0 || is_c || is_simd) {
1901+
let hint_spans = hint_spans.clone().collect();
1902+
self.tcx.sess.emit_err(errors::ReprConflicting { hint_spans });
1903+
}
18971904
// Warn on repr(u8, u16), repr(C, simd), and c-like-enum-repr(C, u8)
18981905
if (int_reprs > 1)
18991906
|| (is_simd && is_c)
@@ -1910,7 +1917,7 @@ impl CheckAttrVisitor<'_> {
19101917
CONFLICTING_REPR_HINTS,
19111918
hir_id,
19121919
hint_spans.collect::<Vec<Span>>(),
1913-
errors::ReprConflicting,
1920+
errors::ReprConflictingLint,
19141921
);
19151922
}
19161923
}

compiler/rustc_passes/src/errors.rs

+8-1
Original file line numberDiff line numberDiff line change
@@ -558,9 +558,16 @@ pub struct ReprIdent {
558558
pub span: Span,
559559
}
560560

561+
#[derive(Diagnostic)]
562+
#[diag(passes_repr_conflicting, code = "E0566")]
563+
pub struct ReprConflicting {
564+
#[primary_span]
565+
pub hint_spans: Vec<Span>,
566+
}
567+
561568
#[derive(LintDiagnostic)]
562569
#[diag(passes_repr_conflicting, code = "E0566")]
563-
pub struct ReprConflicting;
570+
pub struct ReprConflictingLint;
564571

565572
#[derive(Diagnostic)]
566573
#[diag(passes_used_static)]
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,23 @@
1+
#[repr(C, Rust)] //~ ERROR conflicting representation hints
2+
struct S {
3+
a: i32,
4+
}
5+
6+
7+
#[repr(Rust)] //~ ERROR conflicting representation hints
8+
#[repr(C)]
9+
struct T {
10+
a: i32,
11+
}
12+
13+
#[repr(Rust, u64)] //~ ERROR conflicting representation hints
14+
enum U {
15+
V,
16+
}
17+
18+
#[repr(Rust, simd)]
19+
//~^ ERROR conflicting representation hints
20+
//~| ERROR SIMD types are experimental and possibly buggy
21+
struct F32x4(f32, f32, f32, f32);
22+
23+
fn main() {}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,39 @@
1+
error[E0658]: SIMD types are experimental and possibly buggy
2+
--> $DIR/explicit-rust-repr-conflicts.rs:18:1
3+
|
4+
LL | #[repr(Rust, simd)]
5+
| ^^^^^^^^^^^^^^^^^^^
6+
|
7+
= note: see issue #27731 <https://github.com/rust-lang/rust/issues/27731> for more information
8+
= help: add `#![feature(repr_simd)]` to the crate attributes to enable
9+
10+
error[E0566]: conflicting representation hints
11+
--> $DIR/explicit-rust-repr-conflicts.rs:1:8
12+
|
13+
LL | #[repr(C, Rust)]
14+
| ^ ^^^^
15+
16+
error[E0566]: conflicting representation hints
17+
--> $DIR/explicit-rust-repr-conflicts.rs:7:8
18+
|
19+
LL | #[repr(Rust)]
20+
| ^^^^
21+
LL | #[repr(C)]
22+
| ^
23+
24+
error[E0566]: conflicting representation hints
25+
--> $DIR/explicit-rust-repr-conflicts.rs:13:8
26+
|
27+
LL | #[repr(Rust, u64)]
28+
| ^^^^ ^^^
29+
30+
error[E0566]: conflicting representation hints
31+
--> $DIR/explicit-rust-repr-conflicts.rs:18:8
32+
|
33+
LL | #[repr(Rust, simd)]
34+
| ^^^^ ^^^^
35+
36+
error: aborting due to 5 previous errors
37+
38+
Some errors have detailed explanations: E0566, E0658.
39+
For more information about an error, try `rustc --explain E0566`.

0 commit comments

Comments
 (0)