Skip to content

Commit 7d36bfa

Browse files
authored
Rollup merge of #130301 - RalfJung:clashing_extern_declarations, r=compiler-errors
some fixes for clashing_extern_declarations lint There were two issues with the clashing_extern_declarations lint: - It would accept non-`repr(C)` structs as compatible with each other by comparing their fields in declaration order, but the fields could have different memory order (and with `-Zrandomize-layout`, this can really happen). - It would accept two types as compatible if `compare_layouts` returns `true`, but that function actually just compared the *ABI*, not the fully layout -- and all sized structs with more than 2 fields have the same ABI (`Abi::Aggregate`), so this missed a *lot* of cases. We don't currently have a clear spec for what we *want* to consider "clashing" and what is fine, so I otherwise kept the original logic. I hope to have a t-lang discussion about this at some point. But meanwhile, these changes seem like clear bugfixes.
2 parents b0e3235 + f362a59 commit 7d36bfa

File tree

4 files changed

+172
-89
lines changed

4 files changed

+172
-89
lines changed

compiler/rustc_lint/src/foreign_modules.rs

+35-39
Original file line numberDiff line numberDiff line change
@@ -3,8 +3,7 @@ use rustc_data_structures::unord::{UnordMap, UnordSet};
33
use rustc_hir as hir;
44
use rustc_hir::def::DefKind;
55
use rustc_middle::query::Providers;
6-
use rustc_middle::ty::layout::LayoutError;
7-
use rustc_middle::ty::{self, Instance, Ty, TyCtxt};
6+
use rustc_middle::ty::{self, AdtDef, Instance, Ty, TyCtxt};
87
use rustc_session::declare_lint;
98
use rustc_span::{sym, Span, Symbol};
109
use rustc_target::abi::FIRST_VARIANT;
@@ -212,7 +211,17 @@ fn structurally_same_type<'tcx>(
212211
ckind: types::CItemKind,
213212
) -> bool {
214213
let mut seen_types = UnordSet::default();
215-
structurally_same_type_impl(&mut seen_types, tcx, param_env, a, b, ckind)
214+
let result = structurally_same_type_impl(&mut seen_types, tcx, param_env, a, b, ckind);
215+
if cfg!(debug_assertions) && result {
216+
// Sanity-check: must have same ABI, size and alignment.
217+
// `extern` blocks cannot be generic, so we'll always get a layout here.
218+
let a_layout = tcx.layout_of(param_env.and(a)).unwrap();
219+
let b_layout = tcx.layout_of(param_env.and(b)).unwrap();
220+
assert_eq!(a_layout.abi, b_layout.abi);
221+
assert_eq!(a_layout.size, b_layout.size);
222+
assert_eq!(a_layout.align, b_layout.align);
223+
}
224+
result
216225
}
217226

218227
fn structurally_same_type_impl<'tcx>(
@@ -266,30 +275,21 @@ fn structurally_same_type_impl<'tcx>(
266275
// Do a full, depth-first comparison between the two.
267276
use rustc_type_ir::TyKind::*;
268277

269-
let compare_layouts = |a, b| -> Result<bool, &'tcx LayoutError<'tcx>> {
270-
debug!("compare_layouts({:?}, {:?})", a, b);
271-
let a_layout = &tcx.layout_of(param_env.and(a))?.layout.abi();
272-
let b_layout = &tcx.layout_of(param_env.and(b))?.layout.abi();
273-
debug!(
274-
"comparing layouts: {:?} == {:?} = {}",
275-
a_layout,
276-
b_layout,
277-
a_layout == b_layout
278-
);
279-
Ok(a_layout == b_layout)
280-
};
281-
282278
let is_primitive_or_pointer =
283279
|ty: Ty<'tcx>| ty.is_primitive() || matches!(ty.kind(), RawPtr(..) | Ref(..));
284280

