1//! Manages calling a concrete function (with known MIR body) with argument passing,
2//! and returning the return value to the caller.
34use std::assert_matches;
5use std::borrow::Cow;
67use either::{Left, Right};
8use rustc_abi::{selfas abi, ExternAbi, FieldIdx, Integer, VariantIdx};
9use rustc_hir::def_id::DefId;
10use rustc_hir::find_attr;
11use rustc_middle::ty::layout::{IntegerExt, TyAndLayout};
12use rustc_middle::ty::{self, AdtDef, Instance, Ty, VariantDef};
13use rustc_middle::{bug, mir, span_bug};
14use rustc_target::callconv::{ArgAbi, FnAbi};
15use tracing::field::Empty;
16use tracing::{info, instrument, trace};
1718use super::{
19CtfeProvenance, EnteredTraceSpan, FnVal, ImmTy, InterpCx, InterpResult, MPlaceTy, Machine,
20OpTy, PlaceTy, Projectable, Provenance, RetagMode, ReturnAction, ReturnContinuation, Scalar,
21interp_ok, throw_ub, throw_ub_format,
22};
23use crate::enter_trace_span;
2425/// An argument passed to a function.
26#[derive(#[automatically_derived]
impl<'tcx, Prov: ::core::clone::Clone + Provenance> ::core::clone::Clone for
FnArg<'tcx, Prov> {
#[inline]
fn clone(&self) -> FnArg<'tcx, Prov> {
match self {
FnArg::Copy(__self_0) =>
FnArg::Copy(::core::clone::Clone::clone(__self_0)),
FnArg::InPlace(__self_0) =>
FnArg::InPlace(::core::clone::Clone::clone(__self_0)),
}
}
}Clone, #[automatically_derived]
impl<'tcx, Prov: ::core::fmt::Debug + Provenance> ::core::fmt::Debug for
FnArg<'tcx, Prov> {
#[inline]
fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
match self {
FnArg::Copy(__self_0) =>
::core::fmt::Formatter::debug_tuple_field1_finish(f, "Copy",
&__self_0),
FnArg::InPlace(__self_0) =>
::core::fmt::Formatter::debug_tuple_field1_finish(f,
"InPlace", &__self_0),
}
}
}Debug)]
27pub enum FnArg<'tcx, Prov: Provenance = CtfeProvenance> {
28/// Pass a copy of the given operand.
29Copy(OpTy<'tcx, Prov>),
30/// Allow for the argument to be passed in-place: destroy the value originally stored at that
31 /// place and make the place inaccessible for the duration of the function call. This *must* be
32 /// an in-memory place so that we can do the proper alias checks.
33InPlace(MPlaceTy<'tcx, Prov>),
34}
3536impl<'tcx, Prov: Provenance> FnArg<'tcx, Prov> {
37pub fn layout(&self) -> &TyAndLayout<'tcx> {
38match self {
39 FnArg::Copy(op) => &op.layout,
40 FnArg::InPlace(mplace) => &mplace.layout,
41 }
42 }
4344/// Make a copy of the given fn_arg. Any `InPlace` are degenerated to copies, no protection of the
45 /// original memory occurs.
46pub fn copy_fn_arg(&self) -> OpTy<'tcx, Prov> {
47match self {
48 FnArg::Copy(op) => op.clone(),
49 FnArg::InPlace(mplace) => mplace.clone().into(),
50 }
51 }
52}
5354impl<'tcx, M: Machine<'tcx>> InterpCx<'tcx, M> {
55/// Make a copy of the given fn_args. Any `InPlace` are degenerated to copies, no protection of the
56 /// original memory occurs.
57pub fn copy_fn_args(args: &[FnArg<'tcx, M::Provenance>]) -> Vec<OpTy<'tcx, M::Provenance>> {
58args.iter().map(|fn_arg| fn_arg.copy_fn_arg()).collect()
59 }
6061/// Helper function for argument untupling.
62fn fn_arg_project_field(
63&self,
64 arg: &FnArg<'tcx, M::Provenance>,
65 field: FieldIdx,
66 ) -> InterpResult<'tcx, FnArg<'tcx, M::Provenance>> {
67interp_ok(match arg {
68 FnArg::Copy(op) => FnArg::Copy(self.project_field(op, field)?),
69 FnArg::InPlace(mplace) => FnArg::InPlace(self.project_field(mplace, field)?),
70 })
71 }
7273/// Find the wrapped inner type of a transparent wrapper.
74 /// Must not be called on 1-ZST (as they don't have a uniquely defined "wrapped field").
75 ///
76 /// We work with `TyAndLayout` here since that makes it much easier to iterate over all fields.
77fn unfold_transparent(
78&self,
79 layout: TyAndLayout<'tcx>,
80 may_unfold: impl Fn(AdtDef<'tcx>) -> bool,
81 ) -> TyAndLayout<'tcx> {
82match layout.ty.kind() {
83 ty::Adt(adt_def, _) if adt_def.repr().transparent() && may_unfold(*adt_def) => {
84{
match layout.variants {
rustc_abi::Variants::Single { .. } => {}
ref left_val => {
::core::panicking::assert_matches_failed(left_val,
"rustc_abi::Variants::Single { .. }",
::core::option::Option::None);
}
}
};assert_matches!(layout.variants, rustc_abi::Variants::Single { .. });
85// Find the non-1-ZST field, and recurse.
86let (_, field) = layout.non_1zst_field(self).unwrap();
87self.unfold_transparent(field, may_unfold)
88 }
89 ty::Pat(base, _) => self.layout_of(*base).expect(
90"if the layout of a pattern type could be computed, so can the layout of its base",
91 ),
92// Not a transparent type, no further unfolding.
93_ => layout,
94 }
95 }
9697/// Unwrap types that are guaranteed a null-pointer-optimization
98fn unfold_npo(&self, layout: TyAndLayout<'tcx>) -> InterpResult<'tcx, TyAndLayout<'tcx>> {
99// Check if this is an option-like type wrapping some type.
100let ty::Adt(def, args) = layout.ty.kind() else {
101// Not an ADT, so definitely no NPO.
102return interp_ok(layout);
103 };
104if def.variants().len() != 2 {
105// Not a 2-variant enum, so no NPO.
106return interp_ok(layout);
107 }
108if !def.is_enum() {
::core::panicking::panic("assertion failed: def.is_enum()")
};assert!(def.is_enum());
109110let all_fields_1zst = |variant: &VariantDef| -> InterpResult<'tcx, _> {
111for field in &variant.fields {
112let ty = field.ty(*self.tcx, args).skip_norm_wip();
113let layout = self.layout_of(ty)?;
114if !layout.is_1zst() {
115return interp_ok(false);
116 }
117 }
118interp_ok(true)
119 };
120121// If one variant consists entirely of 1-ZST, then the other variant
122 // is the only "relevant" one for this check.
123let var0 = VariantIdx::from_u32(0);
124let var1 = VariantIdx::from_u32(1);
125let relevant_variant = if all_fields_1zst(def.variant(var0))? {
126def.variant(var1)
127 } else if all_fields_1zst(def.variant(var1))? {
128def.variant(var0)
129 } else {
130// No variant is all-1-ZST, so no NPO.
131return interp_ok(layout);
132 };
133// The "relevant" variant must have exactly one field, and its type is the "inner" type.
134if relevant_variant.fields.len() != 1 {
135return interp_ok(layout);
136 }
137let inner =
138relevant_variant.fields[FieldIdx::from_u32(0)].ty(*self.tcx, args).skip_norm_wip();
139let inner = self.layout_of(inner)?;
140141// Check if the inner type is one of the NPO-guaranteed ones.
142 // For that we first unpeel transparent *structs* (but not unions).
143let is_npo =
144 |def: AdtDef<'tcx>| {
{
'done:
{
for i in
::rustc_hir::attrs::HasAttrs::get_attrs(def.did(),
&self.tcx) {
#[allow(unused_imports)]
use rustc_hir::attrs::AttributeKind::*;
let i: &rustc_hir::Attribute = i;
match i {
rustc_hir::Attribute::Parsed(RustcNonnullOptimizationGuaranteed)
=> {
break 'done Some(());
}
rustc_hir::Attribute::Unparsed(..) =>
{}
#[deny(unreachable_patterns)]
_ => {}
}
}
None
}
}
}.is_some()find_attr!(self.tcx, def.did(), RustcNonnullOptimizationGuaranteed);
145let inner = self.unfold_transparent(inner, /* may_unfold */ |def| {
146// Stop at NPO types so that we don't miss that attribute in the check below!
147def.is_struct() && !is_npo(def)
148 });
149interp_ok(match inner.ty.kind() {
150 ty::Ref(..) | ty::FnPtr(..) => {
151// Option<&T> behaves like &T, and same for fn()
152inner153 }
154 ty::Adt(def, _) if is_npo(*def) => {
155// Once we found a `nonnull_optimization_guaranteed` type, further strip off
156 // newtype structs from it to find the underlying ABI type.
157self.unfold_transparent(inner, /* may_unfold */ |def| def.is_struct())
158 }
159_ => {
160// Everything else we do not unfold.
161layout162 }
163 })
164 }
165166/// Check if these two layouts look like they are fn-ABI-compatible.
167 /// (We also compare the `PassMode`, so this doesn't have to check everything. But it turns out
168 /// that only checking the `PassMode` is insufficient.)
169fn layout_compat(
170&self,
171 caller: TyAndLayout<'tcx>,
172 callee: TyAndLayout<'tcx>,
173 ) -> InterpResult<'tcx, bool> {
174// Fast path: equal types are definitely compatible.
175if caller.ty == callee.ty {
176return interp_ok(true);
177 }
178// 1-ZST are compatible with all 1-ZST (and with nothing else).
179if caller.is_1zst() || callee.is_1zst() {
180return interp_ok(caller.is_1zst() && callee.is_1zst());
181 }
182// Unfold newtypes and NPO optimizations.
183let unfold = |layout: TyAndLayout<'tcx>| {
184self.unfold_npo(self.unfold_transparent(layout, /* may_unfold */ |_def| true))
185 };
186let caller = unfold(caller)?;
187let callee = unfold(callee)?;
188// Now see if these inner types are compatible.
189190 // Compatible pointer types. For thin pointers, we have to accept even non-`repr(transparent)`
191 // things as compatible due to `DispatchFromDyn`. For instance, `Rc<i32>` and `*mut i32`
192 // must be compatible. So we just accept everything with Pointer ABI as compatible,
193 // even if this will accept some code that is not stably guaranteed to work.
194 // This also handles function pointers.
195let thin_pointer = |layout: TyAndLayout<'tcx>| match layout.backend_repr {
196 abi::BackendRepr::Scalar(s) => match s.primitive() {
197 abi::Primitive::Pointer(addr_space) => Some(addr_space),
198_ => None,
199 },
200_ => None,
201 };
202if let (Some(caller), Some(callee)) = (thin_pointer(caller), thin_pointer(callee)) {
203return interp_ok(caller == callee);
204 }
205// For wide pointers we have to get the pointee type.
206let pointee_ty = |ty: Ty<'tcx>| -> InterpResult<'tcx, Option<Ty<'tcx>>> {
207// We cannot use `builtin_deref` here since we need to reject `Box<T, MyAlloc>`.
208interp_ok(Some(match ty.kind() {
209 ty::Ref(_, ty, _) => *ty,
210 ty::RawPtr(ty, _) => *ty,
211// We only accept `Box` with the default allocator.
212_ if ty.is_box_global(*self.tcx) => ty.expect_boxed_ty(),
213_ => return interp_ok(None),
214 }))
215 };
216if let (Some(caller), Some(callee)) = (pointee_ty(caller.ty)?, pointee_ty(callee.ty)?) {
217// This is okay if they have the same metadata type.
218let meta_ty = |ty: Ty<'tcx>| {
219// Even if `ty` is normalized, the search for the unsized tail will project
220 // to fields, which can yield non-normalized types. So we need to provide a
221 // normalization function.
222let normalize = |ty| self.tcx.normalize_erasing_regions(self.typing_env, ty);
223ty.ptr_metadata_ty(*self.tcx, normalize)
224 };
225return interp_ok(meta_ty(caller) == meta_ty(callee));
226 }
227228// Compatible integer types (in particular, usize vs ptr-sized-u32/u64).
229 // `char` counts as `u32.`
230let int_ty = |ty: Ty<'tcx>| {
231Some(match ty.kind() {
232 ty::Int(ity) => (Integer::from_int_ty(&self.tcx, *ity), /* signed */ true),
233 ty::Uint(uty) => (Integer::from_uint_ty(&self.tcx, *uty), /* signed */ false),
234 ty::Char => (Integer::I32, /* signed */ false),
235_ => return None,
236 })
237 };
238if let (Some(caller), Some(callee)) = (int_ty(caller.ty), int_ty(callee.ty)) {
239// This is okay if they are the same integer type.
240return interp_ok(caller == callee);
241 }
242243// Fall back to exact equality.
244interp_ok(caller == callee)
245 }
246247/// Returns a `bool` saying whether the two arguments are ABI-compatible.
248pub fn check_argument_compat(
249&self,
250 caller_abi: &ArgAbi<'tcx, Ty<'tcx>>,
251 callee_abi: &ArgAbi<'tcx, Ty<'tcx>>,
252 ) -> InterpResult<'tcx, bool> {
253// We do not want to accept things as ABI-compatible that just "happen to be" compatible on the current target,
254 // so we implement a type-based check that reflects the guaranteed rules for ABI compatibility.
255if self.layout_compat(caller_abi.layout, callee_abi.layout)? {
256// Ensure that our checks imply actual ABI compatibility for this concrete call.
257 // (This can fail e.g. if `#[rustc_nonnull_optimization_guaranteed]` is used incorrectly.)
258if !caller_abi.eq_abi(callee_abi) {
::core::panicking::panic("assertion failed: caller_abi.eq_abi(callee_abi)")
};assert!(caller_abi.eq_abi(callee_abi));
259interp_ok(true)
260 } else {
261{
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/call.rs:261",
"rustc_const_eval::interpret::call",
::tracing::Level::TRACE,
::tracing_core::__macro_support::Option::Some("compiler/rustc_const_eval/src/interpret/call.rs"),
::tracing_core::__macro_support::Option::Some(261u32),
::tracing_core::__macro_support::Option::Some("rustc_const_eval::interpret::call"),
::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!("check_argument_compat: incompatible ABIs:\ncaller: {0:?}\ncallee: {1:?}",
caller_abi, callee_abi) as &dyn Value))])
});
} else { ; }
};trace!(
262"check_argument_compat: incompatible ABIs:\ncaller: {:?}\ncallee: {:?}",
263 caller_abi, callee_abi
264 );
265interp_ok(false)
266 }
267 }
268269/// Initialize a single callee argument, checking the types for compatibility.
270fn pass_argument<'x, 'y>(
271&mut self,
272 caller_args: &mut impl Iterator<
273 Item = (&'x FnArg<'tcx, M::Provenance>, &'y ArgAbi<'tcx, Ty<'tcx>>),
274 >,
275 callee_args_abis: &mut impl Iterator<Item = (usize, &'y ArgAbi<'tcx, Ty<'tcx>>)>,
276 callee_arg: &mir::Place<'tcx>,
277 callee_ty: Ty<'tcx>,
278 already_live: bool,
279 ) -> InterpResult<'tcx>
280where
281'tcx: 'x,
282'tcx: 'y,
283 {
284// Get next callee arg.
285let (callee_arg_idx, callee_abi) = callee_args_abis.next().unwrap();
286{
match (&callee_ty, &callee_abi.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!(callee_ty, callee_abi.layout.ty);
287// Get next caller arg.
288let Some((caller_arg, caller_abi)) = caller_args.next() else {
289do yeet ::rustc_middle::mir::interpret::InterpErrorKind::UndefinedBehavior(::rustc_middle::mir::interpret::UndefinedBehaviorInfo::Ub(::alloc::__export::must_use({
::alloc::fmt::format(format_args!("calling a function with fewer arguments than it requires"))
})));throw_ub_format!("calling a function with fewer arguments than it requires");
290 };
291{
match (&caller_arg.layout().layout, &caller_abi.layout.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);
}
}
}
};assert_eq!(caller_arg.layout().layout, caller_abi.layout.layout);
292// Sadly we cannot assert that `caller_arg.layout().ty` and `caller_abi.layout.ty` are
293 // equal; in closures the types sometimes differ. We just hope that `caller_abi` is the
294 // right type to print to the user.
295296 // Check compatibility
297if !self.check_argument_compat(caller_abi, callee_abi)? {
298do yeet ::rustc_middle::mir::interpret::InterpErrorKind::UndefinedBehavior(::rustc_middle::mir::interpret::UndefinedBehaviorInfo::AbiMismatchArgument {
arg_idx: callee_arg_idx,
caller_ty: caller_abi.layout.ty,
callee_ty: callee_abi.layout.ty,
});throw_ub!(AbiMismatchArgument {
299 arg_idx: callee_arg_idx,
300 caller_ty: caller_abi.layout.ty,
301 callee_ty: callee_abi.layout.ty
302 });
303 }
304// We work with a copy of the argument for now; if this is in-place argument passing, we
305 // will later protect the source it comes from. This means the callee cannot observe if we
306 // did in-place of by-copy argument passing, except for pointer equality tests.
307let caller_arg_copy = caller_arg.copy_fn_arg();
308if !already_live {
309let local = callee_arg.as_local().unwrap();
310let meta = caller_arg_copy.meta();
311// `check_argument_compat` ensures that if metadata is needed, both have the same type,
312 // so we know they will use the metadata the same way.
313if !(!meta.has_meta() || caller_arg_copy.layout.ty == callee_ty) {
::core::panicking::panic("assertion failed: !meta.has_meta() || caller_arg_copy.layout.ty == callee_ty")
};assert!(!meta.has_meta() || caller_arg_copy.layout.ty == callee_ty);
314315self.storage_live_dyn(local, meta)?;
316 }
317// Now we can finally actually evaluate the callee place.
318let callee_arg = self.eval_place(*callee_arg)?;
319// We allow some transmutes here.
320 // FIXME: Depending on the PassMode, this should reset some padding to uninitialized. (This
321 // is true for all `copy_op`, but there are a lot of special cases for argument passing
322 // specifically.)
323self.copy_op_allow_transmute(&caller_arg_copy, &callee_arg)?;
324// If this was an in-place pass, protect the place it comes from for the duration of the call.
325if let FnArg::InPlace(mplace) = caller_arg {
326 M::protect_in_place_function_argument(self, mplace)?;
327 }
328interp_ok(())
329 }
330331/// The main entry point for creating a new stack frame: performs ABI checks and initializes
332 /// arguments.
333#[allow(clippy :: suspicious_else_formatting)]
{
let __tracing_attr_span;
let __tracing_attr_guard;
if ::tracing::Level::TRACE <= ::tracing::level_filters::STATIC_MAX_LEVEL
&&
::tracing::Level::TRACE <=
::tracing::level_filters::LevelFilter::current() ||
{ false } {
__tracing_attr_span =
{
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("init_stack_frame",
"rustc_const_eval::interpret::call",
::tracing::Level::TRACE,
::tracing_core::__macro_support::Option::Some("compiler/rustc_const_eval/src/interpret/call.rs"),
::tracing_core::__macro_support::Option::Some(333u32),
::tracing_core::__macro_support::Option::Some("rustc_const_eval::interpret::call"),
::tracing_core::field::FieldSet::new(&["instance", "body",
"caller_fn_abi", "args", "with_caller_location",
"destination", "cont"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::SPAN)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let mut interest = ::tracing::subscriber::Interest::never();
if ::tracing::Level::TRACE <=
::tracing::level_filters::STATIC_MAX_LEVEL &&
::tracing::Level::TRACE <=
::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(&::tracing::field::debug(&instance)
as &dyn Value)),
(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&::tracing::field::debug(&body)
as &dyn Value)),
(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&::tracing::field::debug(&caller_fn_abi)
as &dyn Value)),
(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&::tracing::field::debug(&args)
as &dyn Value)),
(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&with_caller_location
as &dyn Value)),
(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&::tracing::field::debug(&destination)
as &dyn Value)),
(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&::tracing::field::debug(&cont)
as &dyn Value))])
})
} else {
let span =
::tracing::__macro_support::__disabled_span(__CALLSITE.metadata());
{};
span
}
};
__tracing_attr_guard = __tracing_attr_span.enter();
}
#[warn(clippy :: suspicious_else_formatting)]
{
#[allow(unknown_lints, unreachable_code, clippy ::
diverging_sub_expression, clippy :: empty_loop, clippy ::
let_unit_value, clippy :: let_with_type_underscore, clippy ::
needless_return, clippy :: unreachable)]
if false {
let __tracing_attr_fake_return: InterpResult<'tcx> = loop {};
return __tracing_attr_fake_return;
}
{
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::call", ::tracing::Level::INFO,
::tracing_core::__macro_support::Option::Some("compiler/rustc_const_eval/src/interpret/call.rs"),
::tracing_core::__macro_support::Option::Some(344u32),
::tracing_core::__macro_support::Option::Some("rustc_const_eval::interpret::call"),
::tracing_core::field::FieldSet::new(&["step", "instance",
"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(&"init_stack_frame")
as &dyn Value)),
(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&display(&instance)
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
}
});
let def_id = instance.def_id();
let extra_tys =
if caller_fn_abi.c_variadic {
let fixed_count =
usize::try_from(caller_fn_abi.fixed_count).unwrap();
let extra_tys =
args[fixed_count..].iter().map(|arg| arg.layout().ty);
self.tcx.mk_type_list_from_iter(extra_tys)
} else { ty::List::empty() };
let callee_fn_abi =
self.fn_abi_of_instance_no_deduced_attrs(instance,
extra_tys)?;
if caller_fn_abi.conv != callee_fn_abi.conv {
do yeet ::rustc_middle::mir::interpret::InterpErrorKind::UndefinedBehavior(::rustc_middle::mir::interpret::UndefinedBehaviorInfo::Ub(::alloc::__export::must_use({
::alloc::fmt::format(format_args!("calling a function with calling convention \"{0}\" using calling convention \"{1}\"",
callee_fn_abi.conv, caller_fn_abi.conv))
})))
}
if caller_fn_abi.c_variadic != callee_fn_abi.c_variadic {
do yeet ::rustc_middle::mir::interpret::InterpErrorKind::UndefinedBehavior(::rustc_middle::mir::interpret::UndefinedBehaviorInfo::CVariadicMismatch {
caller_is_c_variadic: caller_fn_abi.c_variadic,
callee_is_c_variadic: callee_fn_abi.c_variadic,
});
}
if caller_fn_abi.c_variadic &&
caller_fn_abi.fixed_count != callee_fn_abi.fixed_count {
do yeet ::rustc_middle::mir::interpret::InterpErrorKind::UndefinedBehavior(::rustc_middle::mir::interpret::UndefinedBehaviorInfo::CVariadicFixedCountMismatch {
caller: caller_fn_abi.fixed_count,
callee: callee_fn_abi.fixed_count,
});
}
M::check_fn_target_features(self, instance)?;
if !callee_fn_abi.can_unwind {
match &mut cont {
ReturnContinuation::Stop { .. } => {}
ReturnContinuation::Goto { unwind, .. } => {
*unwind = mir::UnwindAction::Unreachable;
}
}
}
let destination_mplace =
self.place_to_op(destination)?.as_mplace_or_imm().left();
self.push_stack_frame_raw(instance, body, destination, cont)?;
let preamble_span = self.frame().loc.unwrap_right();
{
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/call.rs:404",
"rustc_const_eval::interpret::call",
::tracing::Level::TRACE,
::tracing_core::__macro_support::Option::Some("compiler/rustc_const_eval/src/interpret/call.rs"),
::tracing_core::__macro_support::Option::Some(404u32),
::tracing_core::__macro_support::Option::Some("rustc_const_eval::interpret::call"),
::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!("caller ABI: {0:#?}, args: {1:#?}",
caller_fn_abi,
args.iter().map(|arg|
(arg.layout().ty,
match arg {
FnArg::Copy(op) =>
::alloc::__export::must_use({
::alloc::fmt::format(format_args!("copy({0:?})", op))
}),
FnArg::InPlace(mplace) =>
::alloc::__export::must_use({
::alloc::fmt::format(format_args!("in-place({0:?})",
mplace))
}),
})).collect::<Vec<_>>()) as &dyn Value))])
});
} else { ; }
};
{
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/call.rs:417",
"rustc_const_eval::interpret::call",
::tracing::Level::TRACE,
::tracing_core::__macro_support::Option::Some("compiler/rustc_const_eval/src/interpret/call.rs"),
::tracing_core::__macro_support::Option::Some(417u32),
::tracing_core::__macro_support::Option::Some("rustc_const_eval::interpret::call"),
::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!("spread_arg: {0:?}, locals: {1:#?}",
body.spread_arg,
body.args_iter().map(|local|
(local,
self.layout_of_local(self.frame(), local,
None).unwrap().ty)).collect::<Vec<_>>()) as &dyn Value))])
});
} else { ; }
};
let va_list_arg =
callee_fn_abi.c_variadic.then(||
mir::Local::from_usize(body.arg_count));
let is_non_capturing_closure =
(#[allow(non_exhaustive_omitted_patterns)] match instance.def
{
ty::InstanceKind::Shim(ty::ShimKind::ClosureOnce { .. }) =>
true,
_ => false,
} || self.tcx.is_closure_like(def_id)) &&
{
let arg = &callee_fn_abi.args[0];
#[allow(non_exhaustive_omitted_patterns)]
match arg.layout.ty.kind() {
ty::Closure(_def, closure_args) if
{ closure_args.as_closure().upvar_tys().is_empty() } =>
true,
_ => false,
}
};
{
match (&(args.len() +
if with_caller_location { 1 } else { 0 }),
&caller_fn_abi.args.len()) {
(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!("mismatch between caller ABI and caller arguments")));
}
}
}
};
let mut caller_args = args.iter().zip(caller_fn_abi.args.iter());
let mut callee_args_abis = callee_fn_abi.args.iter().enumerate();
M::with_retag_mode(self, RetagMode::FnEntry,
|ecx|
{
for local in body.args_iter() {
ecx.frame_mut().loc =
Right(body.local_decls[local].source_info.span);
let dest = mir::Place::from(local);
let ty = ecx.layout_of_local(ecx.frame(), local, None)?.ty;
if is_non_capturing_closure && local == mir::Local::arg(0) {
if !va_list_arg.is_none() {
::core::panicking::panic("assertion failed: va_list_arg.is_none()")
};
if !(Some(local) != body.spread_arg) {
::core::panicking::panic("assertion failed: Some(local) != body.spread_arg")
};
let (callee_arg_idx, callee_abi) =
callee_args_abis.next().unwrap();
if !(callee_abi.layout.is_1zst() && callee_abi.is_ignore())
{
::core::panicking::panic("assertion failed: callee_abi.layout.is_1zst() && callee_abi.is_ignore()")
};
ecx.storage_live(local)?;
if caller_fn_abi.args.len() == callee_fn_abi.args.len() {
let (_caller_arg, caller_abi) = caller_args.next().unwrap();
if !caller_abi.layout.is_1zst() {
do yeet ::rustc_middle::mir::interpret::InterpErrorKind::UndefinedBehavior(::rustc_middle::mir::interpret::UndefinedBehaviorInfo::AbiMismatchArgument {
arg_idx: callee_arg_idx,
caller_ty: caller_abi.layout.ty,
callee_ty: callee_abi.layout.ty,
});
}
if !caller_abi.is_ignore() {
::core::panicking::panic("assertion failed: caller_abi.is_ignore()")
};
}
} else if Some(local) == va_list_arg {
ecx.storage_live(local)?;
let place = ecx.eval_place(dest)?;
let mplace = ecx.force_allocation(&place)?;
let varargs =
M::with_retag_mode(ecx, RetagMode::None,
|ecx|
{
ecx.allocate_varargs(&mut caller_args,
&mut callee_args_abis)
})?;
ecx.frame_mut().va_list = varargs.clone();
let key = ecx.va_list_ptr(varargs.into());
ecx.write_bytes_ptr(mplace.ptr(),
(0..mplace.layout.size.bytes()).map(|_| 0u8))?;
let key_mplace = ecx.va_list_key_field(&mplace)?;
ecx.write_pointer(key, &key_mplace)?;
} else if Some(local) == body.spread_arg {
ecx.storage_live(local)?;
let ty::Tuple(fields) =
ty.kind() else {
::rustc_middle::util::bug::span_bug_fmt(ecx.cur_span(),
format_args!("non-tuple type for `spread_arg`: {0}", ty))
};
for (i, field_ty) in fields.iter().enumerate() {
let dest =
dest.project_deeper(&[mir::ProjectionElem::Field(FieldIdx::from_usize(i),
field_ty)], *ecx.tcx);
ecx.pass_argument(&mut caller_args, &mut callee_args_abis,
&dest, field_ty, true)?;
}
} else {
ecx.pass_argument(&mut caller_args, &mut callee_args_abis,
&dest, ty, false)?;
}
}
interp_ok(())
})?;
self.frame_mut().loc =
Right(body.local_decls[mir::RETURN_PLACE].source_info.span);
if !self.check_argument_compat(&caller_fn_abi.ret,
&callee_fn_abi.ret)? {
do yeet ::rustc_middle::mir::interpret::InterpErrorKind::UndefinedBehavior(::rustc_middle::mir::interpret::UndefinedBehaviorInfo::AbiMismatchReturn {
caller_ty: caller_fn_abi.ret.layout.ty,
callee_ty: callee_fn_abi.ret.layout.ty,
});
}
if let Some(mplace) = destination_mplace {
M::protect_in_place_function_argument(self, &mplace)?;
}
self.frame_mut().loc = Right(preamble_span);
if instance.def.requires_caller_location(*self.tcx) {
callee_args_abis.next().unwrap();
}
if !callee_args_abis.next().is_none() {
{
::core::panicking::panic_fmt(format_args!("mismatch between callee ABI and callee body arguments"));
}
};
if caller_args.next().is_some() {
do yeet ::rustc_middle::mir::interpret::InterpErrorKind::UndefinedBehavior(::rustc_middle::mir::interpret::UndefinedBehaviorInfo::Ub(::alloc::__export::must_use({
::alloc::fmt::format(format_args!("calling a function with more arguments than it expected"))
})));
}
self.push_stack_frame_done()
}
}
}#[instrument(skip(self), level = "trace")]334pub fn init_stack_frame(
335&mut self,
336 instance: Instance<'tcx>,
337 body: &'tcx mir::Body<'tcx>,
338 caller_fn_abi: &FnAbi<'tcx, Ty<'tcx>>,
339 args: &[FnArg<'tcx, M::Provenance>],
340 with_caller_location: bool,
341 destination: &PlaceTy<'tcx, M::Provenance>,
342mut cont: ReturnContinuation,
343 ) -> InterpResult<'tcx> {
344let _trace = enter_trace_span!(M, step::init_stack_frame, %instance, tracing_separate_thread = Empty);
345let def_id = instance.def_id();
346347// The first order of business is to figure out the callee signature.
348 // However, that requires the list of variadic arguments.
349 // We use the *caller* information to determine where to split the list of arguments,
350 // and then later check that the callee indeed has the same number of fixed arguments.
351let extra_tys = if caller_fn_abi.c_variadic {
352let fixed_count = usize::try_from(caller_fn_abi.fixed_count).unwrap();
353let extra_tys = args[fixed_count..].iter().map(|arg| arg.layout().ty);
354self.tcx.mk_type_list_from_iter(extra_tys)
355 } else {
356 ty::List::empty()
357 };
358let callee_fn_abi = self.fn_abi_of_instance_no_deduced_attrs(instance, extra_tys)?;
359360if caller_fn_abi.conv != callee_fn_abi.conv {
361throw_ub_format!(
362"calling a function with calling convention \"{callee_conv}\" using calling convention \"{caller_conv}\"",
363 callee_conv = callee_fn_abi.conv,
364 caller_conv = caller_fn_abi.conv,
365 )
366 }
367368if caller_fn_abi.c_variadic != callee_fn_abi.c_variadic {
369throw_ub!(CVariadicMismatch {
370 caller_is_c_variadic: caller_fn_abi.c_variadic,
371 callee_is_c_variadic: callee_fn_abi.c_variadic,
372 });
373 }
374if caller_fn_abi.c_variadic && caller_fn_abi.fixed_count != callee_fn_abi.fixed_count {
375throw_ub!(CVariadicFixedCountMismatch {
376 caller: caller_fn_abi.fixed_count,
377 callee: callee_fn_abi.fixed_count,
378 });
379 }
380381// Check that all target features required by the callee (i.e., from
382 // the attribute `#[target_feature(enable = ...)]`) are enabled at
383 // compile time.
384M::check_fn_target_features(self, instance)?;
385386if !callee_fn_abi.can_unwind {
387// The callee cannot unwind, so force the `Unreachable` unwind handling.
388match &mut cont {
389 ReturnContinuation::Stop { .. } => {}
390 ReturnContinuation::Goto { unwind, .. } => {
391*unwind = mir::UnwindAction::Unreachable;
392 }
393 }
394 }
395396// *Before* pushing the new frame, determine whether the return destination is in memory.
397 // Need to use `place_to_op` to be *sure* we get the mplace if there is one.
398let destination_mplace = self.place_to_op(destination)?.as_mplace_or_imm().left();
399400// Push the "raw" frame -- this leaves locals uninitialized.
401self.push_stack_frame_raw(instance, body, destination, cont)?;
402let preamble_span = self.frame().loc.unwrap_right(); // the span used for preamble errors
403404trace!(
405"caller ABI: {:#?}, args: {:#?}",
406 caller_fn_abi,
407 args.iter()
408 .map(|arg| (
409 arg.layout().ty,
410match arg {
411 FnArg::Copy(op) => format!("copy({op:?})"),
412 FnArg::InPlace(mplace) => format!("in-place({mplace:?})"),
413 }
414 ))
415 .collect::<Vec<_>>()
416 );
417trace!(
418"spread_arg: {:?}, locals: {:#?}",
419 body.spread_arg,
420 body.args_iter()
421 .map(|local| (local, self.layout_of_local(self.frame(), local, None).unwrap().ty))
422 .collect::<Vec<_>>()
423 );
424425// Determine whether there is a special VaList argument. This is always the
426 // last argument, and since arguments start at index 1 that's `arg_count`.
427let va_list_arg = callee_fn_abi.c_variadic.then(|| mir::Local::from_usize(body.arg_count));
428// Determine whether this is a non-capturing closure. That's relevant as their first
429 // argument can be skipped (and that's the only kind of argument skipping we allow).
430let is_non_capturing_closure =
431 (matches!(instance.def, ty::InstanceKind::Shim(ty::ShimKind::ClosureOnce { .. }))
432 || self.tcx.is_closure_like(def_id))
433 && {
434let arg = &callee_fn_abi.args[0];
435matches!(arg.layout.ty.kind(), ty::Closure (_def, closure_args) if {
436 closure_args.as_closure().upvar_tys().is_empty()
437 })
438 };
439440// In principle, we have two iterators: Where the arguments come from, and where
441 // they go to.
442443 // The "where they come from" part is easy, we expect the caller to do any special handling
444 // that might be required here (e.g. for untupling).
445 // If `with_caller_location` is set we pretend there is an extra argument (that
446 // we will not pass; our `caller_location` intrinsic implementation walks the stack instead).
447assert_eq!(
448 args.len() + if with_caller_location { 1 } else { 0 },
449 caller_fn_abi.args.len(),
450"mismatch between caller ABI and caller arguments",
451 );
452let mut caller_args = args.iter().zip(caller_fn_abi.args.iter());
453454// Now we have to spread them out across the callee's locals,
455 // taking into account the `spread_arg`. If we could write
456 // this is a single iterator (that handles `spread_arg`), then
457 // `pass_argument` would be the loop body.
458let mut callee_args_abis = callee_fn_abi.args.iter().enumerate();
459// During argument passing, we want retagging with protectors.
460M::with_retag_mode(self, RetagMode::FnEntry, |ecx| {
461for local in body.args_iter() {
462// Update the span that we show in case of an error to point to this argument.
463ecx.frame_mut().loc = Right(body.local_decls[local].source_info.span);
464// Construct the destination place for this argument. At this point all
465 // locals are still dead, so we cannot construct a `PlaceTy`.
466let dest = mir::Place::from(local);
467// `layout_of_local` does more than just the instantiation we need to get the
468 // type, but the result gets cached so this avoids calling the instantiation
469 // query *again* the next time this local is accessed.
470let ty = ecx.layout_of_local(ecx.frame(), local, None)?.ty;
471472// Some arguments are special: the first (`self`) argument of a non-capturing
473 // closure; the va_list argument; and the spread_arg.
474if is_non_capturing_closure && local == mir::Local::arg(0) {
475assert!(va_list_arg.is_none());
476assert!(Some(local) != body.spread_arg);
477// This argument might be missing on the caller side. So just initialize it in
478 // the callee.
479let (callee_arg_idx, callee_abi) = callee_args_abis.next().unwrap();
480assert!(callee_abi.layout.is_1zst() && callee_abi.is_ignore());
481 ecx.storage_live(local)?;
482// And skip it in the caller, if present. We can tell whether it is present by
483 // comparing the number of arguments on the caller and callee side.
484if caller_fn_abi.args.len() == callee_fn_abi.args.len() {
485let (_caller_arg, caller_abi) = caller_args.next().unwrap();
486if !caller_abi.layout.is_1zst() {
487// The caller gave us some other, non-ignorable argument.
488throw_ub!(AbiMismatchArgument {
489 arg_idx: callee_arg_idx,
490 caller_ty: caller_abi.layout.ty,
491 callee_ty: callee_abi.layout.ty
492 });
493 }
494assert!(caller_abi.is_ignore());
495 }
496 } else if Some(local) == va_list_arg {
497// This is the last callee-side argument of a variadic function.
498 // This argument is a VaList holding the remaining caller-side arguments.
499ecx.storage_live(local)?;
500501let place = ecx.eval_place(dest)?;
502let mplace = ecx.force_allocation(&place)?;
503504// Consume the remaining arguments by putting them into the variable argument
505 // list. We disable retagging to avoid creating protected tags. Protection should
506 // only use callee-side information, and the varargs have no static callee-side type.
507let varargs = M::with_retag_mode(ecx, RetagMode::None, |ecx| {
508 ecx.allocate_varargs(&mut caller_args, &mut callee_args_abis)
509 })?;
510511// When the frame is dropped, these variable arguments are deallocated.
512ecx.frame_mut().va_list = varargs.clone();
513let key = ecx.va_list_ptr(varargs.into());
514515// Zero the VaList, so it is fully initialized.
516ecx.write_bytes_ptr(
517 mplace.ptr(),
518 (0..mplace.layout.size.bytes()).map(|_| 0u8),
519 )?;
520521// Store the "key" pointer in the right field.
522let key_mplace = ecx.va_list_key_field(&mplace)?;
523 ecx.write_pointer(key, &key_mplace)?;
524 } else if Some(local) == body.spread_arg {
525// Make the local live once, then fill in the value field by field.
526ecx.storage_live(local)?;
527// Must be a tuple
528let ty::Tuple(fields) = ty.kind() else {
529span_bug!(ecx.cur_span(), "non-tuple type for `spread_arg`: {ty}")
530 };
531for (i, field_ty) in fields.iter().enumerate() {
532let dest = dest.project_deeper(
533&[mir::ProjectionElem::Field(FieldIdx::from_usize(i), field_ty)],
534*ecx.tcx,
535 );
536 ecx.pass_argument(
537&mut caller_args,
538&mut callee_args_abis,
539&dest,
540 field_ty,
541/* already_live */ true,
542 )?;
543 }
544 } else {
545// Normal argument. Cannot mark it as live yet, it might be unsized!
546ecx.pass_argument(
547&mut caller_args,
548&mut callee_args_abis,
549&dest,
550 ty,
551/* already_live */ false,
552 )?;
553 }
554 }
555 interp_ok(())
556 })?;
557558// Don't forget to check the return type!
559self.frame_mut().loc = Right(body.local_decls[mir::RETURN_PLACE].source_info.span);
560if !self.check_argument_compat(&caller_fn_abi.ret, &callee_fn_abi.ret)? {
561throw_ub!(AbiMismatchReturn {
562 caller_ty: caller_fn_abi.ret.layout.ty,
563 callee_ty: callee_fn_abi.ret.layout.ty
564 });
565 }
566// Protect return place for in-place return value passing.
567 // We only need to protect anything if this is actually an in-memory place.
568if let Some(mplace) = destination_mplace {
569 M::protect_in_place_function_argument(self, &mplace)?;
570 }
571572// For the final checks, use same span as preamble since it is unclear what else to do.
573self.frame_mut().loc = Right(preamble_span);
574// If the callee needs a caller location, pretend we consume one more argument from the ABI.
575if instance.def.requires_caller_location(*self.tcx) {
576 callee_args_abis.next().unwrap();
577 }
578// Now we should have no more caller args or callee arg ABIs.
579assert!(
580 callee_args_abis.next().is_none(),
581"mismatch between callee ABI and callee body arguments"
582);
583if caller_args.next().is_some() {
584throw_ub_format!("calling a function with more arguments than it expected");
585 }
586587// Done!
588self.push_stack_frame_done()
589 }
590591/// Initiate a call to this function -- pushing the stack frame and initializing the arguments.
592 ///
593 /// `caller_fn_abi` is used to determine if all the arguments are passed the proper way.
594 /// However, we also need `caller_abi` to determine if we need to do untupling of arguments.
595 ///
596 /// `with_caller_location` indicates whether the caller passed a caller location. Miri
597 /// implements caller locations without argument passing, but to match `FnAbi` we need to know
598 /// when those arguments are present.
599pub(super) fn init_fn_call(
600&mut self,
601 fn_val: FnVal<'tcx, M::ExtraFnVal>,
602 (caller_abi, caller_fn_abi): (ExternAbi, &FnAbi<'tcx, Ty<'tcx>>),
603 args: &[FnArg<'tcx, M::Provenance>],
604 with_caller_location: bool,
605 destination: &PlaceTy<'tcx, M::Provenance>,
606 target: Option<mir::BasicBlock>,
607 unwind: mir::UnwindAction,
608 ) -> InterpResult<'tcx> {
609let _trace =
610<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::call", ::tracing::Level::INFO,
::tracing_core::__macro_support::Option::Some("compiler/rustc_const_eval/src/interpret/call.rs"),
::tracing_core::__macro_support::Option::Some(610u32),
::tracing_core::__macro_support::Option::Some("rustc_const_eval::interpret::call"),
::tracing_core::field::FieldSet::new(&["step",
"tracing_separate_thread", "fn_val"],
::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(&"init_fn_call")
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)),
(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&debug(&fn_val) as
&dyn Value))])
})
} else {
let span =
::tracing::__macro_support::__disabled_span(__CALLSITE.metadata());
{};
span
}
})enter_trace_span!(M, step::init_fn_call, tracing_separate_thread = Empty, ?fn_val)611 .or_if_tracing_disabled(|| {
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/call.rs:611",
"rustc_const_eval::interpret::call",
::tracing::Level::TRACE,
::tracing_core::__macro_support::Option::Some("compiler/rustc_const_eval/src/interpret/call.rs"),
::tracing_core::__macro_support::Option::Some(611u32),
::tracing_core::__macro_support::Option::Some("rustc_const_eval::interpret::call"),
::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!("init_fn_call: {0:#?}",
fn_val) as &dyn Value))])
});
} else { ; }
}trace!("init_fn_call: {:#?}", fn_val));
612613let instance = match fn_val {
614 FnVal::Instance(instance) => instance,
615 FnVal::Other(extra) => {
616return M::call_extra_fn(
617self,
618extra,
619caller_fn_abi,
620args,
621destination,
622target,
623unwind,
624 );
625 }
626 };
627628match instance.def {
629 ty::InstanceKind::Intrinsic(def_id) => {
630if !self.tcx.intrinsic(def_id).is_some() {
::core::panicking::panic("assertion failed: self.tcx.intrinsic(def_id).is_some()")
};assert!(self.tcx.intrinsic(def_id).is_some());
631// FIXME: Should `InPlace` arguments be reset to uninit?
632if let Some(fallback) = M::call_intrinsic(
633self,
634 instance,
635&Self::copy_fn_args(args),
636 destination,
637 target,
638 unwind,
639 )? {
640if !!self.tcx.intrinsic(fallback.def_id()).unwrap().must_be_overridden {
::core::panicking::panic("assertion failed: !self.tcx.intrinsic(fallback.def_id()).unwrap().must_be_overridden")
};assert!(!self.tcx.intrinsic(fallback.def_id()).unwrap().must_be_overridden);
641{
match fallback.def {
ty::InstanceKind::Item(_) => {}
ref left_val => {
::core::panicking::assert_matches_failed(left_val,
"ty::InstanceKind::Item(_)", ::core::option::Option::None);
}
}
};assert_matches!(fallback.def, ty::InstanceKind::Item(_));
642return self.init_fn_call(
643 FnVal::Instance(fallback),
644 (caller_abi, caller_fn_abi),
645args,
646with_caller_location,
647destination,
648target,
649unwind,
650 );
651 } else {
652interp_ok(())
653 }
654 }
655 ty::InstanceKind::Shim(ty::ShimKind::VTable(..))
656 | ty::InstanceKind::Shim(ty::ShimKind::Reify(..))
657 | ty::InstanceKind::Shim(ty::ShimKind::ClosureOnce { .. })
658 | ty::InstanceKind::Shim(ty::ShimKind::ConstructCoroutineInClosure { .. })
659 | ty::InstanceKind::Shim(ty::ShimKind::FnPtr(..))
660 | ty::InstanceKind::Shim(ty::ShimKind::DropGlue(..))
661 | ty::InstanceKind::Shim(ty::ShimKind::Clone(..))
662 | ty::InstanceKind::Shim(ty::ShimKind::FnPtrAddr(..))
663 | ty::InstanceKind::Shim(ty::ShimKind::ThreadLocal(..))
664 | ty::InstanceKind::Shim(ty::ShimKind::AsyncDropGlueCtor(..))
665 | ty::InstanceKind::Shim(ty::ShimKind::AsyncDropGlue(..))
666 | ty::InstanceKind::Shim(ty::ShimKind::FutureDropPoll(..))
667 | ty::InstanceKind::Item(_) => {
668// We need MIR for this fn.
669 // Note that this can be an intrinsic, if we are executing its fallback body.
670let Some((body, instance)) = M::find_mir_or_eval_fn(
671self,
672 instance,
673 caller_fn_abi,
674 args,
675 destination,
676 target,
677 unwind,
678 )?
679else {
680return interp_ok(());
681 };
682683// Special handling for the closure ABI: untuple the last argument.
684 // FIXME(splat): un-tuple splatted arguments that were tupled in typecheck
685let args: Cow<'_, [FnArg<'tcx, M::Provenance>]> =
686if caller_abi == ExternAbi::RustCall && !args.is_empty() {
687// Untuple
688let (untuple_arg, args) = args.split_last().unwrap();
689let ty::Tuple(untuple_fields) = untuple_arg.layout().ty.kind() else {
690::rustc_middle::util::bug::span_bug_fmt(self.cur_span(),
format_args!("untuple argument must be a tuple"))span_bug!(self.cur_span(), "untuple argument must be a tuple")691 };
692{
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/call.rs:692",
"rustc_const_eval::interpret::call",
::tracing::Level::TRACE,
::tracing_core::__macro_support::Option::Some("compiler/rustc_const_eval/src/interpret/call.rs"),
::tracing_core::__macro_support::Option::Some(692u32),
::tracing_core::__macro_support::Option::Some("rustc_const_eval::interpret::call"),
::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!("init_fn_call: Will pass last argument by untupling")
as &dyn Value))])
});
} else { ; }
};trace!("init_fn_call: Will pass last argument by untupling");
693Cow::from(
694 args.iter()
695// The regular arguments.
696.map(|a| interp_ok(a.clone()))
697// The fields of the untupled argument.
698.chain((0..untuple_fields.len()).map(|i| {
699self.fn_arg_project_field(untuple_arg, FieldIdx::from_usize(i))
700 }))
701 .collect::<InterpResult<'_, Vec<_>>>()?,
702 )
703 } else {
704// Plain arg passing
705Cow::from(args)
706 };
707708self.init_stack_frame(
709instance,
710body,
711caller_fn_abi,
712&args,
713with_caller_location,
714destination,
715 ReturnContinuation::Goto { ret: target, unwind },
716 )
717 }
718// `InstanceKind::Virtual` does not have callable MIR. Calls to `Virtual` instances must be
719 // codegen'd / interpreted as virtual calls through the vtable.
720ty::InstanceKind::Virtual(def_id, idx) => {
721let mut args = args.to_vec();
722// We have to implement all "dyn-compatible receivers". So we have to go search for a
723 // pointer or `dyn Trait` type, but it could be wrapped in newtypes. So recursively
724 // unwrap those newtypes until we are there.
725 // An `InPlace` does nothing here, we keep the original receiver intact. We can't
726 // really pass the argument in-place anyway, and we are constructing a new
727 // `Immediate` receiver.
728let mut receiver = args[0].copy_fn_arg();
729let receiver_place = loop {
730match receiver.layout.ty.kind() {
731 ty::Ref(..) | ty::RawPtr(..) => {
732// We do *not* use `deref_pointer` here: we don't want to conceptually
733 // create a place that must be dereferenceable, since the receiver might
734 // be a raw pointer and (for `*const dyn Trait`) we don't need to
735 // actually access memory to resolve this method.
736 // Also see <https://github.com/rust-lang/miri/issues/2786>.
737let val = self.read_immediate(&receiver)?;
738break self.imm_ptr_to_mplace(&val)?;
739 }
740 ty::Dynamic(..) => break receiver.assert_mem_place(), // no immediate unsized values
741_ => {
742// Not there yet, search for the only non-ZST field.
743 // (The rules for `DispatchFromDyn` ensure there's exactly one such field.)
744let (idx, _) = receiver.layout.non_1zst_field(self).expect(
745"not exactly one non-1-ZST field in a `DispatchFromDyn` type",
746 );
747receiver = self.project_field(&receiver, idx)?;
748 }
749 }
750 };
751752// Obtain the underlying trait we are working on, and the adjusted receiver argument.
753 // Doesn't have to be a `dyn Trait`, but the unsized tail must be `dyn Trait`.
754 // (For that reason we also cannot use `unpack_dyn_trait`.)
755let receiver_tail =
756self.tcx.struct_tail_for_codegen(receiver_place.layout.ty, self.typing_env);
757let ty::Dynamic(receiver_trait, _) = receiver_tail.kind() else {
758::rustc_middle::util::bug::span_bug_fmt(self.cur_span(),
format_args!("dynamic call on non-`dyn` type {0}", receiver_tail))span_bug!(self.cur_span(), "dynamic call on non-`dyn` type {}", receiver_tail)759 };
760if !receiver_place.layout.is_unsized() {
::core::panicking::panic("assertion failed: receiver_place.layout.is_unsized()")
};assert!(receiver_place.layout.is_unsized());
761762// Get the required information from the vtable.
763let vptr = receiver_place.meta().unwrap_meta().to_pointer(self)?;
764let dyn_ty = self.get_ptr_vtable_ty(vptr, Some(receiver_trait))?;
765let adjusted_recv = receiver_place.ptr();
766767// Now determine the actual method to call. Usually we use the easy way of just
768 // looking up the method at index `idx`.
769let vtable_entries = self.vtable_entries(receiver_trait.principal(), dyn_ty);
770let Some(ty::VtblEntry::Method(fn_inst)) = vtable_entries.get(idx).copied() else {
771// FIXME(fee1-dead) these could be variants of the UB info enum instead of this
772do yeet ::rustc_middle::mir::interpret::InterpErrorKind::UndefinedBehavior(::rustc_middle::mir::interpret::UndefinedBehaviorInfo::Ub(::alloc::__export::must_use({
::alloc::fmt::format(format_args!("`dyn` call trying to call something that is not a method"))
})));throw_ub_format!("`dyn` call trying to call something that is not a method");
773 };
774{
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/call.rs:774",
"rustc_const_eval::interpret::call",
::tracing::Level::TRACE,
::tracing_core::__macro_support::Option::Some("compiler/rustc_const_eval/src/interpret/call.rs"),
::tracing_core::__macro_support::Option::Some(774u32),
::tracing_core::__macro_support::Option::Some("rustc_const_eval::interpret::call"),
::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!("Virtual call dispatches to {0:#?}",
fn_inst) as &dyn Value))])
});
} else { ; }
};trace!("Virtual call dispatches to {fn_inst:#?}");
775// We can also do the lookup based on `def_id` and `dyn_ty`, and check that that
776 // produces the same result.
777self.assert_virtual_instance_matches_concrete(dyn_ty, def_id, instance, fn_inst);
778779// Adjust receiver argument. Layout can be any (thin) ptr.
780let receiver_ty = Ty::new_mut_ptr(self.tcx.tcx, dyn_ty);
781args[0] = FnArg::Copy(
782ImmTy::from_immediate(
783Scalar::from_maybe_pointer(adjusted_recv, self).into(),
784self.layout_of(receiver_ty)?,
785 )
786 .into(),
787 );
788{
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/call.rs:788",
"rustc_const_eval::interpret::call",
::tracing::Level::TRACE,
::tracing_core::__macro_support::Option::Some("compiler/rustc_const_eval/src/interpret/call.rs"),
::tracing_core::__macro_support::Option::Some(788u32),
::tracing_core::__macro_support::Option::Some("rustc_const_eval::interpret::call"),
::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!("Patched receiver operand to {0:#?}",
args[0]) as &dyn Value))])
});
} else { ; }
};trace!("Patched receiver operand to {:#?}", args[0]);
789// Need to also adjust the type in the ABI. Strangely, the layout there is actually
790 // already fine! Just the type is bogus. This is due to what `force_thin_self_ptr`
791 // does in `fn_abi_new_uncached`; supposedly, codegen relies on having the bogus
792 // type, so we just patch this up locally.
793let mut caller_fn_abi = caller_fn_abi.clone();
794caller_fn_abi.args[0].layout.ty = receiver_ty;
795796// recurse with concrete function
797self.init_fn_call(
798 FnVal::Instance(fn_inst),
799 (caller_abi, &caller_fn_abi),
800&args,
801with_caller_location,
802destination,
803target,
804unwind,
805 )
806 }
807 }
808 }
809810fn assert_virtual_instance_matches_concrete(
811&self,
812 dyn_ty: Ty<'tcx>,
813 def_id: DefId,
814 virtual_instance: ty::Instance<'tcx>,
815 concrete_instance: ty::Instance<'tcx>,
816 ) {
817let tcx = *self.tcx;
818819let trait_def_id = tcx.parent(def_id);
820let virtual_trait_ref = ty::TraitRef::from_assoc(tcx, trait_def_id, virtual_instance.args);
821let existential_trait_ref = ty::ExistentialTraitRef::erase_self_ty(tcx, virtual_trait_ref);
822let concrete_trait_ref = existential_trait_ref.with_self_ty(tcx, dyn_ty);
823824let concrete_method = {
825let _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::call", ::tracing::Level::INFO,
::tracing_core::__macro_support::Option::Some("compiler/rustc_const_eval/src/interpret/call.rs"),
::tracing_core::__macro_support::Option::Some(825u32),
::tracing_core::__macro_support::Option::Some("rustc_const_eval::interpret::call"),
::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(&"expect_resolve_for_vtable")
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::expect_resolve_for_vtable, ?def_id);
826Instance::expect_resolve_for_vtable(
827tcx,
828self.typing_env,
829def_id,
830virtual_instance.args.rebase_onto(tcx, trait_def_id, concrete_trait_ref.args),
831self.cur_span(),
832 )
833 };
834{
match (&concrete_instance, &concrete_method) {
(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!(concrete_instance, concrete_method);
835 }
836837/// Initiate a tail call to this function -- popping the current stack frame, pushing the new
838 /// stack frame and initializing the arguments.
839pub(super) fn init_fn_tail_call(
840&mut self,
841 fn_val: FnVal<'tcx, M::ExtraFnVal>,
842 (caller_abi, caller_fn_abi): (ExternAbi, &FnAbi<'tcx, Ty<'tcx>>),
843 args: &[FnArg<'tcx, M::Provenance>],
844 with_caller_location: bool,
845 ) -> InterpResult<'tcx> {
846{
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/call.rs:846",
"rustc_const_eval::interpret::call",
::tracing::Level::TRACE,
::tracing_core::__macro_support::Option::Some("compiler/rustc_const_eval/src/interpret/call.rs"),
::tracing_core::__macro_support::Option::Some(846u32),
::tracing_core::__macro_support::Option::Some("rustc_const_eval::interpret::call"),
::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!("init_fn_tail_call: {0:#?}",
fn_val) as &dyn Value))])
});
} else { ; }
};trace!("init_fn_tail_call: {:#?}", fn_val);
847// This is the "canonical" implementation of tails calls,
848 // a pop of the current stack frame, followed by a normal call
849 // which pushes a new stack frame, with the return address from
850 // the popped stack frame.
851 //
852 // Note that we cannot use `return_from_current_stack_frame`,
853 // as that "executes" the goto to the return block, but we don't want to,
854 // only the tail called function should return to the current return block.
855856 // The arguments need to all be copied since the current stack frame will be removed
857 // before the callee even starts executing.
858 // FIXME(explicit_tail_calls,#144855): does this match what codegen does?
859let args = args.iter().map(|fn_arg| FnArg::Copy(fn_arg.copy_fn_arg())).collect::<Vec<_>>();
860// Remove the frame from the stack.
861let frame = self.pop_stack_frame_raw()?;
862// Remember where this frame would have returned to.
863let ReturnContinuation::Goto { ret, unwind } = frame.return_cont() else {
864::rustc_middle::util::bug::bug_fmt(format_args!("can\'t tailcall as root of the stack"));bug!("can't tailcall as root of the stack");
865 };
866// There's no return value to deal with! Instead, we forward the old return place
867 // to the new function.
868 // FIXME(explicit_tail_calls):
869 // we should check if both caller&callee can/n't unwind,
870 // see <https://github.com/rust-lang/rust/pull/113128#issuecomment-1614979803>
871872 // Now push the new stack frame.
873self.init_fn_call(
874 fn_val,
875 (caller_abi, caller_fn_abi),
876&*args,
877 with_caller_location,
878 frame.return_place(),
879 ret,
880 unwind,
881 )?;
882883// Finally, clear the local variables. Has to be done after pushing to support
884 // non-scalar arguments.
885 // FIXME(explicit_tail_calls,#144855): revisit this once codegen supports indirect
886 // arguments, to ensure the semantics are compatible.
887let return_action = self.cleanup_stack_frame(/* unwinding */ false, frame)?;
888{
match (&return_action, &ReturnAction::Normal) {
(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!(return_action, ReturnAction::Normal);
889890interp_ok(())
891 }
892893pub(super) fn init_drop_in_place_call(
894&mut self,
895 place: &PlaceTy<'tcx, M::Provenance>,
896 instance: ty::Instance<'tcx>,
897 target: mir::BasicBlock,
898 unwind: mir::UnwindAction,
899 ) -> InterpResult<'tcx> {
900{
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/call.rs:900",
"rustc_const_eval::interpret::call",
::tracing::Level::TRACE,
::tracing_core::__macro_support::Option::Some("compiler/rustc_const_eval/src/interpret/call.rs"),
::tracing_core::__macro_support::Option::Some(900u32),
::tracing_core::__macro_support::Option::Some("rustc_const_eval::interpret::call"),
::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!("init_drop_in_place_call: {0:?},\n instance={1:?}",
place, instance) as &dyn Value))])
});
} else { ; }
};trace!("init_drop_in_place_call: {:?},\n instance={:?}", place, instance);
901// We take the address of the object. This may well be unaligned, which is fine
902 // for us here. However, unaligned accesses will probably make the actual drop
903 // implementation fail -- a problem shared by rustc.
904let place = self.force_allocation(place)?;
905906// We behave a bit different from codegen here.
907 // Codegen creates an `InstanceKind::Virtual` with index 0 (the slot of the drop method) and
908 // then dispatches that to the normal call machinery. However, our call machinery currently
909 // only supports calling `VtblEntry::Method`; it would choke on a `MetadataDropInPlace`. So
910 // instead we do the virtual call stuff ourselves. It's easier here than in `eval_fn_call`
911 // since we can just get a place of the underlying type and use `mplace_to_imm_ptr`.
912let place = match place.layout.ty.kind() {
913 ty::Dynamic(data, _) => {
914// Dropping a trait object. Need to find actual drop fn.
915self.unpack_dyn_trait(&place, data)?
916}
917_ => {
918if true {
{
match (&instance,
&ty::Instance::resolve_drop_glue(*self.tcx, place.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);
}
}
}
};
};debug_assert_eq!(
919 instance,
920 ty::Instance::resolve_drop_glue(*self.tcx, place.layout.ty)
921 );
922place923 }
924 };
925926let instance = {
927let _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::call", ::tracing::Level::INFO,
::tracing_core::__macro_support::Option::Some("compiler/rustc_const_eval/src/interpret/call.rs"),
::tracing_core::__macro_support::Option::Some(927u32),
::tracing_core::__macro_support::Option::Some("rustc_const_eval::interpret::call"),
::tracing_core::field::FieldSet::new(&["resolve", "ty"],
::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_drop_glue")
as &dyn Value)),
(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&debug(&place.layout.ty)
as &dyn Value))])
})
} else {
let span =
::tracing::__macro_support::__disabled_span(__CALLSITE.metadata());
{};
span
}
})enter_trace_span!(M, resolve::resolve_drop_glue, ty = ?place.layout.ty);
928 ty::Instance::resolve_drop_glue(*self.tcx, place.layout.ty)
929 };
930let fn_abi = self.fn_abi_of_instance_no_deduced_attrs(instance, ty::List::empty())?;
931932let ref_ty = Ty::new_mut_ref(self.tcx.tcx, self.tcx.lifetimes.re_erased, place.layout.ty);
933let arg = self.mplace_to_imm_ptr(&place, Some(ref_ty))?;
934935let ret = MPlaceTy::fake_alloc_zst(self.layout_of(self.tcx.types.unit)?);
936937self.init_fn_call(
938 FnVal::Instance(instance),
939 (ExternAbi::Rust, fn_abi),
940&[FnArg::Copy(arg.into())],
941false,
942&ret.into(),
943Some(target),
944unwind,
945 )
946 }
947948/// Pops the current frame from the stack, copies the return value to the caller, deallocates
949 /// the memory for allocated locals, and jumps to an appropriate place.
950 ///
951 /// If `unwinding` is `false`, then we are performing a normal return
952 /// from a function. In this case, we jump back into the frame of the caller,
953 /// and continue execution as normal.
954 ///
955 /// If `unwinding` is `true`, then we are in the middle of a panic,
956 /// and need to unwind this frame. In this case, we jump to the
957 /// `cleanup` block for the function, which is responsible for running
958 /// `Drop` impls for any locals that have been initialized at this point.
959 /// The cleanup block ends with a special `Resume` terminator, which will
960 /// cause us to continue unwinding.
961#[allow(clippy :: suspicious_else_formatting)]
{
let __tracing_attr_span;
let __tracing_attr_guard;
if ::tracing::Level::TRACE <= ::tracing::level_filters::STATIC_MAX_LEVEL
&&
::tracing::Level::TRACE <=
::tracing::level_filters::LevelFilter::current() ||
{ false } {
__tracing_attr_span =
{
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("return_from_current_stack_frame",
"rustc_const_eval::interpret::call",
::tracing::Level::TRACE,
::tracing_core::__macro_support::Option::Some("compiler/rustc_const_eval/src/interpret/call.rs"),
::tracing_core::__macro_support::Option::Some(961u32),
::tracing_core::__macro_support::Option::Some("rustc_const_eval::interpret::call"),
::tracing_core::field::FieldSet::new(&["unwinding"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::SPAN)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let mut interest = ::tracing::subscriber::Interest::never();
if ::tracing::Level::TRACE <=
::tracing::level_filters::STATIC_MAX_LEVEL &&
::tracing::Level::TRACE <=
::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(&unwinding as
&dyn Value))])
})
} else {
let span =
::tracing::__macro_support::__disabled_span(__CALLSITE.metadata());
{};
span
}
};
__tracing_attr_guard = __tracing_attr_span.enter();
}
#[warn(clippy :: suspicious_else_formatting)]
{
#[allow(unknown_lints, unreachable_code, clippy ::
diverging_sub_expression, clippy :: empty_loop, clippy ::
let_unit_value, clippy :: let_with_type_underscore, clippy ::
needless_return, clippy :: unreachable)]
if false {
let __tracing_attr_fake_return: InterpResult<'tcx> = loop {};
return __tracing_attr_fake_return;
}
{
{
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/call.rs:966",
"rustc_const_eval::interpret::call", ::tracing::Level::INFO,
::tracing_core::__macro_support::Option::Some("compiler/rustc_const_eval/src/interpret/call.rs"),
::tracing_core::__macro_support::Option::Some(966u32),
::tracing_core::__macro_support::Option::Some("rustc_const_eval::interpret::call"),
::tracing_core::field::FieldSet::new(&["message"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::EVENT)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let enabled =
::tracing::Level::INFO <=
::tracing::level_filters::STATIC_MAX_LEVEL &&
::tracing::Level::INFO <=
::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!("popping stack frame ({0})",
if unwinding {
"during unwinding"
} else { "returning from function" }) as &dyn Value))])
});
} else { ; }
};
{
match (&unwinding,
&match self.frame().loc {
Left(loc) => self.body().basic_blocks[loc.block].is_cleanup,
Right(_) => true,
}) {
(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);
}
}
}
};
if unwinding && self.frame_idx() == 0 {
do yeet ::rustc_middle::mir::interpret::InterpErrorKind::UndefinedBehavior(::rustc_middle::mir::interpret::UndefinedBehaviorInfo::Ub(::alloc::__export::must_use({
::alloc::fmt::format(format_args!("unwinding past the topmost frame of the stack"))
})));
}
let return_op =
self.local_to_op(mir::RETURN_PLACE,
None).expect("return place should always be live");
let frame = self.pop_stack_frame_raw()?;
if !unwinding {
self.copy_op_allow_transmute(&return_op,
frame.return_place())?;
{
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/call.rs:992",
"rustc_const_eval::interpret::call",
::tracing::Level::TRACE,
::tracing_core::__macro_support::Option::Some("compiler/rustc_const_eval/src/interpret/call.rs"),
::tracing_core::__macro_support::Option::Some(992u32),
::tracing_core::__macro_support::Option::Some("rustc_const_eval::interpret::call"),
::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!("return value: {0:?}",
self.dump_place(frame.return_place())) as &dyn Value))])
});
} else { ; }
};
}
let return_cont = frame.return_cont();
let return_action = self.cleanup_stack_frame(unwinding, frame)?;
match return_action {
ReturnAction::Normal => {}
ReturnAction::NoJump => { return interp_ok(()); }
ReturnAction::NoCleanup => {
if !self.stack().is_empty() {
{
::core::panicking::panic_fmt(format_args!("only the topmost frame should ever be leaked"));
}
};
if !!unwinding {
{
::core::panicking::panic_fmt(format_args!("tried to skip cleanup during unwinding"));
}
};
return interp_ok(());
}
}
if unwinding {
match return_cont {
ReturnContinuation::Goto { unwind, .. } => {
self.unwind_to_block(unwind)
}
ReturnContinuation::Stop { .. } => {
{
::core::panicking::panic_fmt(format_args!("encountered ReturnContinuation::Stop when unwinding!"));
}
}
}
} else {
match return_cont {
ReturnContinuation::Goto { ret, .. } =>
self.return_to_block(ret),
ReturnContinuation::Stop { .. } => {
if !self.stack().is_empty() {
{
::core::panicking::panic_fmt(format_args!("only the bottommost frame can have ReturnContinuation::Stop"));
}
};
interp_ok(())
}
}
}
}
}
}#[instrument(skip(self), level = "trace")]962pub(super) fn return_from_current_stack_frame(
963&mut self,
964 unwinding: bool,
965 ) -> InterpResult<'tcx> {
966info!(
967"popping stack frame ({})",
968if unwinding { "during unwinding" } else { "returning from function" }
969 );
970971// Check `unwinding`.
972assert_eq!(
973 unwinding,
974match self.frame().loc {
975 Left(loc) => self.body().basic_blocks[loc.block].is_cleanup,
976 Right(_) => true,
977 }
978 );
979if unwinding && self.frame_idx() == 0 {
980throw_ub_format!("unwinding past the topmost frame of the stack");
981 }
982983// Get out the return value. Must happen *before* the frame is popped as we have to get the
984 // local's value out.
985let return_op =
986self.local_to_op(mir::RETURN_PLACE, None).expect("return place should always be live");
987// Remove the frame from the stack.
988let frame = self.pop_stack_frame_raw()?;
989// Copy the return value and remember the return continuation.
990if !unwinding {
991self.copy_op_allow_transmute(&return_op, frame.return_place())?;
992trace!("return value: {:?}", self.dump_place(frame.return_place()));
993 }
994let return_cont = frame.return_cont();
995// Finish popping the stack frame.
996let return_action = self.cleanup_stack_frame(unwinding, frame)?;
997// Jump to the next block.
998match return_action {
999 ReturnAction::Normal => {}
1000 ReturnAction::NoJump => {
1001// The hook already did everything.
1002return interp_ok(());
1003 }
1004 ReturnAction::NoCleanup => {
1005// If we are not doing cleanup, also skip everything else.
1006assert!(self.stack().is_empty(), "only the topmost frame should ever be leaked");
1007assert!(!unwinding, "tried to skip cleanup during unwinding");
1008// Don't jump anywhere.
1009return interp_ok(());
1010 }
1011 }
10121013// Normal return, figure out where to jump.
1014if unwinding {
1015// Follow the unwind edge.
1016match return_cont {
1017 ReturnContinuation::Goto { unwind, .. } => {
1018// This must be the very last thing that happens, since it can in fact push a new stack frame.
1019self.unwind_to_block(unwind)
1020 }
1021 ReturnContinuation::Stop { .. } => {
1022panic!("encountered ReturnContinuation::Stop when unwinding!")
1023 }
1024 }
1025 } else {
1026// Follow the normal return edge.
1027match return_cont {
1028 ReturnContinuation::Goto { ret, .. } => self.return_to_block(ret),
1029 ReturnContinuation::Stop { .. } => {
1030assert!(
1031self.stack().is_empty(),
1032"only the bottommost frame can have ReturnContinuation::Stop"
1033);
1034 interp_ok(())
1035 }
1036 }
1037 }
1038 }
1039}