mirror of
https://github.com/rust-lang/rust.git
synced 2025-10-23 21:54:16 +00:00
When combined with 143720, this means `rvalue_creates_operand` can just return `true` for *every* `Rvalue`. (A future PR could consider removing it, though just letting it optimize out is fine for now.) It's nicer anyway, IMHO, because it avoids needing the layout checks to be consistent in the two places, and thus is an overall reduction in code. Plus it's a more helpful building block when used in other places this way.
504 lines
14 KiB
Rust
504 lines
14 KiB
Rust
//@ compile-flags: -Copt-level=3 -C no-prepopulate-passes
|
|
//@ only-64bit (so I don't need to worry about usize)
|
|
|
|
#![crate_type = "lib"]
|
|
#![feature(core_intrinsics)]
|
|
#![feature(custom_mir)]
|
|
#![allow(unreachable_code)]
|
|
|
|
// Some of these need custom MIR to not get removed by MIR optimizations.
|
|
use std::intrinsics::mir::*;
|
|
use std::intrinsics::{transmute, transmute_unchecked};
|
|
use std::mem::MaybeUninit;
|
|
use std::num::NonZero;
|
|
|
|
pub enum ZstNever {}
|
|
|
|
#[repr(align(2))]
|
|
pub struct BigNever(ZstNever, u16, ZstNever);
|
|
|
|
#[repr(align(8))]
|
|
pub struct Scalar64(i64);
|
|
|
|
#[repr(C, align(4))]
|
|
pub struct Aggregate64(u16, u8, i8, f32);
|
|
|
|
#[repr(C)]
|
|
pub struct Aggregate8(u8);
|
|
|
|
// CHECK-LABEL: @check_bigger_size(
|
|
#[no_mangle]
|
|
pub unsafe fn check_bigger_size(x: u16) -> u32 {
|
|
// CHECK: store i1 true, ptr poison, align 1
|
|
transmute_unchecked(x)
|
|
}
|
|
|
|
// CHECK-LABEL: @check_smaller_size(
|
|
#[no_mangle]
|
|
pub unsafe fn check_smaller_size(x: u32) -> u16 {
|
|
// CHECK: store i1 true, ptr poison, align 1
|
|
transmute_unchecked(x)
|
|
}
|
|
|
|
// CHECK-LABEL: @check_smaller_array(
|
|
#[no_mangle]
|
|
pub unsafe fn check_smaller_array(x: [u32; 7]) -> [u32; 3] {
|
|
// CHECK: store i1 true, ptr poison, align 1
|
|
transmute_unchecked(x)
|
|
}
|
|
|
|
// CHECK-LABEL: @check_bigger_array(
|
|
#[no_mangle]
|
|
pub unsafe fn check_bigger_array(x: [u32; 3]) -> [u32; 7] {
|
|
// CHECK: store i1 true, ptr poison, align 1
|
|
transmute_unchecked(x)
|
|
}
|
|
|
|
// CHECK-LABEL: @check_to_empty_array(
|
|
#[no_mangle]
|
|
#[custom_mir(dialect = "runtime", phase = "optimized")]
|
|
pub unsafe fn check_to_empty_array(x: [u32; 5]) -> [u32; 0] {
|
|
// CHECK: start
|
|
// CHECK-NEXT: store i1 true, ptr poison, align 1
|
|
// CHECK-NEXT: ret void
|
|
mir! {
|
|
{
|
|
RET = CastTransmute(x);
|
|
Return()
|
|
}
|
|
}
|
|
}
|
|
|
|
// CHECK-LABEL: @check_from_empty_array(
|
|
#[no_mangle]
|
|
#[custom_mir(dialect = "runtime", phase = "optimized")]
|
|
pub unsafe fn check_from_empty_array(x: [u32; 0]) -> [u32; 5] {
|
|
// CHECK: start
|
|
// CHECK-NEXT: store i1 true, ptr poison, align 1
|
|
// CHECK-NEXT: ret void
|
|
mir! {
|
|
{
|
|
RET = CastTransmute(x);
|
|
Return()
|
|
}
|
|
}
|
|
}
|
|
|
|
// CHECK-LABEL: @check_to_uninhabited(
|
|
#[no_mangle]
|
|
#[custom_mir(dialect = "runtime", phase = "optimized")]
|
|
pub unsafe fn check_to_uninhabited(x: u16) {
|
|
// CHECK: start
|
|
// CHECK-NEXT: store i1 true, ptr poison, align 1
|
|
// CHECK-NEXT: ret void
|
|
mir! {
|
|
let temp: BigNever;
|
|
{
|
|
temp = CastTransmute(x);
|
|
Return()
|
|
}
|
|
}
|
|
}
|
|
|
|
// CHECK-LABEL: @check_from_uninhabited(
|
|
#[no_mangle]
|
|
#[custom_mir(dialect = "runtime", phase = "optimized")]
|
|
pub unsafe fn check_from_uninhabited(x: BigNever) -> u16 {
|
|
// CHECK: start
|
|
// CHECK-NEXT: store i1 true, ptr poison, align 1
|
|
// CHECK-NEXT: ret i16 poison
|
|
mir! {
|
|
{
|
|
RET = CastTransmute(x);
|
|
Return()
|
|
}
|
|
}
|
|
}
|
|
|
|
// CHECK-LABEL: @check_intermediate_passthrough(
|
|
#[no_mangle]
|
|
pub unsafe fn check_intermediate_passthrough(x: u32) -> i32 {
|
|
// CHECK: start
|
|
// CHECK: %[[TMP:.+]] = add i32 1, %x
|
|
// CHECK: %[[RET:.+]] = add i32 %[[TMP]], 1
|
|
// CHECK: ret i32 %[[RET]]
|
|
unsafe { transmute::<u32, i32>(1 + x) + 1 }
|
|
}
|
|
|
|
// CHECK-LABEL: @check_nop_pair(
|
|
#[no_mangle]
|
|
pub unsafe fn check_nop_pair(x: (u8, i8)) -> (i8, u8) {
|
|
// CHECK-NOT: alloca
|
|
// CHECK: %0 = insertvalue { i8, i8 } poison, i8 %x.0, 0
|
|
// CHECK: %1 = insertvalue { i8, i8 } %0, i8 %x.1, 1
|
|
// CHECK: ret { i8, i8 } %1
|
|
unsafe { transmute(x) }
|
|
}
|
|
|
|
// CHECK-LABEL: @check_to_newtype(
|
|
#[no_mangle]
|
|
pub unsafe fn check_to_newtype(x: u64) -> Scalar64 {
|
|
// CHECK-NOT: alloca
|
|
// CHECK: ret i64 %x
|
|
transmute(x)
|
|
}
|
|
|
|
// CHECK-LABEL: @check_from_newtype(
|
|
#[no_mangle]
|
|
pub unsafe fn check_from_newtype(x: Scalar64) -> u64 {
|
|
// CHECK-NOT: alloca
|
|
// CHECK: ret i64 %x
|
|
transmute(x)
|
|
}
|
|
|
|
// CHECK-LABEL: @check_aggregate_to_bool(
|
|
#[no_mangle]
|
|
pub unsafe fn check_aggregate_to_bool(x: Aggregate8) -> bool {
|
|
// CHECK: %x = alloca [1 x i8], align 1
|
|
// CHECK: %[[BYTE:.+]] = load i8, ptr %x, align 1
|
|
// CHECK: %[[BOOL:.+]] = trunc nuw i8 %[[BYTE]] to i1
|
|
// CHECK: ret i1 %[[BOOL]]
|
|
transmute(x)
|
|
}
|
|
|
|
// CHECK-LABEL: @check_aggregate_from_bool(
|
|
#[no_mangle]
|
|
pub unsafe fn check_aggregate_from_bool(x: bool) -> Aggregate8 {
|
|
// CHECK: %_0 = alloca [1 x i8], align 1
|
|
// CHECK: %[[BYTE:.+]] = zext i1 %x to i8
|
|
// CHECK: store i8 %[[BYTE]], ptr %_0, align 1
|
|
transmute(x)
|
|
}
|
|
|
|
// CHECK-LABEL: @check_byte_to_bool(
|
|
#[no_mangle]
|
|
pub unsafe fn check_byte_to_bool(x: u8) -> bool {
|
|
// CHECK-NOT: alloca
|
|
// CHECK: %[[R:.+]] = trunc nuw i8 %x to i1
|
|
// CHECK: ret i1 %[[R]]
|
|
transmute(x)
|
|
}
|
|
|
|
// CHECK-LABEL: @check_byte_from_bool(
|
|
#[no_mangle]
|
|
pub unsafe fn check_byte_from_bool(x: bool) -> u8 {
|
|
// CHECK-NOT: alloca
|
|
// CHECK: %[[R:.+]] = zext i1 %x to i8
|
|
// CHECK: ret i8 %[[R:.+]]
|
|
transmute(x)
|
|
}
|
|
|
|
// CHECK-LABEL: @check_to_pair(
|
|
#[no_mangle]
|
|
pub unsafe fn check_to_pair(x: u64) -> Option<i32> {
|
|
// CHECK: %[[TEMP:.+]] = alloca [8 x i8], align 8
|
|
// CHECK: call void @llvm.lifetime.start.p0(i64 8, ptr %[[TEMP]])
|
|
// CHECK: store i64 %x, ptr %[[TEMP]], align 8
|
|
// CHECK: %[[PAIR0:.+]] = load i32, ptr %[[TEMP]], align 8
|
|
// CHECK: %[[PAIR1P:.+]] = getelementptr inbounds i8, ptr %[[TEMP]], i64 4
|
|
// CHECK: %[[PAIR1:.+]] = load i32, ptr %[[PAIR1P]], align 4
|
|
// CHECK: call void @llvm.lifetime.end.p0(i64 8, ptr %[[TEMP]])
|
|
// CHECK: insertvalue {{.+}}, i32 %[[PAIR0]], 0
|
|
// CHECK: insertvalue {{.+}}, i32 %[[PAIR1]], 1
|
|
transmute(x)
|
|
}
|
|
|
|
// CHECK-LABEL: @check_from_pair(
|
|
#[no_mangle]
|
|
pub unsafe fn check_from_pair(x: Option<i32>) -> u64 {
|
|
// CHECK: %[[TEMP:.+]] = alloca [8 x i8], align 8
|
|
// CHECK: call void @llvm.lifetime.start.p0(i64 8, ptr %[[TEMP]])
|
|
// CHECK: store i32 %x.0, ptr %[[TEMP]], align 8
|
|
// CHECK: %[[PAIR1P:.+]] = getelementptr inbounds i8, ptr %[[TEMP]], i64 4
|
|
// CHECK: store i32 %x.1, ptr %[[PAIR1P]], align 4
|
|
// CHECK: %[[R:.+]] = load i64, ptr %[[TEMP]], align 8
|
|
// CHECK: call void @llvm.lifetime.end.p0(i64 8, ptr %[[TEMP]])
|
|
// CHECK: ret i64 %[[R]]
|
|
transmute(x)
|
|
}
|
|
|
|
// CHECK-LABEL: @check_to_float(
|
|
#[no_mangle]
|
|
pub unsafe fn check_to_float(x: u32) -> f32 {
|
|
// CHECK-NOT: alloca
|
|
// CHECK: %_0 = bitcast i32 %x to float
|
|
// CHECK: ret float %_0
|
|
transmute(x)
|
|
}
|
|
|
|
// CHECK-LABEL: @check_from_float(
|
|
#[no_mangle]
|
|
pub unsafe fn check_from_float(x: f32) -> u32 {
|
|
// CHECK-NOT: alloca
|
|
// CHECK: %_0 = bitcast float %x to i32
|
|
// CHECK: ret i32 %_0
|
|
transmute(x)
|
|
}
|
|
|
|
// CHECK-LABEL: @check_to_bytes(
|
|
#[no_mangle]
|
|
pub unsafe fn check_to_bytes(x: u32) -> [u8; 4] {
|
|
// CHECK: %_0 = alloca [4 x i8], align 1
|
|
// CHECK: store i32 %x, ptr %_0, align 1
|
|
transmute(x)
|
|
}
|
|
|
|
// CHECK-LABEL: @check_from_bytes(
|
|
#[no_mangle]
|
|
pub unsafe fn check_from_bytes(x: [u8; 4]) -> u32 {
|
|
// CHECK: %x = alloca [4 x i8], align 1
|
|
// CHECK: %[[VAL:.+]] = load i32, ptr %x, align 1
|
|
// CHECK: ret i32 %[[VAL]]
|
|
transmute(x)
|
|
}
|
|
|
|
// CHECK-LABEL: @check_to_aggregate(
|
|
#[no_mangle]
|
|
pub unsafe fn check_to_aggregate(x: u64) -> Aggregate64 {
|
|
// CHECK: %_0 = alloca [8 x i8], align 4
|
|
// CHECK: store i64 %x, ptr %_0, align 4
|
|
// CHECK: %0 = load i64, ptr %_0, align 4
|
|
// CHECK: ret i64 %0
|
|
transmute(x)
|
|
}
|
|
|
|
// CHECK-LABEL: @check_from_aggregate(
|
|
#[no_mangle]
|
|
pub unsafe fn check_from_aggregate(x: Aggregate64) -> u64 {
|
|
// CHECK: %x = alloca [8 x i8], align 4
|
|
// CHECK: %[[VAL:.+]] = load i64, ptr %x, align 4
|
|
// CHECK: ret i64 %[[VAL]]
|
|
transmute(x)
|
|
}
|
|
|
|
// CHECK-LABEL: @check_long_array_less_aligned(
|
|
#[no_mangle]
|
|
pub unsafe fn check_long_array_less_aligned(x: [u64; 100]) -> [u16; 400] {
|
|
// CHECK-NEXT: start
|
|
// CHECK-NEXT: call void @llvm.memcpy.p0.p0.i64(ptr align 2 %_0, ptr align 8 %x, i64 800, i1 false)
|
|
// CHECK-NEXT: ret void
|
|
transmute(x)
|
|
}
|
|
|
|
// CHECK-LABEL: @check_long_array_more_aligned(
|
|
#[no_mangle]
|
|
pub unsafe fn check_long_array_more_aligned(x: [u8; 100]) -> [u32; 25] {
|
|
// CHECK-NEXT: start
|
|
// CHECK-NEXT: call void @llvm.memcpy.p0.p0.i64(ptr align 4 %_0, ptr align 1 %x, i64 100, i1 false)
|
|
// CHECK-NEXT: ret void
|
|
transmute(x)
|
|
}
|
|
|
|
// CHECK-LABEL: @check_pair_with_bool(
|
|
#[no_mangle]
|
|
pub unsafe fn check_pair_with_bool(x: (u8, bool)) -> (bool, i8) {
|
|
// CHECK-NOT: alloca
|
|
// CHECK: trunc nuw i8 %x.0 to i1
|
|
// CHECK: zext i1 %x.1 to i8
|
|
transmute(x)
|
|
}
|
|
|
|
// CHECK-LABEL: @check_float_to_pointer(
|
|
#[no_mangle]
|
|
pub unsafe fn check_float_to_pointer(x: f64) -> *const () {
|
|
// CHECK-NOT: alloca
|
|
// CHECK: %0 = bitcast double %x to i64
|
|
// CHECK: %_0 = getelementptr i8, ptr null, i64 %0
|
|
// CHECK: ret ptr %_0
|
|
transmute(x)
|
|
}
|
|
|
|
// CHECK-LABEL: @check_float_from_pointer(
|
|
#[no_mangle]
|
|
pub unsafe fn check_float_from_pointer(x: *const ()) -> f64 {
|
|
// CHECK-NOT: alloca
|
|
// CHECK: %0 = ptrtoint ptr %x to i64
|
|
// CHECK: %_0 = bitcast i64 %0 to double
|
|
// CHECK: ret double %_0
|
|
transmute(x)
|
|
}
|
|
|
|
// CHECK-LABEL: @check_array_to_pair(
|
|
#[no_mangle]
|
|
pub unsafe fn check_array_to_pair(x: [u8; 16]) -> (i64, u64) {
|
|
// CHECK-NOT: alloca
|
|
// CHECK: %[[FST:.+]] = load i64, ptr %{{.+}}, align 1, !noundef !
|
|
// CHECK: %[[SND:.+]] = load i64, ptr %{{.+}}, align 1, !noundef !
|
|
// CHECK: %[[PAIR0:.+]] = insertvalue { i64, i64 } poison, i64 %[[FST]], 0
|
|
// CHECK: %[[PAIR01:.+]] = insertvalue { i64, i64 } %[[PAIR0]], i64 %[[SND]], 1
|
|
// CHECK: ret { i64, i64 } %[[PAIR01]]
|
|
transmute(x)
|
|
}
|
|
|
|
// CHECK-LABEL: @check_pair_to_array(
|
|
#[no_mangle]
|
|
pub unsafe fn check_pair_to_array(x: (i64, u64)) -> [u8; 16] {
|
|
// CHECK-NOT: alloca
|
|
// CHECK: store i64 %x.0, ptr %{{.+}}, align 1
|
|
// CHECK: store i64 %x.1, ptr %{{.+}}, align 1
|
|
transmute(x)
|
|
}
|
|
|
|
// CHECK-LABEL: @check_heterogeneous_integer_pair(
|
|
#[no_mangle]
|
|
pub unsafe fn check_heterogeneous_integer_pair(x: (i32, bool)) -> (bool, u32) {
|
|
// CHECK: store i32 %x.0
|
|
// CHECK: %[[WIDER:.+]] = zext i1 %x.1 to i8
|
|
// CHECK: store i8 %[[WIDER]]
|
|
|
|
// CHECK: %[[BYTE:.+]] = load i8
|
|
// CHECK: trunc nuw i8 %[[BYTE:.+]] to i1
|
|
// CHECK: load i32
|
|
transmute(x)
|
|
}
|
|
|
|
// CHECK-LABEL: @check_heterogeneous_float_pair(
|
|
#[no_mangle]
|
|
pub unsafe fn check_heterogeneous_float_pair(x: (f64, f32)) -> (f32, f64) {
|
|
// CHECK: store double %x.0
|
|
// CHECK: store float %x.1
|
|
// CHECK: %[[A:.+]] = load float
|
|
// CHECK: %[[B:.+]] = load double
|
|
// CHECK: %[[P:.+]] = insertvalue { float, double } poison, float %[[A]], 0
|
|
// CHECK: insertvalue { float, double } %[[P]], double %[[B]], 1
|
|
transmute(x)
|
|
}
|
|
|
|
// CHECK-LABEL: @check_issue_110005(
|
|
#[no_mangle]
|
|
pub unsafe fn check_issue_110005(x: (usize, bool)) -> Option<Box<[u8]>> {
|
|
// CHECK: store i64 %x.0
|
|
// CHECK: %[[WIDER:.+]] = zext i1 %x.1 to i8
|
|
// CHECK: store i8 %[[WIDER]]
|
|
// CHECK: load ptr
|
|
// CHECK: load i64
|
|
transmute(x)
|
|
}
|
|
|
|
// CHECK-LABEL: @check_pair_to_dst_ref(
|
|
#[no_mangle]
|
|
pub unsafe fn check_pair_to_dst_ref<'a>(x: (usize, usize)) -> &'a [u8] {
|
|
// CHECK: %_0.0 = getelementptr i8, ptr null, i64 %x.0
|
|
// CHECK: %0 = icmp ne ptr %_0.0, null
|
|
// CHECK: call void @llvm.assume(i1 %0)
|
|
// CHECK: %1 = insertvalue { ptr, i64 } poison, ptr %_0.0, 0
|
|
// CHECK: %2 = insertvalue { ptr, i64 } %1, i64 %x.1, 1
|
|
// CHECK: ret { ptr, i64 } %2
|
|
transmute(x)
|
|
}
|
|
|
|
// CHECK-LABEL: @check_issue_109992(
|
|
#[no_mangle]
|
|
#[custom_mir(dialect = "runtime", phase = "optimized")]
|
|
pub unsafe fn check_issue_109992(x: ()) -> [(); 1] {
|
|
// This uses custom MIR to avoid MIR optimizations having removed ZST ops.
|
|
|
|
// CHECK: start
|
|
// CHECK-NEXT: ret void
|
|
mir! {
|
|
{
|
|
RET = CastTransmute(x);
|
|
Return()
|
|
}
|
|
}
|
|
}
|
|
|
|
// CHECK-LABEL: @check_unit_to_never(
|
|
#[no_mangle]
|
|
#[custom_mir(dialect = "runtime", phase = "optimized")]
|
|
pub unsafe fn check_unit_to_never(x: ()) {
|
|
// This uses custom MIR to avoid MIR optimizations having removed ZST ops.
|
|
|
|
// CHECK: start
|
|
// CHECK-NEXT: store i1 true, ptr poison, align 1
|
|
// CHECK-NEXT: ret void
|
|
mir! {
|
|
let temp: ZstNever;
|
|
{
|
|
temp = CastTransmute(x);
|
|
Return()
|
|
}
|
|
}
|
|
}
|
|
|
|
// CHECK-LABEL: @check_unit_from_never(
|
|
#[no_mangle]
|
|
#[custom_mir(dialect = "runtime", phase = "optimized")]
|
|
pub unsafe fn check_unit_from_never(x: ZstNever) -> () {
|
|
// This uses custom MIR to avoid MIR optimizations having removed ZST ops.
|
|
|
|
// CHECK: start
|
|
// CHECK-NEXT: store i1 true, ptr poison, align 1
|
|
// CHECK-NEXT: ret void
|
|
mir! {
|
|
{
|
|
RET = CastTransmute(x);
|
|
Return()
|
|
}
|
|
}
|
|
}
|
|
|
|
// CHECK-LABEL: @check_maybe_uninit_pair(i16 %x.0, i64 %x.1)
|
|
#[no_mangle]
|
|
pub unsafe fn check_maybe_uninit_pair(
|
|
x: (MaybeUninit<u16>, MaybeUninit<u64>),
|
|
) -> (MaybeUninit<i64>, MaybeUninit<i16>) {
|
|
// Thanks to `MaybeUninit` this is actually defined behaviour,
|
|
// unlike the examples above with pairs of primitives.
|
|
|
|
// CHECK: store i16 %x.0
|
|
// CHECK: store i64 %x.1
|
|
// CHECK: load i64
|
|
// CHECK-NOT: noundef
|
|
// CHECK: load i16
|
|
// CHECK-NOT: noundef
|
|
// CHECK: ret { i64, i16 }
|
|
transmute(x)
|
|
}
|
|
|
|
#[repr(align(8))]
|
|
pub struct HighAlignScalar(u8);
|
|
|
|
// CHECK-LABEL: @check_to_overalign(
|
|
#[no_mangle]
|
|
pub unsafe fn check_to_overalign(x: u64) -> HighAlignScalar {
|
|
// CHECK: %_0 = alloca [8 x i8], align 8
|
|
// CHECK: store i64 %x, ptr %_0, align 8
|
|
// CHECK: %0 = load i64, ptr %_0, align 8
|
|
// CHECK: ret i64 %0
|
|
transmute(x)
|
|
}
|
|
|
|
// CHECK-LABEL: @check_from_overalign(
|
|
#[no_mangle]
|
|
pub unsafe fn check_from_overalign(x: HighAlignScalar) -> u64 {
|
|
// CHECK: %x = alloca [8 x i8], align 8
|
|
// CHECK: %[[VAL:.+]] = load i64, ptr %x, align 8
|
|
// CHECK: ret i64 %[[VAL]]
|
|
transmute(x)
|
|
}
|
|
|
|
#[repr(transparent)]
|
|
struct Level1(std::num::NonZero<u32>);
|
|
#[repr(transparent)]
|
|
struct Level2(Level1);
|
|
#[repr(transparent)]
|
|
struct Level3(Level2);
|
|
|
|
// CHECK-LABEL: @repeatedly_transparent_transmute
|
|
// CHECK-SAME: (i32{{.+}}%[[ARG:[^)]+]])
|
|
#[no_mangle]
|
|
#[custom_mir(dialect = "runtime", phase = "optimized")]
|
|
pub unsafe fn repeatedly_transparent_transmute(x: NonZero<u32>) -> Level3 {
|
|
// CHECK: start
|
|
// CHECK-NEXT: ret i32 %[[ARG]]
|
|
mir! {
|
|
{
|
|
let A = CastTransmute::<NonZero<u32>, Level1>(x);
|
|
let B = CastTransmute::<Level1, Level2>(A);
|
|
RET = CastTransmute::<Level2, Level3>(B);
|
|
Return()
|
|
}
|
|
}
|
|
}
|