285281
ensure_sufficient_stack(|| {
286282
match (a.kind(), b.kind()) {
287-
(Adt(a_def, _), Adt(b_def, _)) => {
288-
// We can immediately rule out these types as structurally same if
289-
// their layouts differ.
290-
match compare_layouts(a, b) {
291-
Ok(false) => return false,
292-
_ => (), // otherwise, continue onto the full, fields comparison
283+
(&Adt(a_def, _), &Adt(b_def, _)) => {
284+
// Only `repr(C)` types can be compared structurally.
285+
if !(a_def.repr().c() && b_def.repr().c()) {
286+
return false;
287+
}
288+
// If the types differ in their packed-ness, align, or simd-ness they conflict.
289+
let repr_characteristica =
290+
|def: AdtDef<'tcx>| (def.repr().pack, def.repr().align, def.repr().simd());
291+
if repr_characteristica(a_def) != repr_characteristica(b_def) {
292+
return false;
293293
}
294294

295295
// Grab a flattened representation of all fields.
@@ -311,9 +311,9 @@ fn structurally_same_type_impl<'tcx>(
311311
},
312312
)
313313
}
314-
(Array(a_ty, a_const), Array(b_ty, b_const)) => {
315-
// For arrays, we also check the constness of the type.
316-
a_const.kind() == b_const.kind()
314+
(Array(a_ty, a_len), Array(b_ty, b_len)) => {
315+
// For arrays, we also check the length.
316+
a_len == b_len
317317
&& structurally_same_type_impl(
318318
seen_types, tcx, param_env, *a_ty, *b_ty, ckind,
319319
)
@@ -357,10 +357,9 @@ fn structurally_same_type_impl<'tcx>(
357357
ckind,
358358
)
359359
}
360-
(Tuple(a_args), Tuple(b_args)) => {
361-
a_args.iter().eq_by(b_args.iter(), |a_ty, b_ty| {
362-
structurally_same_type_impl(seen_types, tcx, param_env, a_ty, b_ty, ckind)
363-
})
360+
(Tuple(..), Tuple(..)) => {
361+
// Tuples are not `repr(C)` so these cannot be compared structurally.
362+
false
364363
}
365364
// For these, it's not quite as easy to define structural-sameness quite so easily.
366365
// For the purposes of this lint, take the conservative approach and mark them as
@@ -380,24 +379,21 @@ fn structurally_same_type_impl<'tcx>(
380379
// An Adt and a primitive or pointer type. This can be FFI-safe if non-null
381380
// enum layout optimisation is being applied.
382381
(Adt(..), _) if is_primitive_or_pointer(b) => {
383-
if let Some(ty) = types::repr_nullable_ptr(tcx, param_env, a, ckind) {
384-
ty == b
382+
if let Some(a_inner) = types::repr_nullable_ptr(tcx, param_env, a, ckind) {
383+
a_inner == b
385384
} else {
386-
compare_layouts(a, b).unwrap_or(false)
385+
false
387386
}
388387
}
389388
(_, Adt(..)) if is_primitive_or_pointer(a) => {
390-
if let Some(ty) = types::repr_nullable_ptr(tcx, param_env, b, ckind) {
391-
ty == a
389+
if let Some(b_inner) = types::repr_nullable_ptr(tcx, param_env, b, ckind) {
390+
b_inner == a
392391
} else {
393-
compare_layouts(a, b).unwrap_or(false)
392+
false
394393
}
395394
}
396395

397-
// Otherwise, just compare the layouts. This may fail to lint for some
398-
// incompatible types, but at the very least, will stop reads into
399-
// uninitialised memory.
400-
_ => compare_layouts(a, b).unwrap_or(false),
396+
_ => false,
401397
}
402398
})
403399
}

tests/ui/lint/clashing-extern-fn-recursion.rs

+2
Original file line numberDiff line numberDiff line change
@@ -92,6 +92,7 @@ mod ref_recursion_once_removed {
9292
reffy: &'a Reffy2<'a>,
9393
}
9494

95+
#[repr(C)]
9596
struct Reffy2<'a> {
9697
reffy: &'a Reffy1<'a>,
9798
}
@@ -107,6 +108,7 @@ mod ref_recursion_once_removed {
107108
reffy: &'a Reffy2<'a>,
108109
}
109110

111+
#[repr(C)]
110112
struct Reffy2<'a> {
111113
reffy: &'a Reffy1<'a>,
112114
}

tests/ui/lint/clashing-extern-fn.rs

+76-23
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,6 @@
11
//@ check-pass
22
//@ aux-build:external_extern_fn.rs
33
#![crate_type = "lib"]
4-
#![warn(clashing_extern_declarations)]
54

