Skip to main content

rustc_const_eval/interpret/
cast.rs

1use rustc_abi::{FieldIdx, Integer};
2use rustc_apfloat::ieee::{Double, Half, Quad, Single};
3use rustc_apfloat::{Float, FloatConvert};
4use rustc_data_structures::assert_matches;
5use rustc_errors::msg;
6use rustc_middle::mir::CastKind;
7use rustc_middle::mir::interpret::{InterpResult, PointerArithmetic, Scalar};
8use rustc_middle::ty::adjustment::PointerCoercion;
9use rustc_middle::ty::layout::{IntegerExt, TyAndLayout};
10use rustc_middle::ty::{self, FloatTy, Ty};
11use rustc_middle::{bug, span_bug};
12use tracing::trace;
13
14use super::util::ensure_monomorphic_enough;
15use super::{
16    FnVal, ImmTy, Immediate, InterpCx, Machine, OpTy, PlaceTy, err_inval, interp_ok, throw_ub,
17    throw_ub_custom,
18};
19use crate::enter_trace_span;
20use crate::interpret::Writeable;
21
22impl<'tcx, M: Machine<'tcx>> InterpCx<'tcx, M> {
23    pub fn cast(
24        &mut self,
25        src: &OpTy<'tcx, M::Provenance>,
26        cast_kind: CastKind,
27        cast_ty: Ty<'tcx>,
28        dest: &PlaceTy<'tcx, M::Provenance>,
29    ) -> InterpResult<'tcx> {
30        // `cast_ty` will often be the same as `dest.ty`, but not always, since subtyping is still
31        // possible.
32        let cast_layout =
33            if cast_ty == dest.layout.ty { dest.layout } else { self.layout_of(cast_ty)? };
34        // FIXME: In which cases should we trigger UB when the source is uninit?
35        match cast_kind {
36            CastKind::PointerCoercion(PointerCoercion::Unsize, _) => {
37                self.unsize_into(src, cast_layout, dest)?;
38            }
39
40            CastKind::PointerExposeProvenance => {
41                let src = self.read_immediate(src)?;
42                let res = self.pointer_expose_provenance_cast(&src, cast_layout)?;
43                self.write_immediate(*res, dest)?;
44            }
45
46            CastKind::PointerWithExposedProvenance => {
47                let src = self.read_immediate(src)?;
48                let res = self.pointer_with_exposed_provenance_cast(&src, cast_layout)?;
49                self.write_immediate(*res, dest)?;
50            }
51
52            CastKind::IntToInt | CastKind::IntToFloat => {
53                let src = self.read_immediate(src)?;
54                let res = self.int_to_int_or_float(&src, cast_layout)?;
55                self.write_immediate(*res, dest)?;
56            }
57
58            CastKind::FloatToFloat | CastKind::FloatToInt => {
59                let src = self.read_immediate(src)?;
60                let res = self.float_to_float_or_int(&src, cast_layout)?;
61                self.write_immediate(*res, dest)?;
62            }
63
64            CastKind::FnPtrToPtr | CastKind::PtrToPtr => {
65                let src = self.read_immediate(src)?;
66                let res = self.ptr_to_ptr(&src, cast_layout)?;
67                self.write_immediate(*res, dest)?;
68            }
69
70            CastKind::PointerCoercion(
71                PointerCoercion::MutToConstPointer | PointerCoercion::ArrayToPointer,
72                _,
73            ) => {
74                ::rustc_middle::util::bug::bug_fmt(format_args!("{0:?} casts are for borrowck only, not runtime MIR",
        cast_kind));bug!("{cast_kind:?} casts are for borrowck only, not runtime MIR");
75            }
76
77            CastKind::PointerCoercion(PointerCoercion::ReifyFnPointer(_), _) => {
78                // All reifications must be monomorphic, bail out otherwise.
79                ensure_monomorphic_enough(*self.tcx, src.layout.ty)?;
80
81                // The src operand does not matter, just its type
82                match *src.layout.ty.kind() {
83                    ty::FnDef(def_id, args) => {
84                        let instance = {
85                            let _trace = <M as
        crate::interpret::Machine>::enter_trace_span(||
        {
            use ::tracing::__macro_support::Callsite as _;
            static __CALLSITE: ::tracing::callsite::DefaultCallsite =
                {
                    static META: ::tracing::Metadata<'static> =
                        {
                            ::tracing_core::metadata::Metadata::new("resolve",
                                "rustc_const_eval::interpret::cast", ::tracing::Level::INFO,
                                ::tracing_core::__macro_support::Option::Some("compiler/rustc_const_eval/src/interpret/cast.rs"),
                                ::tracing_core::__macro_support::Option::Some(85u32),
                                ::tracing_core::__macro_support::Option::Some("rustc_const_eval::interpret::cast"),
                                ::tracing_core::field::FieldSet::new(&["resolve", "def_id"],
                                    ::tracing_core::callsite::Identifier(&__CALLSITE)),
                                ::tracing::metadata::Kind::SPAN)
                        };
                    ::tracing::callsite::DefaultCallsite::new(&META)
                };
            let mut interest = ::tracing::subscriber::Interest::never();
            if ::tracing::Level::INFO <=
                                ::tracing::level_filters::STATIC_MAX_LEVEL &&
                            ::tracing::Level::INFO <=
                                ::tracing::level_filters::LevelFilter::current() &&
                        { interest = __CALLSITE.interest(); !interest.is_never() }
                    &&
                    ::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
                        interest) {
                let meta = __CALLSITE.metadata();
                ::tracing::Span::new(meta,
                    &{
                            #[allow(unused_imports)]
                            use ::tracing::field::{debug, display, Value};
                            let mut iter = meta.fields().iter();
                            meta.fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
                                                ::tracing::__macro_support::Option::Some(&display(&"resolve_for_fn_ptr")
                                                        as &dyn Value)),
                                            (&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
                                                ::tracing::__macro_support::Option::Some(&debug(&def_id) as
                                                        &dyn Value))])
                        })
            } else {
                let span =
                    ::tracing::__macro_support::__disabled_span(__CALLSITE.metadata());
                {};
                span
            }
        })enter_trace_span!(M, resolve::resolve_for_fn_ptr, ?def_id);
86                            ty::Instance::resolve_for_fn_ptr(
87                                *self.tcx,
88                                self.typing_env,
89                                def_id,
90                                args,
91                            )
92                            .ok_or_else(|| ::rustc_middle::mir::interpret::InterpErrorKind::InvalidProgram(::rustc_middle::mir::interpret::InvalidProgramInfo::TooGeneric)err_inval!(TooGeneric))?
93                        };
94
95                        let fn_ptr = self.fn_ptr(FnVal::Instance(instance));
96                        self.write_pointer(fn_ptr, dest)?;
97                    }
98                    _ => ::rustc_middle::util::bug::span_bug_fmt(self.cur_span(),
    format_args!("reify fn pointer on {0}", src.layout.ty))span_bug!(self.cur_span(), "reify fn pointer on {}", src.layout.ty),
