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 block(&mut blocks, TerminatorKind::Goto { target: return_block });
321 block(&mut blocks, TerminatorKind::Return);
322
323 let source = MirSource::from_instance(ty::InstanceKind::DropGlue(def_id, ty));
324 let mut body =
325 new_body(source, blocks, local_decls_for_sig(&sig, span), sig.inputs().len(), span);
326
327 let dropee_ptr = Place::from(Local::arg(0));
329
330 if ty.is_some() {
331 let patch = {
332 let typing_env = ty::TypingEnv::post_analysis(tcx, def_id);
333 let mut elaborator = DropShimElaborator {
334 body: &body,
335 patch: MirPatch::new(&body),
336 tcx,
337 typing_env,
338 produce_async_drops: false,
339 };
340 let dropee = tcx.mk_place_deref(dropee_ptr);
341 let resume_block = elaborator.patch.resume_block();
342 elaborate_drop(
343 &mut elaborator,
344 source_info,
345 dropee,
346 (),
347 return_block,
348 Unwind::To(resume_block),
349 START_BLOCK,
350 None,
351 );
352 elaborator.patch
353 };
354 patch.apply(&mut body);
355 }
356
357 body
358}
359
360fn new_body<'tcx>(
361 source: MirSource<'tcx>,
362 basic_blocks: IndexVec<BasicBlock, BasicBlockData<'tcx>>,
363 local_decls: IndexVec<Local, LocalDecl<'tcx>>,
364 arg_count: usize,
365 span: Span,
366) -> Body<'tcx> {
367 let mut body = Body::new(
368 source,
369 basic_blocks,
370 IndexVec::from_elem_n(
371 SourceScopeData {
372 span,
373 parent_scope: None,
374 inlined: None,
375 inlined_parent_scope: None,
376 local_data: ClearCrossCrate::Clear,
377 },
378 1,
379 ),
380 local_decls,
381 IndexVec::new(),
382 arg_count,
383 ::alloc::vec::Vec::new()vec![],
384 span,
385 None,
386 None,
388 );
389 body.set_required_consts(Vec::new());
391 body
392}
393
394pub(super) struct DropShimElaborator<'a, 'tcx> {
395 pub body: &'a Body<'tcx>,
396 pub patch: MirPatch<'tcx>,
397 pub tcx: TyCtxt<'tcx>,
398 pub typing_env: ty::TypingEnv<'tcx>,
399 pub produce_async_drops: bool,
400}
401
402impl fmt::Debug for DropShimElaborator<'_, '_> {
403 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> Result<(), fmt::Error> {
404 f.debug_struct("DropShimElaborator").finish_non_exhaustive()
405 }
406}
407
408impl<'a, 'tcx> DropElaborator<'a, 'tcx> for DropShimElaborator<'a, 'tcx> {
409 type Path = ();
410
411 fn patch_ref(&self) -> &MirPatch<'tcx> {
412 &self.patch
413 }
414 fn patch(&mut self) -> &mut MirPatch<'tcx> {
415 &mut self.patch
416 }
417 fn body(&self) -> &'a Body<'tcx> {
418 self.body
419 }
420 fn tcx(&self) -> TyCtxt<'tcx> {
421 self.tcx
422 }
423 fn typing_env(&self) -> ty::TypingEnv<'tcx> {
424 self.typing_env
425 }
426
427 fn terminator_loc(&self, bb: BasicBlock) -> Location {
428 self.patch.terminator_loc(self.body, bb)
429 }
430 fn allow_async_drops(&self) -> bool {
431 self.produce_async_drops
432 }
433
434 fn drop_style(&self, _path: Self::Path, mode: DropFlagMode) -> DropStyle {
435 match mode {
436 DropFlagMode::Shallow => {
437 DropStyle::Static
440 }
441 DropFlagMode::Deep => {
442 DropStyle::Open
445 }
446 }
447 }
448
449 fn get_drop_flag(&mut self, _path: Self::Path) -> Option<Operand<'tcx>> {
450 None
451 }
452
453 fn clear_drop_flag(&mut self, _location: Location, _path: Self::Path, _mode: DropFlagMode) {}
454
455 fn field_subpath(&self, _path: Self::Path, _field: FieldIdx) -> Option<Self::Path> {
456 None
457 }
458 fn deref_subpath(&self, _path: Self::Path) -> Option<Self::Path> {
459 None
460 }
461 fn downcast_subpath(&self, _path: Self::Path, _variant: VariantIdx) -> Option<Self::Path> {
462 Some(())
463 }
464 fn array_subpath(&self, _path: Self::Path, _index: u64, _size: u64) -> Option<Self::Path> {
465 None
466 }
467}
468
469fn build_thread_local_shim<'tcx>(
470 tcx: TyCtxt<'tcx>,
471 instance: ty::InstanceKind<'tcx>,
472) -> Body<'tcx> {
473 let def_id = instance.def_id();
474
475 let span = tcx.def_span(def_id);
476 let source_info = SourceInfo::outermost(span);
477
478 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(
479 vec![Statement::new(
480 source_info,
481 StatementKind::Assign(Box::new((
482 Place::return_place(),
483 Rvalue::ThreadLocalRef(def_id),
484 ))),
485 )],
486 Some(Terminator { source_info, kind: TerminatorKind::Return }),
487 false,
488 )]);
489
490 new_body(
491 MirSource::from_instance(instance),
492 blocks,
493 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)]),
494 0,
495 span,
496 )
497}
498
499fn build_clone_shim<'tcx>(tcx: TyCtxt<'tcx>, def_id: DefId, self_ty: Ty<'tcx>) -> Body<'tcx> {
501 {
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:501",
"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(501u32),
::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);
502
503 let mut builder = CloneShimBuilder::new(tcx, def_id, self_ty);
504
505 let dest = Place::return_place();
506 let src = tcx.mk_place_deref(Place::from(Local::arg(0)));
507
508 match self_ty.kind() {
509 ty::FnDef(..) | ty::FnPtr(..) => builder.copy_shim(),
510 ty::Closure(_, args) => builder.tuple_like_shim(dest, src, args.as_closure().upvar_tys()),
511 ty::CoroutineClosure(_, args) => {
512 builder.tuple_like_shim(dest, src, args.as_coroutine_closure().upvar_tys())
513 }
514 ty::Tuple(..) => builder.tuple_like_shim(dest, src, self_ty.tuple_fields()),
515 ty::Coroutine(coroutine_def_id, args) => {
516 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);
517 builder.coroutine_shim(dest, src, *coroutine_def_id, args.as_coroutine())
518 }
519 _ => ::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),
520 };
521
522 builder.into_mir()
523}
524
525struct CloneShimBuilder<'tcx> {
526 tcx: TyCtxt<'tcx>,
527 def_id: DefId,
528 local_decls: IndexVec<Local, LocalDecl<'tcx>>,
529 blocks: IndexVec<BasicBlock, BasicBlockData<'tcx>>,
530 span: Span,
531 sig: ty::FnSig<'tcx>,
532}
533
534impl<'tcx> CloneShimBuilder<'tcx> {
535 fn new(tcx: TyCtxt<'tcx>, def_id: DefId, self_ty: Ty<'tcx>) -> Self {
536 let sig = tcx.fn_sig(def_id).instantiate(tcx, &[self_ty.into()]).skip_norm_wip();
540 let sig = tcx.instantiate_bound_regions_with_erased(sig);
541 let span = tcx.def_span(def_id);
542
543 CloneShimBuilder {
544 tcx,
545 def_id,
546 local_decls: local_decls_for_sig(&sig, span),
547 blocks: IndexVec::new(),
548 span,
549 sig,
550 }
551 }
552
553 fn into_mir(self) -> Body<'tcx> {
554 let source = MirSource::from_instance(ty::InstanceKind::CloneShim(
555 self.def_id,
556 self.sig.inputs_and_output[0],
557 ));
558 new_body(source, self.blocks, self.local_decls, self.sig.inputs().len(), self.span)
559 }
560
561 fn source_info(&self) -> SourceInfo {
562 SourceInfo::outermost(self.span)
563 }
564
565 fn block(
566 &mut self,
567 statements: Vec<Statement<'tcx>>,
568 kind: TerminatorKind<'tcx>,
569 is_cleanup: bool,
570 ) -> BasicBlock {
571 let source_info = self.source_info();
572 self.blocks.push(BasicBlockData::new_stmts(
573 statements,
574 Some(Terminator { source_info, kind }),
575 is_cleanup,
576 ))
577 }
578
579 fn block_index_offset(&self, offset: usize) -> BasicBlock {
584 BasicBlock::new(self.blocks.len() + offset)
585 }
586
587 fn make_statement(&self, kind: StatementKind<'tcx>) -> Statement<'tcx> {
588 Statement::new(self.source_info(), kind)
589 }
590
591 fn copy_shim(&mut self) {
592 let rcvr = self.tcx.mk_place_deref(Place::from(Local::arg(0)));
593 let ret_statement = self.make_statement(StatementKind::Assign(Box::new((
594 Place::return_place(),
595 Rvalue::Use(Operand::Copy(rcvr), WithRetag::Yes),
596 ))));
597 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);
598 }
599
600 fn make_place(&mut self, mutability: Mutability, ty: Ty<'tcx>) -> Place<'tcx> {
601 let span = self.span;
602 let mut local = LocalDecl::new(ty, span);
603 if mutability.is_not() {
604 local = local.immutable();
605 }
606 Place::from(self.local_decls.push(local))
607 }
608
609 fn make_clone_call(
610 &mut self,
611 dest: Place<'tcx>,
612 src: Place<'tcx>,
613 ty: Ty<'tcx>,
614 next: BasicBlock,
615 cleanup: BasicBlock,
616 ) {
617 let tcx = self.tcx;
618
619 let func_ty = Ty::new_fn_def(tcx, self.def_id, [ty]);
621 let func = Operand::Constant(Box::new(ConstOperand {
622 span: self.span,
623 user_ty: None,
624 const_: Const::zero_sized(func_ty),
625 }));
626
627 let ref_loc =
628 self.make_place(Mutability::Not, Ty::new_imm_ref(tcx, tcx.lifetimes.re_erased, ty));
629
630 let statement = self.make_statement(StatementKind::Assign(Box::new((
632 ref_loc,
633 Rvalue::Ref(tcx.lifetimes.re_erased, BorrowKind::Shared, src),
634 ))));
635
636 self.block(
638 ::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[statement]))vec![statement],
639 TerminatorKind::Call {
640 func,
641 args: [Spanned { node: Operand::Move(ref_loc), span: DUMMY_SP }].into(),
642 destination: dest,
643 target: Some(next),
644 unwind: UnwindAction::Cleanup(cleanup),
645 call_source: CallSource::Normal,
646 fn_span: self.span,
647 },
648 false,
649 );
650 }
651
652 fn clone_fields<I>(
653 &mut self,
654 dest: Place<'tcx>,
655 src: Place<'tcx>,
656 target: BasicBlock,
657 mut unwind: BasicBlock,
658 tys: I,
659 ) -> BasicBlock
660 where
661 I: IntoIterator<Item = Ty<'tcx>>,
662 {
663 for (i, ity) in tys.into_iter().enumerate() {
665 let field = FieldIdx::new(i);
676 let src_field = self.tcx.mk_place_field(src, field, ity);
677
678 let dest_field = self.tcx.mk_place_field(dest, field, ity);
679
680 let next_unwind = self.block_index_offset(1);
681 let next_block = self.block_index_offset(2);
682 self.make_clone_call(dest_field, src_field, ity, next_block, unwind);
683 self.block(
684 ::alloc::vec::Vec::new()vec![],
685 TerminatorKind::Drop {
686 place: dest_field,
687 target: unwind,
688 unwind: UnwindAction::Terminate(UnwindTerminateReason::InCleanup),
689 replace: false,
690 drop: None,
691 async_fut: None,
692 },
693 true,
694 );
695 unwind = next_unwind;
696 }
697 self.block(::alloc::vec::Vec::new()vec![], TerminatorKind::Goto { target }, false);
699 unwind
700 }
701
702 fn tuple_like_shim<I>(&mut self, dest: Place<'tcx>, src: Place<'tcx>, tys: I)
703 where
704 I: IntoIterator<Item = Ty<'tcx>>,
705 {
706 self.block(::alloc::vec::Vec::new()vec![], TerminatorKind::Goto { target: self.block_index_offset(3) }, false);
707 let unwind = self.block(::alloc::vec::Vec::new()vec![], TerminatorKind::UnwindResume, true);
708 let target = self.block(::alloc::vec::Vec::new()vec![], TerminatorKind::Return, false);
709
710 let _final_cleanup_block = self.clone_fields(dest, src, target, unwind, tys);
711 }
712
713 fn coroutine_shim(
714 &mut self,
715 dest: Place<'tcx>,
716 src: Place<'tcx>,
717 coroutine_def_id: DefId,
718 args: CoroutineArgs<TyCtxt<'tcx>>,
719 ) {
720 self.block(::alloc::vec::Vec::new()vec![], TerminatorKind::Goto { target: self.block_index_offset(3) }, false);
721 let unwind = self.block(::alloc::vec::Vec::new()vec![], TerminatorKind::UnwindResume, true);
722 let switch = self.block(::alloc::vec::Vec::new()vec![], TerminatorKind::Unreachable, false);
724 let unwind = self.clone_fields(dest, src, switch, unwind, args.upvar_tys());
725 let target = self.block(::alloc::vec::Vec::new()vec![], TerminatorKind::Return, false);
726 let unreachable = self.block(::alloc::vec::Vec::new()vec![], TerminatorKind::Unreachable, false);
727 let mut cases = Vec::with_capacity(args.state_tys(coroutine_def_id, self.tcx).count());
728 for (index, state_tys) in args.state_tys(coroutine_def_id, self.tcx).enumerate() {
729 let variant_index = VariantIdx::new(index);
730 let dest = self.tcx.mk_place_downcast_unnamed(dest, variant_index);
731 let src = self.tcx.mk_place_downcast_unnamed(src, variant_index);
732 let clone_block = self.block_index_offset(1);
733 let start_block = self.block(
734 ::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 {
735 place: Box::new(Place::return_place()),
736 variant_index,
737 })],
738 TerminatorKind::Goto { target: clone_block },
739 false,
740 );
741 cases.push((index as u128, start_block));
742 let _final_cleanup_block = self.clone_fields(dest, src, target, unwind, state_tys);
743 }
744 let discr_ty = args.discr_ty(self.tcx);
745 let temp = self.make_place(Mutability::Mut, discr_ty);
746 let rvalue = Rvalue::Discriminant(src);
747 let statement = self.make_statement(StatementKind::Assign(Box::new((temp, rvalue))));
748 match &mut self.blocks[switch] {
749 BasicBlockData { statements, terminator: Some(Terminator { kind, .. }), .. } => {
750 statements.push(statement);
751 *kind = TerminatorKind::SwitchInt {
752 discr: Operand::Move(temp),
753 targets: SwitchTargets::new(cases.into_iter(), unreachable),
754 };
755 }
756 BasicBlockData { terminator: None, .. } => ::core::panicking::panic("internal error: entered unreachable code")unreachable!(),
757 }
758 }
759}
760
761x;#[instrument(level = "debug", skip(tcx), ret)]
764fn build_call_shim<'tcx>(
765 tcx: TyCtxt<'tcx>,
766 instance: ty::InstanceKind<'tcx>,
767 rcvr_adjustment: Option<Adjustment>,
768 call_kind: CallKind<'tcx>,
769) -> Body<'tcx> {
770 let (sig_args, untuple_args) = if let ty::InstanceKind::FnPtrShim(_, ty) = instance {
774 let sig = tcx.instantiate_bound_regions_with_erased(ty.fn_sig(tcx));
775
776 let untuple_args = sig.inputs();
777
778 let arg_tup = Ty::new_tup(tcx, untuple_args);
780
781 (Some([ty.into(), arg_tup.into()]), Some(untuple_args))
782 } else {
783 (None, None)
784 };
785
786 let def_id = instance.def_id();
787
788 let sig = tcx.fn_sig(def_id);
789 let sig = sig.map_bound(|sig| tcx.instantiate_bound_regions_with_erased(sig));
790
791 assert_eq!(sig_args.is_some(), !instance.has_polymorphic_mir_body());
792 let mut sig = if let Some(sig_args) = sig_args {
793 sig.instantiate(tcx, &sig_args).skip_norm_wip()
794 } else {
795 sig.instantiate_identity().skip_norm_wip()
796 };
797
798 if let CallKind::Indirect(fnty) = call_kind {
799 let mut inputs_and_output = sig.inputs_and_output.to_vec();
805
806 assert_eq!(inputs_and_output.len(), 3);
809
810 let self_arg = &mut inputs_and_output[0];
813 *self_arg = match rcvr_adjustment.unwrap() {
814 Adjustment::Identity => fnty,
815 Adjustment::Deref { source } => match source {
816 DerefSource::ImmRef => Ty::new_imm_ref(tcx, tcx.lifetimes.re_erased, fnty),
817 DerefSource::MutRef => Ty::new_mut_ref(tcx, tcx.lifetimes.re_erased, fnty),
818 DerefSource::MutPtr => Ty::new_mut_ptr(tcx, fnty),
819 },
820 Adjustment::RefMut => bug!("`RefMut` is never used with indirect calls: {instance:?}"),
821 };
822 sig.inputs_and_output = tcx.mk_type_list(&inputs_and_output);
823 }
824
825 if let ty::InstanceKind::VTableShim(..) = instance {
828 let mut inputs_and_output = sig.inputs_and_output.to_vec();
830 let self_arg = &mut inputs_and_output[0];
831 debug_assert!(tcx.generics_of(def_id).has_self && *self_arg == tcx.types.self_param);
832 *self_arg = Ty::new_mut_ptr(tcx, *self_arg);
833 sig.inputs_and_output = tcx.mk_type_list(&inputs_and_output);
834 }
835
836 let span = tcx.def_span(def_id);
837
838 debug!(?sig);
839
840 let mut local_decls = local_decls_for_sig(&sig, span);
841 let source_info = SourceInfo::outermost(span);
842
843 let destination = Place::return_place();
844
845 let rcvr_place = || {
846 assert!(rcvr_adjustment.is_some());
847 Place::from(Local::arg(0))
848 };
849 let mut statements = vec![];
850
851 let rcvr = rcvr_adjustment.map(|rcvr_adjustment| match rcvr_adjustment {
852 Adjustment::Identity => Operand::Move(rcvr_place()),
853 Adjustment::Deref { source: _ } => Operand::Move(tcx.mk_place_deref(rcvr_place())),
854 Adjustment::RefMut => {
855 let ref_rcvr = local_decls.push(
857 LocalDecl::new(
858 Ty::new_mut_ref(tcx, tcx.lifetimes.re_erased, sig.inputs()[0]),
859 span,
860 )
861 .immutable(),
862 );
863 let borrow_kind = BorrowKind::Mut { kind: MutBorrowKind::Default };
864 statements.push(Statement::new(
865 source_info,
866 StatementKind::Assign(Box::new((
867 Place::from(ref_rcvr),
868 Rvalue::Ref(tcx.lifetimes.re_erased, borrow_kind, rcvr_place()),
869 ))),
870 ));
871 Operand::Move(Place::from(ref_rcvr))
872 }
873 });
874
875 let (callee, mut args) = match call_kind {
876 CallKind::Indirect(_) => (rcvr.unwrap(), vec![]),
878
879 CallKind::Direct(def_id) => {
881 let ty = tcx.type_of(def_id).instantiate_identity().skip_norm_wip();
882 (
883 Operand::Constant(Box::new(ConstOperand {
884 span,
885 user_ty: None,
886 const_: Const::zero_sized(ty),
887 })),
888 rcvr.into_iter().collect::<Vec<_>>(),
889 )
890 }
891 };
892
893 let mut arg_range = 0..sig.inputs().len();
894
895 if rcvr_adjustment.is_some() {
897 arg_range.start += 1;
898 }
899
900 if untuple_args.is_some() {
902 arg_range.end -= 1;
903 }
904
905 args.extend(arg_range.map(|i| Operand::Move(Place::from(Local::arg(i)))));
907
908 if let Some(untuple_args) = untuple_args {
910 let tuple_arg = Local::arg(sig.inputs().len() - 1);
911 args.extend(untuple_args.iter().enumerate().map(|(i, ity)| {
912 Operand::Move(tcx.mk_place_field(Place::from(tuple_arg), FieldIdx::new(i), *ity))
913 }));
914 }
915
916 let n_blocks = if let Some(Adjustment::RefMut) = rcvr_adjustment { 5 } else { 2 };
917 let mut blocks = IndexVec::with_capacity(n_blocks);
918 let block = |blocks: &mut IndexVec<_, _>, statements, kind, is_cleanup| {
919 blocks.push(BasicBlockData::new_stmts(
920 statements,
921 Some(Terminator { source_info, kind }),
922 is_cleanup,
923 ))
924 };
925
926 let args = args.into_iter().map(|a| Spanned { node: a, span: DUMMY_SP }).collect();
928 block(
929 &mut blocks,
930 statements,
931 TerminatorKind::Call {
932 func: callee,
933 args,
934 destination,
935 target: Some(BasicBlock::new(1)),
936 unwind: if let Some(Adjustment::RefMut) = rcvr_adjustment {
937 UnwindAction::Cleanup(BasicBlock::new(3))
938 } else {
939 UnwindAction::Continue
940 },
941 call_source: CallSource::Misc,
942 fn_span: span,
943 },
944 false,
945 );
946
947 if let Some(Adjustment::RefMut) = rcvr_adjustment {
948 block(
950 &mut blocks,
951 vec![],
952 TerminatorKind::Drop {
953 place: rcvr_place(),
954 target: BasicBlock::new(2),
955 unwind: UnwindAction::Continue,
956 replace: false,
957 drop: None,
958 async_fut: None,
959 },
960 false,
961 );
962 }
963 let stmts = vec![];
965 block(&mut blocks, stmts, TerminatorKind::Return, false);
966 if let Some(Adjustment::RefMut) = rcvr_adjustment {
967 block(
969 &mut blocks,
970 vec![],
971 TerminatorKind::Drop {
972 place: rcvr_place(),
973 target: BasicBlock::new(4),
974 unwind: UnwindAction::Terminate(UnwindTerminateReason::InCleanup),
975 replace: false,
976 drop: None,
977 async_fut: None,
978 },
979 true,
980 );
981
982 block(&mut blocks, vec![], TerminatorKind::UnwindResume, true);
984 }
985
986 let mut body =
987 new_body(MirSource::from_instance(instance), blocks, local_decls, sig.inputs().len(), span);
988
989 if let ExternAbi::RustCall = sig.abi() {
990 body.spread_arg = Some(Local::new(sig.inputs().len()));
991 }
992
993 body
994}
995
996pub(super) fn build_adt_ctor(tcx: TyCtxt<'_>, ctor_id: DefId) -> Body<'_> {
997 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));
998
999 let typing_env = ty::TypingEnv::post_analysis(tcx, ctor_id);
1000
1001 let sig = tcx
1003 .fn_sig(ctor_id)
1004 .instantiate_identity()
1005 .skip_norm_wip()
1006 .no_bound_vars()
1007 .expect("LBR in ADT constructor signature");
1008 let sig = tcx.normalize_erasing_regions(typing_env, Unnormalized::new_wip(sig));
1009
1010 let ty::Adt(adt_def, args) = sig.output().kind() else {
1011 ::rustc_middle::util::bug::bug_fmt(format_args!("unexpected type for ADT ctor {0:?}",
sig.output()));bug!("unexpected type for ADT ctor {:?}", sig.output());
1012 };
1013
1014 {
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:1014",
"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(1014u32),
::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);
1015
1016 let span = tcx.def_span(ctor_id);
1017
1018 let local_decls = local_decls_for_sig(&sig, span);
1019
1020 let source_info = SourceInfo::outermost(span);
1021
1022 let variant_index =
1023 if adt_def.is_enum() { adt_def.variant_index_with_ctor_id(ctor_id) } else { FIRST_VARIANT };
1024
1025 {
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:1031",
"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(1031u32),
::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);
1032
1033 let kind = AggregateKind::Adt(adt_def.did(), variant_index, args, None, None);
1034 let variant = adt_def.variant(variant_index);
1035 let statement = Statement::new(
1036 source_info,
1037 StatementKind::Assign(Box::new((
1038 Place::return_place(),
1039 Rvalue::Aggregate(
1040 Box::new(kind),
1041 (0..variant.fields.len())
1042 .map(|idx| Operand::Move(Place::from(Local::arg(idx))))
1043 .collect(),
1044 ),
1045 ))),
1046 );
1047
1048 let start_block = BasicBlockData::new_stmts(
1049 ::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[statement]))vec![statement],
1050 Some(Terminator { source_info, kind: TerminatorKind::Return }),
1051 false,
1052 );
1053
1054 let source = MirSource::item(ctor_id);
1055 let mut body = new_body(
1056 source,
1057 IndexVec::from_elem_n(start_block, 1),
1058 local_decls,
1059 sig.inputs().len(),
1060 span,
1061 );
1062 body.set_mentioned_items(Vec::new());
1065
1066 crate::pass_manager::dump_mir_for_phase_change(tcx, &body);
1067
1068 body
1069}
1070
1071fn build_fn_ptr_addr_shim<'tcx>(tcx: TyCtxt<'tcx>, def_id: DefId, self_ty: Ty<'tcx>) -> Body<'tcx> {
1079 {
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}");
1080 let span = tcx.def_span(def_id);
1081 let Some(sig) =
1082 tcx.fn_sig(def_id).instantiate(tcx, &[self_ty.into()]).skip_norm_wip().no_bound_vars()
1083 else {
1084 ::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}`");
1085 };
1086 let locals = local_decls_for_sig(&sig, span);
1087
1088 let source_info = SourceInfo::outermost(span);
1089 let rvalue = Rvalue::Cast(
1092 CastKind::FnPtrToPtr,
1093 Operand::Move(Place::from(Local::arg(0))),
1094 Ty::new_imm_ptr(tcx, tcx.types.unit),
1095 );
1096 let stmt = Statement::new(
1097 source_info,
1098 StatementKind::Assign(Box::new((Place::return_place(), rvalue))),
1099 );
1100 let statements = ::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[stmt]))vec![stmt];
1101 let start_block = BasicBlockData::new_stmts(
1102 statements,
1103 Some(Terminator { source_info, kind: TerminatorKind::Return }),
1104 false,
1105 );
1106 let source = MirSource::from_instance(ty::InstanceKind::FnPtrAddrShim(def_id, self_ty));
1107 new_body(source, IndexVec::from_elem_n(start_block, 1), locals, sig.inputs().len(), span)
1108}
1109
1110fn build_construct_coroutine_by_move_shim<'tcx>(
1111 tcx: TyCtxt<'tcx>,
1112 coroutine_closure_def_id: DefId,
1113 receiver_by_ref: bool,
1114) -> Body<'tcx> {
1115 let mut self_ty = tcx.type_of(coroutine_closure_def_id).instantiate_identity().skip_norm_wip();
1116 let mut self_local: Place<'tcx> = Local::arg(0).into();
1117 let ty::CoroutineClosure(_, args) = *self_ty.kind() else {
1118 ::rustc_middle::util::bug::bug_fmt(format_args!("impossible case reached"));bug!();
1119 };
1120
1121 if receiver_by_ref {
1127 self_local = tcx.mk_place_deref(self_local);
1128 self_ty = Ty::new_imm_ref(tcx, tcx.lifetimes.re_erased, self_ty);
1129 }
1130
1131 let poly_sig = args.as_coroutine_closure().coroutine_closure_sig().map_bound(|sig| {
1132 tcx.mk_fn_sig(
1133 [self_ty].into_iter().chain(sig.tupled_inputs_ty.tuple_fields()),
1134 sig.to_coroutine_given_kind_and_upvars(
1135 tcx,
1136 args.as_coroutine_closure().parent_args(),
1137 tcx.coroutine_for_closure(coroutine_closure_def_id),
1138 ty::ClosureKind::FnOnce,
1139 tcx.lifetimes.re_erased,
1140 args.as_coroutine_closure().tupled_upvars_ty(),
1141 args.as_coroutine_closure().coroutine_captures_by_ref_ty(),
1142 ),
1143 sig.fn_sig_kind,
1144 )
1145 });
1146 let sig = tcx.liberate_late_bound_regions(coroutine_closure_def_id, poly_sig);
1147 let ty::Coroutine(coroutine_def_id, coroutine_args) = *sig.output().kind() else {
1148 ::rustc_middle::util::bug::bug_fmt(format_args!("impossible case reached"));bug!();
1149 };
1150
1151 let span = tcx.def_span(coroutine_closure_def_id);
1152 let locals = local_decls_for_sig(&sig, span);
1153
1154 let mut fields = ::alloc::vec::Vec::new()vec![];
1155
1156 for idx in 1..sig.inputs().len() {
1158 fields.push(Operand::Move(Local::arg(idx).into()));
1159 }
1160
1161 for (idx, ty) in args.as_coroutine_closure().upvar_tys().iter().enumerate() {
1162 if receiver_by_ref {
1163 if !#[allow(non_exhaustive_omitted_patterns)] match ty.kind() {
ty::Ref(_, _, hir::Mutability::Not) => true,
_ => false,
}matches!(ty.kind(), ty::Ref(_, _, hir::Mutability::Not)) {
1167 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!(
1171 "field should be captured by immutable ref if we have \
1172 an `Fn` instance, but it was: {ty}"
1173 ));
1174 }
1175 fields.push(Operand::Copy(tcx.mk_place_field(
1176 self_local,
1177 FieldIdx::from_usize(idx),
1178 ty,
1179 )));
1180 } else {
1181 fields.push(Operand::Move(tcx.mk_place_field(
1182 self_local,
1183 FieldIdx::from_usize(idx),
1184 ty,
1185 )));
1186 }
1187 }
1188
1189 let source_info = SourceInfo::outermost(span);
1190 let rvalue = Rvalue::Aggregate(
1191 Box::new(AggregateKind::Coroutine(coroutine_def_id, coroutine_args)),
1192 IndexVec::from_raw(fields),
1193 );
1194 let stmt = Statement::new(
1195 source_info,
1196 StatementKind::Assign(Box::new((Place::return_place(), rvalue))),
1197 );
1198 let statements = ::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[stmt]))vec![stmt];
1199 let start_block = BasicBlockData::new_stmts(
1200 statements,
1201 Some(Terminator { source_info, kind: TerminatorKind::Return }),
1202 false,
1203 );
1204
1205 let source = MirSource::from_instance(ty::InstanceKind::ConstructCoroutineInClosureShim {
1206 coroutine_closure_def_id,
1207 receiver_by_ref,
1208 });
1209
1210 let body =
1211 new_body(source, IndexVec::from_elem_n(start_block, 1), locals, sig.inputs().len(), span);
1212
1213 let pass_name =
1214 if receiver_by_ref { "coroutine_closure_by_ref" } else { "coroutine_closure_by_move" };
1215 if let Some(dumper) = MirDumper::new(tcx, pass_name, &body) {
1216 dumper.dump_mir(&body);
1217 }
1218
1219 body
1220}