Rollup merge of #132057 - RalfJung:miri-abi-compat, r=wesleywiser
miri: update ABI compat checks to accept Option-like types This implements the t-lang decision described [here](https://github.com/rust-lang/rust/pull/130628#issuecomment-2402761599). Fixes https://github.com/rust-lang/miri/issues/3983
This commit is contained in:
commit
6b5a58c265
@ -4,9 +4,9 @@
|
|||||||
use std::borrow::Cow;
|
use std::borrow::Cow;
|
||||||
|
|
||||||
use either::{Left, Right};
|
use either::{Left, Right};
|
||||||
use rustc_abi::{self as abi, ExternAbi, FieldIdx, Integer};
|
use rustc_abi::{self as abi, ExternAbi, FieldIdx, Integer, VariantIdx};
|
||||||
use rustc_middle::ty::layout::{FnAbiOf, IntegerExt, LayoutOf, TyAndLayout};
|
use rustc_middle::ty::layout::{FnAbiOf, IntegerExt, LayoutOf, TyAndLayout};
|
||||||
use rustc_middle::ty::{self, AdtDef, Instance, Ty};
|
use rustc_middle::ty::{self, AdtDef, Instance, Ty, VariantDef};
|
||||||
use rustc_middle::{bug, mir, span_bug};
|
use rustc_middle::{bug, mir, span_bug};
|
||||||
use rustc_span::sym;
|
use rustc_span::sym;
|
||||||
use rustc_target::callconv::{ArgAbi, FnAbi, PassMode};
|
use rustc_target::callconv::{ArgAbi, FnAbi, PassMode};
|
||||||
@ -92,30 +92,47 @@ fn unfold_transparent(
|
|||||||
|
|
||||||
/// Unwrap types that are guaranteed a null-pointer-optimization
|
/// Unwrap types that are guaranteed a null-pointer-optimization
|
||||||
fn unfold_npo(&self, layout: TyAndLayout<'tcx>) -> InterpResult<'tcx, TyAndLayout<'tcx>> {
|
fn unfold_npo(&self, layout: TyAndLayout<'tcx>) -> InterpResult<'tcx, TyAndLayout<'tcx>> {
|
||||||
// Check if this is `Option` wrapping some type or if this is `Result` wrapping a 1-ZST and
|
// Check if this is an option-like type wrapping some type.
|
||||||
// another type.
|
|
||||||
let ty::Adt(def, args) = layout.ty.kind() else {
|
let ty::Adt(def, args) = layout.ty.kind() else {
|
||||||
// Not an ADT, so definitely no NPO.
|
// Not an ADT, so definitely no NPO.
|
||||||
return interp_ok(layout);
|
return interp_ok(layout);
|
||||||
};
|
};
|
||||||
let inner = if self.tcx.is_diagnostic_item(sym::Option, def.did()) {
|
if def.variants().len() != 2 {
|
||||||
// The wrapped type is the only arg.
|
// Not a 2-variant enum, so no NPO.
|
||||||
self.layout_of(args[0].as_type().unwrap())?
|
return interp_ok(layout);
|
||||||
} else if self.tcx.is_diagnostic_item(sym::Result, def.did()) {
|
}
|
||||||
// We want to extract which (if any) of the args is not a 1-ZST.
|
assert!(def.is_enum());
|
||||||
let lhs = self.layout_of(args[0].as_type().unwrap())?;
|
|
||||||
let rhs = self.layout_of(args[1].as_type().unwrap())?;
|
let all_fields_1zst = |variant: &VariantDef| -> InterpResult<'tcx, _> {
|
||||||
if lhs.is_1zst() {
|
for field in &variant.fields {
|
||||||
rhs
|
let ty = field.ty(*self.tcx, args);
|
||||||
} else if rhs.is_1zst() {
|
let layout = self.layout_of(ty)?;
|
||||||
lhs
|
if !layout.is_1zst() {
|
||||||
} else {
|
return interp_ok(false);
|
||||||
return interp_ok(layout); // no NPO
|
}
|
||||||
}
|
}
|
||||||
} else {
|
interp_ok(true)
|
||||||
return interp_ok(layout); // no NPO
|
|
||||||
};
|
};
|
||||||
|
|
||||||
|
// If one variant consists entirely of 1-ZST, then the other variant
|
||||||
|
// is the only "relevant" one for this check.
|
||||||
|
let var0 = VariantIdx::from_u32(0);
|
||||||
|
let var1 = VariantIdx::from_u32(1);
|
||||||
|
let relevant_variant = if all_fields_1zst(def.variant(var0))? {
|
||||||
|
def.variant(var1)
|
||||||
|
} else if all_fields_1zst(def.variant(var1))? {
|
||||||
|
def.variant(var0)
|
||||||
|
} else {
|
||||||
|
// No varant is all-1-ZST, so no NPO.
|
||||||
|
return interp_ok(layout);
|
||||||
|
};
|
||||||
|
// The "relevant" variant must have exactly one field, and its type is the "inner" type.
|
||||||
|
if relevant_variant.fields.len() != 1 {
|
||||||
|
return interp_ok(layout);
|
||||||
|
}
|
||||||
|
let inner = relevant_variant.fields[FieldIdx::from_u32(0)].ty(*self.tcx, args);
|
||||||
|
let inner = self.layout_of(inner)?;
|
||||||
|
|
||||||
// Check if the inner type is one of the NPO-guaranteed ones.
|
// Check if the inner type is one of the NPO-guaranteed ones.
|
||||||
// For that we first unpeel transparent *structs* (but not unions).
|
// For that we first unpeel transparent *structs* (but not unions).
|
||||||
let is_npo = |def: AdtDef<'tcx>| {
|
let is_npo = |def: AdtDef<'tcx>| {
|
||||||
|
@ -1,3 +1,5 @@
|
|||||||
|
#![feature(never_type)]
|
||||||
|
|
||||||
use std::rc::Rc;
|
use std::rc::Rc;
|
||||||
use std::{mem, num, ptr};
|
use std::{mem, num, ptr};
|
||||||
|
|
||||||
@ -12,6 +14,18 @@ fn id<T>(x: T) -> T {
|
|||||||
x
|
x
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[derive(Copy, Clone)]
|
||||||
|
enum Either<T, U> {
|
||||||
|
Left(T),
|
||||||
|
Right(U),
|
||||||
|
}
|
||||||
|
#[derive(Copy, Clone)]
|
||||||
|
enum Either2<T, U> {
|
||||||
|
Left(T),
|
||||||
|
#[allow(unused)]
|
||||||
|
Right(U, ()),
|
||||||
|
}
|
||||||
|
|
||||||
fn test_abi_compat<T: Clone, U: Clone>(t: T, u: U) {
|
fn test_abi_compat<T: Clone, U: Clone>(t: T, u: U) {
|
||||||
fn id<T>(x: T) -> T {
|
fn id<T>(x: T) -> T {
|
||||||
x
|
x
|
||||||
@ -81,6 +95,8 @@ fn main() {
|
|||||||
test_abi_compat(main as fn(), id::<i32> as fn(i32) -> i32);
|
test_abi_compat(main as fn(), id::<i32> as fn(i32) -> i32);
|
||||||
// - 1-ZST
|
// - 1-ZST
|
||||||
test_abi_compat((), [0u8; 0]);
|
test_abi_compat((), [0u8; 0]);
|
||||||
|
|
||||||
|
// Guaranteed null-pointer-layout optimizations:
|
||||||
// - Guaranteed Option<X> null-pointer-optimizations (RFC 3391).
|
// - Guaranteed Option<X> null-pointer-optimizations (RFC 3391).
|
||||||
test_abi_compat(&0u32 as *const u32, Some(&0u32));
|
test_abi_compat(&0u32 as *const u32, Some(&0u32));
|
||||||
test_abi_compat(main as fn(), Some(main as fn()));
|
test_abi_compat(main as fn(), Some(main as fn()));
|
||||||
@ -89,6 +105,7 @@ fn main() {
|
|||||||
test_abi_compat(0u32, Some(Wrapper(num::NonZeroU32::new(1u32).unwrap())));
|
test_abi_compat(0u32, Some(Wrapper(num::NonZeroU32::new(1u32).unwrap())));
|
||||||
// - Guaranteed Result<X, ZST1> does the same as Option<X> (RFC 3391)
|
// - Guaranteed Result<X, ZST1> does the same as Option<X> (RFC 3391)
|
||||||
test_abi_compat(&0u32 as *const u32, Result::<_, ()>::Ok(&0u32));
|
test_abi_compat(&0u32 as *const u32, Result::<_, ()>::Ok(&0u32));
|
||||||
|
test_abi_compat(&0u32 as *const u32, Result::<_, !>::Ok(&0u32));
|
||||||
test_abi_compat(main as fn(), Result::<_, ()>::Ok(main as fn()));
|
test_abi_compat(main as fn(), Result::<_, ()>::Ok(main as fn()));
|
||||||
test_abi_compat(0u32, Result::<_, ()>::Ok(num::NonZeroU32::new(1).unwrap()));
|
test_abi_compat(0u32, Result::<_, ()>::Ok(num::NonZeroU32::new(1).unwrap()));
|
||||||
test_abi_compat(&0u32 as *const u32, Result::<_, ()>::Ok(Wrapper(&0u32)));
|
test_abi_compat(&0u32 as *const u32, Result::<_, ()>::Ok(Wrapper(&0u32)));
|
||||||
@ -99,6 +116,13 @@ fn main() {
|
|||||||
test_abi_compat(0u32, Result::<(), _>::Err(num::NonZeroU32::new(1).unwrap()));
|
test_abi_compat(0u32, Result::<(), _>::Err(num::NonZeroU32::new(1).unwrap()));
|
||||||
test_abi_compat(&0u32 as *const u32, Result::<(), _>::Err(Wrapper(&0u32)));
|
test_abi_compat(&0u32 as *const u32, Result::<(), _>::Err(Wrapper(&0u32)));
|
||||||
test_abi_compat(0u32, Result::<(), _>::Err(Wrapper(num::NonZeroU32::new(1).unwrap())));
|
test_abi_compat(0u32, Result::<(), _>::Err(Wrapper(num::NonZeroU32::new(1).unwrap())));
|
||||||
|
// - Guaranteed null-pointer-optimizations for custom option-like types
|
||||||
|
test_abi_compat(&0u32 as *const u32, Either::<_, ()>::Left(&0u32));
|
||||||
|
test_abi_compat(&0u32 as *const u32, Either::<_, !>::Left(&0u32));
|
||||||
|
test_abi_compat(&0u32 as *const u32, Either::<(), _>::Right(&0u32));
|
||||||
|
test_abi_compat(&0u32 as *const u32, Either::<!, _>::Right(&0u32));
|
||||||
|
test_abi_compat(&0u32 as *const u32, Either2::<_, ()>::Left(&0u32));
|
||||||
|
test_abi_compat(&0u32 as *const u32, Either2::<_, [u8; 0]>::Left(&0u32));
|
||||||
|
|
||||||
// These must work for *any* type, since we guarantee that `repr(transparent)` is ABI-compatible
|
// These must work for *any* type, since we guarantee that `repr(transparent)` is ABI-compatible
|
||||||
// with the wrapped field.
|
// with the wrapped field.
|
||||||
|
Loading…
Reference in New Issue
Block a user