99                }
100            }
101
102            CastKind::PointerCoercion(PointerCoercion::UnsafeFnPointer, _) => {
103                let src = self.read_immediate(src)?;
104                match cast_ty.kind() {
105                    ty::FnPtr(..) => {
106                        // No change to value
107                        self.write_immediate(*src, dest)?;
108                    }
109                    _ => ::rustc_middle::util::bug::span_bug_fmt(self.cur_span(),
    format_args!("fn to unsafe fn cast on {0}", cast_ty))span_bug!(self.cur_span(), "fn to unsafe fn cast on {}", cast_ty),
110                }
111            }
112
113            CastKind::PointerCoercion(PointerCoercion::ClosureFnPointer(_), _) => {
114                // All reifications must be monomorphic, bail out otherwise.
115                ensure_monomorphic_enough(*self.tcx, src.layout.ty)?;
116
117                // The src operand does not matter, just its type
118                match *src.layout.ty.kind() {
119                    ty::Closure(def_id, args) => {
120                        let instance = {
121                            let _trace = <M as
        crate::interpret::Machine>::enter_trace_span(||
        {
            use ::tracing::__macro_support::Callsite as _;
            static __CALLSITE: ::tracing::callsite::DefaultCallsite =
                {
                    static META: ::tracing::Metadata<'static> =
                        {
                            ::tracing_core::metadata::Metadata::new("resolve",
                                "rustc_const_eval::interpret::cast", ::tracing::Level::INFO,
                                ::tracing_core::__macro_support::Option::Some("compiler/rustc_const_eval/src/interpret/cast.rs"),
                                ::tracing_core::__macro_support::Option::Some(121u32),
                                ::tracing_core::__macro_support::Option::Some("rustc_const_eval::interpret::cast"),
                                ::tracing_core::field::FieldSet::new(&["resolve", "def_id"],
                                    ::tracing_core::callsite::Identifier(&__CALLSITE)),
                                ::tracing::metadata::Kind::SPAN)
                        };
                    ::tracing::callsite::DefaultCallsite::new(&META)
                };
            let mut interest = ::tracing::subscriber::Interest::never();
            if ::tracing::Level::INFO <=
                                ::tracing::level_filters::STATIC_MAX_LEVEL &&
                            ::tracing::Level::INFO <=
                                ::tracing::level_filters::LevelFilter::current() &&
                        { interest = __CALLSITE.interest(); !interest.is_never() }
                    &&
                    ::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
                        interest) {
                let meta = __CALLSITE.metadata();
                ::tracing::Span::new(meta,
                    &{
                            #[allow(unused_imports)]
                            use ::tracing::field::{debug, display, Value};
                            let mut iter = meta.fields().iter();
                            meta.fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
                                                ::tracing::__macro_support::Option::Some(&display(&"resolve_closure")
                                                        as &dyn Value)),
                                            (&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
                                                ::tracing::__macro_support::Option::Some(&debug(&def_id) as
                                                        &dyn Value))])
                        })
            } else {
                let span =
                    ::tracing::__macro_support::__disabled_span(__CALLSITE.metadata());
                {};
                span
            }
        })enter_trace_span!(M, resolve::resolve_closure, ?def_id);
