Rollup merge of #131843 - workingjubilee:thaw-impossible-reprs, r=lukas-code
compiler: Error on layout of enums with invalid reprs Surprising no one, the ICEs with the same message have the same root cause. Invalid reprs can reach layout computation for various reasons. For instance, the compiler may want to use its layout computations to discern if a combination of layout-affecting attributes results in a valid type to begin with by e.g. computing its size. When the input is bad, return an error reflecting that the answer to the question is not a useful one.
This commit is contained in:
commit
6d43de643e
8 changed files with 125 additions and 8 deletions
|
@ -54,6 +54,9 @@ pub enum LayoutCalculatorError<F> {
|
|||
|
||||
/// A union had no fields.
|
||||
EmptyUnion,
|
||||
|
||||
/// The fields or variants have irreconcilable reprs
|
||||
ReprConflict,
|
||||
}
|
||||
|
||||
impl<F> LayoutCalculatorError<F> {
|
||||
|
@ -64,6 +67,7 @@ impl<F> LayoutCalculatorError<F> {
|
|||
}
|
||||
LayoutCalculatorError::SizeOverflow => LayoutCalculatorError::SizeOverflow,
|
||||
LayoutCalculatorError::EmptyUnion => LayoutCalculatorError::EmptyUnion,
|
||||
LayoutCalculatorError::ReprConflict => LayoutCalculatorError::ReprConflict,
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -77,6 +81,7 @@ impl<F> LayoutCalculatorError<F> {
|
|||
}
|
||||
LayoutCalculatorError::SizeOverflow => "size overflow",
|
||||
LayoutCalculatorError::EmptyUnion => "type is a union with no fields",
|
||||
LayoutCalculatorError::ReprConflict => "type has an invalid repr",
|
||||
})
|
||||
}
|
||||
}
|
||||
|
@ -514,6 +519,10 @@ impl<Cx: HasDataLayout> LayoutCalculator<Cx> {
|
|||
}
|
||||
|
||||
let dl = self.cx.data_layout();
|
||||
// bail if the enum has an incoherent repr that cannot be computed
|
||||
if repr.packed() {
|
||||
return Err(LayoutCalculatorError::ReprConflict);
|
||||
}
|
||||
|
||||
let calculate_niche_filling_layout = || -> Option<TmpLayout<FieldIdx, VariantIdx>> {
|
||||
if dont_niche_optimize_enum {
|
||||
|
|
|
@ -30,7 +30,8 @@ use {rustc_abi as abi, rustc_hir as hir};
|
|||
use crate::errors::{
|
||||
MultipleArrayFieldsSimdType, NonPrimitiveSimdType, OversizedSimdType, ZeroLengthSimdType,
|
||||
};
|
||||
use crate::layout_sanity_check::sanity_check_layout;
|
||||
|
||||
mod invariant;
|
||||
|
||||
pub(crate) fn provide(providers: &mut Providers) {
|
||||
*providers = Providers { layout_of, ..*providers };
|
||||
|
@ -79,7 +80,7 @@ fn layout_of<'tcx>(
|
|||
record_layout_for_printing(&cx, layout);
|
||||
}
|
||||
|
||||
sanity_check_layout(&cx, &layout);
|
||||
invariant::partially_check_layout(&cx, &layout);
|
||||
|
||||
Ok(layout)
|
||||
}
|
||||
|
@ -115,6 +116,11 @@ fn map_error<'tcx>(
|
|||
cx.tcx().dcx().delayed_bug(format!("computed layout of empty union: {ty:?}"));
|
||||
LayoutError::Unknown(ty)
|
||||
}
|
||||
LayoutCalculatorError::ReprConflict => {
|
||||
// packed enums are the only known trigger of this, but others might arise
|
||||
cx.tcx().dcx().delayed_bug(format!("computed impossible repr (packed enum?): {ty:?}"));
|
||||
LayoutError::Unknown(ty)
|
||||
}
|
||||
};
|
||||
error(cx, err)
|
||||
}
|
||||
|
|
|
@ -5,7 +5,7 @@ use rustc_middle::ty::layout::{HasTyCtxt, LayoutCx, TyAndLayout};
|
|||
use rustc_target::abi::*;
|
||||
|
||||
/// Enforce some basic invariants on layouts.
|
||||
pub(super) fn sanity_check_layout<'tcx>(cx: &LayoutCx<'tcx>, layout: &TyAndLayout<'tcx>) {
|
||||
pub(super) fn partially_check_layout<'tcx>(cx: &LayoutCx<'tcx>, layout: &TyAndLayout<'tcx>) {
|
||||
let tcx = cx.tcx();
|
||||
|
||||
// Type-level uninhabitedness should always imply ABI uninhabitedness.
|
|
@ -29,7 +29,6 @@ mod errors;
|
|||
mod implied_bounds;
|
||||
mod instance;
|
||||
mod layout;
|
||||
mod layout_sanity_check;
|
||||
mod needs_drop;
|
||||
mod opaque_types;
|
||||
mod representability;
|
||||
|
|
|
@ -1,10 +1,14 @@
|
|||
//@ known-bug: rust-lang/rust#126966
|
||||
#![crate_type = "lib"]
|
||||
|
||||
mod assert {
|
||||
use std::mem::{Assume, TransmuteFrom};
|
||||
//~^ ERROR: use of unstable library feature 'transmutability'
|
||||
//~| ERROR: use of unstable library feature 'transmutability'
|
||||
|
||||
pub fn is_transmutable<Src, Dst>()
|
||||
where
|
||||
Dst: TransmuteFrom<Src>,
|
||||
//~^ ERROR: use of unstable library feature 'transmutability'
|
||||
{
|
||||
}
|
||||
}
|
||||
|
@ -15,6 +19,7 @@ enum Ox00 {
|
|||
}
|
||||
|
||||
#[repr(C, packed(2))]
|
||||
//~^ ERROR: attribute should be applied to a struct
|
||||
enum OxFF {
|
||||
V = 0xFF,
|
||||
}
|
||||
|
@ -22,8 +27,10 @@ enum OxFF {
|
|||
fn test() {
|
||||
union Superset {
|
||||
a: Ox00,
|
||||
//~^ ERROR: field must implement `Copy`
|
||||
b: OxFF,
|
||||
}
|
||||
|
||||
assert::is_transmutable::<Superset, Subset>();
|
||||
//~^ ERROR: cannot find type `Subset`
|
||||
}
|
68
tests/ui/layout/thaw-transmute-invalid-enum.stderr
Normal file
68
tests/ui/layout/thaw-transmute-invalid-enum.stderr
Normal file
|
@ -0,0 +1,68 @@
|
|||
error[E0412]: cannot find type `Subset` in this scope
|
||||
--> $DIR/thaw-transmute-invalid-enum.rs:34:41
|
||||
|
|
||||
LL | assert::is_transmutable::<Superset, Subset>();
|
||||
| ^^^^^^ not found in this scope
|
||||
|
|
||||
help: you might be missing a type parameter
|
||||
|
|
||||
LL | fn test<Subset>() {
|
||||
| ++++++++
|
||||
|
||||
error[E0517]: attribute should be applied to a struct or union
|
||||
--> $DIR/thaw-transmute-invalid-enum.rs:21:11
|
||||
|
|
||||
LL | #[repr(C, packed(2))]
|
||||
| ^^^^^^^^^
|
||||
LL |
|
||||
LL | / enum OxFF {
|
||||
LL | | V = 0xFF,
|
||||
LL | | }
|
||||
| |_- not a struct or union
|
||||
|
||||
error[E0658]: use of unstable library feature 'transmutability'
|
||||
--> $DIR/thaw-transmute-invalid-enum.rs:4:20
|
||||
|
|
||||
LL | use std::mem::{Assume, TransmuteFrom};
|
||||
| ^^^^^^
|
||||
|
|
||||
= note: see issue #99571 <https://github.com/rust-lang/rust/issues/99571> for more information
|
||||
= help: add `#![feature(transmutability)]` to the crate attributes to enable
|
||||
= note: this compiler was built on YYYY-MM-DD; consider upgrading it if it is out of date
|
||||
|
||||
error[E0658]: use of unstable library feature 'transmutability'
|
||||
--> $DIR/thaw-transmute-invalid-enum.rs:4:28
|
||||
|
|
||||
LL | use std::mem::{Assume, TransmuteFrom};
|
||||
| ^^^^^^^^^^^^^
|
||||
|
|
||||
= note: see issue #99571 <https://github.com/rust-lang/rust/issues/99571> for more information
|
||||
= help: add `#![feature(transmutability)]` to the crate attributes to enable
|
||||
= note: this compiler was built on YYYY-MM-DD; consider upgrading it if it is out of date
|
||||
|
||||
error[E0658]: use of unstable library feature 'transmutability'
|
||||
--> $DIR/thaw-transmute-invalid-enum.rs:10:14
|
||||
|
|
||||
LL | Dst: TransmuteFrom<Src>,
|
||||
| ^^^^^^^^^^^^^^^^^^
|
||||
|
|
||||
= note: see issue #99571 <https://github.com/rust-lang/rust/issues/99571> for more information
|
||||
= help: add `#![feature(transmutability)]` to the crate attributes to enable
|
||||
= note: this compiler was built on YYYY-MM-DD; consider upgrading it if it is out of date
|
||||
|
||||
error[E0740]: field must implement `Copy` or be wrapped in `ManuallyDrop<...>` to be used in a union
|
||||
--> $DIR/thaw-transmute-invalid-enum.rs:29:9
|
||||
|
|
||||
LL | a: Ox00,
|
||||
| ^^^^^^^
|
||||
|
|
||||
= note: union fields must not have drop side-effects, which is currently enforced via either `Copy` or `ManuallyDrop<...>`
|
||||
help: wrap the field type in `ManuallyDrop<...>`
|
||||
|
|
||||
LL | a: std::mem::ManuallyDrop<Ox00>,
|
||||
| +++++++++++++++++++++++ +
|
||||
|
||||
error: aborting due to 6 previous errors
|
||||
|
||||
Some errors have detailed explanations: E0412, E0517, E0658, E0740.
|
||||
For more information about an error, try `rustc --explain E0412`.
|
|
@ -1,7 +1,6 @@
|
|||
//@ known-bug: rust-lang/rust#128870
|
||||
//@ compile-flags: -Zvalidate-mir
|
||||
|
||||
#[repr(packed)]
|
||||
#[repr(packed)] //~ ERROR: attribute should be applied to a struct
|
||||
#[repr(u32)]
|
||||
enum E {
|
||||
A,
|
||||
|
@ -12,7 +11,7 @@ enum E {
|
|||
fn main() {
|
||||
union InvalidTag {
|
||||
int: u32,
|
||||
e: E,
|
||||
e: E, //~ ERROR: field must implement `Copy`
|
||||
}
|
||||
let _invalid_tag = InvalidTag { int: 4 };
|
||||
}
|
29
tests/ui/layout/thaw-validate-invalid-enum.stderr
Normal file
29
tests/ui/layout/thaw-validate-invalid-enum.stderr
Normal file
|
@ -0,0 +1,29 @@
|
|||
error[E0517]: attribute should be applied to a struct or union
|
||||
--> $DIR/thaw-validate-invalid-enum.rs:3:8
|
||||
|
|
||||
LL | #[repr(packed)]
|
||||
| ^^^^^^
|
||||
LL | #[repr(u32)]
|
||||
LL | / enum E {
|
||||
LL | | A,
|
||||
LL | | B,
|
||||
LL | | C,
|
||||
LL | | }
|
||||
| |_- not a struct or union
|
||||
|
||||
error[E0740]: field must implement `Copy` or be wrapped in `ManuallyDrop<...>` to be used in a union
|
||||
--> $DIR/thaw-validate-invalid-enum.rs:14:9
|
||||
|
|
||||
LL | e: E,
|
||||
| ^^^^
|
||||
|
|
||||
= note: union fields must not have drop side-effects, which is currently enforced via either `Copy` or `ManuallyDrop<...>`
|
||||
help: wrap the field type in `ManuallyDrop<...>`
|
||||
|
|
||||
LL | e: std::mem::ManuallyDrop<E>,
|
||||
| +++++++++++++++++++++++ +
|
||||
|
||||
error: aborting due to 2 previous errors
|
||||
|
||||
Some errors have detailed explanations: E0517, E0740.
|
||||
For more information about an error, try `rustc --explain E0517`.
|
Loading…
Add table
Reference in a new issue