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