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