Skip to main content

rustc_hir_typeck/
closure.rs

1//! Code for type-checking closure expressions.
2
3use std::iter;
4use std::ops::ControlFlow;
5
6use rustc_abi::ExternAbi;
7use rustc_errors::ErrorGuaranteed;
8use rustc_hir as hir;
9use rustc_hir::lang_items::LangItem;
10use rustc_hir_analysis::hir_ty_lowering::HirTyLowerer;
11use rustc_infer::infer::{BoundRegionConversionTime, DefineOpaqueTypes, InferOk, InferResult};
12use rustc_infer::traits::{ObligationCauseCode, PredicateObligations};
13use rustc_macros::{TypeFoldable, TypeVisitable};
14use rustc_middle::span_bug;
15use rustc_middle::ty::{
16    self, ClosureKind, FnSigKind, GenericArgs, Ty, TyCtxt, TypeSuperVisitable, TypeVisitable,
17    TypeVisitableExt, TypeVisitor, Unnormalized,
18};
19use rustc_span::def_id::LocalDefId;
20use rustc_span::{DUMMY_SP, Span};
21use rustc_trait_selection::error_reporting::traits::ArgKind;
22use rustc_trait_selection::traits;
23use tracing::{debug, instrument, trace};
24
25use super::{CoroutineTypes, Expectation, FnCtxt, check_fn};
26
27/// What signature do we *expect* the closure to have from context?
28#[derive(#[automatically_derived]
impl<'tcx> ::core::fmt::Debug for ExpectedSig<'tcx> {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        ::core::fmt::Formatter::debug_struct_field2_finish(f, "ExpectedSig",
            "cause_span", &self.cause_span, "sig", &&self.sig)
    }
}Debug, #[automatically_derived]
impl<'tcx> ::core::clone::Clone for ExpectedSig<'tcx> {
    #[inline]
    fn clone(&self) -> ExpectedSig<'tcx> {
        ExpectedSig {
            cause_span: ::core::clone::Clone::clone(&self.cause_span),
            sig: ::core::clone::Clone::clone(&self.sig),
        }
    }
}Clone, const _: () =
    {
        impl<'tcx>
            ::rustc_middle::ty::TypeFoldable<::rustc_middle::ty::TyCtxt<'tcx>>
            for ExpectedSig<'tcx> {
            fn try_fold_with<__F: ::rustc_middle::ty::FallibleTypeFolder<::rustc_middle::ty::TyCtxt<'tcx>>>(self,
                __folder: &mut __F) -> Result<Self, __F::Error> {
                Ok(match self {
                        ExpectedSig { cause_span: __binding_0, sig: __binding_1 } =>
                            {
                            ExpectedSig {
                                cause_span: ::rustc_middle::ty::TypeFoldable::try_fold_with(__binding_0,
                                        __folder)?,
                                sig: ::rustc_middle::ty::TypeFoldable::try_fold_with(__binding_1,
                                        __folder)?,
                            }
                        }
                    })
            }
            fn fold_with<__F: ::rustc_middle::ty::TypeFolder<::rustc_middle::ty::TyCtxt<'tcx>>>(self,
                __folder: &mut __F) -> Self {
                match self {
                    ExpectedSig { cause_span: __binding_0, sig: __binding_1 } =>
                        {
                        ExpectedSig {
                            cause_span: ::rustc_middle::ty::TypeFoldable::fold_with(__binding_0,
                                __folder),
                            sig: ::rustc_middle::ty::TypeFoldable::fold_with(__binding_1,
                                __folder),
                        }
                    }
                }
            }
        }
    };TypeFoldable, const _: () =
    {
        impl<'tcx>
            ::rustc_middle::ty::TypeVisitable<::rustc_middle::ty::TyCtxt<'tcx>>
            for ExpectedSig<'tcx> {
            fn visit_with<__V: ::rustc_middle::ty::TypeVisitor<::rustc_middle::ty::TyCtxt<'tcx>>>(&self,
                __visitor: &mut __V) -> __V::Result {
                match *self {
                    ExpectedSig {
                        cause_span: ref __binding_0, sig: ref __binding_1 } => {
                        {
                            match ::rustc_middle::ty::VisitorResult::branch(::rustc_middle::ty::TypeVisitable::visit_with(__binding_0,
                                        __visitor)) {
                                ::core::ops::ControlFlow::Continue(()) => {}
                                ::core::ops::ControlFlow::Break(r) => {
                                    return ::rustc_middle::ty::VisitorResult::from_residual(r);
                                }
                            }
                        }
                        {
                            match ::rustc_middle::ty::VisitorResult::branch(::rustc_middle::ty::TypeVisitable::visit_with(__binding_1,
                                        __visitor)) {
                                ::core::ops::ControlFlow::Continue(()) => {}
                                ::core::ops::ControlFlow::Break(r) => {
                                    return ::rustc_middle::ty::VisitorResult::from_residual(r);
                                }
                            }
                        }
                    }
                }
                <__V::Result as ::rustc_middle::ty::VisitorResult>::output()
            }
        }
    };TypeVisitable)]