122                            ty::Instance::resolve_closure(
123                                *self.tcx,
124                                def_id,
125                                args,
126                                ty::ClosureKind::FnOnce,
127                            )
128                        };
129                        let fn_ptr = self.fn_ptr(FnVal::Instance(instance));
130                        self.write_pointer(fn_ptr, dest)?;
131                    }
132                    _ => ::rustc_middle::util::bug::span_bug_fmt(self.cur_span(),
    format_args!("closure fn pointer on {0}", src.layout.ty))span_bug!(self.cur_span(), "closure fn pointer on {}", src.layout.ty),
133                }
134            }
135
136            CastKind::Transmute | CastKind::Subtype => {
137                if !src.layout.is_sized() {
    ::core::panicking::panic("assertion failed: src.layout.is_sized()")
};assert!(src.layout.is_sized());
138                if !dest.layout.is_sized() {
    ::core::panicking::panic("assertion failed: dest.layout.is_sized()")
};assert!(dest.layout.is_sized());
139                match (&cast_ty, &dest.layout.ty) {
    (left_val, right_val) => {
        if !(*left_val == *right_val) {
            let kind = ::core::panicking::AssertKind::Eq;
            ::core::panicking::assert_failed(kind, &*left_val, &*right_val,
                ::core::option::Option::None);
        }
    }
};assert_eq!(cast_ty, dest.layout.ty); // we otherwise ignore `cast_ty` enirely...
140                if src.layout.size != dest.layout.size {
141                    do yeet {
        let (src_bytes, dest_bytes, src, dest) =
            (src.layout.size.bytes(), dest.layout.size.bytes(), src.layout.ty,
                dest.layout.ty);
        ::rustc_middle::mir::interpret::InterpErrorKind::UndefinedBehavior(::rustc_middle::mir::interpret::UndefinedBehaviorInfo::Custom(::rustc_middle::error::CustomSubdiagnostic {
                    msg: ||
                        rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("transmuting from {$src_bytes}-byte type to {$dest_bytes}-byte type: `{$src}` -> `{$dest}`")),
                    add_args: Box::new(move |mut set_arg|
                            {
                                set_arg("src_bytes".into(),
                                    rustc_errors::IntoDiagArg::into_diag_arg(src_bytes,
                                        &mut None));
                                set_arg("dest_bytes".into(),
                                    rustc_errors::IntoDiagArg::into_diag_arg(dest_bytes,
                                        &mut None));
                                set_arg("src".into(),
                                    rustc_errors::IntoDiagArg::into_diag_arg(src, &mut None));
                                set_arg("dest".into(),
                                    rustc_errors::IntoDiagArg::into_diag_arg(dest, &mut None));
                            }),
                }))
    };throw_ub_custom!(
142                        msg!(
143                            "transmuting from {$src_bytes}-byte type to {$dest_bytes}-byte type: `{$src}` -> `{$dest}`"
144                        ),
145                        src_bytes = src.layout.size.bytes(),
146                        dest_bytes = dest.layout.size.bytes(),
147                        src = src.layout.ty,
148                        dest = dest.layout.ty,
149                    );
150                }
151
152                self.copy_op_allow_transmute(src, dest)?;
153            }
154        }
155        interp_ok(())
156    }
157
158    /// Handles 'IntToInt' and 'IntToFloat' casts.
159    pub fn int_to_int_or_float(
160        &self,
161        src: &ImmTy<'tcx, M::Provenance>,
162        cast_to: TyAndLayout<'tcx>,
163    ) -> InterpResult<'tcx, ImmTy<'tcx, M::Provenance>> {
164        if !(src.layout.ty.is_integral() || src.layout.ty.is_char() ||
            src.layout.ty.is_bool()) {
    ::core::panicking::panic("assertion failed: src.layout.ty.is_integral() || src.layout.ty.is_char() ||\n    src.layout.ty.is_bool()")
};assert!(src.layout.ty.is_integral() || src.layout.ty.is_char() || src.layout.ty.is_bool());
165        if !(cast_to.ty.is_floating_point() || cast_to.ty.is_integral() ||
            cast_to.ty.is_char()) {
    ::core::panicking::panic("assertion failed: cast_to.ty.is_floating_point() || cast_to.ty.is_integral() ||\n    cast_to.ty.is_char()")
};assert!(cast_to.ty.is_floating_point() || cast_to.ty.is_integral() || cast_to.ty.is_char());
166
167        interp_ok(ImmTy::from_scalar(
168            self.cast_from_int_like(src.to_scalar(), src.layout, cast_to.ty)?,
169            cast_to,
170        ))
171    }
172
173    /// Handles 'FloatToFloat' and 'FloatToInt' casts.
174    pub fn float_to_float_or_int(
175        &self,
176        src: &ImmTy<'tcx, M::Provenance>,
177        cast_to: TyAndLayout<'tcx>,
178    ) -> InterpResult<'tcx, ImmTy<'tcx, M::Provenance>> {
179        let ty::Float(fty) = src.layout.ty.kind() else {
180            ::rustc_middle::util::bug::bug_fmt(format_args!("FloatToFloat/FloatToInt cast: source type {0} is not a float type",
        src.layout.ty))bug!("FloatToFloat/FloatToInt cast: source type {} is not a float type", src.layout.ty)
181        };
182        let val = match fty {
183            FloatTy::F16 => self.cast_from_float(src.to_scalar().to_f16()?, cast_to.ty),
184            FloatTy::F32 => self.cast_from_float(src.to_scalar().to_f32()?, cast_to.ty),
185            FloatTy::F64 => self.cast_from_float(src.to_scalar().to_f64()?, cast_to.ty),
186            FloatTy::F128 => self.cast_from_float(src.to_scalar().to_f128()?, cast_to.ty),
187        };
188        interp_ok(ImmTy::from_scalar(val, cast_to))
189    }
190
191    /// Handles 'FnPtrToPtr' and 'PtrToPtr' casts.
192    pub fn ptr_to_ptr(
193        &self,
194        src: &ImmTy<'tcx, M::Provenance>,
195        cast_to: TyAndLayout<'tcx>,
196    ) -> InterpResult<'tcx, ImmTy<'tcx, M::Provenance>> {
197        if !src.layout.ty.is_any_ptr() {
    ::core::panicking::panic("assertion failed: src.layout.ty.is_any_ptr()")
};assert!(src.layout.ty.is_any_ptr());
198        if !cast_to.ty.is_raw_ptr() {
    ::core::panicking::panic("assertion failed: cast_to.ty.is_raw_ptr()")
};assert!(cast_to.ty.is_raw_ptr());
199        // Handle casting any ptr to raw ptr (might be a wide ptr).
200        if cast_to.size == src.layout.size {
201            // Thin or wide pointer that just has the ptr kind of target type changed.
202            return interp_ok(ImmTy::from_immediate(**src, cast_to));
203        } else {
204            // Casting the metadata away from a wide ptr.
205            match (&src.layout.size, &(2 * self.pointer_size())) {
    (left_val, right_val) => {
        if !(*left_val == *right_val) {
            let kind = ::core::panicking::AssertKind::Eq;
            ::core::panicking::assert_failed(kind, &*left_val, &*right_val,
                ::core::option::Option::None);
        }
    }
};assert_eq!(src.layout.size, 2 * self.pointer_size());
206            match (&cast_to.size, &self.pointer_size()) {
    (left_val, right_val) => {
        if !(*left_val == *right_val) {
            let kind = ::core::panicking::AssertKind::Eq;
            ::core::panicking::assert_failed(kind, &*left_val, &*right_val,
                ::core::option::Option::None);
        }
    }
};assert_eq!(cast_to.size, self.pointer_size());
207            if !src.layout.ty.is_raw_ptr() {
    ::core::panicking::panic("assertion failed: src.layout.ty.is_raw_ptr()")
};assert!(src.layout.ty.is_raw_ptr());
208            return match **src {
209                Immediate::ScalarPair(data, _) => interp_ok(ImmTy::from_scalar(data, cast_to)),
210                Immediate::Scalar(..) => ::rustc_middle::util::bug::span_bug_fmt(self.cur_span(),
    format_args!("{0:?} input to a fat-to-thin cast ({1} -> {2})", *src,
        src.layout.ty, cast_to.ty))span_bug!(
211                    self.cur_span(),
212                    "{:?} input to a fat-to-thin cast ({} -> {})",
213                    *src,
214                    src.layout.ty,
215                    cast_to.ty
216                ),
217                Immediate::Uninit => do yeet ::rustc_middle::mir::interpret::InterpErrorKind::UndefinedBehavior(::rustc_middle::mir::interpret::UndefinedBehaviorInfo::InvalidUninitBytes(None))throw_ub!(InvalidUninitBytes(None)),
218            };
219        }
220    }
221
222    pub fn pointer_expose_provenance_cast(
223        &mut self,
224        src: &ImmTy<'tcx, M::Provenance>,
225        cast_to: TyAndLayout<'tcx>,
226    ) -> InterpResult<'tcx, ImmTy<'tcx, M::Provenance>> {
227        match src.layout.ty.kind() {
    ty::RawPtr(_, _) | ty::FnPtr(..) => {}
    ref left_val => {
        ::core::panicking::assert_matches_failed(left_val,
            "ty::RawPtr(_, _) | ty::FnPtr(..)", ::core::option::Option::None);
    }
};assert_matches!(src.layout.ty.kind(), ty::RawPtr(_, _) | ty::FnPtr(..));
228        if !cast_to.ty.is_integral() {
    ::core::panicking::panic("assertion failed: cast_to.ty.is_integral()")
};assert!(cast_to.ty.is_integral());
229
230        let scalar = src.to_scalar();
231        let ptr = scalar.to_pointer(self)?;
232        match ptr.into_pointer_or_addr() {
233            Ok(ptr) => M::expose_provenance(self, ptr.provenance)?,
234            Err(_) => {} // Do nothing, exposing an invalid pointer (`None` provenance) is a NOP.
235        };
236        interp_ok(ImmTy::from_scalar(
237            self.cast_from_int_like(scalar, src.layout, cast_to.ty)?,
238            cast_to,
239        ))
240    }
241
242    pub fn pointer_with_exposed_provenance_cast(
243        &self,
244        src: &ImmTy<'tcx, M::Provenance>,
245        cast_to: TyAndLayout<'tcx>,
246    ) -> InterpResult<'tcx, ImmTy<'tcx, M::Provenance>> {
247        if !src.layout.ty.is_integral() {
    ::core::panicking::panic("assertion failed: src.layout.ty.is_integral()")
};assert!(src.layout.ty.is_integral());
248        match cast_to.ty.kind() {
    ty::RawPtr(_, _) => {}
    ref left_val => {
        ::core::panicking::assert_matches_failed(left_val, "ty::RawPtr(_, _)",
            ::core::option::Option::None);
    }
};assert_matches!(cast_to.ty.kind(), ty::RawPtr(_, _));
249
250        // First cast to usize.
251        let scalar = src.to_scalar();
252        let addr = self.cast_from_int_like(scalar, src.layout, self.tcx.types.usize)?;
253        let addr = addr.to_target_usize(self)?;
254
255        // Then turn address into pointer.
256        let ptr = M::ptr_from_addr_cast(self, addr)?;
257        interp_ok(ImmTy::from_scalar(Scalar::from_maybe_pointer(ptr, self), cast_to))
258    }
259
260    /// Low-level cast helper function. This works directly on scalars and can take 'int-like' input
261    /// type (basically everything with a scalar layout) to int/float/char types.
262    fn cast_from_int_like(
263        &self,
264        scalar: Scalar<M::Provenance>, // input value (there is no ScalarTy so we separate data+layout)
265        src_layout: TyAndLayout<'tcx>,
266        cast_ty: Ty<'tcx>,
267    ) -> InterpResult<'tcx, Scalar<M::Provenance>> {
268        // Let's make sure v is sign-extended *if* it has a signed type.
269        let signed = src_layout.backend_repr.is_signed(); // Also asserts that abi is `Scalar`.
270
271        let v = match src_layout.ty.kind() {
272            ty::Uint(_) | ty::RawPtr(..) | ty::FnPtr(..) => scalar.to_uint(src_layout.size)?,
273            ty::Int(_) => scalar.to_int(src_layout.size)? as u128, // we will cast back to `i128` below if the sign matters
274            ty::Bool => scalar.to_bool()?.into(),
275            ty::Char => scalar.to_char()?.into(),
276            _ => ::rustc_middle::util::bug::span_bug_fmt(self.cur_span(),
    format_args!("invalid int-like cast from {0}", src_layout.ty))span_bug!(self.cur_span(), "invalid int-like cast from {}", src_layout.ty),
277        };
278
279        interp_ok(match *cast_ty.kind() {
280            // int -> int
281            ty::Int(_) | ty::Uint(_) => {
282                let size = match *cast_ty.kind() {
283                    ty::Int(t) => Integer::from_int_ty(self, t).size(),
284                    ty::Uint(t) => Integer::from_uint_ty(self, t).size(),
285                    _ => ::rustc_middle::util::bug::bug_fmt(format_args!("impossible case reached"))bug!(),
286                };
287                let v = size.truncate(v);
288                Scalar::from_uint(v, size)
289            }
290
291            // signed int -> float
292            ty::Float(fty) if signed => {
293                let v = v as i128;
294                match fty {
295                    FloatTy::F16 => Scalar::from_f16(Half::from_i128(v).value),
296                    FloatTy::F32 => Scalar::from_f32(Single::from_i128(v).value),
297                    FloatTy::F64 => Scalar::from_f64(Double::from_i128(v).value),
298                    FloatTy::F128 => Scalar::from_f128(Quad::from_i128(v).value),
299                }
300            }
301            // unsigned int -> float
302            ty::Float(fty) => match fty {
303                FloatTy::F16 => Scalar::from_f16(Half::from_u128(v).value),
304                FloatTy::F32 => Scalar::from_f32(Single::from_u128(v).value),
305                FloatTy::F64 => Scalar::from_f64(Double::from_u128(v).value),
306                FloatTy::F128 => Scalar::from_f128(Quad::from_u128(v).value),
307            },
308
309            // u8 -> char
310            ty::Char => Scalar::from_u32(u8::try_from(v).unwrap().into()),
311
312            // Casts to bool are not permitted by rustc, no need to handle them here.
313            _ => ::rustc_middle::util::bug::span_bug_fmt(self.cur_span(),
    format_args!("invalid int to {0} cast", cast_ty))span_bug!(self.cur_span(), "invalid int to {} cast", cast_ty),
314        })
315    }
316
317    /// Low-level cast helper function. Converts an apfloat `f` into int or float types.
318    fn cast_from_float<F>(&self, f: F, dest_ty: Ty<'tcx>) -> Scalar<M::Provenance>
319    where
320        F: Float
321            + Into<Scalar<M::Provenance>>
322            + FloatConvert<Half>
323            + FloatConvert<Single>
324            + FloatConvert<Double>
325            + FloatConvert<Quad>,
326    {
327        match *dest_ty.kind() {
328            // float -> uint
329            ty::Uint(t) => {
330                let size = Integer::from_uint_ty(self, t).size();
331                // `to_u128` is a saturating cast, which is what we need
332                // (https://doc.rust-lang.org/nightly/nightly-rustc/rustc_apfloat/trait.Float.html#method.to_i128_r).
333                let v = f.to_u128(size.bits_usize()).value;
334                // This should already fit the bit width
335                Scalar::from_uint(v, size)
336            }
337            // float -> int
338            ty::Int(t) => {
339                let size = Integer::from_int_ty(self, t).size();
340                // `to_i128` is a saturating cast, which is what we need
341                // (https://doc.rust-lang.org/nightly/nightly-rustc/rustc_apfloat/trait.Float.html#method.to_i128_r).
342                let v = f.to_i128(size.bits_usize()).value;
343                Scalar::from_int(v, size)
344            }
345            // float -> float
346            ty::Float(fty) => match fty {
347                FloatTy::F16 => {
348                    Scalar::from_f16(self.adjust_nan(f.convert(&mut false).value, &[f]))
349                }
350                FloatTy::F32 => {
351                    Scalar::from_f32(self.adjust_nan(f.convert(&mut false).value, &[f]))
352                }
353                FloatTy::F64 => {
354                    Scalar::from_f64(self.adjust_nan(f.convert(&mut false).value, &[f]))
355                }
356                FloatTy::F128 => {
357                    Scalar::from_f128(self.adjust_nan(f.convert(&mut false).value, &[f]))
358                }
359            },
360            // That's it.
361            _ => ::rustc_middle::util::bug::span_bug_fmt(self.cur_span(),
    format_args!("invalid float to {0} cast", dest_ty))span_bug!(self.cur_span(), "invalid float to {} cast", dest_ty),
362        }
363    }
364
365    /// `src` is a *pointer to* a `source_ty`, and in `dest` we should store a pointer to th same
366    /// data at type `cast_ty`.
367    fn unsize_into_ptr(
368        &mut self,
369        src: &OpTy<'tcx, M::Provenance>,
370        dest: &impl Writeable<'tcx, M::Provenance>,
371        // The pointee types
372        source_ty: Ty<'tcx>,
373        cast_ty: Ty<'tcx>,
374    ) -> InterpResult<'tcx> {
375        // A<Struct> -> A<Trait> conversion
376        let (src_pointee_ty, dest_pointee_ty) =
377            self.tcx.struct_lockstep_tails_for_codegen(source_ty, cast_ty, self.typing_env);
378
379        match (src_pointee_ty.kind(), dest_pointee_ty.kind()) {
380            (&ty::Array(_, length), &ty::Slice(_)) => {
381                let ptr = self.read_pointer(src)?;
382                let val = Immediate::new_slice(
383                    ptr,
384                    length
385                        .try_to_target_usize(*self.tcx)
386                        .expect("expected monomorphic const in const eval"),
387                    self,
388                );
389                self.write_immediate(val, dest)
390            }
391            (ty::Dynamic(data_a, _), ty::Dynamic(data_b, _)) => {
392                let val = self.read_immediate(src)?;
393                // MIR building generates odd NOP casts, prevent them from causing unexpected trouble.
394                // See <https://github.com/rust-lang/rust/issues/128880>.
395                // FIXME: ideally we wouldn't have to do this.
396                if data_a == data_b {
397                    return self.write_immediate(*val, dest);
398                }
399                // Take apart the old pointer, and find the dynamic type.
400                let (old_data, old_vptr) = val.to_scalar_pair();
401                let old_data = old_data.to_pointer(self)?;
402                let old_vptr = old_vptr.to_pointer(self)?;
403                let ty = self.get_ptr_vtable_ty(old_vptr, Some(data_a))?;
404
405                // Sanity-check that `supertrait_vtable_slot` in this type's vtable indeed produces
406                // our destination trait.
407                let vptr_entry_idx =
408                    self.tcx.supertrait_vtable_slot((src_pointee_ty, dest_pointee_ty));
409                let vtable_entries = self.vtable_entries(data_a.principal(), ty);
410                if let Some(entry_idx) = vptr_entry_idx {
411                    let Some(&ty::VtblEntry::TraitVPtr(upcast_trait_ref)) =
412                        vtable_entries.get(entry_idx)
413                    else {
414                        ::rustc_middle::util::bug::span_bug_fmt(self.cur_span(),
    format_args!("invalid vtable entry index in {0} -> {1} upcast",
        src_pointee_ty, dest_pointee_ty));span_bug!(
415                            self.cur_span(),
416                            "invalid vtable entry index in {} -> {} upcast",
417                            src_pointee_ty,
418                            dest_pointee_ty
419                        );
420                    };
421                    let erased_trait_ref =
422                        ty::ExistentialTraitRef::erase_self_ty(*self.tcx, upcast_trait_ref);
423                    match (&data_b.principal().map(|b|
                    {
                        self.tcx.normalize_erasing_late_bound_regions(self.typing_env,
                            b)
                    }), &Some(erased_trait_ref)) {
    (left_val, right_val) => {
        if !(*left_val == *right_val) {
            let kind = ::core::panicking::AssertKind::Eq;
            ::core::panicking::assert_failed(kind, &*left_val, &*right_val,
                ::core::option::Option::None);
        }
    }
};assert_eq!(
424                        data_b.principal().map(|b| {
425                            self.tcx.normalize_erasing_late_bound_regions(self.typing_env, b)
426                        }),
427                        Some(erased_trait_ref),
428                    );
429                } else {
430                    // In this case codegen would keep using the old vtable. We don't want to do
431                    // that as it has the wrong trait. The reason codegen can do this is that
432                    // one vtable is a prefix of the other, so we double-check that.
433                    let vtable_entries_b = self.vtable_entries(data_b.principal(), ty);
434                    if !(&vtable_entries[..vtable_entries_b.len()] == vtable_entries_b) {
    ::core::panicking::panic("assertion failed: &vtable_entries[..vtable_entries_b.len()] == vtable_entries_b")
};assert!(&vtable_entries[..vtable_entries_b.len()] == vtable_entries_b);
435                };
436
437                // Get the destination trait vtable and return that.
438                let new_vptr = self.get_vtable_ptr(ty, data_b)?;
439                self.write_immediate(Immediate::new_dyn_trait(old_data, new_vptr, self), dest)
440            }
441            (_, &ty::Dynamic(data, _)) => {
442                // Initial cast from sized to dyn trait
443                let vtable = self.get_vtable_ptr(src_pointee_ty, data)?;
444                let ptr = self.read_pointer(src)?;
445                let val = Immediate::new_dyn_trait(ptr, vtable, &*self.tcx);
446                self.write_immediate(val, dest)
447            }
448            _ => {
449                // Do not ICE if we are not monomorphic enough.
450                ensure_monomorphic_enough(*self.tcx, src.layout.ty)?;
451                ensure_monomorphic_enough(*self.tcx, cast_ty)?;
452
453                ::rustc_middle::util::bug::span_bug_fmt(self.cur_span(),
    format_args!("invalid pointer unsizing {0} -> {1}", src.layout.ty,
        cast_ty))span_bug!(
454                    self.cur_span(),
455                    "invalid pointer unsizing {} -> {}",
456                    src.layout.ty,
457                    cast_ty
458                )
459            }
460        }
461    }
462
463    pub fn unsize_into(
464        &mut self,
465        src: &OpTy<'tcx, M::Provenance>,
466        cast_ty: TyAndLayout<'tcx>,
467        dest: &impl Writeable<'tcx, M::Provenance>,
468    ) -> InterpResult<'tcx> {
469        {
    use ::tracing::__macro_support::Callsite as _;
    static __CALLSITE: ::tracing::callsite::DefaultCallsite =
        {
            static META: ::tracing::Metadata<'static> =
                {
                    ::tracing_core::metadata::Metadata::new("event compiler/rustc_const_eval/src/interpret/cast.rs:469",
                        "rustc_const_eval::interpret::cast",
                        ::tracing::Level::TRACE,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_const_eval/src/interpret/cast.rs"),
                        ::tracing_core::__macro_support::Option::Some(469u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_const_eval::interpret::cast"),
                        ::tracing_core::field::FieldSet::new(&["message"],
                            ::tracing_core::callsite::Identifier(&__CALLSITE)),
                        ::tracing::metadata::Kind::EVENT)
                };
            ::tracing::callsite::DefaultCallsite::new(&META)
        };
    let enabled =
        ::tracing::Level::TRACE <= ::tracing::level_filters::STATIC_MAX_LEVEL
                &&
                ::tracing::Level::TRACE <=
                    ::tracing::level_filters::LevelFilter::current() &&
            {
                let interest = __CALLSITE.interest();
                !interest.is_never() &&
                    ::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
                        interest)
            };
    if enabled {
        (|value_set: ::tracing::field::ValueSet|
                    {
                        let meta = __CALLSITE.metadata();
                        ::tracing::Event::dispatch(meta, &value_set);
                        ;
                    })({
                #[allow(unused_imports)]
                use ::tracing::field::{debug, display, Value};
                let mut iter = __CALLSITE.metadata().fields().iter();
                __CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
                                    ::tracing::__macro_support::Option::Some(&format_args!("Unsizing {0:?} of type {1} into {2}",
                                                    *src, src.layout.ty, cast_ty.ty) as &dyn Value))])
            });
    } else { ; }
};trace!("Unsizing {:?} of type {} into {}", *src, src.layout.ty, cast_ty.ty);
470        match (src.layout.ty.kind(), cast_ty.ty.kind()) {
471            (&ty::Pat(_, s_pat), &ty::Pat(cast_ty, c_pat)) if s_pat == c_pat => {
472                let src = self.project_field(src, FieldIdx::ZERO)?;
473                let dest = self.project_field(dest, FieldIdx::ZERO)?;
474                let cast_ty = self.layout_of(cast_ty)?;
475                self.unsize_into(&src, cast_ty, &dest)
476            }
477            (&ty::Ref(_, s, _), &ty::Ref(_, c, _) | &ty::RawPtr(c, _))
478            | (&ty::RawPtr(s, _), &ty::RawPtr(c, _)) => self.unsize_into_ptr(src, dest, s, c),
479            (&ty::Adt(def_a, _), &ty::Adt(def_b, _)) => {
480                match (&def_a, &def_b) {
    (left_val, right_val) => {
        if !(*left_val == *right_val) {
            let kind = ::core::panicking::AssertKind::Eq;
            ::core::panicking::assert_failed(kind, &*left_val, &*right_val,
                ::core::option::Option::None);
        }
    }
};assert_eq!(def_a, def_b); // implies same number of fields
481
482                // Unsizing of generic struct with pointer fields, like `Arc<T>` -> `Arc<Trait>`.
483                // There can be extra fields as long as they don't change their type or are 1-ZST.
484                // There might also be no field that actually needs unsizing.
485                let mut found_cast_field = false;
486                for i in 0..src.layout.fields.count() {
487                    let cast_ty_field = cast_ty.field(self, i);
488                    let i = FieldIdx::from_usize(i);
489                    let src_field = self.project_field(src, i)?;
490                    let dst_field = self.project_field(dest, i)?;
491                    if src_field.layout.is_1zst() && cast_ty_field.is_1zst() {
492                        // Skip 1-ZST fields.
493                    } else if src_field.layout.ty == cast_ty_field.ty {
494                        self.copy_op(&src_field, &dst_field)?;
495                    } else {
496                        if found_cast_field {
497                            ::rustc_middle::util::bug::span_bug_fmt(self.cur_span(),
    format_args!("unsize_into: more than one field to cast"));span_bug!(self.cur_span(), "unsize_into: more than one field to cast");
498                        }
499                        found_cast_field = true;
500                        self.unsize_into(&src_field, cast_ty_field, &dst_field)?;
501                    }
502                }
503                interp_ok(())
504            }
505            _ => {
506                // Do not ICE if we are not monomorphic enough.
507                ensure_monomorphic_enough(*self.tcx, src.layout.ty)?;
508                ensure_monomorphic_enough(*self.tcx, cast_ty.ty)?;
509
510                ::rustc_middle::util::bug::span_bug_fmt(self.cur_span(),
    format_args!("unsize_into: invalid conversion: {0:?} -> {1:?}",
        src.layout, dest.layout()))span_bug!(
511                    self.cur_span(),
512                    "unsize_into: invalid conversion: {:?} -> {:?}",
513                    src.layout,
514                    dest.layout()
515                )
516            }
517        }
518    }
519}