Skip to main content

rustc_mir_transform/
elaborate_drop.rs

1use std::{fmt, iter, mem};
2
3use rustc_abi::{FIRST_VARIANT, FieldIdx, VariantIdx};
4use rustc_hir::def::DefKind;
5use rustc_hir::lang_items::LangItem;
6use rustc_index::Idx;
7use rustc_middle::mir::*;
8use rustc_middle::ty::adjustment::PointerCoercion;
9use rustc_middle::ty::util::IntTypeExt;
10use rustc_middle::ty::{self, GenericArg, GenericArgsRef, Ty, TyCtxt, Unnormalized};
11use rustc_middle::{bug, span_bug, traits};
12use rustc_span::{DUMMY_SP, Spanned, dummy_spanned};
13use tracing::{debug, instrument};
14
15use crate::patch::MirPatch;
16
17/// Describes how/if a value should be dropped.
18#[derive(#[automatically_derived]
impl ::core::fmt::Debug for DropStyle {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        ::core::fmt::Formatter::write_str(f,
            match self {
                DropStyle::Dead => "Dead",
                DropStyle::Static => "Static",
                DropStyle::Conditional => "Conditional",
                DropStyle::Open => "Open",
            })
    }
}Debug)]
19pub(crate) enum DropStyle {
20    /// The value is already dead at the drop location, no drop will be executed.
21    Dead,
22
23    /// The value is known to always be initialized at the drop location, drop will always be
24    /// executed.
25    Static,
26
27    /// Whether the value needs to be dropped depends on its drop flag.
28    Conditional,
29
30    /// An "open" drop is one where only the fields of a value are dropped.
31    ///
32    /// For example, this happens when moving out of a struct field: The rest of the struct will be
33    /// dropped in such an "open" drop. It is also used to generate drop glue for the individual
34    /// components of a value, for example for dropping array elements.
35    Open,
36}
37
38/// Which drop flags to affect/check with an operation.
39#[derive(#[automatically_derived]
impl ::core::fmt::Debug for DropFlagMode {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        ::core::fmt::Formatter::write_str(f,
            match self {
                DropFlagMode::Shallow => "Shallow",
                DropFlagMode::Deep => "Deep",
            })
    }
}Debug)]
40pub(crate) enum DropFlagMode {
41    /// Only affect the top-level drop flag, not that of any contained fields.
42    Shallow,
43    /// Affect all nested drop flags in addition to the top-level one.
44    Deep,
45}
46
47/// Describes if unwinding is necessary and where to unwind to if a panic occurs.
48#[derive(#[automatically_derived]
impl ::core::marker::Copy for Unwind { }Copy, #[automatically_derived]
impl ::core::clone::Clone for Unwind {
    #[inline]
    fn clone(&self) -> Unwind {
        let _: ::core::clone::AssertParamIsClone<BasicBlock>;
        *self
    }
}Clone, #[automatically_derived]
impl ::core::fmt::Debug for Unwind {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        match self {
            Unwind::To(__self_0) =>
                ::core::fmt::Formatter::debug_tuple_field1_finish(f, "To",
                    &__self_0),
            Unwind::InCleanup =>
                ::core::fmt::Formatter::write_str(f, "InCleanup"),
        }
    }
}Debug)]
49pub(crate) enum Unwind {
50    /// Unwind to this block.
51    To(BasicBlock),
52    /// Already in an unwind path, any panic will cause an abort.
53    InCleanup,
54}
55
56impl Unwind {
57    fn is_cleanup(self) -> bool {
58        match self {
59            Unwind::To(..) => false,
60            Unwind::InCleanup => true,
61        }
62    }
63
64    fn into_action(self) -> UnwindAction {
65        match self {
66            Unwind::To(bb) => UnwindAction::Cleanup(bb),
67            Unwind::InCleanup => UnwindAction::Terminate(UnwindTerminateReason::InCleanup),
68        }
69    }
70
71    fn map<F>(self, f: F) -> Self
72    where
73        F: FnOnce(BasicBlock) -> BasicBlock,
74    {
75        match self {
76            Unwind::To(bb) => Unwind::To(f(bb)),
77            Unwind::InCleanup => Unwind::InCleanup,
78        }
79    }
80}
81
82pub(crate) trait DropElaborator<'a, 'tcx>: fmt::Debug {
83    /// The type representing paths that can be moved out of.
84    ///
85    /// Users can move out of individual fields of a struct, such as `a.b.c`. This type is used to
86    /// represent such move paths. Sometimes tracking individual move paths is not necessary, in
87    /// which case this may be set to (for example) `()`.
88    type Path: Copy + fmt::Debug;
89
90    // Accessors
91
92    fn patch_ref(&self) -> &MirPatch<'tcx>;
93    fn patch(&mut self) -> &mut MirPatch<'tcx>;
94    fn body(&self) -> &'a Body<'tcx>;
95    fn tcx(&self) -> TyCtxt<'tcx>;
96    fn typing_env(&self) -> ty::TypingEnv<'tcx>;
97    fn allow_async_drops(&self) -> bool;
98
99    fn terminator_loc(&self, bb: BasicBlock) -> Location;
100
101    // Drop logic
102
103    /// Returns how `path` should be dropped, given `mode`.
104    fn drop_style(&self, path: Self::Path, mode: DropFlagMode) -> DropStyle;
105
106    /// Returns the drop flag of `path` as a MIR `Operand` (or `None` if `path` has no drop flag).
107    fn get_drop_flag(&mut self, path: Self::Path) -> Option<Operand<'tcx>>;
108
109    /// Modifies the MIR patch so that the drop flag of `path` (if any) is cleared at `location`.
110    ///
111    /// If `mode` is deep, drop flags of all child paths should also be cleared by inserting
112    /// additional statements.
113    fn clear_drop_flag(&mut self, location: Location, path: Self::Path, mode: DropFlagMode);
114
115    // Subpaths
116
117    /// Returns the subpath of a field of `path` (or `None` if there is no dedicated subpath).
118    ///
119    /// If this returns `None`, `field` will not get a dedicated drop flag.
120    fn field_subpath(&self, path: Self::Path, field: FieldIdx) -> Option<Self::Path>;
121
122    /// Returns the subpath of a dereference of `path` (or `None` if there is no dedicated subpath).
123    ///
124    /// If this returns `None`, `*path` will not get a dedicated drop flag.
125    ///
126    /// This is only relevant for `Box<T>`, where the contained `T` can be moved out of the box.
127    fn deref_subpath(&self, path: Self::Path) -> Option<Self::Path>;
128
129    /// Returns the subpath of downcasting `path` to one of its variants.
130    ///
131    /// If this returns `None`, the downcast of `path` will not get a dedicated drop flag.
132    fn downcast_subpath(&self, path: Self::Path, variant: VariantIdx) -> Option<Self::Path>;
133
134    /// Returns the subpath of indexing a fixed-size array `path`.
135    ///
136    /// If this returns `None`, elements of `path` will not get a dedicated drop flag.
137    ///
138    /// This is only relevant for array patterns, which can move out of individual array elements.
139    fn array_subpath(&self, path: Self::Path, index: u64, size: u64) -> Option<Self::Path>;
140}
141
142#[derive(#[automatically_derived]
impl<'a, 'b, 'tcx, D: ::core::fmt::Debug> ::core::fmt::Debug for
    DropCtxt<'a, 'b, 'tcx, D> where D: DropElaborator<'b, 'tcx>,
    D::Path: ::core::fmt::Debug {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        let names: &'static _ =
            &["elaborator", "source_info", "place", "path", "succ", "unwind",
                        "dropline"];
        let values: &[&dyn ::core::fmt::Debug] =
            &[&self.elaborator, &self.source_info, &self.place, &self.path,
                        &self.succ, &self.unwind, &&self.dropline];
        ::core::fmt::Formatter::debug_struct_fields_finish(f, "DropCtxt",
            names, values)
    }
}Debug)]
143struct DropCtxt<'a, 'b, 'tcx, D>
144where
145    D: DropElaborator<'b, 'tcx>,
146{
147    elaborator: &'a mut D,
148
149    source_info: SourceInfo,
150
151    place: Place<'tcx>,
152    path: D::Path,
153    succ: BasicBlock,
154    unwind: Unwind,
155    dropline: Option<BasicBlock>,
156}
157
158/// "Elaborates" a drop of `place`/`path` and patches `bb`'s terminator to execute it.
159///
160/// The passed `elaborator` is used to determine what should happen at the drop terminator. It
161/// decides whether the drop can be statically determined or whether it needs a dynamic drop flag,
162/// and whether the drop is "open", i.e. should be expanded to drop all subfields of the dropped
163/// value.
164///
165/// When this returns, the MIR patch in the `elaborator` contains the necessary changes.
166pub(crate) fn elaborate_drop<'b, 'tcx, D>(
167    elaborator: &mut D,
168    source_info: SourceInfo,
169    place: Place<'tcx>,
170    path: D::Path,
171    succ: BasicBlock,
172    unwind: Unwind,
173    bb: BasicBlock,
174    dropline: Option<BasicBlock>,
175) where
176    D: DropElaborator<'b, 'tcx>,
177    'tcx: 'b,
178{
179    DropCtxt { elaborator, source_info, place, path, succ, unwind, dropline }.elaborate_drop(bb)
180}
181
182impl<'a, 'b, 'tcx, D> DropCtxt<'a, 'b, 'tcx, D>
183where
184    D: DropElaborator<'b, 'tcx>,
185    'tcx: 'b,
186{
187    x;#[instrument(level = "trace", skip(self), ret)]
188    fn place_ty(&self, place: Place<'tcx>) -> Ty<'tcx> {
189        if place.local < self.elaborator.body().local_decls.next_index() {
190            place.ty(self.elaborator.body(), self.tcx()).ty
191        } else {
192            // We don't have a slice with all the locals, since some are in the patch.
193            PlaceTy::from_ty(self.elaborator.patch_ref().local_ty(place.local))
194                .multi_projection_ty(self.elaborator.tcx(), place.projection)
195                .ty
196        }
197    }
198
199    fn tcx(&self) -> TyCtxt<'tcx> {
200        self.elaborator.tcx()
201    }
202
203    // Generates three blocks:
204    // * #1:pin_obj_bb:   call Pin<ObjTy>::new_unchecked(&mut obj)
205    // * #2:call_drop_bb: fut = call obj.<AsyncDrop::drop>() OR call async_drop_in_place<T>(obj)
206    // * #3:drop_term_bb: drop (obj, fut, ...)
207    // We keep async drop unexpanded to poll-loop here, to expand it later, at StateTransform -
208    //   into states expand.
209    // call_destructor_only - to call only AsyncDrop::drop, not full async_drop_in_place glue
210    fn build_async_drop(
211        &mut self,
212        place: Place<'tcx>,
213        drop_ty: Ty<'tcx>,
214        bb: Option<BasicBlock>,
215        succ: BasicBlock,
216        unwind: Unwind,
217        dropline: Option<BasicBlock>,
218        call_destructor_only: bool,
219    ) -> BasicBlock {
220        let tcx = self.tcx();
221        let span = self.source_info.span;
222
223        let pin_obj_bb = bb.unwrap_or_else(|| {
224            self.elaborator.patch().new_block(BasicBlockData::new(
225                Some(Terminator {
226                    // Temporary terminator, will be replaced by patch
227                    source_info: self.source_info,
228                    kind: TerminatorKind::Return,
229                }),
230                false,
231            ))
232        });
233
234        let (fut_ty, drop_fn_def_id, trait_args) = if call_destructor_only {
235            // Resolving obj.<AsyncDrop::drop>()
236            let trait_ref =
237                ty::TraitRef::new(tcx, tcx.require_lang_item(LangItem::AsyncDrop, span), [drop_ty]);
238            let (drop_trait, trait_args) = match tcx.codegen_select_candidate(
239                ty::TypingEnv::fully_monomorphized().as_query_input(trait_ref),
240            ) {
241                Ok(traits::ImplSource::UserDefined(traits::ImplSourceUserDefinedData {
242                    impl_def_id,
243                    args,
244                    ..
245                })) => (*impl_def_id, *args),
246                impl_source => {
247                    ::rustc_middle::util::bug::span_bug_fmt(span,
    format_args!("invalid `AsyncDrop` impl_source: {0:?}", impl_source));span_bug!(span, "invalid `AsyncDrop` impl_source: {:?}", impl_source);
248                }
249            };
250            // impl_item_refs may be empty if drop fn is not implemented in 'impl AsyncDrop for ...'
251            // (#140974).
252            // Such code will report error, so just generate sync drop here and return
253            let Some(drop_fn_def_id) =
254                tcx.associated_item_def_ids(drop_trait).first().and_then(|&def_id| {
255                    if tcx.def_kind(def_id) == DefKind::AssocFn
256                        && tcx.check_args_compatible(def_id, trait_args)
257                    {
258                        Some(def_id)
259                    } else {
260                        None
261                    }
262                })
263            else {
264                tcx.dcx().span_delayed_bug(
265                    self.elaborator.body().span,
266                    "AsyncDrop type without correct `async fn drop(...)`.",
267                );
268                self.elaborator.patch().patch_terminator(
269                    pin_obj_bb,
270                    TerminatorKind::Drop {
271                        place,
272                        target: succ,
273                        unwind: unwind.into_action(),
274                        replace: false,
275                        drop: None,
276                        async_fut: None,
277                    },
278                );
279                return pin_obj_bb;
280            };
281            let drop_fn = Ty::new_fn_def(tcx, drop_fn_def_id, trait_args);
282            let sig = drop_fn.fn_sig(tcx);
283            let sig = tcx.instantiate_bound_regions_with_erased(sig);
284            (sig.output(), drop_fn_def_id, trait_args)
285        } else {
286            // Resolving async_drop_in_place<T> function for drop_ty
287            let drop_fn_def_id = tcx.require_lang_item(LangItem::AsyncDropInPlace, span);
288            let trait_args = tcx.mk_args(&[drop_ty.into()]);
289            let sig = tcx.fn_sig(drop_fn_def_id).instantiate(tcx, trait_args).skip_norm_wip();
290            let sig = tcx.instantiate_bound_regions_with_erased(sig);
291            (sig.output(), drop_fn_def_id, trait_args)
292        };
293
294        let fut = Place::from(self.new_temp(fut_ty));
295
296        // #1:pin_obj_bb >>> obj_ref = &mut obj
297        let obj_ref_ty = Ty::new_mut_ref(tcx, tcx.lifetimes.re_erased, drop_ty);
298        let obj_ref_place = Place::from(self.new_temp(obj_ref_ty));
299
300        let term_loc = self.elaborator.terminator_loc(pin_obj_bb);
301        self.elaborator.patch().add_assign(
302            term_loc,
303            obj_ref_place,
304            Rvalue::Ref(
305                tcx.lifetimes.re_erased,
306                BorrowKind::Mut { kind: MutBorrowKind::Default },
307                place,
308            ),
309        );
310
311        // pin_obj_place preparation
312        let pin_obj_new_unchecked_fn = Ty::new_fn_def(
313            tcx,
314            tcx.require_lang_item(LangItem::PinNewUnchecked, span),
315            [GenericArg::from(obj_ref_ty)],
316        );
317        let pin_obj_ty = pin_obj_new_unchecked_fn.fn_sig(tcx).output().no_bound_vars().unwrap();
318        let pin_obj_place = Place::from(self.new_temp(pin_obj_ty));
319        let pin_obj_new_unchecked_fn = Operand::Constant(Box::new(ConstOperand {
320            span,
321            user_ty: None,
322            const_: Const::zero_sized(pin_obj_new_unchecked_fn),
323        }));
324
325        // Create an intermediate block that does StorageDead(fut) then jumps to succ.
326        // This is necessary because `succ` may differ from `self.succ` (e.g. when
327        // build_async_drop is called from drop_loop, `succ` is the loop header).
328        // Placing StorageDead directly at `self.succ` would miss the loop-back edge,
329        // causing StorageLive(fut) to fire again without a preceding StorageDead.
330        let succ_with_dead = self.new_block_with_statements(
331            unwind,
332            ::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
        [Statement::new(self.source_info,
                    StatementKind::StorageDead(fut.local))]))vec![Statement::new(self.source_info, StatementKind::StorageDead(fut.local))],
333            TerminatorKind::Goto { target: succ },
334        );
335
336        // #3:drop_term_bb
337        let drop_term_bb = self.new_block(
338            unwind,
339            TerminatorKind::Drop {
340                place,
341                target: succ_with_dead,
342                unwind: unwind.into_action(),
343                replace: false,
344                drop: dropline,
345                async_fut: Some(fut.local),
346            },
347        );
348
349        // #2:call_drop_bb
350        let mut call_statements = Vec::new();
351        let drop_arg = if call_destructor_only {
352            pin_obj_place
353        } else {
354            let ty::Adt(adt_def, adt_args) = pin_obj_ty.kind() else {
355                ::rustc_middle::util::bug::bug_fmt(format_args!("impossible case reached"));bug!();
356            };
357            let unwrap_ty = adt_def.non_enum_variant().fields[FieldIdx::ZERO].ty(tcx, adt_args);
358            let obj_ref_place = Place::from(self.new_temp(unwrap_ty));
359            call_statements.push(self.assign(
360                obj_ref_place,
361                Rvalue::Use(
362                    Operand::Copy(tcx.mk_place_field(pin_obj_place, FieldIdx::ZERO, unwrap_ty)),
363                    WithRetag::Yes,
364                ),
365            ));
366
367            obj_ref_place
368        };
369        call_statements
370            .push(Statement::new(self.source_info, StatementKind::StorageLive(fut.local)));
371
372        let call_drop_bb = self.new_block_with_statements(
373            unwind,
374            call_statements,
375            TerminatorKind::Call {
376                func: Operand::function_handle(tcx, drop_fn_def_id, trait_args, span),
377                args: [Spanned { node: Operand::Move(drop_arg), span: DUMMY_SP }].into(),
378                destination: fut,
379                target: Some(drop_term_bb),
380                unwind: unwind.into_action(),
381                call_source: CallSource::Misc,
382                fn_span: self.source_info.span,
383            },
384        );
385        // StorageDead(fut) in unwind block (at the begin)
386        if let Unwind::To(block) = unwind {
387            self.elaborator.patch().add_statement(
388                Location { block, statement_index: 0 },
389                StatementKind::StorageDead(fut.local),
390            );
391        }
392        // StorageDead(fut) in dropline block (at the begin)
393        if let Some(block) = dropline {
394            self.elaborator.patch().add_statement(
395                Location { block, statement_index: 0 },
396                StatementKind::StorageDead(fut.local),
397            );
398        }
399
400        // #1:pin_obj_bb >>> call Pin<ObjTy>::new_unchecked(&mut obj)
401        self.elaborator.patch().patch_terminator(
402            pin_obj_bb,
403            TerminatorKind::Call {
404                func: pin_obj_new_unchecked_fn,
405                args: [dummy_spanned(Operand::Move(obj_ref_place))].into(),
406                destination: pin_obj_place,
407                target: Some(call_drop_bb),
408                unwind: unwind.into_action(),
409                call_source: CallSource::Misc,
410                fn_span: span,
411            },
412        );
413        pin_obj_bb
414    }
415
416    fn build_drop(&mut self, bb: BasicBlock) {
417        let drop_ty = self.place_ty(self.place);
418        if !self.elaborator.patch_ref().block(self.elaborator.body(), bb).is_cleanup
419            && self.check_if_can_async_drop(drop_ty, false)
420        {
421            self.build_async_drop(
422                self.place,
423                drop_ty,
424                Some(bb),
425                self.succ,
426                self.unwind,
427                self.dropline,
428                false,
429            );
430        } else {
431            self.elaborator.patch().patch_terminator(
432                bb,
433                TerminatorKind::Drop {
434                    place: self.place,
435                    target: self.succ,
436                    unwind: self.unwind.into_action(),
437                    replace: false,
438                    drop: None,
439                    async_fut: None,
440                },
441            );
442        }
443    }
444
445    /// Function to check if we can generate an async drop here
446    fn check_if_can_async_drop(&mut self, drop_ty: Ty<'tcx>, call_destructor_only: bool) -> bool {
447        let is_async_drop_feature_enabled = if self.tcx().features().async_drop() {
448            true
449        } else {
450            // Check if the type needing async drop comes from a dependency crate.
451            if let ty::Adt(adt_def, _) = drop_ty.kind() {
452                !adt_def.did().is_local() && adt_def.async_destructor(self.tcx()).is_some()
453            } else {
454                false
455            }
456        };
457
458        // Short-circuit before calling needs_async_drop/is_async_drop, as those
459        // require the `async_drop` lang item to exist (which may not be present
460        // in minimal/custom core environments like cranelift's mini_core).
461        if !is_async_drop_feature_enabled
462            || !self.elaborator.body().coroutine.is_some()
463            || !self.elaborator.allow_async_drops()
464        {
465            return false;
466        }
467
468        let needs_async_drop = if call_destructor_only {
469            drop_ty.is_async_drop(self.tcx(), self.elaborator.typing_env())
470        } else {
471            drop_ty.needs_async_drop(self.tcx(), self.elaborator.typing_env())
472        };
473
474        // Async drop in libstd/libcore would become insta-stable — catch that mistake.
475        if needs_async_drop && self.tcx().features().staged_api() {
476            ::rustc_middle::util::bug::span_bug_fmt(self.source_info.span,
    format_args!("don\'t use async drop in libstd, it becomes insta-stable"));span_bug!(
477                self.source_info.span,
478                "don't use async drop in libstd, it becomes insta-stable"
479            );
480        }
481
482        needs_async_drop
483    }
484
485    /// This elaborates a single drop instruction, located at `bb`, and
486    /// patches over it.
487    ///
488    /// The elaborated drop checks the drop flags to only drop what
489    /// is initialized.
490    ///
491    /// In addition, the relevant drop flags also need to be cleared
492    /// to avoid double-drops. However, in the middle of a complex
493    /// drop, one must avoid clearing some of the flags before they
494    /// are read, as that would cause a memory leak.
495    ///
496    /// In particular, when dropping an ADT, multiple fields may be
497    /// joined together under the `rest` subpath. They are all controlled
498    /// by the primary drop flag, but only the last rest-field dropped
499    /// should clear it (and it must also not clear anything else).
500    //
501    // FIXME: I think we should just control the flags externally,
502    // and then we do not need this machinery.
503    #[allow(clippy :: suspicious_else_formatting)]
{
    let __tracing_attr_span;
    let __tracing_attr_guard;
    if ::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
                &&
                ::tracing::Level::DEBUG <=
                    ::tracing::level_filters::LevelFilter::current() ||
            { false } {
        __tracing_attr_span =
            {
                use ::tracing::__macro_support::Callsite as _;
                static __CALLSITE: ::tracing::callsite::DefaultCallsite =
                    {
                        static META: ::tracing::Metadata<'static> =
                            {
                                ::tracing_core::metadata::Metadata::new("elaborate_drop",
                                    "rustc_mir_transform::elaborate_drop",
                                    ::tracing::Level::DEBUG,
                                    ::tracing_core::__macro_support::Option::Some("compiler/rustc_mir_transform/src/elaborate_drop.rs"),
                                    ::tracing_core::__macro_support::Option::Some(503u32),
                                    ::tracing_core::__macro_support::Option::Some("rustc_mir_transform::elaborate_drop"),
                                    ::tracing_core::field::FieldSet::new(&["self", "bb"],
                                        ::tracing_core::callsite::Identifier(&__CALLSITE)),
                                    ::tracing::metadata::Kind::SPAN)
                            };
                        ::tracing::callsite::DefaultCallsite::new(&META)
                    };
                let mut interest = ::tracing::subscriber::Interest::never();
                if ::tracing::Level::DEBUG <=
                                    ::tracing::level_filters::STATIC_MAX_LEVEL &&
                                ::tracing::Level::DEBUG <=
                                    ::tracing::level_filters::LevelFilter::current() &&
                            { interest = __CALLSITE.interest(); !interest.is_never() }
                        &&
                        ::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
                            interest) {
                    let meta = __CALLSITE.metadata();
                    ::tracing::Span::new(meta,
                        &{
                                #[allow(unused_imports)]
                                use ::tracing::field::{debug, display, Value};
                                let mut iter = meta.fields().iter();
                                meta.fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
                                                    ::tracing::__macro_support::Option::Some(&::tracing::field::debug(&self)
                                                            as &dyn Value)),
                                                (&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
                                                    ::tracing::__macro_support::Option::Some(&::tracing::field::debug(&bb)
                                                            as &dyn Value))])
                            })
                } else {
                    let span =
                        ::tracing::__macro_support::__disabled_span(__CALLSITE.metadata());
                    {};
                    span
                }
            };
        __tracing_attr_guard = __tracing_attr_span.enter();
    }

    #[warn(clippy :: suspicious_else_formatting)]
    {

        #[allow(unknown_lints, unreachable_code, clippy ::
        diverging_sub_expression, clippy :: empty_loop, clippy ::
        let_unit_value, clippy :: let_with_type_underscore, clippy ::
        needless_return, clippy :: unreachable)]
        if false {
            let __tracing_attr_fake_return: () = loop {};
            return __tracing_attr_fake_return;
        }
        {
            match self.elaborator.drop_style(self.path, DropFlagMode::Deep) {
                DropStyle::Dead => {
                    self.elaborator.patch().patch_terminator(bb,
                        TerminatorKind::Goto { target: self.succ });
                }
                DropStyle::Static => { self.build_drop(bb); }
                DropStyle::Conditional => {
                    let drop_bb = self.complete_drop(self.succ, self.unwind);
                    self.elaborator.patch().patch_terminator(bb,
                        TerminatorKind::Goto { target: drop_bb });
                }
                DropStyle::Open => {
                    let drop_bb = self.open_drop();
                    self.elaborator.patch().patch_terminator(bb,
                        TerminatorKind::Goto { target: drop_bb });
                }
            }
        }
    }
}#[instrument(level = "debug")]
504    fn elaborate_drop(&mut self, bb: BasicBlock) {
505        match self.elaborator.drop_style(self.path, DropFlagMode::Deep) {
506            DropStyle::Dead => {
507                self.elaborator
508                    .patch()
509                    .patch_terminator(bb, TerminatorKind::Goto { target: self.succ });
510            }
511            DropStyle::Static => {
512                self.build_drop(bb);
513            }
514            DropStyle::Conditional => {
515                let drop_bb = self.complete_drop(self.succ, self.unwind);
516                self.elaborator
517                    .patch()
518                    .patch_terminator(bb, TerminatorKind::Goto { target: drop_bb });
519            }
520            DropStyle::Open => {
521                let drop_bb = self.open_drop();
522                self.elaborator
523                    .patch()
524                    .patch_terminator(bb, TerminatorKind::Goto { target: drop_bb });
525            }
526        }
527    }
528
529    /// Returns the place and move path for each field of `variant`,
530    /// (the move path is `None` if the field is a rest field).
531    fn move_paths_for_fields(
532        &self,
533        base_place: Place<'tcx>,
534        variant_path: D::Path,
535        variant: &'tcx ty::VariantDef,
536        args: GenericArgsRef<'tcx>,
537    ) -> Vec<(Place<'tcx>, Option<D::Path>)> {
538        variant
539            .fields
540            .iter_enumerated()
541            .map(|(field_idx, field)| {
542                let subpath = self.elaborator.field_subpath(variant_path, field_idx);
543                let tcx = self.tcx();
544
545                match self.elaborator.typing_env().typing_mode().assert_not_erased() {
546                    ty::TypingMode::PostAnalysis => {}
547                    ty::TypingMode::Coherence
548                    | ty::TypingMode::Analysis { .. }
549                    | ty::TypingMode::Borrowck { .. }
550                    | ty::TypingMode::PostBorrowckAnalysis { .. } => {
551                        ::rustc_middle::util::bug::bug_fmt(format_args!("impossible case reached"))bug!()
552                    }
553                }
554
555                let field_ty = field.ty(tcx, args);
556                // We silently leave an unnormalized type here to support polymorphic drop
557                // elaboration for users of rustc internal APIs
558                let field_ty = tcx
559                    .try_normalize_erasing_regions(
560                        self.elaborator.typing_env(),
561                        Unnormalized::new_wip(field_ty),
562                    )
563                    .unwrap_or(field_ty);
564
565                (tcx.mk_place_field(base_place, field_idx, field_ty), subpath)
566            })
567            .collect()
568    }
569
570    fn drop_subpath(
571        &mut self,
572        place: Place<'tcx>,
573        path: Option<D::Path>,
574        succ: BasicBlock,
575        unwind: Unwind,
576        dropline: Option<BasicBlock>,
577    ) -> BasicBlock {
578        if let Some(path) = path {
579            {
    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/elaborate_drop.rs:579",
                        "rustc_mir_transform::elaborate_drop",
                        ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_mir_transform/src/elaborate_drop.rs"),
                        ::tracing_core::__macro_support::Option::Some(579u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_mir_transform::elaborate_drop"),
                        ::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!("drop_subpath: for std field {0:?}",
                                                    place) as &dyn Value))])
            });
    } else { ; }
};debug!("drop_subpath: for std field {:?}", place);
580
581            DropCtxt {
582                elaborator: self.elaborator,
583                source_info: self.source_info,
584                path,
585                place,
586                succ,
587                unwind,
588                dropline,
589            }
590            .elaborated_drop_block()
591        } else {
592            {
    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/elaborate_drop.rs:592",
                        "rustc_mir_transform::elaborate_drop",
                        ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_mir_transform/src/elaborate_drop.rs"),
                        ::tracing_core::__macro_support::Option::Some(592u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_mir_transform::elaborate_drop"),
                        ::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!("drop_subpath: for rest field {0:?}",
                                                    place) as &dyn Value))])
            });
    } else { ; }
};debug!("drop_subpath: for rest field {:?}", place);
593
594            DropCtxt {
595                elaborator: self.elaborator,
596                source_info: self.source_info,
597                place,
598                succ,
599                unwind,
600                dropline,
601                // Using `self.path` here to condition the drop on
602                // our own drop flag.
603                path: self.path,
604            }
605            .complete_drop(succ, unwind)
606        }
607    }
608
609    /// Creates one-half of the drop ladder for a list of fields, and return
610    /// the list of steps in it in reverse order, with the first step
611    /// dropping 0 fields and so on.
612    ///
613    /// `unwind_ladder` is such a list of steps in reverse order,
614    /// which is called if the matching step of the drop glue panics.
615    ///
616    /// `dropline_ladder` is a similar list of steps in reverse order,
617    /// which is called if the matching step of the drop glue will contain async drop
618    /// (expanded later to Yield) and the containing coroutine will be dropped at this point.
619    fn drop_halfladder(
620        &mut self,
621        unwind_ladder: &[Unwind],
622        dropline_ladder: &[Option<BasicBlock>],
623        mut succ: BasicBlock,
624        fields: &[(Place<'tcx>, Option<D::Path>)],
625    ) -> Vec<BasicBlock> {
626        iter::once(succ)
627            .chain(::itertools::__std_iter::IntoIterator::into_iter(fields.iter().rev()).zip(unwind_ladder).zip(dropline_ladder).map(|((a,
            b), b)| (a, b, b))itertools::izip!(fields.iter().rev(), unwind_ladder, dropline_ladder).map(
628                |(&(place, path), &unwind_succ, &dropline_to)| {
629                    succ = self.drop_subpath(place, path, succ, unwind_succ, dropline_to);
630                    succ
631                },
632            ))
633            .collect()
634    }
635
636    fn drop_ladder_bottom(&mut self) -> (BasicBlock, Unwind, Option<BasicBlock>) {
637        // Clear the "master" drop flag at the end. This is needed
638        // because the "master" drop protects the ADT's discriminant,
639        // which is invalidated after the ADT is dropped.
640        (
641            self.drop_flag_reset_block(DropFlagMode::Shallow, self.succ, self.unwind),
642            self.unwind,
643            self.dropline,
644        )
645    }
646
647    /// Creates a full drop ladder, consisting of 2 connected half-drop-ladders
648    ///
649    /// For example, with 3 fields, the drop ladder is
650    ///
651    /// ```text
652    /// .d0:
653    ///     ELAB(drop location.0 [target=.d1, unwind=.c1])
654    /// .d1:
655    ///     ELAB(drop location.1 [target=.d2, unwind=.c2])
656    /// .d2:
657    ///     ELAB(drop location.2 [target=`self.succ`, unwind=`self.unwind`])
658    /// .c1:
659    ///     ELAB(drop location.1 [target=.c2])
660    /// .c2:
661    ///     ELAB(drop location.2 [target=`self.unwind`])
662    /// ```
663    ///
664    /// For possible-async drops in coroutines we also need dropline ladder
665    /// ```text
666    /// .d0 (mainline):
667    ///     ELAB(drop location.0 [target=.d1, unwind=.c1, drop=.e1])
668    /// .d1 (mainline):
669    ///     ELAB(drop location.1 [target=.d2, unwind=.c2, drop=.e2])
670    /// .d2 (mainline):
671    ///     ELAB(drop location.2 [target=`self.succ`, unwind=`self.unwind`, drop=`self.drop`])
672    /// .c1 (unwind):
673    ///     ELAB(drop location.1 [target=.c2])
674    /// .c2 (unwind):
675    ///     ELAB(drop location.2 [target=`self.unwind`])
676    /// .e1 (dropline):
677    ///     ELAB(drop location.1 [target=.e2, unwind=.c2])
678    /// .e2 (dropline):
679    ///     ELAB(drop location.2 [target=`self.drop`, unwind=`self.unwind`])
680    /// ```
681    ///
682    /// NOTE: this does not clear the master drop flag, so you need
683    /// to point succ/unwind on a `drop_ladder_bottom`.
684    fn drop_ladder(
685        &mut self,
686        fields: Vec<(Place<'tcx>, Option<D::Path>)>,
687        succ: BasicBlock,
688        unwind: Unwind,
689        dropline: Option<BasicBlock>,
690    ) -> (BasicBlock, Unwind, Option<BasicBlock>) {
691        {
    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/elaborate_drop.rs:691",
                        "rustc_mir_transform::elaborate_drop",
                        ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_mir_transform/src/elaborate_drop.rs"),
                        ::tracing_core::__macro_support::Option::Some(691u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_mir_transform::elaborate_drop"),
                        ::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!("drop_ladder({0:?}, {1:?})",
                                                    self, fields) as &dyn Value))])
            });
    } else { ; }
};debug!("drop_ladder({:?}, {:?})", self, fields);
692        if !if unwind.is_cleanup() { dropline.is_none() } else { true } {
    {
        ::core::panicking::panic_fmt(format_args!("Dropline is set for cleanup drop ladder"));
    }
};assert!(
693            if unwind.is_cleanup() { dropline.is_none() } else { true },
694            "Dropline is set for cleanup drop ladder"
695        );
696
697        let mut fields = fields;
698        fields.retain(|&(place, _)| {
699            self.place_ty(place).needs_drop(self.tcx(), self.elaborator.typing_env())
700        });
701
702        {
    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/elaborate_drop.rs:702",
                        "rustc_mir_transform::elaborate_drop",
                        ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_mir_transform/src/elaborate_drop.rs"),
                        ::tracing_core::__macro_support::Option::Some(702u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_mir_transform::elaborate_drop"),
                        ::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!("drop_ladder - fields needing drop: {0:?}",
                                                    fields) as &dyn Value))])
            });
    } else { ; }
};debug!("drop_ladder - fields needing drop: {:?}", fields);
703
704        let dropline_ladder: Vec<Option<BasicBlock>> = ::alloc::vec::from_elem(None, fields.len() + 1)vec![None; fields.len() + 1];
705        let unwind_ladder = ::alloc::vec::from_elem(Unwind::InCleanup, fields.len() + 1)vec![Unwind::InCleanup; fields.len() + 1];
706        let unwind_ladder: Vec<_> = if let Unwind::To(succ) = unwind {
707            let halfladder = self.drop_halfladder(&unwind_ladder, &dropline_ladder, succ, &fields);
708            halfladder.into_iter().map(Unwind::To).collect()
709        } else {
710            unwind_ladder
711        };
712        let dropline_ladder: Vec<_> = if let Some(succ) = dropline {
713            let halfladder = self.drop_halfladder(&unwind_ladder, &dropline_ladder, succ, &fields);
714            halfladder.into_iter().map(Some).collect()
715        } else {
716            dropline_ladder
717        };
718
719        let normal_ladder = self.drop_halfladder(&unwind_ladder, &dropline_ladder, succ, &fields);
720
721        (
722            *normal_ladder.last().unwrap(),
723            *unwind_ladder.last().unwrap(),
724            *dropline_ladder.last().unwrap(),
725        )
726    }
727
728    fn open_drop_for_tuple(&mut self, tys: &[Ty<'tcx>]) -> BasicBlock {
729        {
    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/elaborate_drop.rs:729",
                        "rustc_mir_transform::elaborate_drop",
                        ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_mir_transform/src/elaborate_drop.rs"),
                        ::tracing_core::__macro_support::Option::Some(729u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_mir_transform::elaborate_drop"),
                        ::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!("open_drop_for_tuple({0:?}, {1:?})",
                                                    self, tys) as &dyn Value))])
            });
    } else { ; }
};debug!("open_drop_for_tuple({:?}, {:?})", self, tys);
730
731        let fields = tys
732            .iter()
733            .enumerate()
734            .map(|(i, &ty)| {
735                (
736                    self.tcx().mk_place_field(self.place, FieldIdx::new(i), ty),
737                    self.elaborator.field_subpath(self.path, FieldIdx::new(i)),
738                )
739            })
740            .collect();
741
742        let (succ, unwind, dropline) = self.drop_ladder_bottom();
743        self.drop_ladder(fields, succ, unwind, dropline).0
744    }
745
746    /// Drops the T contained in a `Box<T>` if it has not been moved out of
747    x;#[instrument(level = "debug", ret)]
748    fn open_drop_for_box_contents(
749        &mut self,
750        adt: ty::AdtDef<'tcx>,
751        args: GenericArgsRef<'tcx>,
752        succ: BasicBlock,
753        unwind: Unwind,
754        dropline: Option<BasicBlock>,
755    ) -> BasicBlock {
756        // drop glue is sent straight to codegen
757        // box cannot be directly dereferenced
758        let unique_ty = adt.non_enum_variant().fields[FieldIdx::ZERO].ty(self.tcx(), args);
759        let unique_variant = unique_ty.ty_adt_def().unwrap().non_enum_variant();
760        let nonnull_ty = unique_variant.fields[FieldIdx::ZERO].ty(self.tcx(), args);
761        let ptr_ty = Ty::new_imm_ptr(self.tcx(), args[0].expect_ty());
762
763        let unique_place = self.tcx().mk_place_field(self.place, FieldIdx::ZERO, unique_ty);
764        let nonnull_place = self.tcx().mk_place_field(unique_place, FieldIdx::ZERO, nonnull_ty);
765
766        let ptr_local = self.new_temp(ptr_ty);
767
768        let interior = self.tcx().mk_place_deref(Place::from(ptr_local));
769        let interior_path = self.elaborator.deref_subpath(self.path);
770
771        let do_drop_bb = self.drop_subpath(interior, interior_path, succ, unwind, dropline);
772
773        let setup_bbd = BasicBlockData::new_stmts(
774            vec![self.assign(
775                Place::from(ptr_local),
776                Rvalue::Cast(CastKind::Transmute, Operand::Copy(nonnull_place), ptr_ty),
777            )],
778            Some(Terminator {
779                kind: TerminatorKind::Goto { target: do_drop_bb },
780                source_info: self.source_info,
781            }),
782            unwind.is_cleanup(),
783        );
784        self.elaborator.patch().new_block(setup_bbd)
785    }
786
787    x;#[instrument(level = "debug", ret)]
788    fn open_drop_for_adt(
789        &mut self,
790        adt: ty::AdtDef<'tcx>,
791        args: GenericArgsRef<'tcx>,
792    ) -> BasicBlock {
793        if adt.variants().is_empty() {
794            return self.elaborator.patch().new_block(BasicBlockData::new(
795                Some(Terminator {
796                    source_info: self.source_info,
797                    kind: TerminatorKind::Unreachable,
798                }),
799                self.unwind.is_cleanup(),
800            ));
801        }
802
803        let skip_contents = adt.is_union() || adt.is_manually_drop();
804        let contents_drop = if skip_contents {
805            if adt.has_dtor(self.tcx()) && self.elaborator.get_drop_flag(self.path).is_some() {
806                // the top-level drop flag is usually cleared by open_drop_for_adt_contents
807                // types with destructors would still need an empty drop ladder to clear it
808
809                // however, these types are only open dropped in `DropShimElaborator`
810                // which does not have drop flags
811                // a future box-like "DerefMove" trait would allow for this case to happen
812                span_bug!(self.source_info.span, "open dropping partially moved union");
813            }
814
815            (self.succ, self.unwind, self.dropline)
816        } else {
817            self.open_drop_for_adt_contents(adt, args)
818        };
819
820        if adt.has_dtor(self.tcx()) {
821            let destructor_block = if adt.is_box() {
822                // we need to drop the inside of the box before running the destructor
823                let succ = self.destructor_call_block_sync((contents_drop.0, contents_drop.1));
824                let unwind = contents_drop
825                    .1
826                    .map(|unwind| self.destructor_call_block_sync((unwind, Unwind::InCleanup)));
827                let dropline = contents_drop
828                    .2
829                    .map(|dropline| self.destructor_call_block_sync((dropline, contents_drop.1)));
830                self.open_drop_for_box_contents(adt, args, succ, unwind, dropline)
831            } else {
832                self.destructor_call_block(contents_drop)
833            };
834
835            self.drop_flag_test_block(destructor_block, contents_drop.0, contents_drop.1)
836        } else {
837            contents_drop.0
838        }
839    }
840
841    fn open_drop_for_adt_contents(
842        &mut self,
843        adt: ty::AdtDef<'tcx>,
844        args: GenericArgsRef<'tcx>,
845    ) -> (BasicBlock, Unwind, Option<BasicBlock>) {
846        let (succ, unwind, dropline) = self.drop_ladder_bottom();
847        if !adt.is_enum() {
848            let fields =
849                self.move_paths_for_fields(self.place, self.path, adt.variant(FIRST_VARIANT), args);
850            self.drop_ladder(fields, succ, unwind, dropline)
851        } else {
852            self.open_drop_for_multivariant(adt, args, succ, unwind, dropline)
853        }
854    }
855
856    fn open_drop_for_multivariant(
857        &mut self,
858        adt: ty::AdtDef<'tcx>,
859        args: GenericArgsRef<'tcx>,
860        succ: BasicBlock,
861        unwind: Unwind,
862        dropline: Option<BasicBlock>,
863    ) -> (BasicBlock, Unwind, Option<BasicBlock>) {
864        let mut values = Vec::with_capacity(adt.variants().len());
865        let mut normal_blocks = Vec::with_capacity(adt.variants().len());
866        let mut unwind_blocks =
867            if unwind.is_cleanup() { None } else { Some(Vec::with_capacity(adt.variants().len())) };
868        let mut dropline_blocks =
869            if dropline.is_none() { None } else { Some(Vec::with_capacity(adt.variants().len())) };
870
871        let mut have_otherwise_with_drop_glue = false;
872        let mut have_otherwise = false;
873        let tcx = self.tcx();
874
875        for (variant_index, discr) in adt.discriminants(tcx) {
876            let variant = &adt.variant(variant_index);
877            let subpath = self.elaborator.downcast_subpath(self.path, variant_index);
878
879            if let Some(variant_path) = subpath {
880                let base_place = tcx.mk_place_elem(
881                    self.place,
882                    ProjectionElem::Downcast(Some(variant.name), variant_index),
883                );
884                let fields = self.move_paths_for_fields(base_place, variant_path, variant, args);
885                values.push(discr.val);
886                if let Unwind::To(unwind) = unwind {
887                    // We can't use the half-ladder from the original
888                    // drop ladder, because this breaks the
889                    // "funclet can't have 2 successor funclets"
890                    // requirement from MSVC:
891                    //
892                    //           switch       unwind-switch
893                    //          /      \         /        \
894                    //         v1.0    v2.0  v2.0-unwind  v1.0-unwind
895                    //         |        |      /             |
896                    //    v1.1-unwind  v2.1-unwind           |
897                    //      ^                                |
898                    //       \-------------------------------/
899                    //
900                    // Create a duplicate half-ladder to avoid that. We
901                    // could technically only do this on MSVC, but I
902                    // I want to minimize the divergence between MSVC
903                    // and non-MSVC.
904
905                    let unwind_blocks = unwind_blocks.as_mut().unwrap();
906                    let unwind_ladder = ::alloc::vec::from_elem(Unwind::InCleanup, fields.len() + 1)vec![Unwind::InCleanup; fields.len() + 1];
907                    let dropline_ladder: Vec<Option<BasicBlock>> = ::alloc::vec::from_elem(None, fields.len() + 1)vec![None; fields.len() + 1];
908                    let halfladder =
909                        self.drop_halfladder(&unwind_ladder, &dropline_ladder, unwind, &fields);
910                    unwind_blocks.push(halfladder.last().cloned().unwrap());
911                }
912                let (normal, _, drop_bb) = self.drop_ladder(fields, succ, unwind, dropline);
913                normal_blocks.push(normal);
914                if dropline.is_some() {
915                    dropline_blocks.as_mut().unwrap().push(drop_bb.unwrap());
916                }
917            } else {
918                have_otherwise = true;
919
920                let typing_env = self.elaborator.typing_env();
921                let have_field_with_drop_glue = variant
922                    .fields
923                    .iter()
924                    .any(|field| field.ty(tcx, args).needs_drop(tcx, typing_env));
925                if have_field_with_drop_glue {
926                    have_otherwise_with_drop_glue = true;
927                }
928            }
929        }
930
931        if !have_otherwise {
932            values.pop();
933        } else if !have_otherwise_with_drop_glue {
934            normal_blocks.push(self.goto_block(succ, unwind));
935            if let Unwind::To(unwind) = unwind {
936                unwind_blocks.as_mut().unwrap().push(self.goto_block(unwind, Unwind::InCleanup));
937            }
938        } else {
939            normal_blocks.push(self.drop_block(succ, unwind));
940            if let Unwind::To(unwind) = unwind {
941                unwind_blocks.as_mut().unwrap().push(self.drop_block(unwind, Unwind::InCleanup));
942            }
943        }
944
945        (
946            self.adt_switch_block(adt, normal_blocks, &values, succ, unwind),
947            unwind.map(|unwind| {
948                self.adt_switch_block(
949                    adt,
950                    unwind_blocks.unwrap(),
951                    &values,
952                    unwind,
953                    Unwind::InCleanup,
954                )
955            }),
956            dropline.map(|dropline| {
957                self.adt_switch_block(adt, dropline_blocks.unwrap(), &values, dropline, unwind)
958            }),
959        )
960    }
961
962    fn adt_switch_block(
963        &mut self,
964        adt: ty::AdtDef<'tcx>,
965        blocks: Vec<BasicBlock>,
966        values: &[u128],
967        succ: BasicBlock,
968        unwind: Unwind,
969    ) -> BasicBlock {
970        // If there are multiple variants, then if something
971        // is present within the enum the discriminant, tracked
972        // by the rest path, must be initialized.
973        //
974        // Additionally, we do not want to switch on the
975        // discriminant after it is free-ed, because that
976        // way lies only trouble.
977        let discr_ty = adt.repr().discr_type().to_ty(self.tcx());
978        let discr = Place::from(self.new_temp(discr_ty));
979        let discr_rv = Rvalue::Discriminant(self.place);
980        let switch_block = BasicBlockData::new_stmts(
981            ::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
        [self.assign(discr, discr_rv)]))vec![self.assign(discr, discr_rv)],