29struct ExpectedSig<'tcx> {
30    /// Span that gave us this expectation, if we know that.
31    cause_span: Option<Span>,
32    sig: ty::PolyFnSig<'tcx>,
33}
34
35#[derive(#[automatically_derived]
impl<'tcx> ::core::fmt::Debug for ClosureSignatures<'tcx> {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        ::core::fmt::Formatter::debug_struct_field2_finish(f,
            "ClosureSignatures", "bound_sig", &self.bound_sig,
            "liberated_sig", &&self.liberated_sig)
    }
}Debug)]
36struct ClosureSignatures<'tcx> {
37    /// The signature users of the closure see.
38    bound_sig: ty::PolyFnSig<'tcx>,
39    /// The signature within the function body.
40    /// This mostly differs in the sense that lifetimes are now early bound and any
41    /// opaque types from the signature expectation are overridden in case there are
42    /// explicit hidden types written by the user in the closure signature.
43    liberated_sig: ty::FnSig<'tcx>,
44}
45
46impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
47    #[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("check_expr_closure",
                                    "rustc_hir_typeck::closure", ::tracing::Level::DEBUG,
                                    ::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/closure.rs"),
                                    ::tracing_core::__macro_support::Option::Some(47u32),
                                    ::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::closure"),
                                    ::tracing_core::field::FieldSet::new(&["expr_span",
                                                    "expected"],
                                        ::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(&expr_span)
                                                            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(&expected)
                                                            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: Ty<'tcx> = loop {};
            return __tracing_attr_fake_return;
        }
        {
            let tcx = self.tcx;
            let body = tcx.hir_body(closure.body);
            let expr_def_id = closure.def_id;
            let (expected_sig, expected_kind) =
                match expected.to_option(self) {
                    Some(ty) => {
                        self.deduce_closure_signature(self.resolve_vars_with_obligations(ty),
                            closure.kind)
                    }
                    None => (None, None),
                };
            let ClosureSignatures { bound_sig, mut liberated_sig } =
                self.sig_of_closure(expr_def_id, closure.fn_decl,
                    closure.kind, expected_sig);
            {
                use ::tracing::__macro_support::Callsite as _;
                static __CALLSITE: ::tracing::callsite::DefaultCallsite =
                    {
                        static META: ::tracing::Metadata<'static> =
                            {
                                ::tracing_core::metadata::Metadata::new("event compiler/rustc_hir_typeck/src/closure.rs:71",
                                    "rustc_hir_typeck::closure", ::tracing::Level::DEBUG,
                                    ::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/closure.rs"),
                                    ::tracing_core::__macro_support::Option::Some(71u32),
                                    ::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::closure"),
                                    ::tracing_core::field::FieldSet::new(&["bound_sig",
                                                    "liberated_sig"],
                                        ::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(&debug(&bound_sig)
                                                        as &dyn Value)),
                                            (&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
                                                ::tracing::__macro_support::Option::Some(&debug(&liberated_sig)
                                                        as &dyn Value))])
                        });
                } else { ; }
            };
            let parent_args =
                GenericArgs::identity_for_item(tcx,
                    tcx.typeck_root_def_id_local(expr_def_id));
            let tupled_upvars_ty = self.next_ty_var(expr_span);
            let (closure_ty, coroutine_types) =
                match closure.kind {
                    hir::ClosureKind::Closure => {
                        let sig =
                            bound_sig.map_bound(|sig|
                                    {
                                        tcx.mk_fn_sig([Ty::new_tup(tcx, sig.inputs())],
                                            sig.output(), sig.fn_sig_kind)
                                    });
                        {
                            use ::tracing::__macro_support::Callsite as _;
                            static __CALLSITE: ::tracing::callsite::DefaultCallsite =
                                {
                                    static META: ::tracing::Metadata<'static> =
                                        {
                                            ::tracing_core::metadata::Metadata::new("event compiler/rustc_hir_typeck/src/closure.rs:90",
                                                "rustc_hir_typeck::closure", ::tracing::Level::DEBUG,
                                                ::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/closure.rs"),
                                                ::tracing_core::__macro_support::Option::Some(90u32),
                                                ::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::closure"),
                                                ::tracing_core::field::FieldSet::new(&["sig",
                                                                "expected_kind"],
                                                    ::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(&debug(&sig) as
                                                                    &dyn Value)),
                                                        (&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
                                                            ::tracing::__macro_support::Option::Some(&debug(&expected_kind)
                                                                    as &dyn Value))])
                                    });
                            } else { ; }
                        };
                        let closure_kind_ty =
                            match expected_kind {
                                Some(kind) => Ty::from_closure_kind(tcx, kind),
                                None => self.next_ty_var(expr_span),
                            };
                        let closure_args =
                            ty::ClosureArgs::new(tcx,
                                ty::ClosureArgsParts {
                                    parent_args,
                                    closure_kind_ty,
                                    closure_sig_as_fn_ptr_ty: Ty::new_fn_ptr(tcx, sig),
                                    tupled_upvars_ty,
                                });
                        (Ty::new_closure(tcx, expr_def_id.to_def_id(),
                                closure_args.args), None)
                    }
                    hir::ClosureKind::Coroutine(kind) => {
                        let yield_ty =
                            match kind {
                                hir::CoroutineKind::Desugared(hir::CoroutineDesugaring::Gen,
                                    _) | hir::CoroutineKind::Coroutine(_) => {
                                    let yield_ty = self.next_ty_var(expr_span);
                                    self.require_type_is_sized(yield_ty, expr_span,
                                        ObligationCauseCode::SizedYieldType);
                                    yield_ty
                                }
                                hir::CoroutineKind::Desugared(hir::CoroutineDesugaring::AsyncGen,
                                    _) => {
                                    let yield_ty = self.next_ty_var(expr_span);
                                    self.require_type_is_sized(yield_ty, expr_span,
                                        ObligationCauseCode::SizedYieldType);
                                    Ty::new_adt(tcx,
                                        tcx.adt_def(tcx.require_lang_item(hir::LangItem::Poll,
                                                expr_span)),
                                        tcx.mk_args(&[Ty::new_adt(tcx,
                                                                tcx.adt_def(tcx.require_lang_item(hir::LangItem::Option,
                                                                        expr_span)), tcx.mk_args(&[yield_ty.into()])).into()]))
                                }
                                hir::CoroutineKind::Desugared(hir::CoroutineDesugaring::Async,
                                    _) => {
                                    tcx.types.unit
                                }
                            };
                        let resume_ty =
                            liberated_sig.inputs().get(0).copied().unwrap_or(tcx.types.unit);
                        let kind_ty =
                            match kind {
                                hir::CoroutineKind::Desugared(_,
                                    hir::CoroutineSource::Closure) => {
                                    self.next_ty_var(expr_span)
                                }
                                _ => tcx.types.unit,
                            };
                        let coroutine_args =
                            ty::CoroutineArgs::new(tcx,
                                ty::CoroutineArgsParts {
                                    parent_args,
                                    kind_ty,
                                    resume_ty,
                                    yield_ty,
                                    return_ty: liberated_sig.output(),
                                    tupled_upvars_ty,
                                });
                        (Ty::new_coroutine(tcx, expr_def_id.to_def_id(),
                                coroutine_args.args),
                            Some(CoroutineTypes { resume_ty, yield_ty }))
                    }
                    hir::ClosureKind::CoroutineClosure(kind) => {
                        let (bound_return_ty, bound_yield_ty) =
                            match kind {
                                hir::CoroutineDesugaring::Gen => {
                                    (tcx.types.unit, self.infcx.next_ty_var(expr_span))
                                }
                                hir::CoroutineDesugaring::Async => {
                                    (bound_sig.skip_binder().output(), tcx.types.unit)
                                }
                                hir::CoroutineDesugaring::AsyncGen => {
                                    {
                                        ::core::panicking::panic_fmt(format_args!("not yet implemented: {0}",
                                                format_args!("`async gen` closures not supported yet")));
                                    }
                                }
                            };
                        let resume_ty = self.next_ty_var(expr_span);
                        let closure_kind_ty =
                            match expected_kind {
                                Some(kind) => Ty::from_closure_kind(tcx, kind),
                                None => self.next_ty_var(expr_span),
                            };
                        let coroutine_captures_by_ref_ty =
                            self.next_ty_var(expr_span);
                        let closure_args =
                            ty::CoroutineClosureArgs::new(tcx,
                                ty::CoroutineClosureArgsParts {
                                    parent_args,
                                    closure_kind_ty,
                                    signature_parts_ty: Ty::new_fn_ptr(tcx,
                                        bound_sig.map_bound(|sig|
                                                {
                                                    tcx.mk_fn_sig([resume_ty,
                                                                Ty::new_tup_from_iter(tcx, sig.inputs().iter().copied())],
                                                        Ty::new_tup(tcx, &[bound_yield_ty, bound_return_ty]),
                                                        sig.fn_sig_kind)
                                                })),
                                    tupled_upvars_ty,
                                    coroutine_captures_by_ref_ty,
                                });
                        let coroutine_kind_ty =
                            match expected_kind {
                                Some(kind) => Ty::from_coroutine_closure_kind(tcx, kind),
                                None => self.next_ty_var(expr_span),
                            };
                        let coroutine_upvars_ty = self.next_ty_var(expr_span);
                        let coroutine_output_ty =
                            tcx.liberate_late_bound_regions(expr_def_id.to_def_id(),
                                closure_args.coroutine_closure_sig().map_bound(|sig|
                                        {
                                            sig.to_coroutine(tcx, parent_args, coroutine_kind_ty,
                                                tcx.coroutine_for_closure(expr_def_id), coroutine_upvars_ty)
                                        }));
                        liberated_sig =
                            tcx.mk_fn_sig(liberated_sig.inputs().iter().copied(),
                                coroutine_output_ty, liberated_sig.fn_sig_kind);
                        (Ty::new_coroutine_closure(tcx, expr_def_id.to_def_id(),
                                closure_args.args), None)
                    }
                };
            check_fn(&mut FnCtxt::new(self, self.param_env, closure.def_id),
                liberated_sig, coroutine_types, closure.fn_decl, expr_def_id,
                body, false);
            closure_ty
        }
    }
}#[instrument(skip(self, closure), level = "debug")]
48    pub(crate) fn check_expr_closure(
49        &self,
50        closure: &hir::Closure<'tcx>,
51        expr_span: Span,
52        expected: Expectation<'tcx>,
53    ) -> Ty<'tcx> {
54        let tcx = self.tcx;
55        let body = tcx.hir_body(closure.body);
56        let expr_def_id = closure.def_id;
57
58        // It's always helpful for inference if we know the kind of
59        // closure sooner rather than later, so first examine the expected
60        // type, and see if can glean a closure kind from there.
61        let (expected_sig, expected_kind) = match expected.to_option(self) {
62            Some(ty) => {
63                self.deduce_closure_signature(self.resolve_vars_with_obligations(ty), closure.kind)
64            }
65            None => (None, None),
66        };
67
68        let ClosureSignatures { bound_sig, mut liberated_sig } =
69            self.sig_of_closure(expr_def_id, closure.fn_decl, closure.kind, expected_sig);
70
71        debug!(?bound_sig, ?liberated_sig);
72
73        let parent_args =
74            GenericArgs::identity_for_item(tcx, tcx.typeck_root_def_id_local(expr_def_id));
75
76        let tupled_upvars_ty = self.next_ty_var(expr_span);
77
78        // FIXME: We could probably actually just unify this further --
79        // instead of having a `FnSig` and a `Option<CoroutineTypes>`,
80        // we can have a `ClosureSignature { Coroutine { .. }, Closure { .. } }`,
81        // similar to how `ty::GenSig` is a distinct data structure.
82        let (closure_ty, coroutine_types) = match closure.kind {
83            hir::ClosureKind::Closure => {
84                // Tuple up the arguments and insert the resulting function type into
85                // the `closures` table.
86                let sig = bound_sig.map_bound(|sig| {
87                    tcx.mk_fn_sig([Ty::new_tup(tcx, sig.inputs())], sig.output(), sig.fn_sig_kind)
88                });
89
90                debug!(?sig, ?expected_kind);
91
92                let closure_kind_ty = match expected_kind {
93                    Some(kind) => Ty::from_closure_kind(tcx, kind),
94
95                    // Create a type variable (for now) to represent the closure kind.
96                    // It will be unified during the upvar inference phase (`upvar.rs`)
97                    None => self.next_ty_var(expr_span),
98                };
99
100                let closure_args = ty::ClosureArgs::new(
101                    tcx,
102                    ty::ClosureArgsParts {
103                        parent_args,
104                        closure_kind_ty,
105                        closure_sig_as_fn_ptr_ty: Ty::new_fn_ptr(tcx, sig),
106                        tupled_upvars_ty,
107                    },
108                );
109
110                (Ty::new_closure(tcx, expr_def_id.to_def_id(), closure_args.args), None)
111            }
112            hir::ClosureKind::Coroutine(kind) => {
113                let yield_ty = match kind {
114                    hir::CoroutineKind::Desugared(hir::CoroutineDesugaring::Gen, _)
115                    | hir::CoroutineKind::Coroutine(_) => {
116                        let yield_ty = self.next_ty_var(expr_span);
117                        self.require_type_is_sized(
118                            yield_ty,
119                            expr_span,
120                            ObligationCauseCode::SizedYieldType,
121                        );
122                        yield_ty
123                    }
124                    // HACK(-Ztrait-solver=next): In the *old* trait solver, we must eagerly
125                    // guide inference on the yield type so that we can handle `AsyncIterator`
126                    // in this block in projection correctly. In the new trait solver, it is
127                    // not a problem.
128                    hir::CoroutineKind::Desugared(hir::CoroutineDesugaring::AsyncGen, _) => {
129                        let yield_ty = self.next_ty_var(expr_span);
130                        self.require_type_is_sized(
131                            yield_ty,
132                            expr_span,
133                            ObligationCauseCode::SizedYieldType,
134                        );
135
136                        Ty::new_adt(
137                            tcx,
138                            tcx.adt_def(tcx.require_lang_item(hir::LangItem::Poll, expr_span)),
139                            tcx.mk_args(&[Ty::new_adt(
140                                tcx,
141                                tcx.adt_def(
142                                    tcx.require_lang_item(hir::LangItem::Option, expr_span),
143                                ),
144                                tcx.mk_args(&[yield_ty.into()]),
145                            )
146                            .into()]),
147                        )
148                    }
149                    hir::CoroutineKind::Desugared(hir::CoroutineDesugaring::Async, _) => {
150                        tcx.types.unit
151                    }
152                };
153
154                // Resume type defaults to `()` if the coroutine has no argument.
155                let resume_ty = liberated_sig.inputs().get(0).copied().unwrap_or(tcx.types.unit);
156
157                // Coroutines that come from coroutine closures have not yet determined
158                // their kind ty, so make a fresh infer var which will be constrained
159                // later during upvar analysis. Regular coroutines always have the kind
160                // ty of `().`
161                let kind_ty = match kind {
162                    hir::CoroutineKind::Desugared(_, hir::CoroutineSource::Closure) => {
163                        self.next_ty_var(expr_span)
164                    }
165                    _ => tcx.types.unit,
166                };
167
168                let coroutine_args = ty::CoroutineArgs::new(
169                    tcx,
170                    ty::CoroutineArgsParts {
171                        parent_args,
172                        kind_ty,
173                        resume_ty,
174                        yield_ty,
175                        return_ty: liberated_sig.output(),
176                        tupled_upvars_ty,
177                    },
178                );
179
180                (
181                    Ty::new_coroutine(tcx, expr_def_id.to_def_id(), coroutine_args.args),
182                    Some(CoroutineTypes { resume_ty, yield_ty }),
183                )
184            }
185            hir::ClosureKind::CoroutineClosure(kind) => {
186                let (bound_return_ty, bound_yield_ty) = match kind {
187                    hir::CoroutineDesugaring::Gen => {
188                        // `iter!` closures always return unit and yield the `Iterator::Item` type
189                        // that we have to infer.
190                        (tcx.types.unit, self.infcx.next_ty_var(expr_span))
191                    }
192                    hir::CoroutineDesugaring::Async => {
193                        // async closures always return the type ascribed after the `->` (if present),
194                        // and yield `()`.
195                        (bound_sig.skip_binder().output(), tcx.types.unit)
196                    }
197                    hir::CoroutineDesugaring::AsyncGen => {
198                        todo!("`async gen` closures not supported yet")
199                    }
200                };
201                // Compute all of the variables that will be used to populate the coroutine.
202                let resume_ty = self.next_ty_var(expr_span);
203
204                let closure_kind_ty = match expected_kind {
205                    Some(kind) => Ty::from_closure_kind(tcx, kind),
206
207                    // Create a type variable (for now) to represent the closure kind.
208                    // It will be unified during the upvar inference phase (`upvar.rs`)
209                    None => self.next_ty_var(expr_span),
210                };
211
212                let coroutine_captures_by_ref_ty = self.next_ty_var(expr_span);
213                let closure_args = ty::CoroutineClosureArgs::new(
214                    tcx,
215                    ty::CoroutineClosureArgsParts {
216                        parent_args,
217                        closure_kind_ty,
218                        signature_parts_ty: Ty::new_fn_ptr(
219                            tcx,
220                            bound_sig.map_bound(|sig| {
221                                tcx.mk_fn_sig(
222                                    [
223                                        resume_ty,
224                                        Ty::new_tup_from_iter(tcx, sig.inputs().iter().copied()),
225                                    ],
226                                    Ty::new_tup(tcx, &[bound_yield_ty, bound_return_ty]),
227                                    sig.fn_sig_kind,
228                                )
229                            }),
230                        ),
231                        tupled_upvars_ty,
232                        coroutine_captures_by_ref_ty,
233                    },
234                );
235
236                let coroutine_kind_ty = match expected_kind {
237                    Some(kind) => Ty::from_coroutine_closure_kind(tcx, kind),
238
239                    // Create a type variable (for now) to represent the closure kind.
240                    // It will be unified during the upvar inference phase (`upvar.rs`)
241                    None => self.next_ty_var(expr_span),
242                };
243
244                let coroutine_upvars_ty = self.next_ty_var(expr_span);
245
246                // We need to turn the liberated signature that we got from HIR, which
247                // looks something like `|Args...| -> T`, into a signature that is suitable
248                // for type checking the inner body of the closure, which always returns a
249                // coroutine. To do so, we use the `CoroutineClosureSignature` to compute
250                // the coroutine type, filling in the tupled_upvars_ty and kind_ty with infer
251                // vars which will get constrained during upvar analysis.
252                let coroutine_output_ty = tcx.liberate_late_bound_regions(
253                    expr_def_id.to_def_id(),
254                    closure_args.coroutine_closure_sig().map_bound(|sig| {
255                        sig.to_coroutine(
256                            tcx,
257                            parent_args,
258                            coroutine_kind_ty,
259                            tcx.coroutine_for_closure(expr_def_id),
260                            coroutine_upvars_ty,
261                        )
262                    }),
263                );
264                liberated_sig = tcx.mk_fn_sig(
265                    liberated_sig.inputs().iter().copied(),
266                    coroutine_output_ty,
267                    liberated_sig.fn_sig_kind,
268                );
269
270                (Ty::new_coroutine_closure(tcx, expr_def_id.to_def_id(), closure_args.args), None)
271            }
272        };
273
274        check_fn(
275            &mut FnCtxt::new(self, self.param_env, closure.def_id),
276            liberated_sig,
277            coroutine_types,
278            closure.fn_decl,
279            expr_def_id,
280            body,
281            // Closure "rust-call" ABI doesn't support unsized params
282            false,
283        );
284
285        closure_ty
286    }
287
288    /// Given the expected type, figures out what it can about this closure we
289    /// are about to type check:
290    x;#[instrument(skip(self), level = "debug", ret)]
291    fn deduce_closure_signature(
292        &self,
293        expected_ty: Ty<'tcx>,
294        closure_kind: hir::ClosureKind,
295    ) -> (Option<ExpectedSig<'tcx>>, Option<ty::ClosureKind>) {
296        match *expected_ty.kind() {
297            ty::Alias(ty::AliasTy { kind: ty::Opaque { def_id }, args, .. }) => self
298                .deduce_closure_signature_from_predicates(
299                    expected_ty,
300                    closure_kind,
301                    self.tcx
302                        .explicit_item_self_bounds(def_id)
303                        .iter_instantiated_copied(self.tcx, args)
304                        .map(Unnormalized::skip_norm_wip)
305                        .map(|(c, s)| (c.as_predicate(), s)),
306                ),
307            ty::Dynamic(object_type, ..) => {
308                let sig = object_type.projection_bounds().find_map(|pb| {
309                    let pb = pb.with_self_ty(self.tcx, self.tcx.types.trait_object_dummy_self);
310                    self.deduce_sig_from_projection(None, closure_kind, pb)
311                });
312                let kind = object_type
313                    .principal_def_id()
314                    .and_then(|did| self.tcx.fn_trait_kind_from_def_id(did));
315                (sig, kind)
316            }
317            ty::Infer(ty::TyVar(vid)) => self.deduce_closure_signature_from_predicates(
318                Ty::new_var(self.tcx, self.root_var(vid)),
319                closure_kind,
320                self.obligations_for_self_ty(vid)
321                    .into_iter()
322                    .map(|obl| (obl.predicate, obl.cause.span)),
323            ),
324            ty::FnPtr(sig_tys, hdr) => match closure_kind {
325                hir::ClosureKind::Closure => {
326                    let expected_sig = ExpectedSig { cause_span: None, sig: sig_tys.with(hdr) };
327                    (Some(expected_sig), Some(ty::ClosureKind::Fn))
328                }
329                hir::ClosureKind::Coroutine(_) | hir::ClosureKind::CoroutineClosure(_) => {
330                    (None, None)
331                }
332            },
333            _ => (None, None),
334        }
335    }
336
337    fn deduce_closure_signature_from_predicates(
338        &self,
339        expected_ty: Ty<'tcx>,
340        closure_kind: hir::ClosureKind,
341        predicates: impl DoubleEndedIterator<Item = (ty::Predicate<'tcx>, Span)>,
342    ) -> (Option<ExpectedSig<'tcx>>, Option<ty::ClosureKind>) {
343        let mut expected_sig = None;
344        let mut expected_kind = None;
345
346        for (pred, span) in traits::elaborate(
347            self.tcx,
348            // Reverse the obligations here, since `elaborate_*` uses a stack,
349            // and we want to keep inference generally in the same order of
350            // the registered obligations.
351            predicates.rev(),
352        )
353        // We only care about self bounds
354        .filter_only_self()
355        {
356            {
    use ::tracing::__macro_support::Callsite as _;
    static __CALLSITE: ::tracing::callsite::DefaultCallsite =
        {
            static META: ::tracing::Metadata<'static> =
                {
                    ::tracing_core::metadata::Metadata::new("event compiler/rustc_hir_typeck/src/closure.rs:356",
                        "rustc_hir_typeck::closure", ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/closure.rs"),
                        ::tracing_core::__macro_support::Option::Some(356u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::closure"),
                        ::tracing_core::field::FieldSet::new(&["pred"],
                            ::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(&debug(&pred) as
                                            &dyn Value))])
            });
    } else { ; }
};debug!(?pred);
357            let bound_predicate = pred.kind();
358
359            // Given a Projection predicate, we can potentially infer
360            // the complete signature.
361            if expected_sig.is_none()
362                && let ty::PredicateKind::Clause(ty::ClauseKind::Projection(proj_predicate)) =
363                    bound_predicate.skip_binder()
364            {
365                let inferred_sig = self.normalize(
366                    span,
367                    Unnormalized::new_wip(self.deduce_sig_from_projection(
368                        Some(span),
369                        closure_kind,
370                        bound_predicate.rebind(proj_predicate),
371                    )),
372                );
373
374                // Make sure that we didn't infer a signature that mentions itself.
375                // This can happen when we elaborate certain supertrait bounds that
376                // mention projections containing the `Self` type. See #105401.
377                struct MentionsTy<'tcx> {
378                    expected_ty: Ty<'tcx>,
379                }
380                impl<'tcx> TypeVisitor<TyCtxt<'tcx>> for MentionsTy<'tcx> {
381                    type Result = ControlFlow<()>;
382
383                    fn visit_ty(&mut self, t: Ty<'tcx>) -> Self::Result {
384                        if t == self.expected_ty {
385                            ControlFlow::Break(())
386                        } else {
387                            t.super_visit_with(self)
388                        }
389                    }
390                }
391
392                // Don't infer a closure signature from a goal that names the closure type as this will
393                // (almost always) lead to occurs check errors later in type checking.
394                if self.next_trait_solver()
395                    && let Some(inferred_sig) = inferred_sig
396                {
397                    // In the new solver it is difficult to explicitly normalize the inferred signature as we
398                    // would have to manually handle universes and rewriting bound vars and placeholders back
399                    // and forth.
400                    //
401                    // Instead we take advantage of the fact that we relating an inference variable with an alias
402                    // will only instantiate the variable if the alias is rigid(*not quite). Concretely we:
403                    // - Create some new variable `?sig`
404                    // - Equate `?sig` with the unnormalized signature, e.g. `fn(<Foo<?x> as Trait>::Assoc)`
405                    // - Depending on whether `<Foo<?x> as Trait>::Assoc` is rigid, ambiguous or normalizeable,
406                    //   we will either wind up with `?sig=<Foo<?x> as Trait>::Assoc/?y/ConcreteTy` respectively.
407                    //
408                    // *: In cases where there are ambiguous aliases in the signature that make use of bound vars
409                    //    they will wind up present in `?sig` even though they are non-rigid.
410                    //
411                    //    This is a bit weird and means we may wind up discarding the goal due to it naming `expected_ty`
412                    //    even though the normalized form may not name `expected_ty`. However, this matches the existing
413                    //    behaviour of the old solver and would be technically a breaking change to fix.
414                    let generalized_fnptr_sig = self.next_ty_var(span);
415                    let inferred_fnptr_sig = Ty::new_fn_ptr(self.tcx, inferred_sig.sig);
416                    self.demand_eqtype(span, inferred_fnptr_sig, generalized_fnptr_sig);
417
418                    let resolved_sig = self.resolve_vars_if_possible(generalized_fnptr_sig);
419
420                    if resolved_sig.visit_with(&mut MentionsTy { expected_ty }).is_continue() {
421                        expected_sig = Some(ExpectedSig {
422                            cause_span: inferred_sig.cause_span,
423                            sig: resolved_sig.fn_sig(self.tcx),
424                        });
425                    }
426                } else {
427                    if inferred_sig.visit_with(&mut MentionsTy { expected_ty }).is_continue() {
428                        expected_sig = inferred_sig;
429                    }
430                }
431            }
432
433            // Even if we can't infer the full signature, we may be able to
434            // infer the kind. This can occur when we elaborate a predicate
435            // like `F : Fn<A>`. Note that due to subtyping we could encounter
436            // many viable options, so pick the most restrictive.
437            let trait_def_id = match bound_predicate.skip_binder() {
438                ty::PredicateKind::Clause(ty::ClauseKind::Projection(data)) => {
439                    Some(data.projection_term.trait_def_id(self.tcx))
440                }
441                ty::PredicateKind::Clause(ty::ClauseKind::Trait(data)) => Some(data.def_id()),
442                _ => None,
443            };
444
445            if let Some(trait_def_id) = trait_def_id {
446                let found_kind = match closure_kind {
447                    hir::ClosureKind::Closure
448                    // FIXME(iter_macro): Someday we'll probably want iterator closures instead of
449                    // just using Fn* for iterators.
450                    | hir::ClosureKind::CoroutineClosure(hir::CoroutineDesugaring::Gen) => {
451                        self.tcx.fn_trait_kind_from_def_id(trait_def_id)
452                    }
453                    hir::ClosureKind::CoroutineClosure(hir::CoroutineDesugaring::Async) => self
454                        .tcx
455                        .async_fn_trait_kind_from_def_id(trait_def_id)
456                        .or_else(|| self.tcx.fn_trait_kind_from_def_id(trait_def_id)),
457                    _ => None,
458                };
459
460                if let Some(found_kind) = found_kind {
461                    // always use the closure kind that is more permissive.
462                    match (expected_kind, found_kind) {
463                        (None, _) => expected_kind = Some(found_kind),
464                        (Some(ClosureKind::FnMut), ClosureKind::Fn) => {
465                            expected_kind = Some(ClosureKind::Fn)
466                        }
467                        (Some(ClosureKind::FnOnce), ClosureKind::Fn | ClosureKind::FnMut) => {
468                            expected_kind = Some(found_kind)
469                        }
470                        _ => {}
471                    }
472                }
473            }
474        }
475
476        (expected_sig, expected_kind)
477    }
478
479    /// Given a projection like "<F as Fn(X)>::Result == Y", we can deduce
480    /// everything we need to know about a closure or coroutine.
481    ///
482    /// The `cause_span` should be the span that caused us to
483    /// have this expected signature, or `None` if we can't readily
484    /// know that.
485    x;#[instrument(level = "debug", skip(self, cause_span), ret)]
486    fn deduce_sig_from_projection(
487        &self,
488        cause_span: Option<Span>,
489        closure_kind: hir::ClosureKind,
490        projection: ty::PolyProjectionPredicate<'tcx>,
491    ) -> Option<ExpectedSig<'tcx>> {
492        let def_id = projection.item_def_id();
493
494        // For now, we only do signature deduction based off of the `Fn` and `AsyncFn` traits,
495        // for closures and async closures, respectively.
496        match closure_kind {
497            hir::ClosureKind::Closure if self.tcx.is_lang_item(def_id, LangItem::FnOnceOutput) => {
498                self.extract_sig_from_projection(cause_span, projection)
499            }
500            hir::ClosureKind::CoroutineClosure(hir::CoroutineDesugaring::Async)
501                if self.tcx.is_lang_item(def_id, LangItem::AsyncFnOnceOutput) =>
502            {
503                self.extract_sig_from_projection(cause_span, projection)
504            }
505            // It's possible we've passed the closure to a (somewhat out-of-fashion)
506            // `F: FnOnce() -> Fut, Fut: Future<Output = T>` style bound. Let's still
507            // guide inference here, since it's beneficial for the user.
508            hir::ClosureKind::CoroutineClosure(hir::CoroutineDesugaring::Async)
509                if self.tcx.is_lang_item(def_id, LangItem::FnOnceOutput) =>
510            {
511                self.extract_sig_from_projection_and_future_bound(cause_span, projection)
512            }
513            _ => None,
514        }
515    }
516
517    /// Given an `FnOnce::Output` or `AsyncFn::Output` projection, extract the args
518    /// and return type to infer a [`ty::PolyFnSig`] for the closure.
519    fn extract_sig_from_projection(
520        &self,
521        cause_span: Option<Span>,
522        projection: ty::PolyProjectionPredicate<'tcx>,
523    ) -> Option<ExpectedSig<'tcx>> {
524        let projection = self.resolve_vars_if_possible(projection);
525
526        let arg_param_ty = projection.skip_binder().projection_term.args.type_at(1);
527        {
    use ::tracing::__macro_support::Callsite as _;
    static __CALLSITE: ::tracing::callsite::DefaultCallsite =
        {
            static META: ::tracing::Metadata<'static> =
                {
                    ::tracing_core::metadata::Metadata::new("event compiler/rustc_hir_typeck/src/closure.rs:527",
                        "rustc_hir_typeck::closure", ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/closure.rs"),
                        ::tracing_core::__macro_support::Option::Some(527u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::closure"),
                        ::tracing_core::field::FieldSet::new(&["arg_param_ty"],
                            ::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(&debug(&arg_param_ty)
                                            as &dyn Value))])
            });
    } else { ; }
};debug!(?arg_param_ty);
528
529        let ty::Tuple(input_tys) = *arg_param_ty.kind() else {
530            return None;
531        };
532
533        // Since this is a return parameter type it is safe to unwrap.
534        let ret_param_ty = projection.skip_binder().term.expect_type();
535        {
    use ::tracing::__macro_support::Callsite as _;
    static __CALLSITE: ::tracing::callsite::DefaultCallsite =
        {
            static META: ::tracing::Metadata<'static> =
                {
                    ::tracing_core::metadata::Metadata::new("event compiler/rustc_hir_typeck/src/closure.rs:535",
                        "rustc_hir_typeck::closure", ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/closure.rs"),
                        ::tracing_core::__macro_support::Option::Some(535u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::closure"),
                        ::tracing_core::field::FieldSet::new(&["ret_param_ty"],
                            ::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(&debug(&ret_param_ty)
                                            as &dyn Value))])
            });
    } else { ; }
};debug!(?ret_param_ty);
536
537        let sig = projection.rebind(self.tcx.mk_fn_sig_safe_rust_abi(input_tys, ret_param_ty));
538
539        Some(ExpectedSig { cause_span, sig })
540    }
541
542    /// When an async closure is passed to a function that has a "two-part" `Fn`
543    /// and `Future` trait bound, like:
544    ///
545    /// ```rust
546    /// use std::future::Future;
547    ///
548    /// fn not_exactly_an_async_closure<F, Fut>(_f: F)
549    /// where
550    ///     F: FnOnce(String, u32) -> Fut,
551    ///     Fut: Future<Output = i32>,
552    /// {}
553    /// ```
554    ///
555    /// The we want to be able to extract the signature to guide inference in the async
556    /// closure. We will have two projection predicates registered in this case. First,
557    /// we identify the `FnOnce<Args, Output = ?Fut>` bound, and if the output type is
558    /// an inference variable `?Fut`, we check if that is bounded by a `Future<Output = Ty>`
559    /// projection.
560    ///
561    /// This function is actually best-effort with the return type; if we don't find a
562    /// `Future` projection, we still will return arguments that we extracted from the `FnOnce`
563    /// projection, and the output will be an unconstrained type variable instead.
564    fn extract_sig_from_projection_and_future_bound(
565        &self,
566        cause_span: Option<Span>,
567        projection: ty::PolyProjectionPredicate<'tcx>,
568    ) -> Option<ExpectedSig<'tcx>> {
569        let projection = self.resolve_vars_if_possible(projection);
570
571        let arg_param_ty = projection.skip_binder().projection_term.args.type_at(1);
572        {
    use ::tracing::__macro_support::Callsite as _;
    static __CALLSITE: ::tracing::callsite::DefaultCallsite =
        {
            static META: ::tracing::Metadata<'static> =
                {
                    ::tracing_core::metadata::Metadata::new("event compiler/rustc_hir_typeck/src/closure.rs:572",
                        "rustc_hir_typeck::closure", ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/closure.rs"),
                        ::tracing_core::__macro_support::Option::Some(572u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::closure"),
                        ::tracing_core::field::FieldSet::new(&["arg_param_ty"],
                            ::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(&debug(&arg_param_ty)
                                            as &dyn Value))])
            });
    } else { ; }
};debug!(?arg_param_ty);
573
574        let ty::Tuple(input_tys) = *arg_param_ty.kind() else {
575            return None;
576        };
577
578        // If the return type is a type variable, look for bounds on it.
579        // We could theoretically support other kinds of return types here,
580        // but none of them would be useful, since async closures return
581        // concrete anonymous future types, and their futures are not coerced
582        // into any other type within the body of the async closure.
583        let ty::Infer(ty::TyVar(return_vid)) = *projection.skip_binder().term.expect_type().kind()
584        else {
585            return None;
586        };
587
588        // FIXME: We may want to elaborate here, though I assume this will be exceedingly rare.
589        let mut return_ty = None;
590        for bound in self.obligations_for_self_ty(return_vid) {
591            if let Some(ret_projection) = bound.predicate.as_projection_clause()
592                && let Some(ret_projection) = ret_projection.no_bound_vars()
593                && self.tcx.is_lang_item(ret_projection.def_id(), LangItem::FutureOutput)
594            {
595                return_ty = Some(ret_projection.term.expect_type());
596                break;
597            }
598        }
599
600        // SUBTLE: If we didn't find a `Future<Output = ...>` bound for the return
601        // vid, we still want to attempt to provide inference guidance for the async
602        // closure's arguments. Instantiate a new vid to plug into the output type.
603        //
604        // You may be wondering, what if it's higher-ranked? Well, given that we
605        // found a type variable for the `FnOnce::Output` projection above, we know
606        // that the output can't mention any of the vars.
607        //
608        // Also note that we use a fresh var here for the signature since the signature
609        // records the output of the *future*, and `return_vid` above is the type
610        // variable of the future, not its output.
611        //
612        // FIXME: We probably should store this signature inference output in a way
613        // that does not misuse a `FnSig` type, but that can be done separately.
614        let return_ty =
615            return_ty.unwrap_or_else(|| self.next_ty_var(cause_span.unwrap_or(DUMMY_SP)));
616
617        let sig = projection.rebind(self.tcx.mk_fn_sig_safe_rust_abi(input_tys, return_ty));
618
619        Some(ExpectedSig { cause_span, sig })
620    }
621
622    fn sig_of_closure(
623        &self,
624        expr_def_id: LocalDefId,
625        decl: &hir::FnDecl<'tcx>,
626        closure_kind: hir::ClosureKind,
627        expected_sig: Option<ExpectedSig<'tcx>>,
628    ) -> ClosureSignatures<'tcx> {
629        if let Some(e) = expected_sig {
630            self.sig_of_closure_with_expectation(expr_def_id, decl, closure_kind, e)
631        } else {
632            self.sig_of_closure_no_expectation(expr_def_id, decl, closure_kind)
633        }
634    }
635
636    /// If there is no expected signature, then we will convert the
637    /// types that the user gave into a signature.
638    #[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("sig_of_closure_no_expectation",
                                    "rustc_hir_typeck::closure", ::tracing::Level::DEBUG,
                                    ::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/closure.rs"),
                                    ::tracing_core::__macro_support::Option::Some(638u32),
                                    ::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::closure"),
                                    ::tracing_core::field::FieldSet::new(&["closure_kind"],
                                        ::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(&closure_kind)
                                                            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: ClosureSignatures<'tcx> = loop {};
            return __tracing_attr_fake_return;
        }
        {
            let bound_sig =
                self.supplied_sig_of_closure(expr_def_id, decl, closure_kind);
            self.closure_sigs(expr_def_id, bound_sig)
        }
    }
}#[instrument(skip(self, expr_def_id, decl), level = "debug")]
639    fn sig_of_closure_no_expectation(
640        &self,
641        expr_def_id: LocalDefId,
642        decl: &hir::FnDecl<'tcx>,
643        closure_kind: hir::ClosureKind,
644    ) -> ClosureSignatures<'tcx> {
645        let bound_sig = self.supplied_sig_of_closure(expr_def_id, decl, closure_kind);
646
647        self.closure_sigs(expr_def_id, bound_sig)
648    }
649
650    /// Invoked to compute the signature of a closure expression. This
651    /// combines any user-provided type annotations (e.g., `|x: u32|
652    /// -> u32 { .. }`) with the expected signature.
653    ///
654    /// The approach is as follows:
655    ///
656    /// - Let `S` be the (higher-ranked) signature that we derive from the user's annotations.
657    /// - Let `E` be the (higher-ranked) signature that we derive from the expectations, if any.
658    ///   - If we have no expectation `E`, then the signature of the closure is `S`.
659    ///   - Otherwise, the signature of the closure is E. Moreover:
660    ///     - Skolemize the late-bound regions in `E`, yielding `E'`.
661    ///     - Instantiate all the late-bound regions bound in the closure within `S`
662    ///       with fresh (existential) variables, yielding `S'`
663    ///     - Require that `E' = S'`
664    ///       - We could use some kind of subtyping relationship here,
665    ///         I imagine, but equality is easier and works fine for
666    ///         our purposes.
667    ///
668    /// The key intuition here is that the user's types must be valid
669    /// from "the inside" of the closure, but the expectation
670    /// ultimately drives the overall signature.
671    ///
672    /// # Examples
673    ///
674    /// ```ignore (illustrative)
675    /// fn with_closure<F>(_: F)
676    ///   where F: Fn(&u32) -> &u32 { .. }
677    ///
678    /// with_closure(|x: &u32| { ... })
679    /// ```
680    ///
681    /// Here:
682    /// - E would be `fn(&u32) -> &u32`.
683    /// - S would be `fn(&u32) -> ?T`
684    /// - E' is `&'!0 u32 -> &'!0 u32`
685    /// - S' is `&'?0 u32 -> ?T`
686    ///
687    /// S' can be unified with E' with `['?0 = '!0, ?T = &'!10 u32]`.
688    ///
689    /// # Arguments
690    ///
691    /// - `expr_def_id`: the `LocalDefId` of the closure expression
692    /// - `decl`: the HIR declaration of the closure
693    /// - `body`: the body of the closure
694    /// - `expected_sig`: the expected signature (if any). Note that
695    ///   this is missing a binder: that is, there may be late-bound
696    ///   regions with depth 1, which are bound then by the closure.
697    #[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("sig_of_closure_with_expectation",
                                    "rustc_hir_typeck::closure", ::tracing::Level::DEBUG,
                                    ::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/closure.rs"),
                                    ::tracing_core::__macro_support::Option::Some(697u32),
                                    ::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::closure"),
                                    ::tracing_core::field::FieldSet::new(&["closure_kind",
                                                    "expected_sig"],
                                        ::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(&closure_kind)
                                                            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(&expected_sig)
                                                            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: ClosureSignatures<'tcx> = loop {};
            return __tracing_attr_fake_return;
        }
        {
            if expected_sig.sig.c_variadic() != decl.c_variadic() {
                return self.sig_of_closure_no_expectation(expr_def_id, decl,
                        closure_kind);
            } else if expected_sig.sig.skip_binder().inputs_and_output.len()
                    != decl.inputs.len() + 1 {
                return self.sig_of_closure_with_mismatched_number_of_arguments(expr_def_id,
                        decl, expected_sig);
            }
            if !!expected_sig.sig.skip_binder().has_vars_bound_above(ty::INNERMOST)
                {
                ::core::panicking::panic("assertion failed: !expected_sig.sig.skip_binder().has_vars_bound_above(ty::INNERMOST)")
            };
            let bound_sig =
                expected_sig.sig.map_bound(|sig|
                        {
                            let fn_sig_kind =
                                FnSigKind::default().set_abi(ExternAbi::RustCall).set_safety(hir::Safety::Safe).set_c_variadic(sig.c_variadic());
                            self.tcx.mk_fn_sig(sig.inputs().iter().cloned(),
                                sig.output(), fn_sig_kind)
                        });
            let bound_sig = self.tcx.anonymize_bound_vars(bound_sig);
            let closure_sigs = self.closure_sigs(expr_def_id, bound_sig);
            match self.merge_supplied_sig_with_expectation(expr_def_id, decl,
                    closure_kind, closure_sigs) {
                Ok(infer_ok) => self.register_infer_ok_obligations(infer_ok),
                Err(_) =>
                    self.sig_of_closure_no_expectation(expr_def_id, decl,
                        closure_kind),
            }
        }
    }
}#[instrument(skip(self, expr_def_id, decl), level = "debug")]
698    fn sig_of_closure_with_expectation(
699        &self,
700        expr_def_id: LocalDefId,
701        decl: &hir::FnDecl<'tcx>,
702        closure_kind: hir::ClosureKind,
703        expected_sig: ExpectedSig<'tcx>,
704    ) -> ClosureSignatures<'tcx> {
705        // Watch out for some surprises and just ignore the
706        // expectation if things don't see to match up with what we
707        // expect.
708        if expected_sig.sig.c_variadic() != decl.c_variadic() {
709            return self.sig_of_closure_no_expectation(expr_def_id, decl, closure_kind);
710        } else if expected_sig.sig.skip_binder().inputs_and_output.len() != decl.inputs.len() + 1 {
711            return self.sig_of_closure_with_mismatched_number_of_arguments(
712                expr_def_id,
713                decl,
714                expected_sig,
715            );
716        }
717
718        // Create a `PolyFnSig`. Note the oddity that late bound
719        // regions appearing free in `expected_sig` are now bound up
720        // in this binder we are creating.
721        assert!(!expected_sig.sig.skip_binder().has_vars_bound_above(ty::INNERMOST));
722        let bound_sig = expected_sig.sig.map_bound(|sig| {
723            let fn_sig_kind = FnSigKind::default()
724                .set_abi(ExternAbi::RustCall)
725                .set_safety(hir::Safety::Safe)
726                .set_c_variadic(sig.c_variadic());
727            self.tcx.mk_fn_sig(sig.inputs().iter().cloned(), sig.output(), fn_sig_kind)
728        });
729
730        // `deduce_expectations_from_expected_type` introduces
731        // late-bound lifetimes defined elsewhere, which we now
732        // anonymize away, so as not to confuse the user.
733        let bound_sig = self.tcx.anonymize_bound_vars(bound_sig);
734
735        let closure_sigs = self.closure_sigs(expr_def_id, bound_sig);
736
737        // Up till this point, we have ignored the annotations that the user
738        // gave. This function will check that they unify successfully.
739        // Along the way, it also writes out entries for types that the user
740        // wrote into our typeck results, which are then later used by the privacy
741        // check.
742        match self.merge_supplied_sig_with_expectation(
743            expr_def_id,
744            decl,
745            closure_kind,
746            closure_sigs,
747        ) {
748            Ok(infer_ok) => self.register_infer_ok_obligations(infer_ok),
749            Err(_) => self.sig_of_closure_no_expectation(expr_def_id, decl, closure_kind),
750        }
751    }
752
753    fn sig_of_closure_with_mismatched_number_of_arguments(
754        &self,
755        expr_def_id: LocalDefId,
756        decl: &hir::FnDecl<'tcx>,
757        expected_sig: ExpectedSig<'tcx>,
758    ) -> ClosureSignatures<'tcx> {
759        let expr_map_node = self.tcx.hir_node_by_def_id(expr_def_id);
760        let expected_args: Vec<_> = expected_sig
761            .sig
762            .skip_binder()
763            .inputs()
764            .iter()
765            .map(|ty| ArgKind::from_expected_ty(*ty, None))
766            .collect();
767        let (closure_span, closure_arg_span, found_args) =
768            match self.err_ctxt().get_fn_like_arguments(expr_map_node) {
769                Some((sp, arg_sp, args)) => (Some(sp), arg_sp, args),
770                None => (None, None, Vec::new()),
771            };
772        let expected_span =
773            expected_sig.cause_span.unwrap_or_else(|| self.tcx.def_span(expr_def_id));
774        let guar = self
775            .err_ctxt()
776            .report_arg_count_mismatch(
777                expected_span,
778                closure_span,
779                expected_args,
780                found_args,
781                true,
782                closure_arg_span,
783            )
784            .emit();
785
786        let error_sig = self.error_sig_of_closure(decl, guar);
787
788        self.closure_sigs(expr_def_id, error_sig)
789    }
790
791    /// Enforce the user's types against the expectation. See
792    /// `sig_of_closure_with_expectation` for details on the overall
793    /// strategy.
794    #[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("merge_supplied_sig_with_expectation",
                                    "rustc_hir_typeck::closure", ::tracing::Level::DEBUG,
                                    ::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/closure.rs"),
                                    ::tracing_core::__macro_support::Option::Some(794u32),
                                    ::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::closure"),
                                    ::tracing_core::field::FieldSet::new(&["closure_kind"],
                                        ::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(&closure_kind)
                                                            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:
                    InferResult<'tcx, ClosureSignatures<'tcx>> = loop {};
            return __tracing_attr_fake_return;
        }
        {
            let supplied_sig =
                self.supplied_sig_of_closure(expr_def_id, decl, closure_kind);
            {
                use ::tracing::__macro_support::Callsite as _;
                static __CALLSITE: ::tracing::callsite::DefaultCallsite =
                    {
                        static META: ::tracing::Metadata<'static> =
                            {
                                ::tracing_core::metadata::Metadata::new("event compiler/rustc_hir_typeck/src/closure.rs:808",
                                    "rustc_hir_typeck::closure", ::tracing::Level::DEBUG,
                                    ::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/closure.rs"),
                                    ::tracing_core::__macro_support::Option::Some(808u32),
                                    ::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::closure"),
                                    ::tracing_core::field::FieldSet::new(&["supplied_sig"],
                                        ::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(&debug(&supplied_sig)
                                                        as &dyn Value))])
                        });
                } else { ; }
            };
            self.commit_if_ok(|_|
                    {
                        let mut all_obligations = PredicateObligations::new();
                        let supplied_sig =
                            self.instantiate_binder_with_fresh_vars(self.tcx.def_span(expr_def_id),
                                BoundRegionConversionTime::FnCall, supplied_sig);
                        for ((hir_ty, &supplied_ty), expected_ty) in
                            iter::zip(iter::zip(decl.inputs, supplied_sig.inputs()),
                                expected_sigs.liberated_sig.inputs()) {
                            let cause = self.misc(hir_ty.span);
                            let InferOk { value: (), obligations } =
                                self.at(&cause,
                                            self.param_env).eq(DefineOpaqueTypes::Yes, *expected_ty,
                                        supplied_ty)?;
                            all_obligations.extend(obligations);
                        }
                        let supplied_output_ty = supplied_sig.output();
                        let cause = &self.misc(decl.output.span());
                        let InferOk { value: (), obligations } =
                            self.at(cause,
                                        self.param_env).eq(DefineOpaqueTypes::Yes,
                                    expected_sigs.liberated_sig.output(), supplied_output_ty)?;
                        all_obligations.extend(obligations);
                        let inputs =
                            supplied_sig.inputs().into_iter().map(|&ty|
                                    self.resolve_vars_if_possible(ty));
                        let fn_sig_kind =
                            FnSigKind::default().set_abi(ExternAbi::RustCall).set_safety(hir::Safety::Safe).set_c_variadic(expected_sigs.liberated_sig.c_variadic());
                        expected_sigs.liberated_sig =
                            self.tcx.mk_fn_sig(inputs, supplied_output_ty, fn_sig_kind);
                        Ok(InferOk {
                                value: expected_sigs,
                                obligations: all_obligations,
                            })
                    })
        }
    }
}#[instrument(level = "debug", skip(self, expr_def_id, decl, expected_sigs))]
795    fn merge_supplied_sig_with_expectation(
796        &self,
797        expr_def_id: LocalDefId,
798        decl: &hir::FnDecl<'tcx>,
799        closure_kind: hir::ClosureKind,
800        mut expected_sigs: ClosureSignatures<'tcx>,
801    ) -> InferResult<'tcx, ClosureSignatures<'tcx>> {
802        // Get the signature S that the user gave.
803        //
804        // (See comment on `sig_of_closure_with_expectation` for the
805        // meaning of these letters.)
806        let supplied_sig = self.supplied_sig_of_closure(expr_def_id, decl, closure_kind);
807
808        debug!(?supplied_sig);
809
810        // FIXME(#45727): As discussed in [this comment][c1], naively
811        // forcing equality here actually results in suboptimal error
812        // messages in some cases. For now, if there would have been
813        // an obvious error, we fallback to declaring the type of the
814        // closure to be the one the user gave, which allows other
815        // error message code to trigger.
816        //
817        // However, I think [there is potential to do even better
818        // here][c2], since in *this* code we have the precise span of
819        // the type parameter in question in hand when we report the
820        // error.
821        //
822        // [c1]: https://github.com/rust-lang/rust/pull/45072#issuecomment-341089706
823        // [c2]: https://github.com/rust-lang/rust/pull/45072#issuecomment-341096796
824        self.commit_if_ok(|_| {
825            let mut all_obligations = PredicateObligations::new();
826            let supplied_sig = self.instantiate_binder_with_fresh_vars(
827                self.tcx.def_span(expr_def_id),
828                BoundRegionConversionTime::FnCall,
829                supplied_sig,
830            );
831
832            // The liberated version of this signature should be a subtype
833            // of the liberated form of the expectation.
834            for ((hir_ty, &supplied_ty), expected_ty) in iter::zip(
835                iter::zip(decl.inputs, supplied_sig.inputs()),
836                expected_sigs.liberated_sig.inputs(), // `liberated_sig` is E'.
837            ) {
838                // Check that E' = S'.
839                let cause = self.misc(hir_ty.span);
840                let InferOk { value: (), obligations } = self.at(&cause, self.param_env).eq(
841                    DefineOpaqueTypes::Yes,
842                    *expected_ty,
843                    supplied_ty,
844                )?;
845                all_obligations.extend(obligations);
846            }
847
848            let supplied_output_ty = supplied_sig.output();
849            let cause = &self.misc(decl.output.span());
850            let InferOk { value: (), obligations } = self.at(cause, self.param_env).eq(
851                DefineOpaqueTypes::Yes,
852                expected_sigs.liberated_sig.output(),
853                supplied_output_ty,
854            )?;
855            all_obligations.extend(obligations);
856
857            let inputs =
858                supplied_sig.inputs().into_iter().map(|&ty| self.resolve_vars_if_possible(ty));
859
860            let fn_sig_kind = FnSigKind::default()
861                .set_abi(ExternAbi::RustCall)
862                .set_safety(hir::Safety::Safe)
863                .set_c_variadic(expected_sigs.liberated_sig.c_variadic());
864            expected_sigs.liberated_sig =
865                self.tcx.mk_fn_sig(inputs, supplied_output_ty, fn_sig_kind);
866
867            Ok(InferOk { value: expected_sigs, obligations: all_obligations })
868        })
869    }
870
871    /// If there is no expected signature, then we will convert the
872    /// types that the user gave into a signature.
873    ///
874    /// Also, record this closure signature for later.
875    x;#[instrument(skip(self, decl), level = "debug", ret)]
876    fn supplied_sig_of_closure(
877        &self,
878        expr_def_id: LocalDefId,
879        decl: &hir::FnDecl<'tcx>,
880        closure_kind: hir::ClosureKind,
881    ) -> ty::PolyFnSig<'tcx> {
882        let lowerer = self.lowerer();
883
884        trace!("decl = {:#?}", decl);
885        debug!(?closure_kind);
886
887        let hir_id = self.tcx.local_def_id_to_hir_id(expr_def_id);
888        let bound_vars = self.tcx.late_bound_vars(hir_id);
889
890        // First, convert the types that the user supplied (if any).
891        let supplied_arguments = decl.inputs.iter().map(|a| lowerer.lower_ty(a));
892        let supplied_return = match decl.output {
893            hir::FnRetTy::Return(ref output) => lowerer.lower_ty(output),
894            hir::FnRetTy::DefaultReturn(_) => match closure_kind {
895                // In the case of the async block that we create for a function body,
896                // we expect the return type of the block to match that of the enclosing
897                // function.
898                hir::ClosureKind::Coroutine(hir::CoroutineKind::Desugared(
899                    hir::CoroutineDesugaring::Async,
900                    hir::CoroutineSource::Fn,
901                )) => {
902                    debug!("closure is async fn body");
903                    self.deduce_future_output_from_obligations(expr_def_id).unwrap_or_else(|| {
904                        // AFAIK, deducing the future output
905                        // always succeeds *except* in error cases
906                        // like #65159. I'd like to return Error
907                        // here, but I can't because I can't
908                        // easily (and locally) prove that we
909                        // *have* reported an
910                        // error. --nikomatsakis
911                        lowerer.ty_infer(None, decl.output.span())
912                    })
913                }
914                // All `gen {}` and `async gen {}` must return unit.
915                hir::ClosureKind::Coroutine(hir::CoroutineKind::Desugared(
916                    hir::CoroutineDesugaring::Gen | hir::CoroutineDesugaring::AsyncGen,
917                    _,
918                )) => self.tcx.types.unit,
919
920                // For async blocks, we just fall back to `_` here.
921                // For closures/coroutines, we know nothing about the return
922                // type unless it was supplied.
923                hir::ClosureKind::Coroutine(hir::CoroutineKind::Desugared(
924                    hir::CoroutineDesugaring::Async,
925                    _,
926                ))
927                | hir::ClosureKind::Coroutine(hir::CoroutineKind::Coroutine(_))
928                | hir::ClosureKind::Closure
929                | hir::ClosureKind::CoroutineClosure(_) => {
930                    lowerer.ty_infer(None, decl.output.span())
931                }
932            },
933        };
934
935        let fn_sig_kind = FnSigKind::default()
936            .set_abi(ExternAbi::RustCall)
937            .set_safety(hir::Safety::Safe)
938            .set_c_variadic(decl.c_variadic());
939        let result = ty::Binder::bind_with_vars(
940            self.tcx.mk_fn_sig(supplied_arguments, supplied_return, fn_sig_kind),
941            bound_vars,
942        );
943
944        let c_result = self.infcx.canonicalize_response(result);
945        self.typeck_results.borrow_mut().user_provided_sigs.insert(expr_def_id, c_result);
946
947        // Normalize only after registering in `user_provided_sigs`.
948        self.normalize(self.tcx.def_span(expr_def_id), Unnormalized::new_wip(result))
949    }
950
951    /// Invoked when we are translating the coroutine that results
952    /// from desugaring an `async fn`. Returns the "sugared" return
953    /// type of the `async fn` -- that is, the return type that the
954    /// user specified. The "desugared" return type is an `impl
955    /// Future<Output = T>`, so we do this by searching through the
956    /// obligations to extract the `T`.
957    x;#[instrument(skip(self), level = "debug", ret)]
958    fn deduce_future_output_from_obligations(&self, body_def_id: LocalDefId) -> Option<Ty<'tcx>> {
959        let ret_coercion = self.ret_coercion.as_ref().unwrap_or_else(|| {
960            span_bug!(self.tcx.def_span(body_def_id), "async fn coroutine outside of a fn")
961        });
962
963        let closure_span = self.tcx.def_span(body_def_id);
964        let ret_ty = ret_coercion.borrow().expected_ty();
965        let ret_ty = self.resolve_vars_with_obligations(ret_ty);
966
967        let get_future_output = |predicate: ty::Predicate<'tcx>, span| {
968            // Search for a pending obligation like
969            //
970            // `<R as Future>::Output = T`
971            //
972            // where R is the return type we are expecting. This type `T`
973            // will be our output.
974            let bound_predicate = predicate.kind();
975            if let ty::PredicateKind::Clause(ty::ClauseKind::Projection(proj_predicate)) =
976                bound_predicate.skip_binder()
977            {
978                self.deduce_future_output_from_projection(
979                    span,
980                    bound_predicate.rebind(proj_predicate),
981                )
982            } else {
983                None
984            }
985        };
986
987        let output_ty = match *ret_ty.kind() {
988            ty::Infer(ty::TyVar(ret_vid)) => {
989                self.obligations_for_self_ty(ret_vid).into_iter().find_map(|obligation| {
990                    get_future_output(obligation.predicate, obligation.cause.span)
991                })?
992            }
993            ty::Alias(ty::AliasTy { kind: ty::Projection { .. }, .. }) => {
994                return Some(Ty::new_error_with_message(
995                    self.tcx,
996                    closure_span,
997                    "this projection should have been projected to an opaque type",
998                ));
999            }
1000            ty::Alias(ty::AliasTy { kind: ty::Opaque { def_id }, args, .. }) => self
1001                .tcx
1002                .explicit_item_self_bounds(def_id)
1003                .iter_instantiated_copied(self.tcx, args)
1004                .map(Unnormalized::skip_norm_wip)
1005                .find_map(|(p, s)| get_future_output(p.as_predicate(), s))?,
1006            ty::Error(_) => return Some(ret_ty),
1007            _ => {
1008                span_bug!(closure_span, "invalid async fn coroutine return type: {ret_ty:?}")
1009            }
1010        };
1011
1012        let output_ty = self.normalize(closure_span, Unnormalized::new_wip(output_ty));
1013
1014        // async fn that have opaque types in their return type need to redo the conversion to inference variables
1015        // as they fetch the still opaque version from the signature.
1016        let InferOk { value: output_ty, obligations } = self
1017            .replace_opaque_types_with_inference_vars(
1018                output_ty,
1019                body_def_id,
1020                closure_span,
1021                self.param_env,
1022            );
1023        self.register_predicates(obligations);
1024
1025        Some(output_ty)
1026    }
1027
1028    /// Given a projection like
1029    ///
1030    /// `<X as Future>::Output = T`
1031    ///
1032    /// where `X` is some type that has no late-bound regions, returns
1033    /// `Some(T)`. If the projection is for some other trait, returns
1034    /// `None`.
1035    fn deduce_future_output_from_projection(
1036        &self,
1037        cause_span: Span,
1038        predicate: ty::PolyProjectionPredicate<'tcx>,
1039    ) -> Option<Ty<'tcx>> {
1040        {
    use ::tracing::__macro_support::Callsite as _;
    static __CALLSITE: ::tracing::callsite::DefaultCallsite =
        {
            static META: ::tracing::Metadata<'static> =
                {
                    ::tracing_core::metadata::Metadata::new("event compiler/rustc_hir_typeck/src/closure.rs:1040",
                        "rustc_hir_typeck::closure", ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/closure.rs"),
                        ::tracing_core::__macro_support::Option::Some(1040u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::closure"),
                        ::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!("deduce_future_output_from_projection(predicate={0:?})",
                                                    predicate) as &dyn Value))])
            });
    } else { ; }
};debug!("deduce_future_output_from_projection(predicate={:?})", predicate);
1041
1042        // We do not expect any bound regions in our predicate, so
1043        // skip past the bound vars.
1044        let Some(predicate) = predicate.no_bound_vars() else {
1045            {
    use ::tracing::__macro_support::Callsite as _;
    static __CALLSITE: ::tracing::callsite::DefaultCallsite =
        {
            static META: ::tracing::Metadata<'static> =
                {
                    ::tracing_core::metadata::Metadata::new("event compiler/rustc_hir_typeck/src/closure.rs:1045",
                        "rustc_hir_typeck::closure", ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/closure.rs"),
                        ::tracing_core::__macro_support::Option::Some(1045u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::closure"),
                        ::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!("deduce_future_output_from_projection: has late-bound regions")
                                            as &dyn Value))])
            });
    } else { ; }
};debug!("deduce_future_output_from_projection: has late-bound regions");
1046            return None;
1047        };
1048
1049        // Check that this is a projection from the `Future` trait.
1050        let trait_def_id = predicate.projection_term.trait_def_id(self.tcx);
1051        if !self.tcx.is_lang_item(trait_def_id, LangItem::Future) {
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_hir_typeck/src/closure.rs:1052",
                        "rustc_hir_typeck::closure", ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/closure.rs"),
                        ::tracing_core::__macro_support::Option::Some(1052u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::closure"),
                        ::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!("deduce_future_output_from_projection: not a future")
                                            as &dyn Value))])
            });
    } else { ; }
};debug!("deduce_future_output_from_projection: not a future");
1053            return None;
1054        }
1055
1056        // The `Future` trait has only one associated item, `Output`,
1057        // so check that this is what we see.
1058        let output_assoc_item = self.tcx.associated_item_def_ids(trait_def_id)[0];
1059        if output_assoc_item != predicate.projection_term.def_id() {
1060            ::rustc_middle::util::bug::span_bug_fmt(cause_span,
    format_args!("projecting associated item `{0:?}` from future, which is not Output `{1:?}`",
        predicate.projection_term.kind, output_assoc_item));span_bug!(
1061                cause_span,
1062                "projecting associated item `{:?}` from future, which is not Output `{:?}`",
1063                predicate.projection_term.kind,
1064                output_assoc_item,
1065            );
1066        }
1067
1068        // Extract the type from the projection. Note that there can
1069        // be no bound variables in this type because the "self type"
1070        // does not have any regions in it.
1071        let output_ty = self.resolve_vars_if_possible(predicate.term);
1072        {
    use ::tracing::__macro_support::Callsite as _;
    static __CALLSITE: ::tracing::callsite::DefaultCallsite =
        {
            static META: ::tracing::Metadata<'static> =
                {
                    ::tracing_core::metadata::Metadata::new("event compiler/rustc_hir_typeck/src/closure.rs:1072",
                        "rustc_hir_typeck::closure", ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/closure.rs"),
                        ::tracing_core::__macro_support::Option::Some(1072u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::closure"),
                        ::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!("deduce_future_output_from_projection: output_ty={0:?}",
                                                    output_ty) as &dyn Value))])
            });
    } else { ; }
};debug!("deduce_future_output_from_projection: output_ty={:?}", output_ty);
1073        // This is a projection on a Fn trait so will always be a type.
1074        Some(output_ty.expect_type())
1075    }
1076
1077    /// Converts the types that the user supplied, in case that doing
1078    /// so should yield an error, but returns back a signature where
1079    /// all parameters are of type `ty::Error`.
1080    fn error_sig_of_closure(
1081        &self,
1082        decl: &hir::FnDecl<'tcx>,
1083        guar: ErrorGuaranteed,
1084    ) -> ty::PolyFnSig<'tcx> {
1085        let lowerer = self.lowerer();
1086        let err_ty = Ty::new_error(self.tcx, guar);
1087
1088        let supplied_arguments = decl.inputs.iter().map(|a| {
1089            // Convert the types that the user supplied (if any), but ignore them.
1090            lowerer.lower_ty(a);
1091            err_ty
1092        });
1093
1094        if let hir::FnRetTy::Return(ref output) = decl.output {
1095            lowerer.lower_ty(output);
1096        }
1097
1098        let fn_sig_kind = FnSigKind::default()
1099            .set_abi(ExternAbi::RustCall)
1100            .set_safety(hir::Safety::Safe)
1101            .set_c_variadic(decl.c_variadic());
1102        let result = ty::Binder::dummy(self.tcx.mk_fn_sig(supplied_arguments, err_ty, fn_sig_kind));
1103
1104        {
    use ::tracing::__macro_support::Callsite as _;
    static __CALLSITE: ::tracing::callsite::DefaultCallsite =
        {
            static META: ::tracing::Metadata<'static> =
                {
                    ::tracing_core::metadata::Metadata::new("event compiler/rustc_hir_typeck/src/closure.rs:1104",
                        "rustc_hir_typeck::closure", ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/closure.rs"),
                        ::tracing_core::__macro_support::Option::Some(1104u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::closure"),
                        ::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!("supplied_sig_of_closure: result={0:?}",
                                                    result) as &dyn Value))])
            });
    } else { ; }
};debug!("supplied_sig_of_closure: result={:?}", result);
1105
1106        result
1107    }
1108
1109    x;#[instrument(level = "debug", skip(self), ret)]
1110    fn closure_sigs(
1111        &self,
1112        expr_def_id: LocalDefId,
1113        bound_sig: ty::PolyFnSig<'tcx>,
1114    ) -> ClosureSignatures<'tcx> {
1115        let liberated_sig =
1116            self.tcx().liberate_late_bound_regions(expr_def_id.to_def_id(), bound_sig);
1117        let liberated_sig =
1118            self.normalize(self.tcx.def_span(expr_def_id), Unnormalized::new_wip(liberated_sig));
1119        ClosureSignatures { bound_sig, liberated_sig }
1120    }
1121}