1use std::{assert_matches, fmt, iter};
2
3use rustc_abi::{ExternAbi, FIRST_VARIANT, FieldIdx, VariantIdx};
4use rustc_hir as hir;
5use rustc_hir::def_id::DefId;
6use rustc_hir::lang_items::LangItem;
7use rustc_index::{Idx, IndexVec};
8use rustc_middle::mir::visit::{MutVisitor, PlaceContext};
9use rustc_middle::mir::*;
10use rustc_middle::query::Providers;
11use rustc_middle::ty::{
12 self, CoroutineArgs, CoroutineArgsExt, EarlyBinder, GenericArgs, Ty, TyCtxt, Unnormalized,
13};
14use rustc_middle::{bug, span_bug};
15use rustc_span::{DUMMY_SP, Span, Spanned, dummy_spanned};
16use tracing::{debug, instrument};
17
18use crate::deref_separator::deref_finder;
19use crate::elaborate_drop::{DropElaborator, DropFlagMode, DropStyle, Unwind, elaborate_drop};
20use crate::patch::MirPatch;
21use crate::{
22 abort_unwinding_calls, add_call_guards, add_moves_for_packed_drops, inline, instsimplify,
23 mentioned_items, pass_manager as pm, remove_noop_landing_pads, run_optimization_passes,
24 simplify,
25};
26
27mod async_destructor_ctor;
28
29pub(super) fn provide(providers: &mut Providers) {
30 providers.mir_shims = make_shim;
31}
32
33struct FixProxyFutureDropVisitor<'tcx> {
35 tcx: TyCtxt<'tcx>,
36 replace_to: Local,
37}
38
39impl<'tcx> MutVisitor<'tcx> for FixProxyFutureDropVisitor<'tcx> {
40 fn tcx(&self) -> TyCtxt<'tcx> {
41 self.tcx
42 }
43
44 fn visit_place(
45 &mut self,
46 place: &mut Place<'tcx>,
47 _context: PlaceContext,
48 _location: Location,
49 ) {
50 if place.local == Local::from_u32(1) {
51 if place.projection.len() == 1 {
52 if !#[allow(non_exhaustive_omitted_patterns)] match place.projection.first() {
Some(ProjectionElem::Field(FieldIdx::ZERO, _)) => true,
_ => false,
} {
::core::panicking::panic("assertion failed: matches!(place.projection.first(),\n Some(ProjectionElem::Field(FieldIdx::ZERO, _)))")
};assert!(matches!(
53 place.projection.first(),
54 Some(ProjectionElem::Field(FieldIdx::ZERO, _))
55 ));
56 *place = Place::from(self.replace_to);
57 } else if place.projection.len() == 2 {
58 if !#[allow(non_exhaustive_omitted_patterns)] match place.projection[0] {
ProjectionElem::Field(FieldIdx::ZERO, _) => true,
_ => false,
} {
::core::panicking::panic("assertion failed: matches!(place.projection[0], ProjectionElem::Field(FieldIdx::ZERO, _))")
};assert!(matches!(place.projection[0], ProjectionElem::Field(FieldIdx::ZERO, _)));
59 if !#[allow(non_exhaustive_omitted_patterns)] match place.projection[1] {
ProjectionElem::Deref => true,
_ => false,
} {
::core::panicking::panic("assertion failed: matches!(place.projection[1], ProjectionElem::Deref)")
};assert!(matches!(place.projection[1], ProjectionElem::Deref));
60 *place =
61 Place::from(self.replace_to).project_deeper(&[ProjectionElem::Deref], self.tcx);
62 }
63 }
64 }
65}
66
67fn make_shim<'tcx>(tcx: TyCtxt<'tcx>, instance: ty::InstanceKind<'tcx>) -> Body<'tcx> {
68 {
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_mir_transform/src/shim.rs:68",
"rustc_mir_transform::shim", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_mir_transform/src/shim.rs"),
::tracing_core::__macro_support::Option::Some(68u32),
::tracing_core::__macro_support::Option::Some("rustc_mir_transform::shim"),
::tracing_core::field::FieldSet::new(&["message"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::EVENT)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let enabled =
::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
&&
::tracing::Level::DEBUG <=
::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!("make_shim({0:?})",
instance) as &dyn Value))])
});
} else { ; }
};debug!("make_shim({:?})", instance);
69
70 let mut result = match instance {
71 ty::InstanceKind::Item(..) => ::rustc_middle::util::bug::bug_fmt(format_args!("item {0:?} passed to make_shim",
instance))bug!("item {:?} passed to make_shim", instance),
72 ty::InstanceKind::VTableShim(def_id) => {
73 let adjustment = Adjustment::Deref { source: DerefSource::MutPtr };
74 build_call_shim(tcx, instance, Some(adjustment), CallKind::Direct(def_id))
75 }
76 ty::InstanceKind::FnPtrShim(def_id, ty) => {
77 let trait_ = tcx.parent(def_id);
78 let adjustment = match tcx
80 .fn_trait_kind_from_def_id(trait_)
81 .or_else(|| tcx.async_fn_trait_kind_from_def_id(trait_))
82 {
83 Some(ty::ClosureKind::FnOnce) => Adjustment::Identity,
84 Some(ty::ClosureKind::Fn) => Adjustment::Deref { source: DerefSource::ImmRef },
85 Some(ty::ClosureKind::FnMut) => Adjustment::Deref { source: DerefSource::MutRef },
86 None => ::rustc_middle::util::bug::bug_fmt(format_args!("fn pointer {0:?} is not an fn",
ty))bug!("fn pointer {:?} is not an fn", ty),
87 };
88
89 build_call_shim(tcx, instance, Some(adjustment), CallKind::Indirect(ty))
90 }
91 ty::InstanceKind::ReifyShim(def_id, _) => {
97 build_call_shim(tcx, instance, None, CallKind::Direct(def_id))
98 }
99 ty::InstanceKind::ClosureOnceShim { call_once: _, track_caller: _ } => {
100 let fn_mut = tcx.require_lang_item(LangItem::FnMut, DUMMY_SP);
101 let call_mut = tcx
102 .associated_items(fn_mut)
103 .in_definition_order()
104 .find(|it| it.is_fn())
105 .unwrap()
106 .def_id;
107
108 build_call_shim(tcx, instance, Some(Adjustment::RefMut), CallKind::Direct(call_mut))
109 }
110
111 ty::InstanceKind::ConstructCoroutineInClosureShim {
112 coroutine_closure_def_id,
113 receiver_by_ref,
114 } => build_construct_coroutine_by_move_shim(tcx, coroutine_closure_def_id, receiver_by_ref),
115
116 ty::InstanceKind::DropGlue(def_id, ty) => {
117 if let Some(&ty::Coroutine(coroutine_def_id, args)) = ty.map(Ty::kind) {
120 let coroutine_body = tcx.optimized_mir(coroutine_def_id);
121
122 let ty::Coroutine(_, id_args) = *tcx.type_of(coroutine_def_id).skip_binder().kind()
123 else {
124 ::rustc_middle::util::bug::bug_fmt(format_args!("impossible case reached"))bug!()
125 };
126
127 let body = if id_args.as_coroutine().kind_ty() == args.as_coroutine().kind_ty() {
133 coroutine_body.coroutine_drop().unwrap()
134 } else {
135 match (&args.as_coroutine().kind_ty().to_opt_closure_kind().unwrap(),
&ty::ClosureKind::FnOnce) {
(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!(
136 args.as_coroutine().kind_ty().to_opt_closure_kind().unwrap(),
137 ty::ClosureKind::FnOnce
138 );
139 tcx.optimized_mir(tcx.coroutine_by_move_body_def_id(coroutine_def_id))
140 .coroutine_drop()
141 .unwrap()
142 };
143
144 let mut body =
145 EarlyBinder::bind(body.clone()).instantiate(tcx, args).skip_norm_wip();
146 {
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_mir_transform/src/shim.rs:146",
"rustc_mir_transform::shim", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_mir_transform/src/shim.rs"),
::tracing_core::__macro_support::Option::Some(146u32),
::tracing_core::__macro_support::Option::Some("rustc_mir_transform::shim"),
::tracing_core::field::FieldSet::new(&["message"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::EVENT)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let enabled =
::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
&&
::tracing::Level::DEBUG <=
::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!("make_shim({0:?}) = {1:?}",
instance, body) as &dyn Value))])
});
} else { ; }
};debug!("make_shim({:?}) = {:?}", instance, body);
147
148 pm::run_passes(
149 tcx,
150 &mut body,
151 &[
152 &mentioned_items::MentionedItems,
153 &abort_unwinding_calls::AbortUnwindingCalls,
154 &add_call_guards::CriticalCallEdges,
155 ],
156 Some(MirPhase::Runtime(RuntimePhase::Optimized)),
157 pm::Optimizations::Allowed,
158 );
159
160 return body;
161 }
162
163 build_drop_shim(tcx, def_id, ty)
164 }
165 ty::InstanceKind::ThreadLocalShim(..) => build_thread_local_shim(tcx, instance),
166 ty::InstanceKind::CloneShim(def_id, ty) => build_clone_shim(tcx, def_id, ty),
167 ty::InstanceKind::FnPtrAddrShim(def_id, ty) => build_fn_ptr_addr_shim(tcx, def_id, ty),
168 ty::InstanceKind::FutureDropPollShim(def_id, proxy_ty, impl_ty) => {
169 let mut body =
170 async_destructor_ctor::build_future_drop_poll_shim(tcx, def_id, proxy_ty, impl_ty);
171
172 pm::run_passes(
173 tcx,
174 &mut body,
175 &[
176 &mentioned_items::MentionedItems,
177 &abort_unwinding_calls::AbortUnwindingCalls,
178 &add_call_guards::CriticalCallEdges,
179 ],
180 Some(MirPhase::Runtime(RuntimePhase::PostCleanup)),
181 pm::Optimizations::Allowed,
182 );
183 run_optimization_passes(tcx, &mut body);
184 {
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_mir_transform/src/shim.rs:184",
"rustc_mir_transform::shim", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_mir_transform/src/shim.rs"),
::tracing_core::__macro_support::Option::Some(184u32),
::tracing_core::__macro_support::Option::Some("rustc_mir_transform::shim"),
::tracing_core::field::FieldSet::new(&["message"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::EVENT)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let enabled =
::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
&&
::tracing::Level::DEBUG <=
::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!("make_shim({0:?}) = {1:?}",
instance, body) as &dyn Value))])
});
} else { ; }
};debug!("make_shim({:?}) = {:?}", instance, body);
185 return body;
186 }
187 ty::InstanceKind::AsyncDropGlue(def_id, ty) => {
188 let mut body = async_destructor_ctor::build_async_drop_shim(tcx, def_id, ty);
189
190 pm::run_passes(
194 tcx,
195 &mut body,
196 &[
197 &mentioned_items::MentionedItems,
198 &abort_unwinding_calls::AbortUnwindingCalls,
199 &add_call_guards::CriticalCallEdges,
200 &simplify::SimplifyCfg::MakeShim,
201 &crate::coroutine::StateTransform,
202 ],
203 Some(MirPhase::Runtime(RuntimePhase::PostCleanup)),
204 pm::Optimizations::Allowed,
205 );
206 run_optimization_passes(tcx, &mut body);
207 {
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_mir_transform/src/shim.rs:207",
"rustc_mir_transform::shim", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_mir_transform/src/shim.rs"),
::tracing_core::__macro_support::Option::Some(207u32),
::tracing_core::__macro_support::Option::Some("rustc_mir_transform::shim"),
::tracing_core::field::FieldSet::new(&["message"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::EVENT)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let enabled =
::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
&&
::tracing::Level::DEBUG <=
::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!("make_shim({0:?}) = {1:?}",
instance, body) as &dyn Value))])
});
} else { ; }
};debug!("make_shim({:?}) = {:?}", instance, body);
208 return body;
209 }
210
211 ty::InstanceKind::AsyncDropGlueCtorShim(def_id, ty) => {
212 let body = async_destructor_ctor::build_async_destructor_ctor_shim(tcx, def_id, ty);
213 {
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_mir_transform/src/shim.rs:213",
"rustc_mir_transform::shim", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_mir_transform/src/shim.rs"),
::tracing_core::__macro_support::Option::Some(213u32),
::tracing_core::__macro_support::Option::Some("rustc_mir_transform::shim"),
::tracing_core::field::FieldSet::new(&["message"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::EVENT)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let enabled =
::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
&&
::tracing::Level::DEBUG <=
::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!("make_shim({0:?}) = {1:?}",
instance, body) as &dyn Value))])
});
} else { ; }
};debug!("make_shim({:?}) = {:?}", instance, body);
214 return body;
215 }
216 ty::InstanceKind::Virtual(..) => {
217 ::rustc_middle::util::bug::bug_fmt(format_args!("InstanceKind::Virtual ({0:?}) is for direct calls only",
instance))bug!("InstanceKind::Virtual ({:?}) is for direct calls only", instance)
218 }
219 ty::InstanceKind::Intrinsic(_) => {
220 ::rustc_middle::util::bug::bug_fmt(format_args!("creating shims from intrinsics ({0:?}) is unsupported",
instance))bug!("creating shims from intrinsics ({:?}) is unsupported", instance)
221 }
222 };
223 {
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_mir_transform/src/shim.rs:223",
"rustc_mir_transform::shim", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_mir_transform/src/shim.rs"),
::tracing_core::__macro_support::Option::Some(223u32),
::tracing_core::__macro_support::Option::Some("rustc_mir_transform::shim"),
::tracing_core::field::FieldSet::new(&["message"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::EVENT)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let enabled =
::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
&&
::tracing::Level::DEBUG <=
::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!("make_shim({0:?}) = untransformed {1:?}",
instance, result) as &dyn Value))])
});
} else { ; }
};debug!("make_shim({:?}) = untransformed {:?}", instance, result);
224
225 deref_finder(tcx, &mut result, false);
226
227 pm::run_passes_no_validate(
232 tcx,
233 &mut result,
234 &[
235 &mentioned_items::MentionedItems,
236 &add_moves_for_packed_drops::AddMovesForPackedDrops,
237 &remove_noop_landing_pads::RemoveNoopLandingPads,
238 &simplify::SimplifyCfg::MakeShim,
239 &instsimplify::InstSimplify::BeforeInline,
240 &inline::ForceInline,
242 &abort_unwinding_calls::AbortUnwindingCalls,
243 &add_call_guards::CriticalCallEdges,
244 ],
245 Some(MirPhase::Runtime(RuntimePhase::Optimized)),
246 );
247
248 {
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_mir_transform/src/shim.rs:248",
"rustc_mir_transform::shim", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_mir_transform/src/shim.rs"),
::tracing_core::__macro_support::Option::Some(248u32),
::tracing_core::__macro_support::Option::Some("rustc_mir_transform::shim"),
::tracing_core::field::FieldSet::new(&["message"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::EVENT)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let enabled =
::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
&&
::tracing::Level::DEBUG <=
::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!("make_shim({0:?}) = {1:?}",
instance, result) as &dyn Value))])
});
} else { ; }
};debug!("make_shim({:?}) = {:?}", instance, result);
249
250 result
251}
252
253#[derive(#[automatically_derived]
impl ::core::marker::Copy for DerefSource { }Copy, #[automatically_derived]
impl ::core::clone::Clone for DerefSource {
#[inline]
fn clone(&self) -> DerefSource { *self }
}Clone, #[automatically_derived]
impl ::core::fmt::Debug for DerefSource {
#[inline]
fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
::core::fmt::Formatter::write_str(f,
match self {
DerefSource::ImmRef => "ImmRef",
DerefSource::MutRef => "MutRef",
DerefSource::MutPtr => "MutPtr",
})
}
}Debug, #[automatically_derived]
impl ::core::cmp::PartialEq for DerefSource {
#[inline]
fn eq(&self, other: &DerefSource) -> bool {
let __self_discr = ::core::intrinsics::discriminant_value(self);
let __arg1_discr = ::core::intrinsics::discriminant_value(other);
__self_discr == __arg1_discr
}
}PartialEq)]
254enum DerefSource {
255 ImmRef,
257 MutRef,
259 MutPtr,
261}
262
263#[derive(#[automatically_derived]
impl ::core::marker::Copy for Adjustment { }Copy, #[automatically_derived]
impl ::core::clone::Clone for Adjustment {
#[inline]
fn clone(&self) -> Adjustment {
let _: ::core::clone::AssertParamIsClone<DerefSource>;
*self
}
}Clone, #[automatically_derived]
impl ::core::fmt::Debug for Adjustment {
#[inline]
fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
match self {
Adjustment::Identity =>
::core::fmt::Formatter::write_str(f, "Identity"),
Adjustment::Deref { source: __self_0 } =>
::core::fmt::Formatter::debug_struct_field1_finish(f, "Deref",
"source", &__self_0),
Adjustment::RefMut =>
::core::fmt::Formatter::write_str(f, "RefMut"),
}
}
}Debug, #[automatically_derived]
impl ::core::cmp::PartialEq for Adjustment {
#[inline]
fn eq(&self, other: &Adjustment) -> bool {
let __self_discr = ::core::intrinsics::discriminant_value(self);
let __arg1_discr = ::core::intrinsics::discriminant_value(other);
__self_discr == __arg1_discr &&
match (self, other) {
(Adjustment::Deref { source: __self_0 }, Adjustment::Deref {
source: __arg1_0 }) => __self_0 == __arg1_0,
_ => true,
}
}
}PartialEq)]
264enum Adjustment {
265 Identity,
267
268 Deref { source: DerefSource },
273
274 RefMut,
279}
280
281#[derive(#[automatically_derived]
impl<'tcx> ::core::marker::Copy for CallKind<'tcx> { }Copy, #[automatically_derived]
impl<'tcx> ::core::clone::Clone for CallKind<'tcx> {
#[inline]
fn clone(&self) -> CallKind<'tcx> {
let _: ::core::clone::AssertParamIsClone<Ty<'tcx>>;
let _: ::core::clone::AssertParamIsClone<DefId>;
*self
}
}Clone, #[automatically_derived]
impl<'tcx> ::core::fmt::Debug for CallKind<'tcx> {
#[inline]
fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
match self {
CallKind::Indirect(__self_0) =>
::core::fmt::Formatter::debug_tuple_field1_finish(f,
"Indirect", &__self_0),
CallKind::Direct(__self_0) =>
::core::fmt::Formatter::debug_tuple_field1_finish(f, "Direct",
&__self_0),
}
}
}Debug, #[automatically_derived]
impl<'tcx> ::core::cmp::PartialEq for CallKind<'tcx> {
#[inline]
fn eq(&self, other: &CallKind<'tcx>) -> bool {
let __self_discr = ::core::intrinsics::discriminant_value(self);
let __arg1_discr = ::core::intrinsics::discriminant_value(other);
__self_discr == __arg1_discr &&
match (self, other) {
(CallKind::Indirect(__self_0), CallKind::Indirect(__arg1_0))
=> __self_0 == __arg1_0,
(CallKind::Direct(__self_0), CallKind::Direct(__arg1_0)) =>
__self_0 == __arg1_0,
_ => unsafe { ::core::intrinsics::unreachable() }
}
}
}PartialEq)]
282enum CallKind<'tcx> {
283 Indirect(Ty<'tcx>),
285
286 Direct(DefId),
288}
289
290fn local_decls_for_sig<'tcx>(
291 sig: &ty::FnSig<'tcx>,
292 span: Span,
293) -> IndexVec<Local, LocalDecl<'tcx>> {
294 iter::once(LocalDecl::new(sig.output(), span))
295 .chain(sig.inputs().iter().map(|ity| LocalDecl::new(*ity, span).immutable()))
296 .collect()
297}
298
299fn build_drop_shim<'tcx>(tcx: TyCtxt<'tcx>, def_id: DefId, ty: Option<Ty<'tcx>>) -> Body<'tcx> {
300 {
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_mir_transform/src/shim.rs:300",
"rustc_mir_transform::shim", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_mir_transform/src/shim.rs"),
::tracing_core::__macro_support::Option::Some(300u32),
::tracing_core::__macro_support::Option::Some("rustc_mir_transform::shim"),
::tracing_core::field::FieldSet::new(&["message"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::EVENT)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let enabled =
::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
&&
::tracing::Level::DEBUG <=
::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!("build_drop_shim(def_id={0:?}, ty={1:?})",
def_id, ty) as &dyn Value))])
});
} else { ; }
};debug!("build_drop_shim(def_id={:?}, ty={:?})", def_id, ty);
301
302 if !!#[allow(non_exhaustive_omitted_patterns)] match ty {
Some(ty) if ty.is_coroutine() => true,
_ => false,
} {
::core::panicking::panic("assertion failed: !matches!(ty, Some(ty) if ty.is_coroutine())")
};assert!(!matches!(ty, Some(ty) if ty.is_coroutine()));
303
304 let args = if let Some(ty) = ty {
305 tcx.mk_args(&[ty.into()])
306 } else {
307 GenericArgs::identity_for_item(tcx, def_id)
308 };
309 let sig = tcx.fn_sig(def_id).instantiate(tcx, args).skip_norm_wip();
310 let sig = tcx.instantiate_bound_regions_with_erased(sig);
311 let span = tcx.def_span(def_id);
312
313 let source_info = SourceInfo::outermost(span);
314
315 let return_block = BasicBlock::new(1);
316 let mut blocks = IndexVec::with_capacity(2);
317 let block = |blocks: &mut IndexVec<_, _>, kind| {
318 blocks.push(BasicBlockData::new(Some(Terminator { source_info, kind }), false))
319 };
320 if ty.is_some() {
321 block(&mut blocks, TerminatorKind::Goto { target: return_block });
322 }
323 block(&mut blocks, TerminatorKind::Return);
324
325 let source = MirSource::from_instance(ty::InstanceKind::DropGlue(def_id, ty));
326 let mut body =
327 new_body(source, blocks, local_decls_for_sig(&sig, span), sig.inputs().len(), span);
328
329 let Some(ty) = ty else {
330 return body;
331 };
332
333 let dropee_ptr = Place::from(Local::arg(0));
334
335 if let ty::Array(ety, _len) = *ty.kind() {
336 let slice_ty = Ty::new_slice(tcx, ety);
339 let mut_slice_ty = Ty::new_ref(tcx, tcx.lifetimes.re_erased, slice_ty, ty::Mutability::Mut);
340 let erased_local = body.local_decls.push(LocalDecl::new(mut_slice_ty, span));
341
342 let start = &mut body.basic_blocks_mut()[START_BLOCK];
343 start.statements.push(Statement::new(
344 source_info,
345 StatementKind::Assign(Box::new((
346 Place::from(erased_local),
347 Rvalue::Cast(
348 CastKind::PointerCoercion(
349 ty::adjustment::PointerCoercion::Unsize,
350 CoercionSource::Implicit,
351 ),
352 Operand::Move(dropee_ptr),
353 mut_slice_ty,
354 ),
355 ))),
356 ));
357 start.terminator = Some(Terminator {
358 source_info,
359 kind: TerminatorKind::Call {
360 func: Operand::function_handle(tcx, def_id, [ty::GenericArg::from(slice_ty)], span),
361 args: Box::new([Spanned { span, node: Operand::Move(Place::from(erased_local)) }]),
362 destination: Place::from(RETURN_PLACE),
363 target: Some(return_block),
364 unwind: UnwindAction::Continue,
365 call_source: CallSource::Misc,
366 fn_span: span,
367 },
368 });
369 } else {
370 let patch = {
371 let typing_env = ty::TypingEnv::post_analysis(tcx, def_id);
372 let mut elaborator = DropShimElaborator {
373 body: &body,
374 patch: MirPatch::new(&body),
375 tcx,
376 typing_env,
377 produce_async_drops: false,
378 };
379 let dropee = tcx.mk_place_deref(dropee_ptr);
380 let resume_block = elaborator.patch.resume_block();
381 elaborate_drop(
382 &mut elaborator,
383 source_info,
384 dropee,
385 (),
386 return_block,
387 Unwind::To(resume_block),
388 START_BLOCK,
389 None,
390 );
391 elaborator.patch
392 };
393 patch.apply(&mut body);
394 }
395
396 body
397}
398
399fn new_body<'tcx>(
400 source: MirSource<'tcx>,
401 basic_blocks: IndexVec<BasicBlock, BasicBlockData<'tcx>>,
402 local_decls: IndexVec<Local, LocalDecl<'tcx>>,
403 arg_count: usize,
404 span: Span,
405) -> Body<'tcx> {
406 let mut body = Body::new(
407 source,
408 basic_blocks,
409 IndexVec::from_elem_n(
410 SourceScopeData {
411 span,
412 parent_scope: None,
413 inlined: None,
414 inlined_parent_scope: None,
415 local_data: ClearCrossCrate::Clear,
416 },
417 1,
418 ),
419 local_decls,
420 IndexVec::new(),
421 arg_count,
422 ::alloc::vec::Vec::new()vec![],
423 span,
424 None,
425 None,
427 );
428 body.set_required_consts(Vec::new());
430 body
431}
432
433pub(super) struct DropShimElaborator<'a, 'tcx> {
434 pub body: &'a Body<'tcx>,
435 pub patch: MirPatch<'tcx>,
436 pub tcx: TyCtxt<'tcx>,
437 pub typing_env: ty::TypingEnv<'tcx>,
438 pub produce_async_drops: bool,
439}
440
441impl fmt::Debug for DropShimElaborator<'_, '_> {
442 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> Result<(), fmt::Error> {
443 f.debug_struct("DropShimElaborator").finish_non_exhaustive()
444 }
445}
446
447impl<'a, 'tcx> DropElaborator<'a, 'tcx> for DropShimElaborator<'a, 'tcx> {
448 type Path = ();
449
450 fn patch_ref(&self) -> &MirPatch<'tcx> {
451 &self.patch
452 }
453 fn patch(&mut self) -> &mut MirPatch<'tcx> {
454 &mut self.patch
455 }
456 fn body(&self) -> &'a Body<'tcx> {
457 self.body
458 }
459 fn tcx(&self) -> TyCtxt<'tcx> {
460 self.tcx
461 }
462 fn typing_env(&self) -> ty::TypingEnv<'tcx> {
463 self.typing_env
464 }
465
466 fn terminator_loc(&self, bb: BasicBlock) -> Location {
467 self.patch.terminator_loc(self.body, bb)
468 }
469 fn allow_async_drops(&self) -> bool {
470 self.produce_async_drops
471 }
472
473 fn drop_style(&self, _path: Self::Path, mode: DropFlagMode) -> DropStyle {
474 match mode {
475 DropFlagMode::Shallow => {
476 DropStyle::Static
479 }
480 DropFlagMode::Deep => {
481 DropStyle::Open
484 }
485 }
486 }
487
488 fn get_drop_flag(&mut self, _path: Self::Path) -> Option<Operand<'tcx>> {
489 None
490 }
491
492 fn clear_drop_flag(&mut self, _location: Location, _path: Self::Path, _mode: DropFlagMode) {}
493
494 fn field_subpath(&self, _path: Self::Path, _field: FieldIdx) -> Option<Self::Path> {
495 None
496 }
497 fn deref_subpath(&self, _path: Self::Path) -> Option<Self::Path> {
498 None
499 }
500 fn downcast_subpath(&self, _path: Self::Path, _variant: VariantIdx) -> Option<Self::Path> {
501 Some(())
502 }
503 fn array_subpath(&self, _path: Self::Path, _index: u64, _size: u64) -> Option<Self::Path> {
504 None
505 }
506}
507
508fn build_thread_local_shim<'tcx>(
509 tcx: TyCtxt<'tcx>,
510 instance: ty::InstanceKind<'tcx>,
511) -> Body<'tcx> {
512 let def_id = instance.def_id();
513
514 let span = tcx.def_span(def_id);
515 let source_info = SourceInfo::outermost(span);
516
517 let blocks = IndexVec::from_raw(::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[BasicBlockData::new_stmts(::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[Statement::new(source_info,
StatementKind::Assign(Box::new((Place::return_place(),
Rvalue::ThreadLocalRef(def_id)))))])),
Some(Terminator {
source_info,
kind: TerminatorKind::Return,
}), false)]))vec![BasicBlockData::new_stmts(
518 vec![Statement::new(
519 source_info,
520 StatementKind::Assign(Box::new((
521 Place::return_place(),
522 Rvalue::ThreadLocalRef(def_id),
523 ))),
524 )],
525 Some(Terminator { source_info, kind: TerminatorKind::Return }),
526 false,
527 )]);
528
529 new_body(
530 MirSource::from_instance(instance),
531 blocks,
532 IndexVec::from_raw(::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[LocalDecl::new(tcx.thread_local_ptr_ty(def_id), span)]))vec![LocalDecl::new(tcx.thread_local_ptr_ty(def_id), span)]),
533 0,
534 span,
535 )
536}
537
538fn build_clone_shim<'tcx>(tcx: TyCtxt<'tcx>, def_id: DefId, self_ty: Ty<'tcx>) -> Body<'tcx> {
540 {
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_mir_transform/src/shim.rs:540",
"rustc_mir_transform::shim", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_mir_transform/src/shim.rs"),
::tracing_core::__macro_support::Option::Some(540u32),
::tracing_core::__macro_support::Option::Some("rustc_mir_transform::shim"),
::tracing_core::field::FieldSet::new(&["message"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::EVENT)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let enabled =
::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
&&
::tracing::Level::DEBUG <=
::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!("build_clone_shim(def_id={0:?})",
def_id) as &dyn Value))])
});
} else { ; }
};debug!("build_clone_shim(def_id={:?})", def_id);
541
542 let mut builder = CloneShimBuilder::new(tcx, def_id, self_ty);
543
544 let dest = Place::return_place();
545 let src = tcx.mk_place_deref(Place::from(Local::arg(0)));
546
547 match self_ty.kind() {
548 ty::FnDef(..) | ty::FnPtr(..) => builder.copy_shim(),
549 ty::Closure(_, args) => builder.tuple_like_shim(dest, src, args.as_closure().upvar_tys()),
550 ty::CoroutineClosure(_, args) => {
551 builder.tuple_like_shim(dest, src, args.as_coroutine_closure().upvar_tys())
552 }
553 ty::Tuple(..) => builder.tuple_like_shim(dest, src, self_ty.tuple_fields()),
554 ty::Coroutine(coroutine_def_id, args) => {
555 match (&tcx.coroutine_movability(*coroutine_def_id),
&hir::Movability::Movable) {
(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!(tcx.coroutine_movability(*coroutine_def_id), hir::Movability::Movable);
556 builder.coroutine_shim(dest, src, *coroutine_def_id, args.as_coroutine())
557 }
558 _ => ::rustc_middle::util::bug::bug_fmt(format_args!("clone shim for `{0:?}` which is not `Copy` and is not an aggregate",
self_ty))bug!("clone shim for `{:?}` which is not `Copy` and is not an aggregate", self_ty),
559 };
560
561 builder.into_mir()
562}
563
564struct CloneShimBuilder<'tcx> {
565 tcx: TyCtxt<'tcx>,
566 def_id: DefId,
567 local_decls: IndexVec<Local, LocalDecl<'tcx>>,
568 blocks: IndexVec<BasicBlock, BasicBlockData<'tcx>>,
569 span: Span,
570 sig: ty::FnSig<'tcx>,
571}
572
573impl<'tcx> CloneShimBuilder<'tcx> {
574 fn new(tcx: TyCtxt<'tcx>, def_id: DefId, self_ty: Ty<'tcx>) -> Self {
575 let sig = tcx.fn_sig(def_id).instantiate(tcx, &[self_ty.into()]).skip_norm_wip();
579 let sig = tcx.instantiate_bound_regions_with_erased(sig);
580 let span = tcx.def_span(def_id);
581
582 CloneShimBuilder {
583 tcx,
584 def_id,
585 local_decls: local_decls_for_sig(&sig, span),
586 blocks: IndexVec::new(),
587 span,
588 sig,
589 }
590 }
591
592 fn into_mir(self) -> Body<'tcx> {
593 let source = MirSource::from_instance(ty::InstanceKind::CloneShim(
594 self.def_id,
595 self.sig.inputs_and_output[0],
596 ));
597 new_body(source, self.blocks, self.local_decls, self.sig.inputs().len(), self.span)
598 }
599
600 fn source_info(&self) -> SourceInfo {
601 SourceInfo::outermost(self.span)
602 }
603
604 fn block(
605 &mut self,
606 statements: Vec<Statement<'tcx>>,
607 kind: TerminatorKind<'tcx>,
608 is_cleanup: bool,
609 ) -> BasicBlock {
610 let source_info = self.source_info();
611 self.blocks.push(BasicBlockData::new_stmts(
612 statements,
613 Some(Terminator { source_info, kind }),
614 is_cleanup,
615 ))
616 }
617
618 fn block_index_offset(&self, offset: usize) -> BasicBlock {
623 BasicBlock::new(self.blocks.len() + offset)
624 }
625
626 fn make_statement(&self, kind: StatementKind<'tcx>) -> Statement<'tcx> {
627 Statement::new(self.source_info(), kind)
628 }
629
630 fn copy_shim(&mut self) {
631 let rcvr = self.tcx.mk_place_deref(Place::from(Local::arg(0)));
632 let ret_statement = self.make_statement(StatementKind::Assign(Box::new((
633 Place::return_place(),
634 Rvalue::Use(Operand::Copy(rcvr), WithRetag::Yes),
635 ))));
636 self.block(::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[ret_statement]))vec![ret_statement], TerminatorKind::Return, false);
637 }
638
639 fn make_place(&mut self, mutability: Mutability, ty: Ty<'tcx>) -> Place<'tcx> {
640 let span = self.span;
641 let mut local = LocalDecl::new(ty, span);
642 if mutability.is_not() {
643 local = local.immutable();
644 }
645 Place::from(self.local_decls.push(local))
646 }
647
648 fn make_clone_call(
649 &mut self,
650 dest: Place<'tcx>,
651 src: Place<'tcx>,
652 ty: Ty<'tcx>,
653 next: BasicBlock,
654 cleanup: BasicBlock,
655 ) {
656 let tcx = self.tcx;
657
658 let func_ty = Ty::new_fn_def(tcx, self.def_id, [ty]);
660 let func = Operand::Constant(Box::new(ConstOperand {
661 span: self.span,
662 user_ty: None,
663 const_: Const::zero_sized(func_ty),
664 }));
665
666 let ref_loc =
667 self.make_place(Mutability::Not, Ty::new_imm_ref(tcx, tcx.lifetimes.re_erased, ty));
668
669 let statement = self.make_statement(StatementKind::Assign(Box::new((
671 ref_loc,
672 Rvalue::Ref(tcx.lifetimes.re_erased, BorrowKind::Shared, src),
673 ))));
674
675 self.block(
677 ::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[statement]))vec![statement],
678 TerminatorKind::Call {
679 func,
680 args: [Spanned { node: Operand::Move(ref_loc), span: DUMMY_SP }].into(),
681 destination: dest,
682 target: Some(next),
683 unwind: UnwindAction::Cleanup(cleanup),
684 call_source: CallSource::Normal,
685 fn_span: self.span,
686 },
687 false,
688 );
689 }
690
691 fn clone_fields<I>(
692 &mut self,
693 dest: Place<'tcx>,
694 src: Place<'tcx>,
695 target: BasicBlock,
696 mut unwind: BasicBlock,
697 tys: I,
698 ) -> BasicBlock
699 where
700 I: IntoIterator<Item = Ty<'tcx>>,
701 {
702 for (i, ity) in tys.into_iter().enumerate() {
704 let field = FieldIdx::new(i);
715 let src_field = self.tcx.mk_place_field(src, field, ity);
716
717 let dest_field = self.tcx.mk_place_field(dest, field, ity);
718
719 let next_unwind = self.block_index_offset(1);
720 let next_block = self.block_index_offset(2);
721 self.make_clone_call(dest_field, src_field, ity, next_block, unwind);
722 self.block(
723 ::alloc::vec::Vec::new()vec![],
724 TerminatorKind::Drop {
725 place: dest_field,
726 target: unwind,
727 unwind: UnwindAction::Terminate(UnwindTerminateReason::InCleanup),
728 replace: false,
729 drop: None,
730 async_fut: None,
731 },
732 true,
733 );
734 unwind = next_unwind;
735 }
736 self.block(::alloc::vec::Vec::new()vec![], TerminatorKind::Goto { target }, false);
738 unwind
739 }
740
741 fn tuple_like_shim<I>(&mut self, dest: Place<'tcx>, src: Place<'tcx>, tys: I)
742 where
743 I: IntoIterator<Item = Ty<'tcx>>,
744 {
745 self.block(::alloc::vec::Vec::new()vec![], TerminatorKind::Goto { target: self.block_index_offset(3) }, false);
746 let unwind = self.block(::alloc::vec::Vec::new()vec![], TerminatorKind::UnwindResume, true);
747 let target = self.block(::alloc::vec::Vec::new()vec![], TerminatorKind::Return, false);
748
749 let _final_cleanup_block = self.clone_fields(dest, src, target, unwind, tys);
750 }
751
752 fn coroutine_shim(
753 &mut self,
754 dest: Place<'tcx>,
755 src: Place<'tcx>,
756 coroutine_def_id: DefId,
757 args: CoroutineArgs<TyCtxt<'tcx>>,
758 ) {
759 self.block(::alloc::vec::Vec::new()vec![], TerminatorKind::Goto { target: self.block_index_offset(3) }, false);
760 let unwind = self.block(::alloc::vec::Vec::new()vec![], TerminatorKind::UnwindResume, true);
761 let switch = self.block(::alloc::vec::Vec::new()vec![], TerminatorKind::Unreachable, false);
763 let unwind = self.clone_fields(dest, src, switch, unwind, args.upvar_tys());
764 let target = self.block(::alloc::vec::Vec::new()vec![], TerminatorKind::Return, false);
765 let unreachable = self.block(::alloc::vec::Vec::new()vec![], TerminatorKind::Unreachable, false);
766 let mut cases = Vec::with_capacity(args.state_tys(coroutine_def_id, self.tcx).count());
767 for (index, state_tys) in args.state_tys(coroutine_def_id, self.tcx).enumerate() {
768 let variant_index = VariantIdx::new(index);
769 let dest = self.tcx.mk_place_downcast_unnamed(dest, variant_index);
770 let src = self.tcx.mk_place_downcast_unnamed(src, variant_index);
771 let clone_block = self.block_index_offset(1);
772 let start_block = self.block(
773 ::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[self.make_statement(StatementKind::SetDiscriminant {
place: Box::new(Place::return_place()),
variant_index,
})]))vec![self.make_statement(StatementKind::SetDiscriminant {
774 place: Box::new(Place::return_place()),
775 variant_index,
776 })],
777 TerminatorKind::Goto { target: clone_block },
778 false,
779 );
780 cases.push((index as u128, start_block));
781 let _final_cleanup_block = self.clone_fields(dest, src, target, unwind, state_tys);
782 }
783 let discr_ty = args.discr_ty(self.tcx);
784 let temp = self.make_place(Mutability::Mut, discr_ty);
785 let rvalue = Rvalue::Discriminant(src);
786 let statement = self.make_statement(StatementKind::Assign(Box::new((temp, rvalue))));
787 match &mut self.blocks[switch] {
788 BasicBlockData { statements, terminator: Some(Terminator { kind, .. }), .. } => {
789 statements.push(statement);
790 *kind = TerminatorKind::SwitchInt {
791 discr: Operand::Move(temp),
792 targets: SwitchTargets::new(cases.into_iter(), unreachable),
793 };
794 }
795 BasicBlockData { terminator: None, .. } => ::core::panicking::panic("internal error: entered unreachable code")unreachable!(),
796 }
797 }
798}
799
800x;#[instrument(level = "debug", skip(tcx), ret)]
803fn build_call_shim<'tcx>(
804 tcx: TyCtxt<'tcx>,
805 instance: ty::InstanceKind<'tcx>,
806 rcvr_adjustment: Option<Adjustment>,
807 call_kind: CallKind<'tcx>,
808) -> Body<'tcx> {
809 let (sig_args, untuple_args) = if let ty::InstanceKind::FnPtrShim(_, ty) = instance {
813 let sig = tcx.instantiate_bound_regions_with_erased(ty.fn_sig(tcx));
814
815 let untuple_args = sig.inputs();
816
817 let arg_tup = Ty::new_tup(tcx, untuple_args);
819
820 (Some([ty.into(), arg_tup.into()]), Some(untuple_args))
821 } else {
822 (None, None)
823 };
824
825 let def_id = instance.def_id();
826
827 let sig = tcx.fn_sig(def_id);
828 let sig = sig.map_bound(|sig| tcx.instantiate_bound_regions_with_erased(sig));
829
830 assert_eq!(sig_args.is_some(), !instance.has_polymorphic_mir_body());
831 let mut sig = if let Some(sig_args) = sig_args {
832 sig.instantiate(tcx, &sig_args).skip_norm_wip()
833 } else {
834 sig.instantiate_identity().skip_norm_wip()
835 };
836
837 if let CallKind::Indirect(fnty) = call_kind {
838 let mut inputs_and_output = sig.inputs_and_output.to_vec();
844
845 assert_eq!(inputs_and_output.len(), 3);
848
849 let self_arg = &mut inputs_and_output[0];
852 *self_arg = match rcvr_adjustment.unwrap() {
853 Adjustment::Identity => fnty,
854 Adjustment::Deref { source } => match source {
855 DerefSource::ImmRef => Ty::new_imm_ref(tcx, tcx.lifetimes.re_erased, fnty),
856 DerefSource::MutRef => Ty::new_mut_ref(tcx, tcx.lifetimes.re_erased, fnty),
857 DerefSource::MutPtr => Ty::new_mut_ptr(tcx, fnty),
858 },
859 Adjustment::RefMut => bug!("`RefMut` is never used with indirect calls: {instance:?}"),
860 };
861 sig.inputs_and_output = tcx.mk_type_list(&inputs_and_output);
862 }
863
864 if let ty::InstanceKind::VTableShim(..) = instance {
867 let mut inputs_and_output = sig.inputs_and_output.to_vec();
869 let self_arg = &mut inputs_and_output[0];
870 debug_assert!(tcx.generics_of(def_id).has_self && *self_arg == tcx.types.self_param);
871 *self_arg = Ty::new_mut_ptr(tcx, *self_arg);
872 sig.inputs_and_output = tcx.mk_type_list(&inputs_and_output);
873 }
874
875 let span = tcx.def_span(def_id);
876
877 debug!(?sig);
878
879 let mut local_decls = local_decls_for_sig(&sig, span);
880 let source_info = SourceInfo::outermost(span);
881
882 let destination = Place::return_place();
883
884 let rcvr_place = || {
885 assert!(rcvr_adjustment.is_some());
886 Place::from(Local::arg(0))
887 };
888 let mut statements = vec![];
889
890 let rcvr = rcvr_adjustment.map(|rcvr_adjustment| match rcvr_adjustment {
891 Adjustment::Identity => Operand::Move(rcvr_place()),
892 Adjustment::Deref { source: _ } => Operand::Move(tcx.mk_place_deref(rcvr_place())),
893 Adjustment::RefMut => {
894 let ref_rcvr = local_decls.push(
896 LocalDecl::new(
897 Ty::new_mut_ref(tcx, tcx.lifetimes.re_erased, sig.inputs()[0]),
898 span,
899 )
900 .immutable(),
901 );
902 let borrow_kind = BorrowKind::Mut { kind: MutBorrowKind::Default };
903 statements.push(Statement::new(
904 source_info,
905 StatementKind::Assign(Box::new((
906 Place::from(ref_rcvr),
907 Rvalue::Ref(tcx.lifetimes.re_erased, borrow_kind, rcvr_place()),
908 ))),
909 ));
910 Operand::Move(Place::from(ref_rcvr))
911 }
912 });
913
914 let (callee, mut args) = match call_kind {
915 CallKind::Indirect(_) => (rcvr.unwrap(), vec![]),
917
918 CallKind::Direct(def_id) => {
920 let ty = tcx.type_of(def_id).instantiate_identity().skip_norm_wip();
921 (
922 Operand::Constant(Box::new(ConstOperand {
923 span,
924 user_ty: None,
925 const_: Const::zero_sized(ty),
926 })),
927 rcvr.into_iter().collect::<Vec<_>>(),
928 )
929 }
930 };
931
932 let mut arg_range = 0..sig.inputs().len();
933
934 if rcvr_adjustment.is_some() {
936 arg_range.start += 1;
937 }
938
939 if untuple_args.is_some() {
941 arg_range.end -= 1;
942 }
943
944 args.extend(arg_range.map(|i| Operand::Move(Place::from(Local::arg(i)))));
946
947 if let Some(untuple_args) = untuple_args {
949 let tuple_arg = Local::arg(sig.inputs().len() - 1);
950 args.extend(untuple_args.iter().enumerate().map(|(i, ity)| {
951 Operand::Move(tcx.mk_place_field(Place::from(tuple_arg), FieldIdx::new(i), *ity))
952 }));
953 }
954
955 let n_blocks = if let Some(Adjustment::RefMut) = rcvr_adjustment { 5 } else { 2 };
956 let mut blocks = IndexVec::with_capacity(n_blocks);
957 let block = |blocks: &mut IndexVec<_, _>, statements, kind, is_cleanup| {
958 blocks.push(BasicBlockData::new_stmts(
959 statements,
960 Some(Terminator { source_info, kind }),
961 is_cleanup,
962 ))
963 };
964
965 let args = args.into_iter().map(|a| Spanned { node: a, span: DUMMY_SP }).collect();
967 block(
968 &mut blocks,
969 statements,
970 TerminatorKind::Call {
971 func: callee,
972 args,
973 destination,
974 target: Some(BasicBlock::new(1)),
975 unwind: if let Some(Adjustment::RefMut) = rcvr_adjustment {
976 UnwindAction::Cleanup(BasicBlock::new(3))
977 } else {
978 UnwindAction::Continue
979 },
980 call_source: CallSource::Misc,
981 fn_span: span,
982 },
983 false,
984 );
985
986 if let Some(Adjustment::RefMut) = rcvr_adjustment {
987 block(
989 &mut blocks,
990 vec![],
991 TerminatorKind::Drop {
992 place: rcvr_place(),
993 target: BasicBlock::new(2),
994 unwind: UnwindAction::Continue,
995 replace: false,
996 drop: None,
997 async_fut: None,
998 },
999 false,
1000 );
1001 }
1002 let stmts = vec![];
1004 block(&mut blocks, stmts, TerminatorKind::Return, false);
1005 if let Some(Adjustment::RefMut) = rcvr_adjustment {
1006 block(
1008 &mut blocks,
1009 vec![],
1010 TerminatorKind::Drop {
1011 place: rcvr_place(),
1012 target: BasicBlock::new(4),
1013 unwind: UnwindAction::Terminate(UnwindTerminateReason::InCleanup),
1014 replace: false,
1015 drop: None,
1016 async_fut: None,
1017 },
1018 true,
1019 );
1020
1021 block(&mut blocks, vec![], TerminatorKind::UnwindResume, true);
1023 }
1024
1025 let mut body =
1026 new_body(MirSource::from_instance(instance), blocks, local_decls, sig.inputs().len(), span);
1027
1028 if let ExternAbi::RustCall = sig.abi() {
1029 body.spread_arg = Some(Local::new(sig.inputs().len()));
1030 }
1031
1032 body
1033}
1034
1035pub(super) fn build_adt_ctor(tcx: TyCtxt<'_>, ctor_id: DefId) -> Body<'_> {
1036 if true {
if !tcx.is_constructor(ctor_id) {
::core::panicking::panic("assertion failed: tcx.is_constructor(ctor_id)")
};
};debug_assert!(tcx.is_constructor(ctor_id));
1037
1038 let typing_env = ty::TypingEnv::post_analysis(tcx, ctor_id);
1039
1040 let sig = tcx
1042 .fn_sig(ctor_id)
1043 .instantiate_identity()
1044 .skip_norm_wip()
1045 .no_bound_vars()
1046 .expect("LBR in ADT constructor signature");
1047 let sig = tcx.normalize_erasing_regions(typing_env, Unnormalized::new_wip(sig));
1048
1049 let ty::Adt(adt_def, args) = sig.output().kind() else {
1050 ::rustc_middle::util::bug::bug_fmt(format_args!("unexpected type for ADT ctor {0:?}",
sig.output()));bug!("unexpected type for ADT ctor {:?}", sig.output());
1051 };
1052
1053 {
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_mir_transform/src/shim.rs:1053",
"rustc_mir_transform::shim", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_mir_transform/src/shim.rs"),
::tracing_core::__macro_support::Option::Some(1053u32),
::tracing_core::__macro_support::Option::Some("rustc_mir_transform::shim"),
::tracing_core::field::FieldSet::new(&["message"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::EVENT)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let enabled =
::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
&&
::tracing::Level::DEBUG <=
::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!("build_ctor: ctor_id={0:?} sig={1:?}",
ctor_id, sig) as &dyn Value))])
});
} else { ; }
};debug!("build_ctor: ctor_id={:?} sig={:?}", ctor_id, sig);
1054
1055 let span = tcx.def_span(ctor_id);
1056
1057 let local_decls = local_decls_for_sig(&sig, span);
1058
1059 let source_info = SourceInfo::outermost(span);
1060
1061 let variant_index =
1062 if adt_def.is_enum() { adt_def.variant_index_with_ctor_id(ctor_id) } else { FIRST_VARIANT };
1063
1064 {
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_mir_transform/src/shim.rs:1070",
"rustc_mir_transform::shim", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_mir_transform/src/shim.rs"),
::tracing_core::__macro_support::Option::Some(1070u32),
::tracing_core::__macro_support::Option::Some("rustc_mir_transform::shim"),
::tracing_core::field::FieldSet::new(&["message"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::EVENT)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let enabled =
::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
&&
::tracing::Level::DEBUG <=
::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!("build_ctor: variant_index={0:?}",
variant_index) as &dyn Value))])
});
} else { ; }
};debug!("build_ctor: variant_index={:?}", variant_index);
1071
1072 let kind = AggregateKind::Adt(adt_def.did(), variant_index, args, None, None);
1073 let variant = adt_def.variant(variant_index);
1074 let statement = Statement::new(
1075 source_info,
1076 StatementKind::Assign(Box::new((
1077 Place::return_place(),
1078 Rvalue::Aggregate(
1079 Box::new(kind),
1080 (0..variant.fields.len())
1081 .map(|idx| Operand::Move(Place::from(Local::arg(idx))))
1082 .collect(),
1083 ),
1084 ))),
1085 );
1086
1087 let start_block = BasicBlockData::new_stmts(
1088 ::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[statement]))vec![statement],
1089 Some(Terminator { source_info, kind: TerminatorKind::Return }),
1090 false,
1091 );
1092
1093 let source = MirSource::item(ctor_id);
1094 let mut body = new_body(
1095 source,
1096 IndexVec::from_elem_n(start_block, 1),
1097 local_decls,
1098 sig.inputs().len(),
1099 span,
1100 );
1101 body.set_mentioned_items(Vec::new());
1104
1105 crate::pass_manager::dump_mir_for_phase_change(tcx, &body);
1106
1107 body
1108}
1109
1110fn build_fn_ptr_addr_shim<'tcx>(tcx: TyCtxt<'tcx>, def_id: DefId, self_ty: Ty<'tcx>) -> Body<'tcx> {
1118 {
match self_ty.kind() {
ty::FnPtr(..) => {}
ref left_val => {
::core::panicking::assert_matches_failed(left_val,
"ty::FnPtr(..)",
::core::option::Option::Some(format_args!("expected fn ptr, found {0}",
self_ty)));
}
}
};assert_matches!(self_ty.kind(), ty::FnPtr(..), "expected fn ptr, found {self_ty}");
1119 let span = tcx.def_span(def_id);
1120 let Some(sig) =
1121 tcx.fn_sig(def_id).instantiate(tcx, &[self_ty.into()]).skip_norm_wip().no_bound_vars()
1122 else {
1123 ::rustc_middle::util::bug::span_bug_fmt(span,
format_args!("FnPtr::addr with bound vars for `{0}`", self_ty));span_bug!(span, "FnPtr::addr with bound vars for `{self_ty}`");
1124 };
1125 let locals = local_decls_for_sig(&sig, span);
1126
1127 let source_info = SourceInfo::outermost(span);
1128 let rvalue = Rvalue::Cast(
1131 CastKind::FnPtrToPtr,
1132 Operand::Move(Place::from(Local::arg(0))),
1133 Ty::new_imm_ptr(tcx, tcx.types.unit),
1134 );
1135 let stmt = Statement::new(
1136 source_info,
1137 StatementKind::Assign(Box::new((Place::return_place(), rvalue))),
1138 );
1139 let statements = ::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[stmt]))vec![stmt];
1140 let start_block = BasicBlockData::new_stmts(
1141 statements,
1142 Some(Terminator { source_info, kind: TerminatorKind::Return }),
1143 false,
1144 );
1145 let source = MirSource::from_instance(ty::InstanceKind::FnPtrAddrShim(def_id, self_ty));
1146 new_body(source, IndexVec::from_elem_n(start_block, 1), locals, sig.inputs().len(), span)
1147}
1148
1149fn build_construct_coroutine_by_move_shim<'tcx>(
1150 tcx: TyCtxt<'tcx>,
1151 coroutine_closure_def_id: DefId,
1152 receiver_by_ref: bool,
1153) -> Body<'tcx> {
1154 let mut self_ty = tcx.type_of(coroutine_closure_def_id).instantiate_identity().skip_norm_wip();
1155 let mut self_local: Place<'tcx> = Local::arg(0).into();
1156 let ty::CoroutineClosure(_, args) = *self_ty.kind() else {
1157 ::rustc_middle::util::bug::bug_fmt(format_args!("impossible case reached"));bug!();
1158 };
1159
1160 if receiver_by_ref {
1166 self_local = tcx.mk_place_deref(self_local);
1167 self_ty = Ty::new_imm_ref(tcx, tcx.lifetimes.re_erased, self_ty);
1168 }
1169
1170 let poly_sig = args.as_coroutine_closure().coroutine_closure_sig().map_bound(|sig| {
1171 tcx.mk_fn_sig(
1172 [self_ty].into_iter().chain(sig.tupled_inputs_ty.tuple_fields()),
1173 sig.to_coroutine_given_kind_and_upvars(
1174 tcx,
1175 args.as_coroutine_closure().parent_args(),
1176 tcx.coroutine_for_closure(coroutine_closure_def_id),
1177 ty::ClosureKind::FnOnce,
1178 tcx.lifetimes.re_erased,
1179 args.as_coroutine_closure().tupled_upvars_ty(),
1180 args.as_coroutine_closure().coroutine_captures_by_ref_ty(),
1181 ),
1182 sig.fn_sig_kind,
1183 )
1184 });
1185 let sig = tcx.liberate_late_bound_regions(coroutine_closure_def_id, poly_sig);
1186 let ty::Coroutine(coroutine_def_id, coroutine_args) = *sig.output().kind() else {
1187 ::rustc_middle::util::bug::bug_fmt(format_args!("impossible case reached"));bug!();
1188 };
1189
1190 let span = tcx.def_span(coroutine_closure_def_id);
1191 let locals = local_decls_for_sig(&sig, span);
1192
1193 let mut fields = ::alloc::vec::Vec::new()vec![];
1194
1195 for idx in 1..sig.inputs().len() {
1197 fields.push(Operand::Move(Local::arg(idx).into()));
1198 }
1199
1200 for (idx, ty) in args.as_coroutine_closure().upvar_tys().iter().enumerate() {
1201 if receiver_by_ref {
1202 if !#[allow(non_exhaustive_omitted_patterns)] match ty.kind() {
ty::Ref(_, _, hir::Mutability::Not) => true,
_ => false,
}matches!(ty.kind(), ty::Ref(_, _, hir::Mutability::Not)) {
1206 tcx.dcx().delayed_bug(::alloc::__export::must_use({
::alloc::fmt::format(format_args!("field should be captured by immutable ref if we have an `Fn` instance, but it was: {0}",
ty))
})format!(
1210 "field should be captured by immutable ref if we have \
1211 an `Fn` instance, but it was: {ty}"
1212 ));
1213 }
1214 fields.push(Operand::Copy(tcx.mk_place_field(
1215 self_local,
1216 FieldIdx::from_usize(idx),
1217 ty,
1218 )));
1219 } else {
1220 fields.push(Operand::Move(tcx.mk_place_field(
1221 self_local,
1222 FieldIdx::from_usize(idx),
1223 ty,
1224 )));
1225 }
1226 }
1227
1228 let source_info = SourceInfo::outermost(span);
1229 let rvalue = Rvalue::Aggregate(
1230 Box::new(AggregateKind::Coroutine(coroutine_def_id, coroutine_args)),
1231 IndexVec::from_raw(fields),
1232 );
1233 let stmt = Statement::new(
1234 source_info,
1235 StatementKind::Assign(Box::new((Place::return_place(), rvalue))),
1236 );
1237 let statements = ::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[stmt]))vec![stmt];
1238 let start_block = BasicBlockData::new_stmts(
1239 statements,
1240 Some(Terminator { source_info, kind: TerminatorKind::Return }),
1241 false,
1242 );
1243
1244 let source = MirSource::from_instance(ty::InstanceKind::ConstructCoroutineInClosureShim {
1245 coroutine_closure_def_id,
1246 receiver_by_ref,
1247 });
1248
1249 let body =
1250 new_body(source, IndexVec::from_elem_n(start_block, 1), locals, sig.inputs().len(), span);
1251
1252 let pass_name =
1253 if receiver_by_ref { "coroutine_closure_by_ref" } else { "coroutine_closure_by_move" };
1254 if let Some(dumper) = MirDumper::new(tcx, pass_name, &body) {
1255 dumper.dump_mir(&body);
1256 }
1257
1258 body
1259}