65
mod redeclared_different_signature {
76
mod a {
@@ -132,7 +131,7 @@ mod banana {
132131
mod three {
133132
// This _should_ trigger the lint, because repr(packed) should generate a struct that has a
134133
// different layout.
135-
#[repr(packed)]
134+
#[repr(C, packed)]
136135
struct Banana {
137136
weight: u32,
138137
length: u16,
@@ -143,6 +142,79 @@ mod banana {
143142
//~^ WARN `weigh_banana` redeclared with a different signature
144143
}
145144
}
145+
146+
mod four {
147+
// This _should_ trigger the lint, because the type is not repr(C).
148+
struct Banana {
149+
weight: u32,
150+
length: u16,
151+
}
152+
#[allow(improper_ctypes)]
153+
extern "C" {
154+
fn weigh_banana(count: *const Banana) -> u64;
155+
//~^ WARN `weigh_banana` redeclared with a different signature
156+
}
157+
}
158+
}
159+
160+
// 3-field structs can't be distinguished by ScalarPair, side-stepping some shortucts
161+
// the logic used to (incorrectly) take.
162+
mod banana3 {
163+
mod one {
164+
#[repr(C)]
165+
struct Banana {
166+
weight: u32,
167+
length: u16,
168+
color: u8,
169+
}
170+
extern "C" {
171+
fn weigh_banana3(count: *const Banana) -> u64;
172+
}
173+
}
174+
175+
mod two {
176+
#[repr(C)]
177+
struct Banana {
178+
weight: u32,
179+
length: u16,
180+
color: u8,
181+
} // note: distinct type
182+
// This should not trigger the lint because two::Banana is structurally equivalent to
183+
// one::Banana.
184+
extern "C" {
185+
fn weigh_banana3(count: *const Banana) -> u64;
186+
}
187+
}
188+
189+
mod three {
190+
// This _should_ trigger the lint, because repr(packed) should generate a struct that has a
191+
// different layout.
192+
#[repr(C, packed)]
193+
struct Banana {
194+
weight: u32,
195+
length: u16,
196+
color: u8,
197+
}
198+
#[allow(improper_ctypes)]
199+
extern "C" {
200+
fn weigh_banana3(count: *const Banana) -> u64;
201+
//~^ WARN `weigh_banana3` redeclared with a different signature
202+
}
203+
}
204+
205+
mod four {
206+
// This _should_ trigger the lint, because the type is not repr(C).
207+
struct Banana {
208+
weight: u32,
209+
length: u16,
210+
color: u8,
211+
}
212+
#[allow(improper_ctypes)]
213+
extern "C" {
214+
fn weigh_banana3(count: *const Banana) -> u64;
215+
//~^ WARN `weigh_banana3` redeclared with a different signature
216+
}
217+
}
146218
}
147219

148220
mod sameish_members {
@@ -223,27 +295,6 @@ mod transparent {
223295
}
224296
}
225297

226-
#[allow(improper_ctypes)]
227-
mod zst {
228-
mod transparent {
229-
#[repr(transparent)]
230-
struct TransparentZst(());
231-
extern "C" {
232-
fn zst() -> ();
233-
fn transparent_zst() -> TransparentZst;
234-
}
235-
}
236-
237-
mod not_transparent {
238-
struct NotTransparentZst(());
239-
extern "C" {
240-
// These shouldn't warn since all return types are zero sized
241-
fn zst() -> NotTransparentZst;
242-
fn transparent_zst() -> NotTransparentZst;
243-
}
244-
}
245-
}
246-
247298
mod missing_return_type {
248299
mod a {
249300
extern "C" {
@@ -384,6 +435,7 @@ mod unknown_layout {
384435
extern "C" {
385436
pub fn generic(l: Link<u32>);
386437
}
438+
#[repr(C)]
387439
pub struct Link<T> {
388440
pub item: T,
389441
pub next: *const Link<T>,
@@ -394,6 +446,7 @@ mod unknown_layout {
394446
extern "C" {
395447
pub fn generic(l: Link<u32>);
396448
}
449+
#[repr(C)]
397450
pub struct Link<T> {
398451
pub item: T,
399452
pub next: *const Link<T>,

0 commit comments

Comments
 (0)