982            Some(Terminator {
983                source_info: self.source_info,
984                kind: TerminatorKind::SwitchInt {
985                    discr: Operand::Move(discr),
986                    targets: SwitchTargets::new(
987                        values.iter().copied().zip(blocks.iter().copied()),
988                        *blocks.last().unwrap(),
989                    ),
990                },
991            }),
992            unwind.is_cleanup(),
993        );
994        let switch_block = self.elaborator.patch().new_block(switch_block);
995        self.drop_flag_test_block(switch_block, succ, unwind)
996    }
997
998    fn destructor_call_block_sync(&mut self, (succ, unwind): (BasicBlock, Unwind)) -> BasicBlock {
999        {
    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/elaborate_drop.rs:999",
                        "rustc_mir_transform::elaborate_drop",
                        ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_mir_transform/src/elaborate_drop.rs"),
                        ::tracing_core::__macro_support::Option::Some(999u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_mir_transform::elaborate_drop"),
                        ::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!("destructor_call_block_sync({0:?}, {1:?})",
                                                    self, succ) as &dyn Value))])
            });
    } else { ; }
};debug!("destructor_call_block_sync({:?}, {:?})", self, succ);
1000        let tcx = self.tcx();
1001        let drop_trait = tcx.require_lang_item(LangItem::Drop, DUMMY_SP);
1002        let drop_fn = tcx.associated_item_def_ids(drop_trait)[0];
1003        let ty = self.place_ty(self.place);
1004
1005        let ref_ty = Ty::new_mut_ref(tcx, tcx.lifetimes.re_erased, ty);
1006        let ref_place = self.new_temp(ref_ty);
1007        let unit_temp = Place::from(self.new_temp(tcx.types.unit));
1008
1009        let result = BasicBlockData::new_stmts(
1010            ::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
        [self.assign(Place::from(ref_place),
                    Rvalue::Ref(tcx.lifetimes.re_erased,
                        BorrowKind::Mut { kind: MutBorrowKind::Default },
                        self.place))]))vec![self.assign(
1011                Place::from(ref_place),
1012                Rvalue::Ref(
1013                    tcx.lifetimes.re_erased,
1014                    BorrowKind::Mut { kind: MutBorrowKind::Default },
1015                    self.place,
1016                ),
1017            )],
1018            Some(Terminator {
1019                kind: TerminatorKind::Call {
1020                    func: Operand::function_handle(
1021                        tcx,
1022                        drop_fn,
1023                        [ty.into()],
1024                        self.source_info.span,
1025                    ),
1026                    args: [Spanned { node: Operand::Move(Place::from(ref_place)), span: DUMMY_SP }]
1027                        .into(),
1028                    destination: unit_temp,
1029                    target: Some(succ),
1030                    unwind: unwind.into_action(),
1031                    call_source: CallSource::Misc,
1032                    fn_span: self.source_info.span,
1033                },
1034                source_info: self.source_info,
1035            }),
1036            unwind.is_cleanup(),
1037        );
1038
1039        self.elaborator.patch().new_block(result)
1040    }
1041
1042    fn destructor_call_block(
1043        &mut self,
1044        (succ, unwind, dropline): (BasicBlock, Unwind, Option<BasicBlock>),
1045    ) -> BasicBlock {
1046        {
    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/elaborate_drop.rs:1046",
                        "rustc_mir_transform::elaborate_drop",
                        ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_mir_transform/src/elaborate_drop.rs"),
                        ::tracing_core::__macro_support::Option::Some(1046u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_mir_transform::elaborate_drop"),
                        ::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!("destructor_call_block({0:?}, {1:?})",
                                                    self, succ) as &dyn Value))])
            });
    } else { ; }
};debug!("destructor_call_block({:?}, {:?})", self, succ);
1047        let ty = self.place_ty(self.place);
1048        if !unwind.is_cleanup() && self.check_if_can_async_drop(ty, true) {
1049            self.build_async_drop(self.place, ty, None, succ, unwind, dropline, true)
1050        } else {
1051            self.destructor_call_block_sync((succ, unwind))
1052        }
1053    }
1054
1055    /// Create a loop that drops an array:
1056    ///
1057    /// ```text
1058    /// loop-block:
1059    ///    can_go = cur == len
1060    ///    if can_go then succ else drop-block
1061    /// drop-block:
1062    ///    ptr = &raw mut P[cur]
1063    ///    cur = cur + 1
1064    ///    drop(ptr)
1065    /// ```
1066    fn drop_loop(
1067        &mut self,
1068        succ: BasicBlock,
1069        cur: Local,
1070        len: Local,
1071        ety: Ty<'tcx>,
1072        unwind: Unwind,
1073        dropline: Option<BasicBlock>,
1074    ) -> BasicBlock {
1075        let copy = |place: Place<'tcx>| Operand::Copy(place);
1076        let move_ = |place: Place<'tcx>| Operand::Move(place);
1077        let tcx = self.tcx();
1078
1079        let ptr_ty = Ty::new_mut_ptr(tcx, ety);
1080        let ptr = Place::from(self.new_temp(ptr_ty));
1081        let can_go = Place::from(self.new_temp(tcx.types.bool));
1082        let one = self.constant_usize(1);
1083
1084        let drop_block = BasicBlockData::new_stmts(
1085            ::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
        [self.assign(ptr,
                    Rvalue::RawPtr(RawPtrKind::Mut,
                        tcx.mk_place_index(self.place, cur))),
                self.assign(cur.into(),
                    Rvalue::BinaryOp(BinOp::Add,
                        Box::new((move_(cur.into()), one))))]))vec![
1086                self.assign(
1087                    ptr,
1088                    Rvalue::RawPtr(RawPtrKind::Mut, tcx.mk_place_index(self.place, cur)),
1089                ),
1090                self.assign(
1091                    cur.into(),
1092                    Rvalue::BinaryOp(BinOp::Add, Box::new((move_(cur.into()), one))),
1093                ),
1094            ],
1095            Some(Terminator {
1096                source_info: self.source_info,
1097                // this gets overwritten by drop elaboration.
1098                kind: TerminatorKind::Unreachable,
1099            }),
1100            unwind.is_cleanup(),
1101        );
1102        let drop_block = self.elaborator.patch().new_block(drop_block);
1103
1104        let loop_block = BasicBlockData::new_stmts(
1105            ::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
        [self.assign(can_go,
                    Rvalue::BinaryOp(BinOp::Eq,
                        Box::new((copy(Place::from(cur)), copy(len.into())))))]))vec![self.assign(
1106                can_go,
1107                Rvalue::BinaryOp(BinOp::Eq, Box::new((copy(Place::from(cur)), copy(len.into())))),
1108            )],
1109            Some(Terminator {
1110                source_info: self.source_info,
1111                kind: TerminatorKind::if_(move_(can_go), succ, drop_block),
1112            }),
1113            unwind.is_cleanup(),
1114        );
1115        let loop_block = self.elaborator.patch().new_block(loop_block);
1116
1117        let place = tcx.mk_place_deref(ptr);
1118        if !unwind.is_cleanup() && self.check_if_can_async_drop(ety, false) {
1119            self.build_async_drop(
1120                place,
1121                ety,
1122                Some(drop_block),
1123                loop_block,
1124                unwind,
1125                dropline,
1126                false,
1127            );
1128        } else {
1129            self.elaborator.patch().patch_terminator(
1130                drop_block,
1131                TerminatorKind::Drop {
1132                    place,
1133                    target: loop_block,
1134                    unwind: unwind.into_action(),
1135                    replace: false,
1136                    drop: None,
1137                    async_fut: None,
1138                },
1139            );
1140        }
1141        loop_block
1142    }
1143
1144    fn open_drop_for_array(
1145        &mut self,
1146        array_ty: Ty<'tcx>,
1147        ety: Ty<'tcx>,
1148        opt_size: Option<u64>,
1149    ) -> BasicBlock {
1150        {
    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/elaborate_drop.rs:1150",
                        "rustc_mir_transform::elaborate_drop",
                        ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_mir_transform/src/elaborate_drop.rs"),
                        ::tracing_core::__macro_support::Option::Some(1150u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_mir_transform::elaborate_drop"),
                        ::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!("open_drop_for_array({0:?}, {1:?}, {2:?})",
                                                    array_ty, ety, opt_size) as &dyn Value))])
            });
    } else { ; }
};debug!("open_drop_for_array({:?}, {:?}, {:?})", array_ty, ety, opt_size);
1151        let tcx = self.tcx();
1152
1153        if let Some(size) = opt_size {
1154            enum ProjectionKind<Path> {
1155                Drop(std::ops::Range<u64>),
1156                Keep(u64, Path),
1157            }
1158            // Previously, we'd make a projection for every element in the array and create a drop
1159            // ladder if any `array_subpath` was `Some`, i.e. moving out with an array pattern.
1160            // This caused huge memory usage when generating the drops for large arrays, so we instead
1161            // record the *subslices* which are dropped and the *indexes* which are kept
1162            let mut drop_ranges = ::alloc::vec::Vec::new()vec![];
1163            let mut dropping = true;
1164            let mut start = 0;
1165            for i in 0..size {
1166                let path = self.elaborator.array_subpath(self.path, i, size);
1167                if dropping && path.is_some() {
1168                    drop_ranges.push(ProjectionKind::Drop(start..i));
1169                    dropping = false;
1170                } else if !dropping && path.is_none() {
1171                    dropping = true;
1172                    start = i;
1173                }
1174                if let Some(path) = path {
1175                    drop_ranges.push(ProjectionKind::Keep(i, path));
1176                }
1177            }
1178            if !drop_ranges.is_empty() {
1179                if dropping {
1180                    drop_ranges.push(ProjectionKind::Drop(start..size));
1181                }
1182                let fields = drop_ranges
1183                    .iter()
1184                    .rev()
1185                    .map(|p| {
1186                        let (project, path) = match p {
1187                            ProjectionKind::Drop(r) => (
1188                                ProjectionElem::Subslice {
1189                                    from: r.start,
1190                                    to: r.end,
1191                                    from_end: false,
1192                                },
1193                                None,
1194                            ),
1195                            &ProjectionKind::Keep(offset, path) => (
1196                                ProjectionElem::ConstantIndex {
1197                                    offset,
1198                                    min_length: size,
1199                                    from_end: false,
1200                                },
1201                                Some(path),
1202                            ),
1203                        };
1204                        (tcx.mk_place_elem(self.place, project), path)
1205                    })
1206                    .collect::<Vec<_>>();
1207                let (succ, unwind, dropline) = self.drop_ladder_bottom();
1208                return self.drop_ladder(fields, succ, unwind, dropline).0;
1209            }
1210        }
1211
1212        let array_ptr_ty = Ty::new_mut_ptr(tcx, array_ty);
1213        let array_ptr = self.new_temp(array_ptr_ty);
1214
1215        let slice_ty = Ty::new_slice(tcx, ety);
1216        let slice_ptr_ty = Ty::new_mut_ptr(tcx, slice_ty);
1217        let slice_ptr = self.new_temp(slice_ptr_ty);
1218
1219        let mut delegate_block = BasicBlockData::new_stmts(
1220            ::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
        [self.assign(Place::from(array_ptr),
                    Rvalue::RawPtr(RawPtrKind::Mut, self.place)),
                self.assign(Place::from(slice_ptr),
                    Rvalue::Cast(CastKind::PointerCoercion(PointerCoercion::Unsize,
                            CoercionSource::Implicit),
                        Operand::Move(Place::from(array_ptr)), slice_ptr_ty))]))vec![
1221                self.assign(Place::from(array_ptr), Rvalue::RawPtr(RawPtrKind::Mut, self.place)),
1222                self.assign(
1223                    Place::from(slice_ptr),
1224                    Rvalue::Cast(
1225                        CastKind::PointerCoercion(
1226                            PointerCoercion::Unsize,
1227                            CoercionSource::Implicit,
1228                        ),
1229                        Operand::Move(Place::from(array_ptr)),
1230                        slice_ptr_ty,
1231                    ),
1232                ),
1233            ],
1234            None,
1235            self.unwind.is_cleanup(),
1236        );
1237
1238        let array_place = mem::replace(
1239            &mut self.place,
1240            Place::from(slice_ptr).project_deeper(&[PlaceElem::Deref], tcx),
1241        );
1242        let slice_block = self.drop_loop_trio_for_slice(ety);
1243        self.place = array_place;
1244
1245        delegate_block.terminator = Some(Terminator {
1246            source_info: self.source_info,
1247            kind: TerminatorKind::Goto { target: slice_block },
1248        });
1249        self.elaborator.patch().new_block(delegate_block)
1250    }
1251
1252    /// Creates a trio of drop-loops of `place`, which drops its contents, even
1253    /// in the case of 1 panic or in the case of coroutine drop
1254    fn drop_loop_trio_for_slice(&mut self, ety: Ty<'tcx>) -> BasicBlock {
1255        {
    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/elaborate_drop.rs:1255",
                        "rustc_mir_transform::elaborate_drop",
                        ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_mir_transform/src/elaborate_drop.rs"),
                        ::tracing_core::__macro_support::Option::Some(1255u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_mir_transform::elaborate_drop"),
                        ::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!("drop_loop_trio_for_slice({0:?})",
                                                    ety) as &dyn Value))])
            });
    } else { ; }
};debug!("drop_loop_trio_for_slice({:?})", ety);
1256        let tcx = self.tcx();
1257        let len = self.new_temp(tcx.types.usize);
1258        let cur = self.new_temp(tcx.types.usize);
1259
1260        let unwind = self
1261            .unwind
1262            .map(|unwind| self.drop_loop(unwind, cur, len, ety, Unwind::InCleanup, None));
1263
1264        let dropline =
1265            self.dropline.map(|dropline| self.drop_loop(dropline, cur, len, ety, unwind, None));
1266
1267        let loop_block = self.drop_loop(self.succ, cur, len, ety, unwind, dropline);
1268
1269        let [PlaceElem::Deref] = self.place.projection.as_slice() else {
1270            ::rustc_middle::util::bug::span_bug_fmt(self.source_info.span,
    format_args!("Expected place for slice drop shim to be *_n, but it\'s {0:?}",
        self.place));span_bug!(
1271                self.source_info.span,
1272                "Expected place for slice drop shim to be *_n, but it's {:?}",
1273                self.place,
1274            );
1275        };
1276
1277        let zero = self.constant_usize(0);
1278        let block = BasicBlockData::new_stmts(
1279            ::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
        [self.assign(len.into(),
                    Rvalue::UnaryOp(UnOp::PtrMetadata,
                        Operand::Copy(Place::from(self.place.local)))),
                self.assign(cur.into(), Rvalue::Use(zero, WithRetag::Yes))]))vec![
1280                self.assign(
1281                    len.into(),
1282                    Rvalue::UnaryOp(
1283                        UnOp::PtrMetadata,
1284                        Operand::Copy(Place::from(self.place.local)),
1285                    ),
1286                ),
1287                self.assign(cur.into(), Rvalue::Use(zero, WithRetag::Yes)),
1288            ],
1289            Some(Terminator {
1290                source_info: self.source_info,
1291                kind: TerminatorKind::Goto { target: loop_block },
1292            }),
1293            unwind.is_cleanup(),
1294        );
1295
1296        let drop_block = self.elaborator.patch().new_block(block);
1297        // FIXME(#34708): handle partially-dropped array/slice elements.
1298        let reset_block = self.drop_flag_reset_block(DropFlagMode::Deep, drop_block, unwind);
1299        self.drop_flag_test_block(reset_block, self.succ, unwind)
1300    }
1301
1302    /// The slow-path - create an "open", elaborated drop for a type
1303    /// which is moved-out-of only partially, and patch `bb` to a jump
1304    /// to it. This must not be called on ADTs with a destructor,
1305    /// as these can't be moved-out-of, except for `Box<T>`, which is
1306    /// special-cased.
1307    ///
1308    /// This creates a "drop ladder" that drops the needed fields of the
1309    /// ADT, both in the success case or if one of the destructors fail.
1310    fn open_drop(&mut self) -> BasicBlock {
1311        let ty = self.place_ty(self.place);
1312        match ty.kind() {
1313            ty::Closure(_, args) => self.open_drop_for_tuple(args.as_closure().upvar_tys()),
1314            ty::CoroutineClosure(_, args) => {
1315                self.open_drop_for_tuple(args.as_coroutine_closure().upvar_tys())
1316            }
1317            // Note that `elaborate_drops` only drops the upvars of a coroutine,
1318            // and this is ok because `open_drop` here can only be reached
1319            // within that own coroutine's resume function.
1320            // This should only happen for the self argument on the resume function.
1321            // It effectively only contains upvars until the coroutine transformation runs.
1322            // See librustc_body/transform/coroutine.rs for more details.
1323            ty::Coroutine(_, args) => self.open_drop_for_tuple(args.as_coroutine().upvar_tys()),
1324            ty::Tuple(fields) => self.open_drop_for_tuple(fields),
1325            ty::Adt(def, args) => self.open_drop_for_adt(*def, args),
1326            ty::Dynamic(..) => self.complete_drop(self.succ, self.unwind),
1327            ty::Array(ety, size) => {
1328                let size = size.try_to_target_usize(self.tcx());
1329                self.open_drop_for_array(ty, *ety, size)
1330            }
1331            ty::Slice(ety) => self.drop_loop_trio_for_slice(*ety),
1332
1333            ty::UnsafeBinder(_) => {
1334                // Unsafe binders may elaborate drops if their inner type isn't copy.
1335                // This is enforced in typeck, so this should never happen.
1336                self.tcx().dcx().span_delayed_bug(
1337                    self.source_info.span,
1338                    "open drop for unsafe binder shouldn't be encountered",
1339                );
1340                self.elaborator.patch().new_block(BasicBlockData::new(
1341                    Some(Terminator {
1342                        source_info: self.source_info,
1343                        kind: TerminatorKind::Unreachable,
1344                    }),
1345                    self.unwind.is_cleanup(),
1346                ))
1347            }
1348
1349            _ => ::rustc_middle::util::bug::span_bug_fmt(self.source_info.span,
    format_args!("open drop from non-ADT `{0:?}`", ty))span_bug!(self.source_info.span, "open drop from non-ADT `{:?}`", ty),
1350        }
1351    }
1352
1353    fn complete_drop(&mut self, succ: BasicBlock, unwind: Unwind) -> BasicBlock {
1354        {
    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/elaborate_drop.rs:1354",
                        "rustc_mir_transform::elaborate_drop",
                        ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_mir_transform/src/elaborate_drop.rs"),
                        ::tracing_core::__macro_support::Option::Some(1354u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_mir_transform::elaborate_drop"),
                        ::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!("complete_drop(succ={0:?}, unwind={1:?})",
                                                    succ, unwind) as &dyn Value))])
            });
    } else { ; }
};debug!("complete_drop(succ={:?}, unwind={:?})", succ, unwind);
1355
1356        let drop_block = self.drop_block(succ, unwind);
1357
1358        self.drop_flag_test_block(drop_block, succ, unwind)
1359    }
1360
1361    /// Creates a block that resets the drop flag. If `mode` is deep, all children drop flags will
1362    /// also be cleared.
1363    fn drop_flag_reset_block(
1364        &mut self,
1365        mode: DropFlagMode,
1366        succ: BasicBlock,
1367        unwind: Unwind,
1368    ) -> BasicBlock {
1369        {
    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/elaborate_drop.rs:1369",
                        "rustc_mir_transform::elaborate_drop",
                        ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_mir_transform/src/elaborate_drop.rs"),
                        ::tracing_core::__macro_support::Option::Some(1369u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_mir_transform::elaborate_drop"),
                        ::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!("drop_flag_reset_block({0:?},{1:?})",
                                                    self, mode) as &dyn Value))])
            });
    } else { ; }
};debug!("drop_flag_reset_block({:?},{:?})", self, mode);
1370
1371        if unwind.is_cleanup() {
1372            // The drop flag isn't read again on the unwind path, so don't
1373            // bother setting it.
1374            return succ;
1375        }
1376        let block = self.new_block(unwind, TerminatorKind::Goto { target: succ });
1377        let block_start = Location { block, statement_index: 0 };
1378        self.elaborator.clear_drop_flag(block_start, self.path, mode);
1379        block
1380    }
1381
1382    fn elaborated_drop_block(&mut self) -> BasicBlock {
1383        {
    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/elaborate_drop.rs:1383",
                        "rustc_mir_transform::elaborate_drop",
                        ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_mir_transform/src/elaborate_drop.rs"),
                        ::tracing_core::__macro_support::Option::Some(1383u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_mir_transform::elaborate_drop"),
                        ::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!("elaborated_drop_block({0:?})",
                                                    self) as &dyn Value))])
            });
    } else { ; }
};debug!("elaborated_drop_block({:?})", self);
1384        let blk = self.drop_block_simple(self.succ, self.unwind);
1385        self.elaborate_drop(blk);
1386        blk
1387    }
1388
1389    fn drop_block_simple(&mut self, target: BasicBlock, unwind: Unwind) -> BasicBlock {
1390        let block = TerminatorKind::Drop {
1391            place: self.place,
1392            target,
1393            unwind: unwind.into_action(),
1394            replace: false,
1395            drop: self.dropline,
1396            async_fut: None,
1397        };
1398        self.new_block(unwind, block)
1399    }
1400
1401    fn drop_block(&mut self, target: BasicBlock, unwind: Unwind) -> BasicBlock {
1402        let drop_ty = self.place_ty(self.place);
1403        if !unwind.is_cleanup() && self.check_if_can_async_drop(drop_ty, false) {
1404            self.build_async_drop(
1405                self.place,
1406                drop_ty,
1407                None,
1408                self.succ,
1409                unwind,
1410                self.dropline,
1411                false,
1412            )
1413        } else {
1414            let block = TerminatorKind::Drop {
1415                place: self.place,
1416                target,
1417                unwind: unwind.into_action(),
1418                replace: false,
1419                drop: None,
1420                async_fut: None,
1421            };
1422            self.new_block(unwind, block)
1423        }
1424    }
1425
1426    fn goto_block(&mut self, target: BasicBlock, unwind: Unwind) -> BasicBlock {
1427        let block = TerminatorKind::Goto { target };
1428        self.new_block(unwind, block)
1429    }
1430
1431    /// Returns the block to jump to in order to test the drop flag and execute the drop.
1432    ///
1433    /// Depending on the required `DropStyle`, this might be a generated block with an `if`
1434    /// terminator (for dynamic/open drops), or it might be `on_set` or `on_unset` itself, in case
1435    /// the drop can be statically determined.
1436    fn drop_flag_test_block(
1437        &mut self,
1438        on_set: BasicBlock,
1439        on_unset: BasicBlock,
1440        unwind: Unwind,
1441    ) -> BasicBlock {
1442        let style = self.elaborator.drop_style(self.path, DropFlagMode::Shallow);
1443        {
    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/elaborate_drop.rs:1443",
                        "rustc_mir_transform::elaborate_drop",
                        ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_mir_transform/src/elaborate_drop.rs"),
                        ::tracing_core::__macro_support::Option::Some(1443u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_mir_transform::elaborate_drop"),
                        ::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!("drop_flag_test_block({0:?},{1:?},{2:?},{3:?}) - {4:?}",
                                                    self, on_set, on_unset, unwind, style) as &dyn Value))])
            });
    } else { ; }
};debug!(
1444            "drop_flag_test_block({:?},{:?},{:?},{:?}) - {:?}",
1445            self, on_set, on_unset, unwind, style
1446        );
1447
1448        match style {
1449            DropStyle::Dead => on_unset,
1450            DropStyle::Static => on_set,
1451            DropStyle::Conditional | DropStyle::Open => {
1452                let flag = self.elaborator.get_drop_flag(self.path).unwrap();
1453                let term = TerminatorKind::if_(flag, on_set, on_unset);
1454                self.new_block(unwind, term)
1455            }
1456        }
1457    }
1458
1459    fn new_block(&mut self, unwind: Unwind, k: TerminatorKind<'tcx>) -> BasicBlock {
1460        self.elaborator.patch().new_block(BasicBlockData::new(
1461            Some(Terminator { source_info: self.source_info, kind: k }),
1462            unwind.is_cleanup(),
1463        ))
1464    }
1465
1466    fn new_block_with_statements(
1467        &mut self,
1468        unwind: Unwind,
1469        statements: Vec<Statement<'tcx>>,
1470        k: TerminatorKind<'tcx>,
1471    ) -> BasicBlock {
1472        self.elaborator.patch().new_block(BasicBlockData::new_stmts(
1473            statements,
1474            Some(Terminator { source_info: self.source_info, kind: k }),
1475            unwind.is_cleanup(),
1476        ))
1477    }
1478
1479    fn new_temp(&mut self, ty: Ty<'tcx>) -> Local {
1480        self.elaborator.patch().new_temp(ty, self.source_info.span)
1481    }
1482
1483    fn constant_usize(&self, val: u16) -> Operand<'tcx> {
1484        Operand::Constant(Box::new(ConstOperand {
1485            span: self.source_info.span,
1486            user_ty: None,
1487            const_: Const::from_usize(self.tcx(), val.into()),
1488        }))
1489    }
1490
1491    fn assign(&self, lhs: Place<'tcx>, rhs: Rvalue<'tcx>) -> Statement<'tcx> {
1492        Statement::new(self.source_info, StatementKind::Assign(Box::new((lhs, rhs))))
1493    }
1494}