1use std::assert_matches;
5
6use either::{Either, Left, Right};
7use rustc_abi as abi;
8use rustc_abi::{BackendRepr, HasDataLayout, Size};
9use rustc_hir::def::Namespace;
10use rustc_middle::mir::interpret::ScalarSizeMismatch;
11use rustc_middle::ty::layout::{HasTyCtxt, HasTypingEnv, TyAndLayout};
12use rustc_middle::ty::print::{FmtPrinter, PrettyPrinter};
13use rustc_middle::ty::{ConstInt, ScalarInt, Ty, TyCtxt};
14use rustc_middle::{bug, mir, span_bug, ty};
15use rustc_span::DUMMY_SP;
16use tracing::field::Empty;
17use tracing::trace;
18
19use super::{
20 CtfeProvenance, Frame, InterpCx, InterpResult, MPlaceTy, Machine, MemPlace, MemPlaceMeta,
21 OffsetMode, PlaceTy, Pointer, Projectable, Provenance, Scalar, alloc_range, err_ub,
22 from_known_layout, interp_ok, mir_assign_valid_types, throw_ub,
23};
24use crate::enter_trace_span;
25
26#[derive(#[automatically_derived]
impl<Prov: ::core::marker::Copy + Provenance> ::core::marker::Copy for
Immediate<Prov> {
}Copy, #[automatically_derived]
impl<Prov: ::core::clone::Clone + Provenance> ::core::clone::Clone for
Immediate<Prov> {
#[inline]
fn clone(&self) -> Immediate<Prov> {
match self {
Immediate::Scalar(__self_0) =>
Immediate::Scalar(::core::clone::Clone::clone(__self_0)),
Immediate::ScalarPair(__self_0, __self_1) =>
Immediate::ScalarPair(::core::clone::Clone::clone(__self_0),
::core::clone::Clone::clone(__self_1)),
Immediate::Uninit => Immediate::Uninit,
}
}
}Clone, #[automatically_derived]
impl<Prov: ::core::fmt::Debug + Provenance> ::core::fmt::Debug for
Immediate<Prov> {
#[inline]
fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
match self {
Immediate::Scalar(__self_0) =>
::core::fmt::Formatter::debug_tuple_field1_finish(f, "Scalar",
&__self_0),
Immediate::ScalarPair(__self_0, __self_1) =>
::core::fmt::Formatter::debug_tuple_field2_finish(f,
"ScalarPair", __self_0, &__self_1),
Immediate::Uninit =>
::core::fmt::Formatter::write_str(f, "Uninit"),
}
}
}Debug)]
34pub enum Immediate<Prov: Provenance = CtfeProvenance> {
35 Scalar(Scalar<Prov>),
37 ScalarPair(Scalar<Prov>, Scalar<Prov>),
40 Uninit,
42}
43
44impl<Prov: Provenance> From<Scalar<Prov>> for Immediate<Prov> {
45 #[inline(always)]
46 fn from(val: Scalar<Prov>) -> Self {
47 Immediate::Scalar(val)
48 }
49}
50
51impl<Prov: Provenance> Immediate<Prov> {
52 pub fn new_pointer_with_meta(
53 ptr: Pointer<Option<Prov>>,
54 meta: MemPlaceMeta<Prov>,
55 cx: &impl HasDataLayout,
56 ) -> Self {
57 let ptr = Scalar::from_maybe_pointer(ptr, cx);
58 match meta {
59 MemPlaceMeta::None => Immediate::from(ptr),
60 MemPlaceMeta::Meta(meta) => Immediate::ScalarPair(ptr, meta),
61 }
62 }
63
64 pub fn new_slice(ptr: Pointer<Option<Prov>>, len: u64, cx: &impl HasDataLayout) -> Self {
65 Immediate::ScalarPair(
66 Scalar::from_maybe_pointer(ptr, cx),
67 Scalar::from_target_usize(len, cx),
68 )
69 }
70
71 pub fn new_dyn_trait(
72 val: Pointer<Option<Prov>>,
73 vtable: Pointer<Option<Prov>>,
74 cx: &impl HasDataLayout,
75 ) -> Self {
76 Immediate::ScalarPair(
77 Scalar::from_maybe_pointer(val, cx),
78 Scalar::from_maybe_pointer(vtable, cx),
79 )
80 }
81
82 #[inline]
83 #[cfg_attr(debug_assertions, track_caller)] pub fn to_scalar(self) -> Scalar<Prov> {
85 match self {
86 Immediate::Scalar(val) => val,
87 Immediate::ScalarPair(..) => ::rustc_middle::util::bug::bug_fmt(format_args!("Got a scalar pair where a scalar was expected"))bug!("Got a scalar pair where a scalar was expected"),
88 Immediate::Uninit => ::rustc_middle::util::bug::bug_fmt(format_args!("Got uninit where a scalar was expected"))bug!("Got uninit where a scalar was expected"),
89 }
90 }
91
92 #[inline]
93 #[cfg_attr(debug_assertions, track_caller)] pub fn to_scalar_int(self) -> ScalarInt {
95 self.to_scalar().try_to_scalar_int().unwrap()
96 }
97
98 #[inline]
99 #[cfg_attr(debug_assertions, track_caller)] pub fn to_scalar_pair(self) -> (Scalar<Prov>, Scalar<Prov>) {
101 match self {
102 Immediate::ScalarPair(val1, val2) => (val1, val2),
103 Immediate::Scalar(..) => ::rustc_middle::util::bug::bug_fmt(format_args!("Got a scalar where a scalar pair was expected"))bug!("Got a scalar where a scalar pair was expected"),
104 Immediate::Uninit => ::rustc_middle::util::bug::bug_fmt(format_args!("Got uninit where a scalar pair was expected"))bug!("Got uninit where a scalar pair was expected"),
105 }
106 }
107
108 #[inline]
110 #[cfg_attr(debug_assertions, track_caller)] pub fn to_scalar_and_meta(self) -> (Scalar<Prov>, MemPlaceMeta<Prov>) {
112 match self {
113 Immediate::ScalarPair(val1, val2) => (val1, MemPlaceMeta::Meta(val2)),
114 Immediate::Scalar(val) => (val, MemPlaceMeta::None),
115 Immediate::Uninit => ::rustc_middle::util::bug::bug_fmt(format_args!("Got uninit where a scalar or scalar pair was expected"))bug!("Got uninit where a scalar or scalar pair was expected"),
116 }
117 }
118
119 pub fn assert_matches_abi(self, abi: BackendRepr, msg: &str, cx: &impl HasDataLayout) {
121 match (self, abi) {
122 (Immediate::Scalar(scalar), BackendRepr::Scalar(s)) => {
123 match (&scalar.size(), &s.size(cx)) {
(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::Some(format_args!("{0}: scalar value has wrong size",
msg)));
}
}
};assert_eq!(scalar.size(), s.size(cx), "{msg}: scalar value has wrong size");
124 if !#[allow(non_exhaustive_omitted_patterns)] match s.primitive() {
abi::Primitive::Pointer(..) => true,
_ => false,
}matches!(s.primitive(), abi::Primitive::Pointer(..)) {
125 if !#[allow(non_exhaustive_omitted_patterns)] match scalar {
Scalar::Int(..) => true,
_ => false,
} {
{
::core::panicking::panic_fmt(format_args!("{0}: scalar value should be an integer, but has provenance",
msg));
}
};assert!(
127 matches!(scalar, Scalar::Int(..)),
128 "{msg}: scalar value should be an integer, but has provenance"
129 );
130 }
131 }
132 (Immediate::ScalarPair(a_val, b_val), BackendRepr::ScalarPair(a, b)) => {
133 match (&a_val.size(), &a.size(cx)) {
(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::Some(format_args!("{0}: first component of scalar pair has wrong size",
msg)));
}
}
};assert_eq!(
134 a_val.size(),
135 a.size(cx),
136 "{msg}: first component of scalar pair has wrong size"
137 );
138 if !#[allow(non_exhaustive_omitted_patterns)] match a.primitive() {
abi::Primitive::Pointer(..) => true,
_ => false,
}matches!(a.primitive(), abi::Primitive::Pointer(..)) {
139 if !#[allow(non_exhaustive_omitted_patterns)] match a_val {
Scalar::Int(..) => true,
_ => false,
} {
{
::core::panicking::panic_fmt(format_args!("{0}: first component of scalar pair should be an integer, but has provenance",
msg));
}
};assert!(
140 matches!(a_val, Scalar::Int(..)),
141 "{msg}: first component of scalar pair should be an integer, but has provenance"
142 );
143 }
144 match (&b_val.size(), &b.size(cx)) {
(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::Some(format_args!("{0}: second component of scalar pair has wrong size",
msg)));
}
}
};assert_eq!(
145 b_val.size(),
146 b.size(cx),
147 "{msg}: second component of scalar pair has wrong size"
148 );
149 if !#[allow(non_exhaustive_omitted_patterns)] match b.primitive() {
abi::Primitive::Pointer(..) => true,
_ => false,
}matches!(b.primitive(), abi::Primitive::Pointer(..)) {
150 if !#[allow(non_exhaustive_omitted_patterns)] match b_val {
Scalar::Int(..) => true,
_ => false,
} {
{
::core::panicking::panic_fmt(format_args!("{0}: second component of scalar pair should be an integer, but has provenance",
msg));
}
};assert!(
151 matches!(b_val, Scalar::Int(..)),
152 "{msg}: second component of scalar pair should be an integer, but has provenance"
153 );
154 }
155 }
156 (Immediate::Uninit, _) => {
157 if !abi.is_sized() {
{
::core::panicking::panic_fmt(format_args!("{0}: unsized immediates are not a thing",
msg));
}
};assert!(abi.is_sized(), "{msg}: unsized immediates are not a thing");
158 }
159 _ => {
160 ::rustc_middle::util::bug::bug_fmt(format_args!("{0}: value {1:?} does not match ABI {2:?})",
msg, self, abi))bug!("{msg}: value {self:?} does not match ABI {abi:?})",)
161 }
162 }
163 }
164
165 pub fn clear_provenance<'tcx>(&mut self) -> InterpResult<'tcx> {
166 match self {
167 Immediate::Scalar(s) => {
168 s.clear_provenance()?;
169 }
170 Immediate::ScalarPair(a, b) => {
171 a.clear_provenance()?;
172 b.clear_provenance()?;
173 }
174 Immediate::Uninit => {}
175 }
176 interp_ok(())
177 }
178
179 pub fn has_provenance(&self) -> bool {
180 match self {
181 Immediate::Scalar(scalar) => #[allow(non_exhaustive_omitted_patterns)] match scalar {
Scalar::Ptr { .. } => true,
_ => false,
}matches!(scalar, Scalar::Ptr { .. }),
182 Immediate::ScalarPair(s1, s2) => {
183 #[allow(non_exhaustive_omitted_patterns)] match s1 {
Scalar::Ptr { .. } => true,
_ => false,
}matches!(s1, Scalar::Ptr { .. }) || #[allow(non_exhaustive_omitted_patterns)] match s2 {
Scalar::Ptr { .. } => true,
_ => false,
}matches!(s2, Scalar::Ptr { .. })
184 }
185 Immediate::Uninit => false,
186 }
187 }
188}
189
190#[derive(#[automatically_derived]
impl<'tcx, Prov: ::core::clone::Clone + Provenance> ::core::clone::Clone for
ImmTy<'tcx, Prov> {
#[inline]
fn clone(&self) -> ImmTy<'tcx, Prov> {
ImmTy {
imm: ::core::clone::Clone::clone(&self.imm),
layout: ::core::clone::Clone::clone(&self.layout),
}
}
}Clone)]
193pub struct ImmTy<'tcx, Prov: Provenance = CtfeProvenance> {
194 imm: Immediate<Prov>,
195 pub layout: TyAndLayout<'tcx>,
196}
197
198impl<Prov: Provenance> std::fmt::Display for ImmTy<'_, Prov> {
199 fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
200 fn print_scalar<'a, 'tcx, Prov: Provenance>(
202 p: &mut FmtPrinter<'a, 'tcx>,
203 s: Scalar<Prov>,
204 ty: Ty<'tcx>,
205 ) -> Result<(), std::fmt::Error> {
206 match s {
207 Scalar::Int(int) => p.pretty_print_const_scalar_int(int, ty, true),
208 Scalar::Ptr(ptr, _sz) => {
209 p.pretty_print_const_pointer(ptr, ty)
213 }
214 }
215 }
216 ty::tls::with(|tcx| {
217 match self.imm {
218 Immediate::Scalar(s) => {
219 let ty = tcx.lift(self.layout.ty);
220 let s = FmtPrinter::print_string(tcx, Namespace::ValueNS, |p| {
221 print_scalar(p, s, ty)
222 })?;
223 f.write_str(&s)
224 }
225 Immediate::ScalarPair(a, b) => {
226 f.write_fmt(format_args!("({0:x}, {1:x}): {2}", a, b, self.layout.ty))write!(f, "({:x}, {:x}): {}", a, b, self.layout.ty)
228 }
229 Immediate::Uninit => {
230 f.write_fmt(format_args!("uninit: {0}", self.layout.ty))write!(f, "uninit: {}", self.layout.ty)
231 }
232 }
233 })
234 }
235}
236
237impl<Prov: Provenance> std::fmt::Debug for ImmTy<'_, Prov> {
238 fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
239 f.debug_struct("ImmTy")
241 .field("imm", &self.imm)
242 .field("ty", &format_args!("{0}", self.layout.ty)format_args!("{}", self.layout.ty))
243 .finish()
244 }
245}
246
247impl<'tcx, Prov: Provenance> std::ops::Deref for ImmTy<'tcx, Prov> {
248 type Target = Immediate<Prov>;
249 #[inline(always)]
250 fn deref(&self) -> &Immediate<Prov> {
251 &self.imm
252 }
253}
254
255impl<'tcx, Prov: Provenance> ImmTy<'tcx, Prov> {
256 #[inline]
257 pub fn from_scalar(val: Scalar<Prov>, layout: TyAndLayout<'tcx>) -> Self {
258 if true {
if !layout.backend_repr.is_scalar() {
{
::core::panicking::panic_fmt(format_args!("`ImmTy::from_scalar` on non-scalar layout"));
}
};
};debug_assert!(layout.backend_repr.is_scalar(), "`ImmTy::from_scalar` on non-scalar layout");
259 if true {
match (&val.size(), &layout.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);
}
}
};
};debug_assert_eq!(val.size(), layout.size);
260 ImmTy { imm: val.into(), layout }
261 }
262
263 #[inline]
264 pub fn from_scalar_pair(a: Scalar<Prov>, b: Scalar<Prov>, layout: TyAndLayout<'tcx>) -> Self {
265 if true {
if !#[allow(non_exhaustive_omitted_patterns)] match layout.backend_repr {
BackendRepr::ScalarPair(..) => true,
_ => false,
} {
{
::core::panicking::panic_fmt(format_args!("`ImmTy::from_scalar_pair` on non-scalar-pair layout"));
}
};
};debug_assert!(
266 matches!(layout.backend_repr, BackendRepr::ScalarPair(..)),
267 "`ImmTy::from_scalar_pair` on non-scalar-pair layout"
268 );
269 let imm = Immediate::ScalarPair(a, b);
270 ImmTy { imm, layout }
271 }
272
273 #[inline(always)]
274 pub fn from_immediate(imm: Immediate<Prov>, layout: TyAndLayout<'tcx>) -> Self {
275 if true {
if !match (imm, layout.backend_repr) {
(Immediate::Scalar(..), BackendRepr::Scalar(..)) => true,
(Immediate::ScalarPair(..), BackendRepr::ScalarPair(..)) =>
true,
(Immediate::Uninit, _) if layout.is_sized() => true,
_ => false,
} {
{
::core::panicking::panic_fmt(format_args!("immediate {0:?} does not fit to layout {1:?}",
imm, layout));
}
};
};debug_assert!(
277 match (imm, layout.backend_repr) {
278 (Immediate::Scalar(..), BackendRepr::Scalar(..)) => true,
279 (Immediate::ScalarPair(..), BackendRepr::ScalarPair(..)) => true,
280 (Immediate::Uninit, _) if layout.is_sized() => true,
281 _ => false,
282 },
283 "immediate {imm:?} does not fit to layout {layout:?}",
284 );
285 ImmTy { imm, layout }
286 }
287
288 #[inline]
289 pub fn uninit(layout: TyAndLayout<'tcx>) -> Self {
290 if true {
if !layout.is_sized() {
{
::core::panicking::panic_fmt(format_args!("immediates must be sized"));
}
};
};debug_assert!(layout.is_sized(), "immediates must be sized");
291 ImmTy { imm: Immediate::Uninit, layout }
292 }
293
294 #[inline]
295 pub fn from_scalar_int(s: ScalarInt, layout: TyAndLayout<'tcx>) -> Self {
296 Self::from_scalar(Scalar::from(s), layout)
297 }
298
299 #[inline]
300 pub fn from_uint(i: impl Into<u128>, layout: TyAndLayout<'tcx>) -> Self {
301 Self::from_scalar(Scalar::from_uint(i, layout.size), layout)
302 }
303
304 #[inline]
305 pub fn from_int(i: impl Into<i128>, layout: TyAndLayout<'tcx>) -> Self {
306 Self::from_scalar(Scalar::from_int(i, layout.size), layout)
307 }
308
309 #[inline]
310 pub fn from_bool(b: bool, tcx: TyCtxt<'tcx>) -> Self {
311 let layout = tcx
313 .layout_of(ty::TypingEnv::fully_monomorphized().as_query_input(tcx.types.bool))
314 .unwrap();
315 Self::from_scalar(Scalar::from_bool(b), layout)
316 }
317
318 #[inline]
319 pub fn from_ordering(c: std::cmp::Ordering, tcx: TyCtxt<'tcx>) -> Self {
320 let ty = tcx.ty_ordering_enum(DUMMY_SP);
322 let layout =
323 tcx.layout_of(ty::TypingEnv::fully_monomorphized().as_query_input(ty)).unwrap();
324 Self::from_scalar(Scalar::Int(c.into()), layout)
325 }
326
327 pub fn from_pair(a: Self, b: Self, cx: &(impl HasTypingEnv<'tcx> + HasTyCtxt<'tcx>)) -> Self {
328 let layout = cx
329 .tcx()
330 .layout_of(
331 cx.typing_env().as_query_input(Ty::new_tup(cx.tcx(), &[a.layout.ty, b.layout.ty])),
332 )
333 .unwrap();
334 Self::from_scalar_pair(a.to_scalar(), b.to_scalar(), layout)
335 }
336
337 #[inline]
340 pub fn to_scalar_int(&self) -> InterpResult<'tcx, ScalarInt> {
341 let s = self.to_scalar().to_scalar_int()?;
342 if s.size() != self.layout.size {
343 do yeet ::rustc_middle::mir::interpret::InterpErrorKind::UndefinedBehavior(::rustc_middle::mir::interpret::UndefinedBehaviorInfo::ScalarSizeMismatch(ScalarSizeMismatch {
target_size: self.layout.size.bytes(),
data_size: s.size().bytes(),
}));throw_ub!(ScalarSizeMismatch(ScalarSizeMismatch {
344 target_size: self.layout.size.bytes(),
345 data_size: s.size().bytes(),
346 }));
347 }
348 interp_ok(s)
349 }
350
351 #[inline]
352 pub fn to_const_int(self) -> ConstInt {
353 if !self.layout.ty.is_integral() {
::core::panicking::panic("assertion failed: self.layout.ty.is_integral()")
};assert!(self.layout.ty.is_integral());
354 let int = self.imm.to_scalar_int();
355 match (&int.size(), &self.layout.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!(int.size(), self.layout.size);
356 ConstInt::new(int, self.layout.ty.is_signed(), self.layout.ty.is_ptr_sized_integral())
357 }
358
359 #[inline]
360 #[cfg_attr(debug_assertions, track_caller)] pub fn to_pair(self, cx: &(impl HasTyCtxt<'tcx> + HasTypingEnv<'tcx>)) -> (Self, Self) {
362 let layout = self.layout;
363 let (val0, val1) = self.to_scalar_pair();
364 (
365 ImmTy::from_scalar(val0, layout.field(cx, 0)),
366 ImmTy::from_scalar(val1, layout.field(cx, 1)),
367 )
368 }
369
370 fn offset_(&self, offset: Size, layout: TyAndLayout<'tcx>, cx: &impl HasDataLayout) -> Self {
374 if truecfg!(debug_assertions) {
376 self.assert_matches_abi(
377 self.layout.backend_repr,
378 "invalid input to Immediate::offset",
379 cx,
380 );
381 }
382 if !(offset + layout.size <= self.layout.size) {
{
::core::panicking::panic_fmt(format_args!("attempting to project to field at offset {0} with size {1} into immediate with layout {2:#?}",
offset.bytes(), layout.size.bytes(), self.layout));
}
};assert!(
386 offset + layout.size <= self.layout.size,
387 "attempting to project to field at offset {} with size {} into immediate with layout {:#?}",
388 offset.bytes(),
389 layout.size.bytes(),
390 self.layout,
391 );
392 let inner_val: Immediate<_> = match (**self, self.layout.backend_repr) {
395 (Immediate::Uninit, _) => Immediate::Uninit,
397 _ if layout.is_uninhabited() => Immediate::Uninit,
401 _ if layout.is_zst() => Immediate::Uninit,
404 _ if #[allow(non_exhaustive_omitted_patterns)] match layout.backend_repr {
BackendRepr::Memory { .. } => true,
_ => false,
}matches!(layout.backend_repr, BackendRepr::Memory { .. })
407 && #[allow(non_exhaustive_omitted_patterns)] match layout.variants {
abi::Variants::Single { .. } => true,
_ => false,
}matches!(layout.variants, abi::Variants::Single { .. })
408 && #[allow(non_exhaustive_omitted_patterns)] match &layout.fields {
abi::FieldsShape::Arbitrary { offsets, .. } if offsets.len() == 0 => true,
_ => false,
}matches!(&layout.fields, abi::FieldsShape::Arbitrary { offsets, .. } if offsets.len() == 0) =>
409 {
410 Immediate::Uninit
411 }
412 _ if layout.size == self.layout.size => {
414 match (&offset.bytes(), &0) {
(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!(offset.bytes(), 0);
415 **self
416 }
417 (Immediate::ScalarPair(a_val, b_val), BackendRepr::ScalarPair(a, b)) => {
419 Immediate::from(if offset.bytes() == 0 {
420 a_val
421 } else {
422 match (&offset, &a.size(cx).align_to(b.align(cx).abi)) {
(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!(offset, a.size(cx).align_to(b.align(cx).abi));
423 b_val
424 })
425 }
426 _ => ::rustc_middle::util::bug::bug_fmt(format_args!("invalid field access on immediate {0} at offset {1}, original layout {2:#?}",
self, offset.bytes(), self.layout))bug!(
428 "invalid field access on immediate {} at offset {}, original layout {:#?}",
429 self,
430 offset.bytes(),
431 self.layout
432 ),
433 };
434 inner_val.assert_matches_abi(
436 layout.backend_repr,
437 "invalid field type in Immediate::offset",
438 cx,
439 );
440
441 ImmTy::from_immediate(inner_val, layout)
442 }
443}
444
445impl<'tcx, Prov: Provenance> Projectable<'tcx, Prov> for ImmTy<'tcx, Prov> {
446 #[inline(always)]
447 fn layout(&self) -> TyAndLayout<'tcx> {
448 self.layout
449 }
450
451 #[inline(always)]
452 fn meta(&self) -> MemPlaceMeta<Prov> {
453 if true {
if !self.layout.is_sized() {
::core::panicking::panic("assertion failed: self.layout.is_sized()")
};
};debug_assert!(self.layout.is_sized()); MemPlaceMeta::None
455 }
456
457 fn offset_with_meta<M: Machine<'tcx, Provenance = Prov>>(
458 &self,
459 offset: Size,
460 _mode: OffsetMode,
461 meta: MemPlaceMeta<Prov>,
462 layout: TyAndLayout<'tcx>,
463 ecx: &InterpCx<'tcx, M>,
464 ) -> InterpResult<'tcx, Self> {
465 {
match meta {
MemPlaceMeta::None => {}
ref left_val => {
::core::panicking::assert_matches_failed(left_val,
"MemPlaceMeta::None", ::core::option::Option::None);
}
}
};assert_matches!(meta, MemPlaceMeta::None); interp_ok(self.offset_(offset, layout, ecx))
467 }
468
469 #[inline(always)]
470 fn to_op<M: Machine<'tcx, Provenance = Prov>>(
471 &self,
472 _ecx: &InterpCx<'tcx, M>,
473 ) -> InterpResult<'tcx, OpTy<'tcx, M::Provenance>> {
474 interp_ok(self.clone().into())
475 }
476}
477
478#[derive(#[automatically_derived]
impl<Prov: ::core::marker::Copy + Provenance> ::core::marker::Copy for
Operand<Prov> {
}Copy, #[automatically_derived]
impl<Prov: ::core::clone::Clone + Provenance> ::core::clone::Clone for
Operand<Prov> {
#[inline]
fn clone(&self) -> Operand<Prov> {
match self {
Operand::Immediate(__self_0) =>
Operand::Immediate(::core::clone::Clone::clone(__self_0)),
Operand::Indirect(__self_0) =>
Operand::Indirect(::core::clone::Clone::clone(__self_0)),
}
}
}Clone, #[automatically_derived]
impl<Prov: ::core::fmt::Debug + Provenance> ::core::fmt::Debug for
Operand<Prov> {
#[inline]
fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
match self {
Operand::Immediate(__self_0) =>
::core::fmt::Formatter::debug_tuple_field1_finish(f,
"Immediate", &__self_0),
Operand::Indirect(__self_0) =>
::core::fmt::Formatter::debug_tuple_field1_finish(f,
"Indirect", &__self_0),
}
}
}Debug)]
482pub(super) enum Operand<Prov: Provenance = CtfeProvenance> {
483 Immediate(Immediate<Prov>),
484 Indirect(MemPlace<Prov>),
485}
486
487#[derive(#[automatically_derived]
impl<'tcx, Prov: ::core::clone::Clone + Provenance> ::core::clone::Clone for
OpTy<'tcx, Prov> {
#[inline]
fn clone(&self) -> OpTy<'tcx, Prov> {
OpTy {
op: ::core::clone::Clone::clone(&self.op),
layout: ::core::clone::Clone::clone(&self.layout),
}
}
}Clone)]
488pub struct OpTy<'tcx, Prov: Provenance = CtfeProvenance> {
489 op: Operand<Prov>, pub layout: TyAndLayout<'tcx>,
491}
492
493impl<Prov: Provenance> std::fmt::Debug for OpTy<'_, Prov> {
494 fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
495 f.debug_struct("OpTy")
497 .field("op", &self.op)
498 .field("ty", &format_args!("{0}", self.layout.ty)format_args!("{}", self.layout.ty))
499 .finish()
500 }
501}
502
503impl<'tcx, Prov: Provenance> From<ImmTy<'tcx, Prov>> for OpTy<'tcx, Prov> {
504 #[inline(always)]
505 fn from(val: ImmTy<'tcx, Prov>) -> Self {
506 OpTy { op: Operand::Immediate(val.imm), layout: val.layout }
507 }
508}
509
510impl<'tcx, Prov: Provenance> From<MPlaceTy<'tcx, Prov>> for OpTy<'tcx, Prov> {
511 #[inline(always)]
512 fn from(mplace: MPlaceTy<'tcx, Prov>) -> Self {
513 OpTy { op: Operand::Indirect(*mplace.mplace()), layout: mplace.layout }
514 }
515}
516
517impl<'tcx, Prov: Provenance> OpTy<'tcx, Prov> {
518 #[inline(always)]
519 pub(super) fn op(&self) -> &Operand<Prov> {
520 &self.op
521 }
522
523 pub fn is_immediate_uninit(&self) -> bool {
524 #[allow(non_exhaustive_omitted_patterns)] match self.op {
Operand::Immediate(Immediate::Uninit) => true,
_ => false,
}matches!(self.op, Operand::Immediate(Immediate::Uninit))
525 }
526}
527
528impl<'tcx, Prov: Provenance> Projectable<'tcx, Prov> for OpTy<'tcx, Prov> {
529 #[inline(always)]
530 fn layout(&self) -> TyAndLayout<'tcx> {
531 self.layout
532 }
533
534 #[inline]
535 fn meta(&self) -> MemPlaceMeta<Prov> {
536 match self.as_mplace_or_imm() {
537 Left(mplace) => mplace.meta(),
538 Right(_) => {
539 if true {
if !self.layout.is_sized() {
{
::core::panicking::panic_fmt(format_args!("unsized immediates are not a thing"));
}
};
};debug_assert!(self.layout.is_sized(), "unsized immediates are not a thing");
540 MemPlaceMeta::None
541 }
542 }
543 }
544
545 fn offset_with_meta<M: Machine<'tcx, Provenance = Prov>>(
546 &self,
547 offset: Size,
548 mode: OffsetMode,
549 meta: MemPlaceMeta<Prov>,
550 layout: TyAndLayout<'tcx>,
551 ecx: &InterpCx<'tcx, M>,
552 ) -> InterpResult<'tcx, Self> {
553 match self.as_mplace_or_imm() {
554 Left(mplace) => {
555 interp_ok(mplace.offset_with_meta(offset, mode, meta, layout, ecx)?.into())
556 }
557 Right(imm) => {
558 {
match meta {
MemPlaceMeta::None => {}
ref left_val => {
::core::panicking::assert_matches_failed(left_val,
"MemPlaceMeta::None", ::core::option::Option::None);
}
}
};assert_matches!(meta, MemPlaceMeta::None); interp_ok(imm.offset_(offset, layout, ecx).into())
561 }
562 }
563 }
564
565 #[inline(always)]
566 fn to_op<M: Machine<'tcx, Provenance = Prov>>(
567 &self,
568 _ecx: &InterpCx<'tcx, M>,
569 ) -> InterpResult<'tcx, OpTy<'tcx, M::Provenance>> {
570 interp_ok(self.clone())
571 }
572}
573
574impl<'tcx, M: Machine<'tcx>> InterpCx<'tcx, M> {
575 fn read_immediate_from_mplace_raw(
580 &self,
581 mplace: &MPlaceTy<'tcx, M::Provenance>,
582 ) -> InterpResult<'tcx, Option<ImmTy<'tcx, M::Provenance>>> {
583 if mplace.layout.is_unsized() {
584 return interp_ok(None);
586 }
587
588 let Some(alloc) = self.get_place_alloc(mplace)? else {
589 return interp_ok(Some(ImmTy::uninit(mplace.layout)));
591 };
592
593 interp_ok(match mplace.layout.backend_repr {
600 BackendRepr::Scalar(abi::Scalar::Initialized { value: s, .. }) => {
601 let size = s.size(self);
602 match (&size, &mplace.layout.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::Some(format_args!("abi::Scalar size does not match layout size")));
}
}
};assert_eq!(size, mplace.layout.size, "abi::Scalar size does not match layout size");
603 let scalar = alloc.read_scalar(
604 alloc_range(Size::ZERO, size),
605 #[allow(non_exhaustive_omitted_patterns)] match s {
abi::Primitive::Pointer(_) => true,
_ => false,
}matches!(s, abi::Primitive::Pointer(_)),
606 )?;
607 Some(ImmTy::from_scalar(scalar, mplace.layout))
608 }
609 BackendRepr::ScalarPair(
610 abi::Scalar::Initialized { value: a, .. },
611 abi::Scalar::Initialized { value: b, .. },
612 ) => {
613 let (a_size, b_size) = (a.size(self), b.size(self));
617 let b_offset = a_size.align_to(b.align(self).abi);
618 if !(b_offset.bytes() > 0) {
::core::panicking::panic("assertion failed: b_offset.bytes() > 0")
};assert!(b_offset.bytes() > 0); let a_val = alloc.read_scalar(
620 alloc_range(Size::ZERO, a_size),
621 #[allow(non_exhaustive_omitted_patterns)] match a {
abi::Primitive::Pointer(_) => true,
_ => false,
}matches!(a, abi::Primitive::Pointer(_)),
622 )?;
623 let b_val = alloc.read_scalar(
624 alloc_range(b_offset, b_size),
625 #[allow(non_exhaustive_omitted_patterns)] match b {
abi::Primitive::Pointer(_) => true,
_ => false,
}matches!(b, abi::Primitive::Pointer(_)),
626 )?;
627 Some(ImmTy::from_immediate(Immediate::ScalarPair(a_val, b_val), mplace.layout))
628 }
629 _ => {
630 None
632 }
633 })
634 }
635
636 pub fn read_immediate_raw(
645 &self,
646 src: &impl Projectable<'tcx, M::Provenance>,
647 ) -> InterpResult<'tcx, Either<MPlaceTy<'tcx, M::Provenance>, ImmTy<'tcx, M::Provenance>>> {
648 interp_ok(match src.to_op(self)?.as_mplace_or_imm() {
649 Left(ref mplace) => {
650 if let Some(val) = self.read_immediate_from_mplace_raw(mplace)? {
651 Right(val)
652 } else {
653 Left(mplace.clone())
654 }
655 }
656 Right(val) => Right(val),
657 })
658 }
659
660 #[inline(always)]
664 pub fn read_immediate(
665 &self,
666 op: &impl Projectable<'tcx, M::Provenance>,
667 ) -> InterpResult<'tcx, ImmTy<'tcx, M::Provenance>> {
668 if !#[allow(non_exhaustive_omitted_patterns)] match op.layout().backend_repr {
BackendRepr::Scalar(abi::Scalar::Initialized { .. }) |
BackendRepr::ScalarPair(abi::Scalar::Initialized { .. },
abi::Scalar::Initialized { .. }) => true,
_ => false,
}matches!(
669 op.layout().backend_repr,
670 BackendRepr::Scalar(abi::Scalar::Initialized { .. })
671 | BackendRepr::ScalarPair(
672 abi::Scalar::Initialized { .. },
673 abi::Scalar::Initialized { .. }
674 )
675 ) {
676 ::rustc_middle::util::bug::span_bug_fmt(self.cur_span(),
format_args!("primitive read not possible for type: {0}",
op.layout().ty));span_bug!(self.cur_span(), "primitive read not possible for type: {}", op.layout().ty);
677 }
678 let imm = self.read_immediate_raw(op)?.right().unwrap();
679 if #[allow(non_exhaustive_omitted_patterns)] match *imm {
Immediate::Uninit => true,
_ => false,
}matches!(*imm, Immediate::Uninit) {
680 do yeet ::rustc_middle::mir::interpret::InterpErrorKind::UndefinedBehavior(::rustc_middle::mir::interpret::UndefinedBehaviorInfo::InvalidUninitBytes(None));throw_ub!(InvalidUninitBytes(None));
681 }
682 interp_ok(imm)
683 }
684
685 pub fn read_scalar(
687 &self,
688 op: &impl Projectable<'tcx, M::Provenance>,
689 ) -> InterpResult<'tcx, Scalar<M::Provenance>> {
690 interp_ok(self.read_immediate(op)?.to_scalar())
691 }
692
693 pub fn read_pointer(
698 &self,
699 op: &impl Projectable<'tcx, M::Provenance>,
700 ) -> InterpResult<'tcx, Pointer<Option<M::Provenance>>> {
701 self.read_scalar(op)?.to_pointer(self)
702 }
703 pub fn read_target_usize(
705 &self,
706 op: &impl Projectable<'tcx, M::Provenance>,
707 ) -> InterpResult<'tcx, u64> {
708 self.read_scalar(op)?.to_target_usize(self)
709 }
710 pub fn read_target_isize(
712 &self,
713 op: &impl Projectable<'tcx, M::Provenance>,
714 ) -> InterpResult<'tcx, i64> {
715 self.read_scalar(op)?.to_target_isize(self)
716 }
717
718 pub fn read_str(&self, mplace: &MPlaceTy<'tcx, M::Provenance>) -> InterpResult<'tcx, &str> {
720 let len = mplace.len(self)?;
721 let bytes = self.read_bytes_ptr_strip_provenance(mplace.ptr(), Size::from_bytes(len))?;
722 let s = std::str::from_utf8(bytes).map_err(|err| ::rustc_middle::mir::interpret::InterpErrorKind::UndefinedBehavior(::rustc_middle::mir::interpret::UndefinedBehaviorInfo::InvalidStr(err))err_ub!(InvalidStr(err)))?;
723 interp_ok(s)
724 }
725
726 pub fn local_to_op(
728 &self,
729 local: mir::Local,
730 layout: Option<TyAndLayout<'tcx>>,
731 ) -> InterpResult<'tcx, OpTy<'tcx, M::Provenance>> {
732 self.local_at_frame_to_op(self.frame(), local, layout)
733 }
734
735 pub fn local_at_frame_to_op(
741 &self,
742 frame: &Frame<'tcx, M::Provenance, M::FrameExtra>,
743 local: mir::Local,
744 layout: Option<TyAndLayout<'tcx>>,
745 ) -> InterpResult<'tcx, OpTy<'tcx, M::Provenance>> {
746 let layout = self.layout_of_local(frame, local, layout)?;
747 let op = *frame.locals[local].access()?;
748 if #[allow(non_exhaustive_omitted_patterns)] match op {
Operand::Immediate(_) => true,
_ => false,
}matches!(op, Operand::Immediate(_)) {
749 if !!layout.is_unsized() {
::core::panicking::panic("assertion failed: !layout.is_unsized()")
};assert!(!layout.is_unsized());
750 }
751 M::after_local_read(self, frame, local)?;
752 interp_ok(OpTy { op, layout })
753 }
754
755 pub fn place_to_op(
759 &self,
760 place: &PlaceTy<'tcx, M::Provenance>,
761 ) -> InterpResult<'tcx, OpTy<'tcx, M::Provenance>> {
762 match place.as_mplace_or_local() {
763 Left(mplace) => interp_ok(mplace.into()),
764 Right((local, offset, locals_addr, _)) => {
765 if true {
if !place.layout.is_sized() {
::core::panicking::panic("assertion failed: place.layout.is_sized()")
};
};debug_assert!(place.layout.is_sized()); if true {
match (&locals_addr, &self.frame().locals_addr()) {
(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);
}
}
};
};debug_assert_eq!(locals_addr, self.frame().locals_addr());
767 let base = self.local_to_op(local, None)?;
768 interp_ok(match offset {
769 Some(offset) => base.offset(offset, place.layout, self)?,
770 None => {
771 if true {
match (&place.layout, &base.layout) {
(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);
}
}
};
};debug_assert_eq!(place.layout, base.layout);
773 base
774 }
775 })
776 }
777 }
778 }
779
780 pub fn eval_place_to_op(
783 &self,
784 mir_place: mir::Place<'tcx>,
785 layout: Option<TyAndLayout<'tcx>>,
786 ) -> InterpResult<'tcx, OpTy<'tcx, M::Provenance>> {
787 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("step",
"rustc_const_eval::interpret::operand",
::tracing::Level::INFO,
::tracing_core::__macro_support::Option::Some("compiler/rustc_const_eval/src/interpret/operand.rs"),
::tracing_core::__macro_support::Option::Some(787u32),
::tracing_core::__macro_support::Option::Some("rustc_const_eval::interpret::operand"),
::tracing_core::field::FieldSet::new(&["step", "mir_place",
"tracing_separate_thread"],
::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(&"eval_place_to_op")
as &dyn Value)),
(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&debug(&mir_place)
as &dyn Value)),
(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&Empty as
&dyn Value))])
})
} else {
let span =
::tracing::__macro_support::__disabled_span(__CALLSITE.metadata());
{};
span
}
})enter_trace_span!(
788 M,
789 step::eval_place_to_op,
790 ?mir_place,
791 tracing_separate_thread = Empty
792 );
793
794 let layout = if mir_place.projection.is_empty() { layout } else { None };
797
798 let mut op = self.local_to_op(mir_place.local, layout)?;
799 for elem in mir_place.projection.iter() {
801 op = self.project(&op, elem)?
802 }
803
804 {
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/operand.rs:804",
"rustc_const_eval::interpret::operand",
::tracing::Level::TRACE,
::tracing_core::__macro_support::Option::Some("compiler/rustc_const_eval/src/interpret/operand.rs"),
::tracing_core::__macro_support::Option::Some(804u32),
::tracing_core::__macro_support::Option::Some("rustc_const_eval::interpret::operand"),
::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!("eval_place_to_op: got {0:?}",
op) as &dyn Value))])
});
} else { ; }
};trace!("eval_place_to_op: got {:?}", op);
805 if truecfg!(debug_assertions) {
807 let normalized_place_ty = self
808 .instantiate_from_current_frame_and_normalize_erasing_regions(
809 mir_place.ty(&self.frame().body.local_decls, *self.tcx).ty,
810 )?;
811 if !mir_assign_valid_types(
812 *self.tcx,
813 self.typing_env(),
814 self.layout_of(normalized_place_ty)?,
815 op.layout,
816 ) {
817 ::rustc_middle::util::bug::span_bug_fmt(self.cur_span(),
format_args!("eval_place of a MIR place with type {0} produced an interpreter operand with type {1}",
normalized_place_ty, op.layout.ty))span_bug!(
818 self.cur_span(),
819 "eval_place of a MIR place with type {} produced an interpreter operand with type {}",
820 normalized_place_ty,
821 op.layout.ty,
822 )
823 }
824 }
825 interp_ok(op)
826 }
827
828 #[inline]
832 pub fn eval_operand(
833 &self,
834 mir_op: &mir::Operand<'tcx>,
835 layout: Option<TyAndLayout<'tcx>>,
836 ) -> InterpResult<'tcx, OpTy<'tcx, M::Provenance>> {
837 let _trace =
838 <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("step",
"rustc_const_eval::interpret::operand",
::tracing::Level::INFO,
::tracing_core::__macro_support::Option::Some("compiler/rustc_const_eval/src/interpret/operand.rs"),
::tracing_core::__macro_support::Option::Some(838u32),
::tracing_core::__macro_support::Option::Some("rustc_const_eval::interpret::operand"),
::tracing_core::field::FieldSet::new(&["step", "mir_op",
"tracing_separate_thread"],
::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(&"eval_operand")
as &dyn Value)),
(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&debug(&mir_op) as
&dyn Value)),
(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&Empty as
&dyn Value))])
})
} else {
let span =
::tracing::__macro_support::__disabled_span(__CALLSITE.metadata());
{};
span
}
})enter_trace_span!(M, step::eval_operand, ?mir_op, tracing_separate_thread = Empty);
839
840 use rustc_middle::mir::Operand::*;
841 let op = match mir_op {
842 &Copy(place) | &Move(place) => self.eval_place_to_op(place, layout)?,
844
845 &RuntimeChecks(checks) => {
846 let val = M::runtime_checks(self, checks)?;
847 ImmTy::from_bool(val, self.tcx()).into()
848 }
849
850 Constant(constant) => {
851 let c = self.instantiate_from_current_frame_and_normalize_erasing_regions(
852 constant.const_,
853 )?;
854
855 self.eval_mir_constant(&c, constant.span, layout)?
860 }
861 };
862 {
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/operand.rs:862",
"rustc_const_eval::interpret::operand",
::tracing::Level::TRACE,
::tracing_core::__macro_support::Option::Some("compiler/rustc_const_eval/src/interpret/operand.rs"),
::tracing_core::__macro_support::Option::Some(862u32),
::tracing_core::__macro_support::Option::Some("rustc_const_eval::interpret::operand"),
::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!("{0:?}: {1:?}",
mir_op, op) as &dyn Value))])
});
} else { ; }
};trace!("{:?}: {:?}", mir_op, op);
863 interp_ok(op)
864 }
865
866 pub(crate) fn const_val_to_op(
867 &self,
868 val_val: mir::ConstValue,
869 ty: Ty<'tcx>,
870 layout: Option<TyAndLayout<'tcx>>,
871 ) -> InterpResult<'tcx, OpTy<'tcx, M::Provenance>> {
872 let adjust_scalar = |scalar| -> InterpResult<'tcx, _> {
874 interp_ok(match scalar {
875 Scalar::Ptr(ptr, size) => Scalar::Ptr(self.global_root_pointer(ptr)?, size),
876 Scalar::Int(int) => Scalar::Int(int),
877 })
878 };
879 let layout =
880 from_known_layout(self.tcx, self.typing_env(), layout, || self.layout_of(ty).into())?;
881 let imm = match val_val {
882 mir::ConstValue::Indirect { alloc_id, offset } => {
883 let ptr = self.global_root_pointer(Pointer::new(
885 CtfeProvenance::from(alloc_id).as_immutable(),
886 offset,
887 ))?;
888 return interp_ok(self.ptr_to_mplace(ptr.into(), layout).into());
889 }
890 mir::ConstValue::Scalar(x) => adjust_scalar(x)?.into(),
891 mir::ConstValue::ZeroSized => Immediate::Uninit,
892 mir::ConstValue::Slice { alloc_id, meta } => {
893 let ptr = Pointer::new(CtfeProvenance::from(alloc_id).as_immutable(), Size::ZERO);
895 Immediate::new_slice(self.global_root_pointer(ptr)?.into(), meta, self)
896 }
897 };
898 interp_ok(OpTy { op: Operand::Immediate(imm), layout })
899 }
900}
901
902#[cfg(target_pointer_width = "64")]
904mod size_asserts {
905 use rustc_data_structures::static_assert_size;
906
907 use super::*;
908 const _: [(); 64] = [(); ::std::mem::size_of::<ImmTy<'_>>()];static_assert_size!(ImmTy<'_>, 64);
910 const _: [(); 48] = [(); ::std::mem::size_of::<Immediate>()];static_assert_size!(Immediate, 48);
911 const _: [(); 72] = [(); ::std::mem::size_of::<OpTy<'_>>()];static_assert_size!(OpTy<'_>, 72);
912 const _: [(); 56] = [(); ::std::mem::size_of::<Operand>()];static_assert_size!(Operand, 56);
913 }