Skip to main content

rustc_hir_typeck/
expr.rs

1// ignore-tidy-filelength
2// FIXME: we should move the field error reporting code somewhere else.
3
4//! Type checking expressions.
5//!
6//! See [`rustc_hir_analysis::check`] for more context on type checking in general.
7
8use rustc_abi::{FIRST_VARIANT, FieldIdx};
9use rustc_ast as ast;
10use rustc_ast::util::parser::ExprPrecedence;
11use rustc_data_structures::fx::{FxHashMap, FxHashSet};
12use rustc_data_structures::stack::ensure_sufficient_stack;
13use rustc_data_structures::unord::UnordMap;
14use rustc_errors::codes::*;
15use rustc_errors::{
16    Applicability, Diag, ErrorGuaranteed, MultiSpan, StashKey, Subdiagnostic, listify, pluralize,
17    struct_span_code_err,
18};
19use rustc_hir as hir;
20use rustc_hir::def::{CtorKind, DefKind, Res};
21use rustc_hir::def_id::DefId;
22use rustc_hir::lang_items::LangItem;
23use rustc_hir::{ExprKind, HirId, QPath, find_attr, is_range_literal};
24use rustc_hir_analysis::NoVariantNamed;
25use rustc_hir_analysis::errors::NoFieldOnType;
26use rustc_hir_analysis::hir_ty_lowering::HirTyLowerer as _;
27use rustc_infer::infer::{self, DefineOpaqueTypes, InferOk, RegionVariableOrigin};
28use rustc_infer::traits::query::NoSolution;
29use rustc_middle::ty::adjustment::{Adjust, Adjustment, AllowTwoPhase};
30use rustc_middle::ty::error::{ExpectedFound, TypeError};
31use rustc_middle::ty::{self, AdtKind, GenericArgsRef, Ty, TypeVisitableExt};
32use rustc_middle::{bug, span_bug};
33use rustc_session::errors::ExprParenthesesNeeded;
34use rustc_session::parse::feature_err;
35use rustc_span::edit_distance::find_best_match_for_name;
36use rustc_span::hygiene::DesugaringKind;
37use rustc_span::source_map::Spanned;
38use rustc_span::{Ident, Span, Symbol, kw, sym};
39use rustc_trait_selection::infer::InferCtxtExt;
40use rustc_trait_selection::traits::{self, ObligationCauseCode, ObligationCtxt};
41use tracing::{debug, instrument, trace};
42
43use crate::Expectation::{self, ExpectCastableToType, ExpectHasType, NoExpectation};
44use crate::coercion::CoerceMany;
45use crate::errors::{
46    AddressOfTemporaryTaken, BaseExpressionDoubleDot, BaseExpressionDoubleDotAddExpr,
47    BaseExpressionDoubleDotRemove, CantDereference, FieldMultiplySpecifiedInInitializer,
48    FunctionalRecordUpdateOnNonStruct, HelpUseLatestEdition, NakedAsmOutsideNakedFn,
49    NoFieldOnVariant, ReturnLikeStatementKind, ReturnStmtOutsideOfFnBody, StructExprNonExhaustive,
50    TypeMismatchFruTypo, YieldExprOutsideOfCoroutine,
51};
52use crate::op::contains_let_in_chain;
53use crate::{
54    BreakableCtxt, CoroutineTypes, Diverges, FnCtxt, GatherLocalsVisitor, Needs,
55    TupleArgumentsFlag, cast, fatally_break_rust, report_unexpected_variant_res, type_error_struct,
56};
57
58impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
59    pub(crate) fn precedence(&self, expr: &hir::Expr<'_>) -> ExprPrecedence {
60        let has_attr = |id: HirId| -> bool {
61            for attr in self.tcx.hir_attrs(id) {
62                // For the purpose of rendering suggestions, disregard attributes
63                // that originate from desugaring of any kind. For example, `x?`
64                // desugars to `#[allow(unreachable_code)] match ...`. Failing to
65                // ignore the prefix attribute in the desugaring would cause this
66                // suggestion:
67                //
68                //     let y: u32 = x?.try_into().unwrap();
69                //                    ++++++++++++++++++++
70                //
71                // to be rendered as:
72                //
73                //     let y: u32 = (x?).try_into().unwrap();
74                //                  +  +++++++++++++++++++++
75                if attr.span().desugaring_kind().is_none() {
76                    return true;
77                }
78            }
79            false
80        };
81
82        // Special case: range expressions are desugared to struct literals in HIR,
83        // so they would normally return `Unambiguous` precedence in expr.precedence.
84        // we should return `Range` precedence for correct parenthesization in suggestions.
85        if is_range_literal(expr) {
86            return ExprPrecedence::Range;
87        }
88
89        expr.precedence(&has_attr)
90    }
91
92    /// Check an expr with an expectation type, and also demand that the expr's
93    /// evaluated type is a subtype of the expectation at the end. This is a
94    /// *hard* requirement.
95    pub(crate) fn check_expr_has_type_or_error(
96        &self,
97        expr: &'tcx hir::Expr<'tcx>,
98        expected_ty: Ty<'tcx>,
99        extend_err: impl FnOnce(&mut Diag<'_>),
100    ) -> Ty<'tcx> {
101        let mut ty = self.check_expr_with_expectation(expr, ExpectHasType(expected_ty));
102
103        // While we don't allow *arbitrary* coercions here, we *do* allow
104        // coercions from ! to `expected`.
105        if self.try_structurally_resolve_type(expr.span, ty).is_never()
106            && self.tcx.expr_guaranteed_to_constitute_read_for_never(expr)
107        {
108            if let Some(adjustments) = self.typeck_results.borrow().adjustments().get(expr.hir_id) {
109                let reported = self.dcx().span_delayed_bug(
110                    expr.span,
111                    "expression with never type wound up being adjusted",
112                );
113
114                return if let [Adjustment { kind: Adjust::NeverToAny, target }] = &adjustments[..] {
115                    target.to_owned()
116                } else {
117                    Ty::new_error(self.tcx(), reported)
118                };
119            }
120
121            let adj_ty = self.next_ty_var(expr.span);
122            self.apply_adjustments(
123                expr,
124                ::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
        [Adjustment { kind: Adjust::NeverToAny, target: adj_ty }]))vec![Adjustment { kind: Adjust::NeverToAny, target: adj_ty }],
125            );
126            ty = adj_ty;
127        }
128
129        if let Err(mut err) = self.demand_suptype_diag(expr.span, expected_ty, ty) {
130            let _ = self.emit_type_mismatch_suggestions(
131                &mut err,
132                expr.peel_drop_temps(),
133                ty,
134                expected_ty,
135                None,
136                None,
137            );
138            extend_err(&mut err);
139            err.emit();
140        }
141        ty
142    }
143
144    /// Check an expr with an expectation type, and also demand that the expr's
145    /// evaluated type is a coercible to the expectation at the end. This is a
146    /// *hard* requirement.
147    pub(super) fn check_expr_coercible_to_type(
148        &self,
149        expr: &'tcx hir::Expr<'tcx>,
150        expected: Ty<'tcx>,
151        expected_ty_expr: Option<&'tcx hir::Expr<'tcx>>,
152    ) -> Ty<'tcx> {
153        self.check_expr_coercible_to_type_or_error(expr, expected, expected_ty_expr, |_, _| {})
154    }
155
156    pub(crate) fn check_expr_coercible_to_type_or_error(
157        &self,
158        expr: &'tcx hir::Expr<'tcx>,
159        expected: Ty<'tcx>,
160        expected_ty_expr: Option<&'tcx hir::Expr<'tcx>>,
161        extend_err: impl FnOnce(&mut Diag<'_>, Ty<'tcx>),
162    ) -> Ty<'tcx> {
163        let ty = self.check_expr_with_hint(expr, expected);
164        // checks don't need two phase
165        match self.demand_coerce_diag(expr, ty, expected, expected_ty_expr, AllowTwoPhase::No) {
166            Ok(ty) => ty,
167            Err(mut err) => {
168                extend_err(&mut err, ty);
169                err.emit();
170                // Return the original type instead of an error type here, otherwise the type of `x` in
171                // `let x: u32 = ();` will be a type error, causing all subsequent usages of `x` to not
172                // report errors, even though `x` is definitely `u32`.
173                expected
174            }
175        }
176    }
177
178    /// Check an expr with an expectation type. Don't actually enforce that expectation
179    /// is related to the expr's evaluated type via subtyping or coercion. This is
180    /// usually called because we want to do that subtype/coerce call manually for better
181    /// diagnostics.
182    pub(super) fn check_expr_with_hint(
183        &self,
184        expr: &'tcx hir::Expr<'tcx>,
185        expected: Ty<'tcx>,
186    ) -> Ty<'tcx> {
187        self.check_expr_with_expectation(expr, ExpectHasType(expected))
188    }
189
190    /// Check an expr with an expectation type, and also [`Needs`] which will
191    /// prompt typeck to convert any implicit immutable derefs to mutable derefs.
192    fn check_expr_with_expectation_and_needs(
193        &self,
194        expr: &'tcx hir::Expr<'tcx>,
195        expected: Expectation<'tcx>,
196        needs: Needs,
197    ) -> Ty<'tcx> {
198        let ty = self.check_expr_with_expectation(expr, expected);
199
200        // If the expression is used in a place whether mutable place is required
201        // e.g. LHS of assignment, perform the conversion.
202        if let Needs::MutPlace = needs {
203            self.convert_place_derefs_to_mutable(expr);
204        }
205
206        ty
207    }
208
209    /// Check an expr with no expectations.
210    pub(super) fn check_expr(&self, expr: &'tcx hir::Expr<'tcx>) -> Ty<'tcx> {
211        self.check_expr_with_expectation(expr, NoExpectation)
212    }
213
214    /// Check an expr with no expectations, but with [`Needs`] which will
215    /// prompt typeck to convert any implicit immutable derefs to mutable derefs.
216    pub(super) fn check_expr_with_needs(
217        &self,
218        expr: &'tcx hir::Expr<'tcx>,
219        needs: Needs,
220    ) -> Ty<'tcx> {
221        self.check_expr_with_expectation_and_needs(expr, NoExpectation, needs)
222    }
223
224    /// Check an expr with an expectation type which may be used to eagerly
225    /// guide inference when evaluating that expr.
226    #[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_with_expectation",
                                    "rustc_hir_typeck::expr", ::tracing::Level::DEBUG,
                                    ::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/expr.rs"),
                                    ::tracing_core::__macro_support::Option::Some(226u32),
                                    ::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::expr"),
                                    ::tracing_core::field::FieldSet::new(&["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(&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;
        }
        { self.check_expr_with_expectation_and_args(expr, expected, None) }
    }
}#[instrument(skip(self, expr), level = "debug")]
227    pub(super) fn check_expr_with_expectation(
228        &self,
229        expr: &'tcx hir::Expr<'tcx>,
230        expected: Expectation<'tcx>,
231    ) -> Ty<'tcx> {
232        self.check_expr_with_expectation_and_args(expr, expected, None)
233    }
234
235    /// Same as [`Self::check_expr_with_expectation`], but allows us to pass in
236    /// the arguments of a [`ExprKind::Call`] when evaluating its callee that
237    /// is an [`ExprKind::Path`]. We use this to refine the spans for certain
238    /// well-formedness guarantees for the path expr.
239    pub(super) fn check_expr_with_expectation_and_args(
240        &self,
241        expr: &'tcx hir::Expr<'tcx>,
242        expected: Expectation<'tcx>,
243        call_expr_and_args: Option<(&'tcx hir::Expr<'tcx>, &'tcx [hir::Expr<'tcx>])>,
244    ) -> Ty<'tcx> {
245        if self.tcx().sess.verbose_internals() {
246            // make this code only run with -Zverbose-internals because it is probably slow
247            if let Ok(lint_str) = self.tcx.sess.source_map().span_to_snippet(expr.span) {
248                if !lint_str.contains('\n') {
249                    {
    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/expr.rs:249",
                        "rustc_hir_typeck::expr", ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/expr.rs"),
                        ::tracing_core::__macro_support::Option::Some(249u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::expr"),
                        ::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!("expr text: {0}",
                                                    lint_str) as &dyn Value))])
            });
    } else { ; }
};debug!("expr text: {lint_str}");
250                } else {
251                    let mut lines = lint_str.lines();
252                    if let Some(line0) = lines.next() {
253                        let remaining_lines = lines.count();
254                        {
    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/expr.rs:254",
                        "rustc_hir_typeck::expr", ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/expr.rs"),
                        ::tracing_core::__macro_support::Option::Some(254u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::expr"),
                        ::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!("expr text: {0}",
                                                    line0) as &dyn Value))])
            });
    } else { ; }
};debug!("expr text: {line0}");
255                        {
    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/expr.rs:255",
                        "rustc_hir_typeck::expr", ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/expr.rs"),
                        ::tracing_core::__macro_support::Option::Some(255u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::expr"),
                        ::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!("expr text: ...(and {0} more lines)",
                                                    remaining_lines) as &dyn Value))])
            });
    } else { ; }
};debug!("expr text: ...(and {remaining_lines} more lines)");
256                    }
257                }
258            }
259        }
260
261        // True if `expr` is a `Try::from_ok(())` that is a result of desugaring a try block
262        // without the final expr (e.g. `try { return; }`). We don't want to generate an
263        // unreachable_code lint for it since warnings for autogenerated code are confusing.
264        let is_try_block_generated_unit_expr = match expr.kind {
265            ExprKind::Call(_, [arg]) => {
266                expr.span.is_desugaring(DesugaringKind::TryBlock)
267                    && arg.span.is_desugaring(DesugaringKind::TryBlock)
268            }
269            _ => false,
270        };
271
272        // Warn for expressions after diverging siblings.
273        if !is_try_block_generated_unit_expr {
274            self.warn_if_unreachable(expr.hir_id, expr.span, "expression");
275        }
276
277        // Whether a past expression diverges doesn't affect typechecking of this expression, so we
278        // reset `diverges` while checking `expr`.
279        let old_diverges = self.diverges.replace(Diverges::Maybe);
280
281        if self.is_whole_body.replace(false) {
282            // If this expression is the whole body and the function diverges because of its
283            // arguments, we check this here to ensure the body is considered to diverge.
284            self.diverges.set(self.function_diverges_because_of_empty_arguments.get())
285        };
286
287        let ty = ensure_sufficient_stack(|| match &expr.kind {
288            // Intercept the callee path expr and give it better spans.
289            hir::ExprKind::Path(
290                qpath @ (hir::QPath::Resolved(..) | hir::QPath::TypeRelative(..)),
291            ) => self.check_expr_path(qpath, expr, call_expr_and_args),
292            _ => self.check_expr_kind(expr, expected),
293        });
294        let ty = self.resolve_vars_if_possible(ty);
295
296        // Warn for non-block expressions with diverging children.
297        match expr.kind {
298            ExprKind::Block(..)
299            | ExprKind::If(..)
300            | ExprKind::Let(..)
301            | ExprKind::Loop(..)
302            | ExprKind::Match(..) => {}
303            // Do not warn on `as` casts from never to any,
304            // they are sometimes required to appeal typeck.
305            ExprKind::Cast(_, _) => {}
306            // If `expr` is a result of desugaring the try block and is an ok-wrapped
307            // diverging expression (e.g. it arose from desugaring of `try { return }`),
308            // we skip issuing a warning because it is autogenerated code.
309            ExprKind::Call(..) if expr.span.is_desugaring(DesugaringKind::TryBlock) => {}
310            // Likewise, do not lint unreachable code injected via contracts desugaring.
311            ExprKind::Call(..) if expr.span.is_desugaring(DesugaringKind::Contract) => {}
312            ExprKind::Call(callee, _) => self.warn_if_unreachable(expr.hir_id, callee.span, "call"),
313            ExprKind::MethodCall(segment, ..) => {
314                self.warn_if_unreachable(expr.hir_id, segment.ident.span, "call")
315            }
316            _ => self.warn_if_unreachable(expr.hir_id, expr.span, "expression"),
317        }
318
319        // Any expression that produces a value of type `!` must have diverged,
320        // unless it's a place expression that isn't being read from, in which case
321        // diverging would be unsound since we may never actually read the `!`.
322        // e.g. `let _ = *never_ptr;` with `never_ptr: *const !`.
323        if self.try_structurally_resolve_type(expr.span, ty).is_never()
324            && self.tcx.expr_guaranteed_to_constitute_read_for_never(expr)
325        {
326            self.diverges.set(self.diverges.get() | Diverges::always(expr.span));
327        }
328
329        // Record the type, which applies it effects.
330        // We need to do this after the warning above, so that
331        // we don't warn for the diverging expression itself.
332        self.write_ty(expr.hir_id, ty);
333
334        // Combine the diverging and has_error flags.
335        self.diverges.set(self.diverges.get() | old_diverges);
336
337        {
    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/expr.rs:337",
                        "rustc_hir_typeck::expr", ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/expr.rs"),
                        ::tracing_core::__macro_support::Option::Some(337u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::expr"),
                        ::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!("type of {0} is...",
                                                    self.tcx.hir_id_to_string(expr.hir_id)) as &dyn Value))])
            });
    } else { ; }
};debug!("type of {} is...", self.tcx.hir_id_to_string(expr.hir_id));
338        {
    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/expr.rs:338",
                        "rustc_hir_typeck::expr", ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/expr.rs"),
                        ::tracing_core::__macro_support::Option::Some(338u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::expr"),
                        ::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!("... {0:?}, expected is {1:?}",
                                                    ty, expected) as &dyn Value))])
            });
    } else { ; }
};debug!("... {:?}, expected is {:?}", ty, expected);
339
340        ty
341    }
342
343    #[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_kind",
                                    "rustc_hir_typeck::expr", ::tracing::Level::DEBUG,
                                    ::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/expr.rs"),
                                    ::tracing_core::__macro_support::Option::Some(343u32),
                                    ::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::expr"),
                                    ::tracing_core::field::FieldSet::new(&["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(&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;
        }
        {
            {
                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/expr.rs:349",
                                    "rustc_hir_typeck::expr", ::tracing::Level::TRACE,
                                    ::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/expr.rs"),
                                    ::tracing_core::__macro_support::Option::Some(349u32),
                                    ::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::expr"),
                                    ::tracing_core::field::FieldSet::new(&["message"],
                                        ::tracing_core::callsite::Identifier(&__CALLSITE)),
                                    ::tracing::metadata::Kind::EVENT)
                            };
                        ::tracing::callsite::DefaultCallsite::new(&META)
                    };
                let enabled =
                    ::tracing::Level::TRACE <=
                                ::tracing::level_filters::STATIC_MAX_LEVEL &&
                            ::tracing::Level::TRACE <=
                                ::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!("expr={0:#?}",
                                                                expr) as &dyn Value))])
                        });
                } else { ; }
            };
            let tcx = self.tcx;
            match expr.kind {
                ExprKind::Lit(ref lit) => self.check_expr_lit(lit, expected),
                ExprKind::Binary(op, lhs, rhs) =>
                    self.check_expr_binop(expr, op, lhs, rhs, expected),
                ExprKind::Assign(lhs, rhs, span) => {
                    self.check_expr_assign(expr, expected, lhs, rhs, span)
                }
                ExprKind::AssignOp(op, lhs, rhs) => {
                    self.check_expr_assign_op(expr, op, lhs, rhs, expected)
                }
                ExprKind::Unary(unop, oprnd) =>
                    self.check_expr_unop(unop, oprnd, expected, expr),
                ExprKind::AddrOf(kind, mutbl, oprnd) => {
                    self.check_expr_addr_of(kind, mutbl, oprnd, expected, expr)
                }
                ExprKind::Path(ref qpath) =>
                    self.check_expr_path(qpath, expr, None),
                ExprKind::InlineAsm(asm) => {
                    self.deferred_asm_checks.borrow_mut().push((asm,
                            expr.hir_id));
                    self.check_expr_asm(asm, expr.span)
                }
                ExprKind::OffsetOf(container, fields) => {
                    self.check_expr_offset_of(container, fields, expr)
                }
                ExprKind::Break(destination, ref expr_opt) => {
                    self.check_expr_break(destination, expr_opt.as_deref(),
                        expr)
                }
                ExprKind::Continue(destination) =>
                    self.check_expr_continue(destination, expr),
                ExprKind::Ret(ref expr_opt) =>
                    self.check_expr_return(expr_opt.as_deref(), expr),
                ExprKind::Become(call) => self.check_expr_become(call, expr),
                ExprKind::Let(let_expr) =>
                    self.check_expr_let(let_expr, expr.hir_id),
                ExprKind::Loop(body, _, source, _) => {
                    self.check_expr_loop(body, source, expected, expr)
                }
                ExprKind::Match(discrim, arms, match_src) => {
                    self.check_expr_match(expr, discrim, arms, expected,
                        match_src)
                }
                ExprKind::Closure(closure) =>
                    self.check_expr_closure(closure, expr.span, expected),
                ExprKind::Block(body, _) =>
                    self.check_expr_block(body, expected),
                ExprKind::Call(callee, args) =>
                    self.check_expr_call(expr, callee, args, expected),
                ExprKind::Use(used_expr, _) =>
                    self.check_expr_use(used_expr, expected),
                ExprKind::MethodCall(segment, receiver, args, _) => {
                    self.check_expr_method_call(expr, segment, receiver, args,
                        expected)
                }
                ExprKind::Cast(e, t) => self.check_expr_cast(e, t, expr),
                ExprKind::Type(e, t) => {
                    let ascribed_ty = self.lower_ty_saving_user_provided_ty(t);
                    let ty = self.check_expr_with_hint(e, ascribed_ty);
                    self.demand_eqtype(e.span, ascribed_ty, ty);
                    ascribed_ty
                }
                ExprKind::If(cond, then_expr, opt_else_expr) => {
                    self.check_expr_if(expr.hir_id, cond, then_expr,
                        opt_else_expr, expr.span, expected)
                }
                ExprKind::DropTemps(e) =>
                    self.check_expr_with_expectation(e, expected),
                ExprKind::Array(args) =>
                    self.check_expr_array(args, expected, expr),
                ExprKind::ConstBlock(ref block) =>
                    self.check_expr_const_block(block, expected),
                ExprKind::Repeat(element, ref count) => {
                    self.check_expr_repeat(element, count, expected, expr)
                }
                ExprKind::Tup(elts) =>
                    self.check_expr_tuple(elts, expected, expr),
                ExprKind::Struct(qpath, fields, ref base_expr) => {
                    self.check_expr_struct(expr, expected, qpath, fields,
                        base_expr)
                }
                ExprKind::Field(base, field) =>
                    self.check_expr_field(expr, base, field, expected),
                ExprKind::Index(base, idx, brackets_span) => {
                    self.check_expr_index(base, idx, expr, brackets_span)
                }
                ExprKind::Yield(value, _) =>
                    self.check_expr_yield(value, expr),
                ExprKind::UnsafeBinderCast(kind, inner_expr, ty) => {
                    self.check_expr_unsafe_binder_cast(expr.span, kind,
                        inner_expr, ty, expected)
                }
                ExprKind::Err(guar) => Ty::new_error(tcx, guar),
            }
        }
    }
}#[instrument(skip(self, expr), level = "debug")]
344    fn check_expr_kind(
345        &self,
346        expr: &'tcx hir::Expr<'tcx>,
347        expected: Expectation<'tcx>,
348    ) -> Ty<'tcx> {
349        trace!("expr={:#?}", expr);
350
351        let tcx = self.tcx;
352        match expr.kind {
353            ExprKind::Lit(ref lit) => self.check_expr_lit(lit, expected),
354            ExprKind::Binary(op, lhs, rhs) => self.check_expr_binop(expr, op, lhs, rhs, expected),
355            ExprKind::Assign(lhs, rhs, span) => {
356                self.check_expr_assign(expr, expected, lhs, rhs, span)
357            }
358            ExprKind::AssignOp(op, lhs, rhs) => {
359                self.check_expr_assign_op(expr, op, lhs, rhs, expected)
360            }
361            ExprKind::Unary(unop, oprnd) => self.check_expr_unop(unop, oprnd, expected, expr),
362            ExprKind::AddrOf(kind, mutbl, oprnd) => {
363                self.check_expr_addr_of(kind, mutbl, oprnd, expected, expr)
364            }
365            ExprKind::Path(ref qpath) => self.check_expr_path(qpath, expr, None),
366            ExprKind::InlineAsm(asm) => {
367                // We defer some asm checks as we may not have resolved the input and output types yet (they may still be infer vars).
368                self.deferred_asm_checks.borrow_mut().push((asm, expr.hir_id));
369                self.check_expr_asm(asm, expr.span)
370            }
371            ExprKind::OffsetOf(container, fields) => {
372                self.check_expr_offset_of(container, fields, expr)
373            }
374            ExprKind::Break(destination, ref expr_opt) => {
375                self.check_expr_break(destination, expr_opt.as_deref(), expr)
376            }
377            ExprKind::Continue(destination) => self.check_expr_continue(destination, expr),
378            ExprKind::Ret(ref expr_opt) => self.check_expr_return(expr_opt.as_deref(), expr),
379            ExprKind::Become(call) => self.check_expr_become(call, expr),
380            ExprKind::Let(let_expr) => self.check_expr_let(let_expr, expr.hir_id),
381            ExprKind::Loop(body, _, source, _) => {
382                self.check_expr_loop(body, source, expected, expr)
383            }
384            ExprKind::Match(discrim, arms, match_src) => {
385                self.check_expr_match(expr, discrim, arms, expected, match_src)
386            }
387            ExprKind::Closure(closure) => self.check_expr_closure(closure, expr.span, expected),
388            ExprKind::Block(body, _) => self.check_expr_block(body, expected),
389            ExprKind::Call(callee, args) => self.check_expr_call(expr, callee, args, expected),
390            ExprKind::Use(used_expr, _) => self.check_expr_use(used_expr, expected),
391            ExprKind::MethodCall(segment, receiver, args, _) => {
392                self.check_expr_method_call(expr, segment, receiver, args, expected)
393            }
394            ExprKind::Cast(e, t) => self.check_expr_cast(e, t, expr),
395            ExprKind::Type(e, t) => {
396                let ascribed_ty = self.lower_ty_saving_user_provided_ty(t);
397                let ty = self.check_expr_with_hint(e, ascribed_ty);
398                self.demand_eqtype(e.span, ascribed_ty, ty);
399                ascribed_ty
400            }
401            ExprKind::If(cond, then_expr, opt_else_expr) => {
402                self.check_expr_if(expr.hir_id, cond, then_expr, opt_else_expr, expr.span, expected)
403            }
404            ExprKind::DropTemps(e) => self.check_expr_with_expectation(e, expected),
405            ExprKind::Array(args) => self.check_expr_array(args, expected, expr),
406            ExprKind::ConstBlock(ref block) => self.check_expr_const_block(block, expected),
407            ExprKind::Repeat(element, ref count) => {
408                self.check_expr_repeat(element, count, expected, expr)
409            }
410            ExprKind::Tup(elts) => self.check_expr_tuple(elts, expected, expr),
411            ExprKind::Struct(qpath, fields, ref base_expr) => {
412                self.check_expr_struct(expr, expected, qpath, fields, base_expr)
413            }
414            ExprKind::Field(base, field) => self.check_expr_field(expr, base, field, expected),
415            ExprKind::Index(base, idx, brackets_span) => {
416                self.check_expr_index(base, idx, expr, brackets_span)
417            }
418            ExprKind::Yield(value, _) => self.check_expr_yield(value, expr),
419            ExprKind::UnsafeBinderCast(kind, inner_expr, ty) => {
420                self.check_expr_unsafe_binder_cast(expr.span, kind, inner_expr, ty, expected)
421            }
422            ExprKind::Err(guar) => Ty::new_error(tcx, guar),
423        }
424    }
425
426    fn check_expr_unop(
427        &self,
428        unop: hir::UnOp,
429        oprnd: &'tcx hir::Expr<'tcx>,
430        expected: Expectation<'tcx>,
431        expr: &'tcx hir::Expr<'tcx>,
432    ) -> Ty<'tcx> {
433        let tcx = self.tcx;
434        let expected_inner = match unop {
435            hir::UnOp::Not | hir::UnOp::Neg => expected,
436            hir::UnOp::Deref => NoExpectation,
437        };
438        let oprnd_t = self.check_expr_with_expectation(oprnd, expected_inner);
439
440        if let Err(guar) = oprnd_t.error_reported() {
441            return Ty::new_error(tcx, guar);
442        }
443
444        let oprnd_t = self.structurally_resolve_type(expr.span, oprnd_t);
445        match unop {
446            hir::UnOp::Deref => self.lookup_derefing(expr, oprnd, oprnd_t).unwrap_or_else(|| {
447                let mut err =
448                    self.dcx().create_err(CantDereference { span: expr.span, ty: oprnd_t });
449                let sp = tcx.sess.source_map().start_point(expr.span).with_parent(None);
450                if let Some(sp) = tcx.sess.psess.ambiguous_block_expr_parse.borrow().get(&sp) {
451                    err.subdiagnostic(ExprParenthesesNeeded::surrounding(*sp));
452                }
453                Ty::new_error(tcx, err.emit())
454            }),
455            hir::UnOp::Not => {
456                let result = self.check_user_unop(expr, oprnd_t, unop, expected_inner);
457                // If it's builtin, we can reuse the type, this helps inference.
458                if oprnd_t.is_integral() || *oprnd_t.kind() == ty::Bool { oprnd_t } else { result }
459            }
460            hir::UnOp::Neg => {
461                let result = self.check_user_unop(expr, oprnd_t, unop, expected_inner);
462                // If it's builtin, we can reuse the type, this helps inference.
463                if oprnd_t.is_numeric() { oprnd_t } else { result }
464            }
465        }
466    }
467
468    fn check_expr_addr_of(
469        &self,
470        kind: hir::BorrowKind,
471        mutbl: hir::Mutability,
472        oprnd: &'tcx hir::Expr<'tcx>,
473        expected: Expectation<'tcx>,
474        expr: &'tcx hir::Expr<'tcx>,
475    ) -> Ty<'tcx> {
476        let hint = expected.only_has_type(self).map_or(NoExpectation, |ty| {
477            match self.try_structurally_resolve_type(expr.span, ty).kind() {
478                ty::Ref(_, ty, _) | ty::RawPtr(ty, _) => {
479                    if oprnd.is_syntactic_place_expr() {
480                        // Places may legitimately have unsized types.
481                        // For example, dereferences of a wide pointer and
482                        // the last field of a struct can be unsized.
483                        ExpectHasType(*ty)
484                    } else {
485                        Expectation::rvalue_hint(self, *ty)
486                    }
487                }
488                _ => NoExpectation,
489            }
490        });
491        let ty =
492            self.check_expr_with_expectation_and_needs(oprnd, hint, Needs::maybe_mut_place(mutbl));
493        if let Err(guar) = ty.error_reported() {
494            return Ty::new_error(self.tcx, guar);
495        }
496
497        match kind {
498            hir::BorrowKind::Raw => {
499                self.check_named_place_expr(oprnd);
500                Ty::new_ptr(self.tcx, ty, mutbl)
501            }
502            hir::BorrowKind::Ref | hir::BorrowKind::Pin => {
503                // Note: at this point, we cannot say what the best lifetime
504                // is to use for resulting pointer. We want to use the
505                // shortest lifetime possible so as to avoid spurious borrowck
506                // errors. Moreover, the longest lifetime will depend on the
507                // precise details of the value whose address is being taken
508                // (and how long it is valid), which we don't know yet until
509                // type inference is complete.
510                //
511                // Therefore, here we simply generate a region variable. The
512                // region inferencer will then select a suitable value.
513                // Finally, borrowck will infer the value of the region again,
514                // this time with enough precision to check that the value
515                // whose address was taken can actually be made to live as long
516                // as it needs to live.
517                let region = self.next_region_var(RegionVariableOrigin::BorrowRegion(expr.span));
518                match kind {
519                    hir::BorrowKind::Ref => Ty::new_ref(self.tcx, region, ty, mutbl),
520                    hir::BorrowKind::Pin => Ty::new_pinned_ref(self.tcx, region, ty, mutbl),
521                    _ => ::core::panicking::panic("internal error: entered unreachable code")unreachable!(),
522                }
523            }
524        }
525    }
526
527    /// Does this expression refer to a place that either:
528    /// * Is based on a local or static.
529    /// * Contains a dereference
530    /// Note that the adjustments for the children of `expr` should already
531    /// have been resolved.
532    fn check_named_place_expr(&self, oprnd: &'tcx hir::Expr<'tcx>) {
533        let is_named = oprnd.is_place_expr(|base| {
534            // Allow raw borrows if there are any deref adjustments.
535            //
536            // const VAL: (i32,) = (0,);
537            // const REF: &(i32,) = &(0,);
538            //
539            // &raw const VAL.0;            // ERROR
540            // &raw const REF.0;            // OK, same as &raw const (*REF).0;
541            //
542            // This is maybe too permissive, since it allows
543            // `let u = &raw const Box::new((1,)).0`, which creates an
544            // immediately dangling raw pointer.
545            self.typeck_results
546                .borrow()
547                .adjustments()
548                .get(base.hir_id)
549                .is_some_and(|x| x.iter().any(|adj| #[allow(non_exhaustive_omitted_patterns)] match adj.kind {
    Adjust::Deref(_) => true,
    _ => false,
}matches!(adj.kind, Adjust::Deref(_))))
550        });
551        if !is_named {
552            self.dcx().emit_err(AddressOfTemporaryTaken { span: oprnd.span });
553        }
554    }
555
556    pub(crate) fn check_expr_path(
557        &self,
558        qpath: &'tcx hir::QPath<'tcx>,
559        expr: &'tcx hir::Expr<'tcx>,
560        call_expr_and_args: Option<(&'tcx hir::Expr<'tcx>, &'tcx [hir::Expr<'tcx>])>,
561    ) -> Ty<'tcx> {
562        let tcx = self.tcx;
563
564        if let Some((_, [arg])) = call_expr_and_args
565            && let QPath::Resolved(_, path) = qpath
566            && let Res::Def(_, def_id) = path.res
567            && let Some(lang_item) = tcx.lang_items().from_def_id(def_id)
568        {
569            let code = match lang_item {
570                LangItem::IntoFutureIntoFuture
571                    if expr.span.is_desugaring(DesugaringKind::Await) =>
572                {
573                    Some(ObligationCauseCode::AwaitableExpr(arg.hir_id))
574                }
575                LangItem::IntoIterIntoIter | LangItem::IteratorNext
576                    if expr.span.is_desugaring(DesugaringKind::ForLoop) =>
577                {
578                    Some(ObligationCauseCode::ForLoopIterator)
579                }
580                LangItem::TryTraitFromOutput
581                    if expr.span.is_desugaring(DesugaringKind::TryBlock) =>
582                {
583                    // FIXME it's a try block, not a question mark
584                    Some(ObligationCauseCode::QuestionMark)
585                }
586                LangItem::TryTraitBranch | LangItem::TryTraitFromResidual
587                    if expr.span.is_desugaring(DesugaringKind::QuestionMark) =>
588                {
589                    Some(ObligationCauseCode::QuestionMark)
590                }
591                _ => None,
592            };
593            if let Some(code) = code {
594                let args = self.fresh_args_for_item(expr.span, def_id);
595                self.add_required_obligations_with_code(expr.span, def_id, args, |_, _| {
596                    code.clone()
597                });
598                return tcx.type_of(def_id).instantiate(tcx, args);
599            }
600        }
601
602        let (res, opt_ty, segs) =
603            self.resolve_ty_and_res_fully_qualified_call(qpath, expr.hir_id, expr.span);
604        let ty = match res {
605            Res::Err => {
606                self.suggest_assoc_method_call(segs);
607                let e =
608                    self.dcx().span_delayed_bug(qpath.span(), "`Res::Err` but no error emitted");
609                Ty::new_error(tcx, e)
610            }
611            Res::Def(DefKind::Variant, _) => {
612                let e = report_unexpected_variant_res(
613                    tcx,
614                    res,
615                    Some(expr),
616                    qpath,
617                    expr.span,
618                    E0533,
619                    "value",
620                );
621                Ty::new_error(tcx, e)
622            }
623            _ => {
624                self.instantiate_value_path(
625                    segs,
626                    opt_ty,
627                    res,
628                    call_expr_and_args.map_or(expr.span, |(e, _)| e.span),
629                    expr.span,
630                    expr.hir_id,
631                )
632                .0
633            }
634        };
635
636        if let ty::FnDef(did, _) = *ty.kind() {
637            let fn_sig = ty.fn_sig(tcx);
638
639            if tcx.is_intrinsic(did, sym::transmute) {
640                let Some(from) = fn_sig.inputs().skip_binder().get(0) else {
641                    ::rustc_middle::util::bug::span_bug_fmt(tcx.def_span(did),
    format_args!("intrinsic fn `transmute` defined with no parameters"));span_bug!(
642                        tcx.def_span(did),
643                        "intrinsic fn `transmute` defined with no parameters"
644                    );
645                };
646                let to = fn_sig.output().skip_binder();
647                // We defer the transmute to the end of typeck, once all inference vars have
648                // been resolved or we errored. This is important as we can only check transmute
649                // on concrete types, but the output type may not be known yet (it would only
650                // be known if explicitly specified via turbofish).
651                self.deferred_transmute_checks.borrow_mut().push((*from, to, expr.hir_id));
652            }
653            if !tcx.features().unsized_fn_params() {
654                // We want to remove some Sized bounds from std functions,
655                // but don't want to expose the removal to stable Rust.
656                // i.e., we don't want to allow
657                //
658                // ```rust
659                // drop as fn(str);
660                // ```
661                //
662                // to work in stable even if the Sized bound on `drop` is relaxed.
663                for i in 0..fn_sig.inputs().skip_binder().len() {
664                    // We just want to check sizedness, so instead of introducing
665                    // placeholder lifetimes with probing, we just replace higher lifetimes
666                    // with fresh vars.
667                    let span = call_expr_and_args
668                        .and_then(|(_, args)| args.get(i))
669                        .map_or(expr.span, |arg| arg.span);
670                    let input = self.instantiate_binder_with_fresh_vars(
671                        span,
672                        infer::BoundRegionConversionTime::FnCall,
673                        fn_sig.input(i),
674                    );
675                    self.require_type_is_sized_deferred(
676                        input,
677                        span,
678                        ObligationCauseCode::SizedArgumentType(None),
679                    );
680                }
681            }
682            // Here we want to prevent struct constructors from returning unsized types,
683            // which can happen with fn pointer coercion on stable.
684            // Also, as we just want to check sizedness, instead of introducing
685            // placeholder lifetimes with probing, we just replace higher lifetimes
686            // with fresh vars.
687            let output = self.instantiate_binder_with_fresh_vars(
688                expr.span,
689                infer::BoundRegionConversionTime::FnCall,
690                fn_sig.output(),
691            );
692            self.require_type_is_sized_deferred(
693                output,
694                call_expr_and_args.map_or(expr.span, |(e, _)| e.span),
695                ObligationCauseCode::SizedCallReturnType,
696            );
697        }
698
699        // We always require that the type provided as the value for
700        // a type parameter outlives the moment of instantiation.
701        let args = self.typeck_results.borrow().node_args(expr.hir_id);
702        self.add_wf_bounds(args, expr.span);
703
704        ty
705    }
706
707    fn check_expr_break(
708        &self,
709        destination: hir::Destination,
710        expr_opt: Option<&'tcx hir::Expr<'tcx>>,
711        expr: &'tcx hir::Expr<'tcx>,
712    ) -> Ty<'tcx> {
713        let tcx = self.tcx;
714        if let Ok(target_id) = destination.target_id {
715            let (e_ty, cause);
716            if let Some(e) = expr_opt {
717                // If this is a break with a value, we need to type-check
718                // the expression. Get an expected type from the loop context.
719                let opt_coerce_to = {
720                    // We should release `enclosing_breakables` before the `check_expr_with_hint`
721                    // below, so can't move this block of code to the enclosing scope and share
722                    // `ctxt` with the second `enclosing_breakables` borrow below.
723                    let mut enclosing_breakables = self.enclosing_breakables.borrow_mut();
724                    match enclosing_breakables.opt_find_breakable(target_id) {
725                        Some(ctxt) => ctxt.coerce.as_ref().map(|coerce| coerce.expected_ty()),
726                        None => {
727                            // Avoid ICE when `break` is inside a closure (#65383).
728                            return Ty::new_error_with_message(
729                                tcx,
730                                expr.span,
731                                "break was outside loop, but no error was emitted",
732                            );
733                        }
734                    }
735                };
736
737                // If the loop context is not a `loop { }`, then break with
738                // a value is illegal, and `opt_coerce_to` will be `None`.
739                // Set expectation to error in that case and set tainted
740                // by error (#114529)
741                let coerce_to = opt_coerce_to.unwrap_or_else(|| {
742                    let guar = self.dcx().span_delayed_bug(
743                        expr.span,
744                        "illegal break with value found but no error reported",
745                    );
746                    self.set_tainted_by_errors(guar);
747                    Ty::new_error(tcx, guar)
748                });
749
750                // Recurse without `enclosing_breakables` borrowed.
751                e_ty = self.check_expr_with_hint(e, coerce_to);
752                cause = self.misc(e.span);
753            } else {
754                // Otherwise, this is a break *without* a value. That's
755                // always legal, and is equivalent to `break ()`.
756                e_ty = tcx.types.unit;
757                cause = self.misc(expr.span);
758            }
759
760            // Now that we have type-checked `expr_opt`, borrow
761            // the `enclosing_loops` field and let's coerce the
762            // type of `expr_opt` into what is expected.
763            let mut enclosing_breakables = self.enclosing_breakables.borrow_mut();
764            let Some(ctxt) = enclosing_breakables.opt_find_breakable(target_id) else {
765                // Avoid ICE when `break` is inside a closure (#65383).
766                return Ty::new_error_with_message(
767                    tcx,
768                    expr.span,
769                    "break was outside loop, but no error was emitted",
770                );
771            };
772
773            if let Some(ref mut coerce) = ctxt.coerce {
774                if let Some(e) = expr_opt {
775                    coerce.coerce(self, &cause, e, e_ty);
776                } else {
777                    if !e_ty.is_unit() {
    ::core::panicking::panic("assertion failed: e_ty.is_unit()")
};assert!(e_ty.is_unit());
778                    let ty = coerce.expected_ty();
779                    coerce.coerce_forced_unit(
780                        self,
781                        &cause,
782                        |mut err| {
783                            self.suggest_missing_semicolon(&mut err, expr, e_ty, false, false);
784                            self.suggest_mismatched_types_on_tail(
785                                &mut err, expr, ty, e_ty, target_id,
786                            );
787                            let error =
788                                Some(TypeError::Sorts(ExpectedFound { expected: ty, found: e_ty }));
789                            self.annotate_loop_expected_due_to_inference(err, expr, error);
790                            if let Some(val) =
791                                self.err_ctxt().ty_kind_suggestion(self.param_env, ty)
792                            {
793                                err.span_suggestion_verbose(
794                                    expr.span.shrink_to_hi(),
795                                    "give the `break` a value of the expected type",
796                                    ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!(" {0}", val))
    })format!(" {val}"),
797                                    Applicability::HasPlaceholders,
798                                );
799                            }
800                        },
801                        false,
802                    );
803                }
804            } else {
805                // If `ctxt.coerce` is `None`, we can just ignore
806                // the type of the expression. This is because
807                // either this was a break *without* a value, in
808                // which case it is always a legal type (`()`), or
809                // else an error would have been flagged by the
810                // `loops` pass for using break with an expression
811                // where you are not supposed to.
812                if !(expr_opt.is_none() || self.tainted_by_errors().is_some()) {
    ::core::panicking::panic("assertion failed: expr_opt.is_none() || self.tainted_by_errors().is_some()")
};assert!(expr_opt.is_none() || self.tainted_by_errors().is_some());
813            }
814
815            // If we encountered a `break`, then (no surprise) it may be possible to break from the
816            // loop... unless the value being returned from the loop diverges itself, e.g.
817            // `break return 5` or `break loop {}`.
818            ctxt.may_break |= !self.diverges.get().is_always();
819
820            // the type of a `break` is always `!`, since it diverges
821            tcx.types.never
822        } else {
823            // Otherwise, we failed to find the enclosing loop;
824            // this can only happen if the `break` was not
825            // inside a loop at all, which is caught by the
826            // loop-checking pass.
827            let err = Ty::new_error_with_message(
828                self.tcx,
829                expr.span,
830                "break was outside loop, but no error was emitted",
831            );
832
833            // We still need to assign a type to the inner expression to
834            // prevent the ICE in #43162.
835            if let Some(e) = expr_opt {
836                self.check_expr_with_hint(e, err);
837
838                // ... except when we try to 'break rust;'.
839                // ICE this expression in particular (see #43162).
840                if let ExprKind::Path(QPath::Resolved(_, path)) = e.kind {
841                    if let [segment] = path.segments
842                        && segment.ident.name == sym::rust
843                    {
844                        fatally_break_rust(self.tcx, expr.span);
845                    }
846                }
847            }
848
849            // There was an error; make type-check fail.
850            err
851        }
852    }
853
854    fn check_expr_continue(
855        &self,
856        destination: hir::Destination,
857        expr: &'tcx hir::Expr<'tcx>,
858    ) -> Ty<'tcx> {
859        if let Ok(target_id) = destination.target_id {
860            if let hir::Node::Expr(hir::Expr { kind: ExprKind::Loop(..), .. }) =
861                self.tcx.hir_node(target_id)
862            {
863                self.tcx.types.never
864            } else {
865                // Liveness linting assumes `continue`s all point to loops. We'll report an error
866                // in `check_mod_loops`, but make sure we don't run liveness (#113379, #121623).
867                let guar = self.dcx().span_delayed_bug(
868                    expr.span,
869                    "found `continue` not pointing to loop, but no error reported",
870                );
871                Ty::new_error(self.tcx, guar)
872            }
873        } else {
874            // There was an error; make type-check fail.
875            Ty::new_misc_error(self.tcx)
876        }
877    }
878
879    fn check_expr_return(
880        &self,
881        expr_opt: Option<&'tcx hir::Expr<'tcx>>,
882        expr: &'tcx hir::Expr<'tcx>,
883    ) -> Ty<'tcx> {
884        if self.ret_coercion.is_none() {
885            self.emit_return_outside_of_fn_body(expr, ReturnLikeStatementKind::Return);
886
887            if let Some(e) = expr_opt {
888                // We still have to type-check `e` (issue #86188), but calling
889                // `check_return_expr` only works inside fn bodies.
890                self.check_expr(e);
891            }
892        } else if let Some(e) = expr_opt {
893            if self.ret_coercion_span.get().is_none() {
894                self.ret_coercion_span.set(Some(e.span));
895            }
896            self.check_return_or_body_tail(e, true);
897        } else {
898            let mut coercion = self.ret_coercion.as_ref().unwrap().borrow_mut();
899            if self.ret_coercion_span.get().is_none() {
900                self.ret_coercion_span.set(Some(expr.span));
901            }
902            let cause = self.cause(expr.span, ObligationCauseCode::ReturnNoExpression);
903            if let Some((_, fn_decl)) = self.get_fn_decl(expr.hir_id) {
904                coercion.coerce_forced_unit(
905                    self,
906                    &cause,
907                    |db| {
908                        let span = fn_decl.output.span();
909                        if let Ok(snippet) = self.tcx.sess.source_map().span_to_snippet(span) {
910                            db.span_label(
911                                span,
912                                ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("expected `{0}` because of this return type",
                snippet))
    })format!("expected `{snippet}` because of this return type"),
913                            );
914                        }
915                    },
916                    true,
917                );
918            } else {
919                coercion.coerce_forced_unit(self, &cause, |_| (), true);
920            }
921        }
922        self.tcx.types.never
923    }
924
925    fn check_expr_become(
926        &self,
927        call: &'tcx hir::Expr<'tcx>,
928        expr: &'tcx hir::Expr<'tcx>,
929    ) -> Ty<'tcx> {
930        match &self.ret_coercion {
931            Some(ret_coercion) => {
932                let ret_ty = ret_coercion.borrow().expected_ty();
933                let call_expr_ty = self.check_expr_with_hint(call, ret_ty);
934
935                // N.B. don't coerce here, as tail calls can't support most/all coercions
936                // FIXME(explicit_tail_calls): add a diagnostic note that `become` doesn't allow coercions
937                self.demand_suptype(expr.span, ret_ty, call_expr_ty);
938            }
939            None => {
940                self.emit_return_outside_of_fn_body(expr, ReturnLikeStatementKind::Become);
941
942                // Fallback to simply type checking `call` without hint/demanding the right types.
943                // Best effort to highlight more errors.
944                self.check_expr(call);
945            }
946        }
947
948        self.tcx.types.never
949    }
950
951    /// Check an expression that _is being returned_.
952    /// For example, this is called with `return_expr: $expr` when `return $expr`
953    /// is encountered.
954    ///
955    /// Note that this function must only be called in function bodies.
956    ///
957    /// `explicit_return` is `true` if we're checking an explicit `return expr`,
958    /// and `false` if we're checking a trailing expression.
959    pub(super) fn check_return_or_body_tail(
960        &self,
961        return_expr: &'tcx hir::Expr<'tcx>,
962        explicit_return: bool,
963    ) {
964        let ret_coercion = self.ret_coercion.as_ref().unwrap_or_else(|| {
965            ::rustc_middle::util::bug::span_bug_fmt(return_expr.span,
    format_args!("check_return_expr called outside fn body"))span_bug!(return_expr.span, "check_return_expr called outside fn body")
966        });
967
968        let ret_ty = ret_coercion.borrow().expected_ty();
969        let return_expr_ty = self.check_expr_with_hint(return_expr, ret_ty);
970        let mut span = return_expr.span;
971        let mut hir_id = return_expr.hir_id;
972        // Use the span of the trailing expression for our cause,
973        // not the span of the entire function
974        if !explicit_return
975            && let ExprKind::Block(body, _) = return_expr.kind
976            && let Some(last_expr) = body.expr
977        {
978            span = last_expr.span;
979            hir_id = last_expr.hir_id;
980        }
981        ret_coercion.borrow_mut().coerce(
982            self,
983            &self.cause(span, ObligationCauseCode::ReturnValue(return_expr.hir_id)),
984            return_expr,
985            return_expr_ty,
986        );
987
988        if let Some(fn_sig) = self.body_fn_sig()
989            && fn_sig.output().has_opaque_types()
990        {
991            // Point any obligations that were registered due to opaque type
992            // inference at the return expression.
993            self.select_obligations_where_possible(|errors| {
994                self.point_at_return_for_opaque_ty_error(
995                    errors,
996                    hir_id,
997                    span,
998                    return_expr_ty,
999                    return_expr.span,
1000                );
1001            });
1002        }
1003    }
1004
1005    /// Emit an error because `return` or `become` is used outside of a function body.
1006    ///
1007    /// `expr` is the `return` (`become`) "statement", `kind` is the kind of the statement
1008    /// either `Return` or `Become`.
1009    fn emit_return_outside_of_fn_body(&self, expr: &hir::Expr<'_>, kind: ReturnLikeStatementKind) {
1010        let mut err = ReturnStmtOutsideOfFnBody {
1011            span: expr.span,
1012            encl_body_span: None,
1013            encl_fn_span: None,
1014            statement_kind: kind,
1015        };
1016
1017        let encl_item_id = self.tcx.hir_get_parent_item(expr.hir_id);
1018
1019        if let hir::Node::Item(hir::Item {
1020            kind: hir::ItemKind::Fn { .. },
1021            span: encl_fn_span,
1022            ..
1023        })
1024        | hir::Node::TraitItem(hir::TraitItem {
1025            kind: hir::TraitItemKind::Fn(_, hir::TraitFn::Provided(_)),
1026            span: encl_fn_span,
1027            ..
1028        })
1029        | hir::Node::ImplItem(hir::ImplItem {
1030            kind: hir::ImplItemKind::Fn(..),
1031            span: encl_fn_span,
1032            ..
1033        }) = self.tcx.hir_node_by_def_id(encl_item_id.def_id)
1034        {
1035            // We are inside a function body, so reporting "return statement
1036            // outside of function body" needs an explanation.
1037
1038            let encl_body_owner_id = self.tcx.hir_enclosing_body_owner(expr.hir_id);
1039
1040            // If this didn't hold, we would not have to report an error in
1041            // the first place.
1042            match (&encl_item_id.def_id, &encl_body_owner_id) {
    (left_val, right_val) => {
        if *left_val == *right_val {
            let kind = ::core::panicking::AssertKind::Ne;
            ::core::panicking::assert_failed(kind, &*left_val, &*right_val,
                ::core::option::Option::None);
        }
    }
};assert_ne!(encl_item_id.def_id, encl_body_owner_id);
1043
1044            let encl_body = self.tcx.hir_body_owned_by(encl_body_owner_id);
1045
1046            err.encl_body_span = Some(encl_body.value.span);
1047            err.encl_fn_span = Some(*encl_fn_span);
1048        }
1049
1050        self.dcx().emit_err(err);
1051    }
1052
1053    fn point_at_return_for_opaque_ty_error(
1054        &self,
1055        errors: &mut Vec<traits::FulfillmentError<'tcx>>,
1056        hir_id: HirId,
1057        span: Span,
1058        return_expr_ty: Ty<'tcx>,
1059        return_span: Span,
1060    ) {
1061        // Don't point at the whole block if it's empty
1062        if span == return_span {
1063            return;
1064        }
1065        for err in errors {
1066            let cause = &mut err.obligation.cause;
1067            if let ObligationCauseCode::OpaqueReturnType(None) = cause.code() {
1068                let new_cause = self.cause(
1069                    cause.span,
1070                    ObligationCauseCode::OpaqueReturnType(Some((return_expr_ty, hir_id))),
1071                );
1072                *cause = new_cause;
1073            }
1074        }
1075    }
1076
1077    pub(crate) fn check_lhs_assignable(
1078        &self,
1079        lhs: &'tcx hir::Expr<'tcx>,
1080        code: ErrCode,
1081        op_span: Span,
1082        adjust_err: impl FnOnce(&mut Diag<'_>),
1083    ) {
1084        if lhs.is_syntactic_place_expr() {
1085            return;
1086        }
1087
1088        // Skip suggestion if LHS contains a let-chain at this would likely be spurious
1089        // cc: https://github.com/rust-lang/rust/issues/147664
1090        if contains_let_in_chain(lhs) {
1091            return;
1092        }
1093
1094        let mut err = self.dcx().struct_span_err(op_span, "invalid left-hand side of assignment");
1095        err.code(code);
1096        err.span_label(lhs.span, "cannot assign to this expression");
1097
1098        self.comes_from_while_condition(lhs.hir_id, |expr| {
1099            err.span_suggestion_verbose(
1100                expr.span.shrink_to_lo(),
1101                "you might have meant to use pattern destructuring",
1102                "let ",
1103                Applicability::MachineApplicable,
1104            );
1105        });
1106        self.check_for_missing_semi(lhs, &mut err);
1107
1108        adjust_err(&mut err);
1109
1110        err.emit();
1111    }
1112
1113    /// Check if the expression that could not be assigned to was a typoed expression that
1114    pub(crate) fn check_for_missing_semi(
1115        &self,
1116        expr: &'tcx hir::Expr<'tcx>,
1117        err: &mut Diag<'_>,
1118    ) -> bool {
1119        if let hir::ExprKind::Binary(binop, lhs, rhs) = expr.kind
1120            && let hir::BinOpKind::Mul = binop.node
1121            && self.tcx.sess.source_map().is_multiline(lhs.span.between(rhs.span))
1122            && rhs.is_syntactic_place_expr()
1123        {
1124            //      v missing semicolon here
1125            // foo()
1126            // *bar = baz;
1127            // (#80446).
1128            err.span_suggestion_verbose(
1129                lhs.span.shrink_to_hi(),
1130                "you might have meant to write a semicolon here",
1131                ";",
1132                Applicability::MachineApplicable,
1133            );
1134            return true;
1135        }
1136        false
1137    }
1138
1139    // Check if an expression `original_expr_id` comes from the condition of a while loop,
1140    /// as opposed from the body of a while loop, which we can naively check by iterating
1141    /// parents until we find a loop...
1142    pub(super) fn comes_from_while_condition(
1143        &self,
1144        original_expr_id: HirId,
1145        then: impl FnOnce(&hir::Expr<'_>),
1146    ) {
1147        let mut parent = self.tcx.parent_hir_id(original_expr_id);
1148        loop {
1149            let node = self.tcx.hir_node(parent);
1150            match node {
1151                hir::Node::Expr(hir::Expr {
1152                    kind:
1153                        hir::ExprKind::Loop(
1154                            hir::Block {
1155                                expr:
1156                                    Some(hir::Expr {
1157                                        kind:
1158                                            hir::ExprKind::Match(expr, ..) | hir::ExprKind::If(expr, ..),
1159                                        ..
1160                                    }),
1161                                ..
1162                            },
1163                            _,
1164                            hir::LoopSource::While,
1165                            _,
1166                        ),
1167                    ..
1168                }) => {
1169                    // Check if our original expression is a child of the condition of a while loop.
1170                    // If it is, then we have a situation like `while Some(0) = value.get(0) {`,
1171                    // where `while let` was more likely intended.
1172                    if self.tcx.hir_parent_id_iter(original_expr_id).any(|id| id == expr.hir_id) {
1173                        then(expr);
1174                    }
1175                    break;
1176                }
1177                hir::Node::Item(_)
1178                | hir::Node::ImplItem(_)
1179                | hir::Node::TraitItem(_)
1180                | hir::Node::Crate(_) => break,
1181                _ => {
1182                    parent = self.tcx.parent_hir_id(parent);
1183                }
1184            }
1185        }
1186    }
1187
1188    // A generic function for checking the 'then' and 'else' clauses in an 'if'
1189    // or 'if-else' expression.
1190    fn check_expr_if(
1191        &self,
1192        expr_id: HirId,
1193        cond_expr: &'tcx hir::Expr<'tcx>,
1194        then_expr: &'tcx hir::Expr<'tcx>,
1195        opt_else_expr: Option<&'tcx hir::Expr<'tcx>>,
1196        sp: Span,
1197        orig_expected: Expectation<'tcx>,
1198    ) -> Ty<'tcx> {
1199        let cond_ty = self.check_expr_has_type_or_error(cond_expr, self.tcx.types.bool, |_| {});
1200
1201        self.warn_if_unreachable(
1202            cond_expr.hir_id,
1203            then_expr.span,
1204            "block in `if` or `while` expression",
1205        );
1206
1207        let cond_diverges = self.diverges.get();
1208        self.diverges.set(Diverges::Maybe);
1209
1210        let expected = orig_expected.try_structurally_resolve_and_adjust_for_branches(self, sp);
1211        let then_ty = self.check_expr_with_expectation(then_expr, expected);
1212        let then_diverges = self.diverges.get();
1213        self.diverges.set(Diverges::Maybe);
1214
1215        // We've already taken the expected type's preferences
1216        // into account when typing the `then` branch. To figure
1217        // out the initial shot at a LUB, we thus only consider
1218        // `expected` if it represents a *hard* constraint
1219        // (`only_has_type`); otherwise, we just go with a
1220        // fresh type variable.
1221        let coerce_to_ty = expected.coercion_target_type(self, sp);
1222        let mut coerce = CoerceMany::with_capacity(coerce_to_ty, 2);
1223
1224        coerce.coerce(self, &self.misc(sp), then_expr, then_ty);
1225
1226        if let Some(else_expr) = opt_else_expr {
1227            let else_ty = self.check_expr_with_expectation(else_expr, expected);
1228            let else_diverges = self.diverges.get();
1229
1230            let tail_defines_return_position_impl_trait =
1231                self.return_position_impl_trait_from_match_expectation(orig_expected);
1232            let if_cause =
1233                self.if_cause(expr_id, else_expr, tail_defines_return_position_impl_trait);
1234
1235            coerce.coerce(self, &if_cause, else_expr, else_ty);
1236
1237            // We won't diverge unless both branches do (or the condition does).
1238            self.diverges.set(cond_diverges | then_diverges & else_diverges);
1239        } else {
1240            self.if_fallback_coercion(sp, cond_expr, then_expr, &mut coerce);
1241
1242            // If the condition is false we can't diverge.
1243            self.diverges.set(cond_diverges);
1244        }
1245
1246        let result_ty = coerce.complete(self);
1247        if let Err(guar) = cond_ty.error_reported() {
1248            Ty::new_error(self.tcx, guar)
1249        } else {
1250            result_ty
1251        }
1252    }
1253
1254    /// Type check assignment expression `expr` of form `lhs = rhs`.
1255    /// The expected type is `()` and is passed to the function for the purposes of diagnostics.
1256    fn check_expr_assign(
1257        &self,
1258        expr: &'tcx hir::Expr<'tcx>,
1259        expected: Expectation<'tcx>,
1260        lhs: &'tcx hir::Expr<'tcx>,
1261        rhs: &'tcx hir::Expr<'tcx>,
1262        span: Span,
1263    ) -> Ty<'tcx> {
1264        let expected_ty = expected.only_has_type(self);
1265        if expected_ty == Some(self.tcx.types.bool) {
1266            let guar = self.expr_assign_expected_bool_error(expr, lhs, rhs, span);
1267            return Ty::new_error(self.tcx, guar);
1268        }
1269
1270        let lhs_ty = self.check_expr_with_needs(lhs, Needs::MutPlace);
1271
1272        let suggest_deref_binop = |err: &mut Diag<'_>, rhs_ty: Ty<'tcx>| {
1273            if let Some(lhs_deref_ty) = self.deref_once_mutably_for_diagnostic(lhs_ty) {
1274                // Can only assign if the type is sized, so if `DerefMut` yields a type that is
1275                // unsized, do not suggest dereferencing it.
1276                let lhs_deref_ty_is_sized = self
1277                    .infcx
1278                    .type_implements_trait(
1279                        self.tcx.require_lang_item(LangItem::Sized, span),
1280                        [lhs_deref_ty],
1281                        self.param_env,
1282                    )
1283                    .may_apply();
1284                if lhs_deref_ty_is_sized && self.may_coerce(rhs_ty, lhs_deref_ty) {
1285                    err.span_suggestion_verbose(
1286                        lhs.span.shrink_to_lo(),
1287                        "consider dereferencing here to assign to the mutably borrowed value",
1288                        "*",
1289                        Applicability::MachineApplicable,
1290                    );
1291                }
1292            }
1293        };
1294
1295        // This is (basically) inlined `check_expr_coercible_to_type`, but we want
1296        // to suggest an additional fixup here in `suggest_deref_binop`.
1297        let rhs_ty = self.check_expr_with_hint(rhs, lhs_ty);
1298        if let Err(mut diag) =
1299            self.demand_coerce_diag(rhs, rhs_ty, lhs_ty, Some(lhs), AllowTwoPhase::No)
1300        {
1301            suggest_deref_binop(&mut diag, rhs_ty);
1302            diag.emit();
1303        }
1304
1305        self.check_lhs_assignable(lhs, E0070, span, |err| {
1306            if let Some(rhs_ty) = self.typeck_results.borrow().expr_ty_opt(rhs) {
1307                suggest_deref_binop(err, rhs_ty);
1308            }
1309        });
1310
1311        self.require_type_is_sized(lhs_ty, lhs.span, ObligationCauseCode::AssignmentLhsSized);
1312
1313        if let Err(guar) = (lhs_ty, rhs_ty).error_reported() {
1314            Ty::new_error(self.tcx, guar)
1315        } else {
1316            self.tcx.types.unit
1317        }
1318    }
1319
1320    /// The expected type is `bool` but this will result in `()` so we can reasonably
1321    /// say that the user intended to write `lhs == rhs` instead of `lhs = rhs`.
1322    /// The likely cause of this is `if foo = bar { .. }`.
1323    fn expr_assign_expected_bool_error(
1324        &self,
1325        expr: &'tcx hir::Expr<'tcx>,
1326        lhs: &'tcx hir::Expr<'tcx>,
1327        rhs: &'tcx hir::Expr<'tcx>,
1328        span: Span,
1329    ) -> ErrorGuaranteed {
1330        let actual_ty = self.tcx.types.unit;
1331        let expected_ty = self.tcx.types.bool;
1332        let mut err = self.demand_suptype_diag(expr.span, expected_ty, actual_ty).unwrap_err();
1333        let lhs_ty = self.check_expr(lhs);
1334        let rhs_ty = self.check_expr(rhs);
1335        let refs_can_coerce = |lhs: Ty<'tcx>, rhs: Ty<'tcx>| {
1336            let lhs = Ty::new_imm_ref(self.tcx, self.tcx.lifetimes.re_erased, lhs.peel_refs());
1337            let rhs = Ty::new_imm_ref(self.tcx, self.tcx.lifetimes.re_erased, rhs.peel_refs());
1338            self.may_coerce(rhs, lhs)
1339        };
1340        let (applicability, eq) = if self.may_coerce(rhs_ty, lhs_ty) {
1341            (Applicability::MachineApplicable, true)
1342        } else if refs_can_coerce(rhs_ty, lhs_ty) {
1343            // The lhs and rhs are likely missing some references in either side. Subsequent
1344            // suggestions will show up.
1345            (Applicability::MaybeIncorrect, true)
1346        } else if let ExprKind::Binary(
1347            Spanned { node: hir::BinOpKind::And | hir::BinOpKind::Or, .. },
1348            _,
1349            rhs_expr,
1350        ) = lhs.kind
1351        {
1352            // if x == 1 && y == 2 { .. }
1353            //                 +
1354            let actual_lhs = self.check_expr(rhs_expr);
1355            let may_eq = self.may_coerce(rhs_ty, actual_lhs) || refs_can_coerce(rhs_ty, actual_lhs);
1356            (Applicability::MaybeIncorrect, may_eq)
1357        } else if let ExprKind::Binary(
1358            Spanned { node: hir::BinOpKind::And | hir::BinOpKind::Or, .. },
1359            lhs_expr,
1360            _,
1361        ) = rhs.kind
1362        {
1363            // if x == 1 && y == 2 { .. }
1364            //       +
1365            let actual_rhs = self.check_expr(lhs_expr);
1366            let may_eq = self.may_coerce(actual_rhs, lhs_ty) || refs_can_coerce(actual_rhs, lhs_ty);
1367            (Applicability::MaybeIncorrect, may_eq)
1368        } else {
1369            (Applicability::MaybeIncorrect, false)
1370        };
1371
1372        if !lhs.is_syntactic_place_expr()
1373            && lhs.is_approximately_pattern()
1374            && !#[allow(non_exhaustive_omitted_patterns)] match lhs.kind {
    hir::ExprKind::Lit(_) => true,
    _ => false,
}matches!(lhs.kind, hir::ExprKind::Lit(_))
1375        {
1376            // Do not suggest `if let x = y` as `==` is way more likely to be the intention.
1377            if let hir::Node::Expr(hir::Expr { kind: ExprKind::If { .. }, .. }) =
1378                self.tcx.parent_hir_node(expr.hir_id)
1379            {
1380                err.span_suggestion_verbose(
1381                    expr.span.shrink_to_lo(),
1382                    "you might have meant to use pattern matching",
1383                    "let ",
1384                    applicability,
1385                );
1386            };
1387        }
1388        if eq {
1389            err.span_suggestion_verbose(
1390                span.shrink_to_hi(),
1391                "you might have meant to compare for equality",
1392                '=',
1393                applicability,
1394            );
1395        }
1396
1397        // If the assignment expression itself is ill-formed, don't
1398        // bother emitting another error
1399        err.emit_unless_delay(lhs_ty.references_error() || rhs_ty.references_error())
1400    }
1401
1402    pub(super) fn check_expr_let(
1403        &self,
1404        let_expr: &'tcx hir::LetExpr<'tcx>,
1405        hir_id: HirId,
1406    ) -> Ty<'tcx> {
1407        GatherLocalsVisitor::gather_from_let_expr(self, let_expr, hir_id);
1408
1409        // for let statements, this is done in check_stmt
1410        let init = let_expr.init;
1411        self.warn_if_unreachable(init.hir_id, init.span, "block in `let` expression");
1412
1413        // otherwise check exactly as a let statement
1414        self.check_decl((let_expr, hir_id).into());
1415
1416        // but return a bool, for this is a boolean expression
1417        if let ast::Recovered::Yes(error_guaranteed) = let_expr.recovered {
1418            self.set_tainted_by_errors(error_guaranteed);
1419            Ty::new_error(self.tcx, error_guaranteed)
1420        } else {
1421            self.tcx.types.bool
1422        }
1423    }
1424
1425    fn check_expr_loop(
1426        &self,
1427        body: &'tcx hir::Block<'tcx>,
1428        source: hir::LoopSource,
1429        expected: Expectation<'tcx>,
1430        expr: &'tcx hir::Expr<'tcx>,
1431    ) -> Ty<'tcx> {
1432        let coerce = match source {
1433            // you can only use break with a value from a normal `loop { }`
1434            hir::LoopSource::Loop => {
1435                let coerce_to = expected.coercion_target_type(self, body.span);
1436                Some(CoerceMany::new(coerce_to))
1437            }
1438
1439            hir::LoopSource::While | hir::LoopSource::ForLoop => None,
1440        };
1441
1442        let ctxt = BreakableCtxt {
1443            coerce,
1444            may_break: false, // Will get updated if/when we find a `break`.
1445        };
1446
1447        let (ctxt, ()) = self.with_breakable_ctxt(expr.hir_id, ctxt, || {
1448            self.check_block_no_value(body);
1449        });
1450
1451        if ctxt.may_break {
1452            // No way to know whether it's diverging because
1453            // of a `break` or an outer `break` or `return`.
1454            self.diverges.set(Diverges::Maybe);
1455        } else {
1456            self.diverges.set(self.diverges.get() | Diverges::always(expr.span));
1457        }
1458
1459        // If we permit break with a value, then result type is
1460        // the LUB of the breaks (possibly ! if none); else, it
1461        // is nil. This makes sense because infinite loops
1462        // (which would have type !) are only possible iff we
1463        // permit break with a value.
1464        if ctxt.coerce.is_none() && !ctxt.may_break {
1465            self.dcx().span_bug(body.span, "no coercion, but loop may not break");
1466        }
1467        ctxt.coerce.map(|c| c.complete(self)).unwrap_or_else(|| self.tcx.types.unit)
1468    }
1469
1470    /// Checks a method call.
1471    fn check_expr_method_call(
1472        &self,
1473        expr: &'tcx hir::Expr<'tcx>,
1474        segment: &'tcx hir::PathSegment<'tcx>,
1475        rcvr: &'tcx hir::Expr<'tcx>,
1476        args: &'tcx [hir::Expr<'tcx>],
1477        expected: Expectation<'tcx>,
1478    ) -> Ty<'tcx> {
1479        let rcvr_t = self.check_expr(rcvr);
1480        let rcvr_t = self.try_structurally_resolve_type(rcvr.span, rcvr_t);
1481
1482        match self.lookup_method(rcvr_t, segment, segment.ident.span, expr, rcvr, args) {
1483            Ok(method) => {
1484                self.write_method_call_and_enforce_effects(expr.hir_id, expr.span, method);
1485
1486                self.check_argument_types(
1487                    segment.ident.span,
1488                    expr,
1489                    &method.sig.inputs()[1..],
1490                    method.sig.output(),
1491                    expected,
1492                    args,
1493                    method.sig.c_variadic,
1494                    TupleArgumentsFlag::DontTupleArguments,
1495                    Some(method.def_id),
1496                );
1497
1498                self.check_call_abi(method.sig.abi, expr.span);
1499
1500                method.sig.output()
1501            }
1502            Err(error) => {
1503                let guar = self.report_method_error(expr.hir_id, rcvr_t, error, expected, false);
1504
1505                let err_inputs = self.err_args(args.len(), guar);
1506                let err_output = Ty::new_error(self.tcx, guar);
1507
1508                self.check_argument_types(
1509                    segment.ident.span,
1510                    expr,
1511                    &err_inputs,
1512                    err_output,
1513                    NoExpectation,
1514                    args,
1515                    false,
1516                    TupleArgumentsFlag::DontTupleArguments,
1517                    None,
1518                );
1519
1520                err_output
1521            }
1522        }
1523    }
1524
1525    /// Checks use `x.use`.
1526    fn check_expr_use(
1527        &self,
1528        used_expr: &'tcx hir::Expr<'tcx>,
1529        expected: Expectation<'tcx>,
1530    ) -> Ty<'tcx> {
1531        self.check_expr_with_expectation(used_expr, expected)
1532    }
1533
1534    fn check_expr_cast(
1535        &self,
1536        e: &'tcx hir::Expr<'tcx>,
1537        t: &'tcx hir::Ty<'tcx>,
1538        expr: &'tcx hir::Expr<'tcx>,
1539    ) -> Ty<'tcx> {
1540        // Find the type of `e`. Supply hints based on the type we are casting to,
1541        // if appropriate.
1542        let t_cast = self.lower_ty_saving_user_provided_ty(t);
1543        let t_cast = self.resolve_vars_if_possible(t_cast);
1544        let t_expr = self.check_expr_with_expectation(e, ExpectCastableToType(t_cast));
1545        let t_expr = self.resolve_vars_if_possible(t_expr);
1546
1547        // Eagerly check for some obvious errors.
1548        if let Err(guar) = (t_expr, t_cast).error_reported() {
1549            Ty::new_error(self.tcx, guar)
1550        } else {
1551            // Defer other checks until we're done type checking.
1552            let mut deferred_cast_checks = self.deferred_cast_checks.borrow_mut();
1553            match cast::CastCheck::new(self, e, t_expr, t_cast, t.span, expr.span) {
1554                Ok(cast_check) => {
1555                    {
    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/expr.rs:1555",
                        "rustc_hir_typeck::expr", ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/expr.rs"),
                        ::tracing_core::__macro_support::Option::Some(1555u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::expr"),
                        ::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!("check_expr_cast: deferring cast from {0:?} to {1:?}: {2:?}",
                                                    t_cast, t_expr, cast_check) as &dyn Value))])
            });
    } else { ; }
};debug!(
1556                        "check_expr_cast: deferring cast from {:?} to {:?}: {:?}",
1557                        t_cast, t_expr, cast_check,
1558                    );
1559                    deferred_cast_checks.push(cast_check);
1560                    t_cast
1561                }
1562                Err(guar) => Ty::new_error(self.tcx, guar),
1563            }
1564        }
1565    }
1566
1567    fn check_expr_unsafe_binder_cast(
1568        &self,
1569        span: Span,
1570        kind: ast::UnsafeBinderCastKind,
1571        inner_expr: &'tcx hir::Expr<'tcx>,
1572        hir_ty: Option<&'tcx hir::Ty<'tcx>>,
1573        expected: Expectation<'tcx>,
1574    ) -> Ty<'tcx> {
1575        match kind {
1576            ast::UnsafeBinderCastKind::Wrap => {
1577                let ascribed_ty =
1578                    hir_ty.map(|hir_ty| self.lower_ty_saving_user_provided_ty(hir_ty));
1579                let expected_ty = expected.only_has_type(self);
1580                let binder_ty = match (ascribed_ty, expected_ty) {
1581                    (Some(ascribed_ty), Some(expected_ty)) => {
1582                        self.demand_eqtype(inner_expr.span, expected_ty, ascribed_ty);
1583                        expected_ty
1584                    }
1585                    (Some(ty), None) | (None, Some(ty)) => ty,
1586                    // This will always cause a structural resolve error, but we do it
1587                    // so we don't need to manually report an E0282 both on this codepath
1588                    // and in the others; it all happens in `structurally_resolve_type`.
1589                    (None, None) => self.next_ty_var(inner_expr.span),
1590                };
1591
1592                let binder_ty = self.structurally_resolve_type(inner_expr.span, binder_ty);
1593                let hint_ty = match *binder_ty.kind() {
1594                    ty::UnsafeBinder(binder) => self.instantiate_binder_with_fresh_vars(
1595                        inner_expr.span,
1596                        infer::BoundRegionConversionTime::HigherRankedType,
1597                        binder.into(),
1598                    ),
1599                    ty::Error(e) => Ty::new_error(self.tcx, e),
1600                    _ => {
1601                        let guar = self
1602                            .dcx()
1603                            .struct_span_err(
1604                                hir_ty.map_or(span, |hir_ty| hir_ty.span),
1605                                ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("`wrap_binder!()` can only wrap into unsafe binder, not {0}",
                binder_ty.sort_string(self.tcx)))
    })format!(
1606                                    "`wrap_binder!()` can only wrap into unsafe binder, not {}",
1607                                    binder_ty.sort_string(self.tcx)
1608                                ),
1609                            )
1610                            .with_note("unsafe binders are the only valid output of wrap")
1611                            .emit();
1612                        Ty::new_error(self.tcx, guar)
1613                    }
1614                };
1615
1616                self.check_expr_has_type_or_error(inner_expr, hint_ty, |_| {});
1617
1618                binder_ty
1619            }
1620            ast::UnsafeBinderCastKind::Unwrap => {
1621                let ascribed_ty =
1622                    hir_ty.map(|hir_ty| self.lower_ty_saving_user_provided_ty(hir_ty));
1623                let hint_ty = ascribed_ty.unwrap_or_else(|| self.next_ty_var(inner_expr.span));
1624                // FIXME(unsafe_binders): coerce here if needed?
1625                let binder_ty = self.check_expr_has_type_or_error(inner_expr, hint_ty, |_| {});
1626
1627                // Unwrap the binder. This will be ambiguous if it's an infer var, and will error
1628                // if it's not an unsafe binder.
1629                let binder_ty = self.structurally_resolve_type(inner_expr.span, binder_ty);
1630                match *binder_ty.kind() {
1631                    ty::UnsafeBinder(binder) => self.instantiate_binder_with_fresh_vars(
1632                        inner_expr.span,
1633                        infer::BoundRegionConversionTime::HigherRankedType,
1634                        binder.into(),
1635                    ),
1636                    ty::Error(e) => Ty::new_error(self.tcx, e),
1637                    _ => {
1638                        let guar = self
1639                            .dcx()
1640                            .struct_span_err(
1641                                hir_ty.map_or(inner_expr.span, |hir_ty| hir_ty.span),
1642                                ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("expected unsafe binder, found {0} as input of `unwrap_binder!()`",
                binder_ty.sort_string(self.tcx)))
    })format!(
1643                                    "expected unsafe binder, found {} as input of \
1644                                    `unwrap_binder!()`",
1645                                    binder_ty.sort_string(self.tcx)
1646                                ),
1647                            )
1648                            .with_note("only an unsafe binder type can be unwrapped")
1649                            .emit();
1650                        Ty::new_error(self.tcx, guar)
1651                    }
1652                }
1653            }
1654        }
1655    }
1656
1657    fn check_expr_array(
1658        &self,
1659        args: &'tcx [hir::Expr<'tcx>],
1660        expected: Expectation<'tcx>,
1661        expr: &'tcx hir::Expr<'tcx>,
1662    ) -> Ty<'tcx> {
1663        let element_ty = if !args.is_empty() {
1664            // This shouldn't happen unless there's another error
1665            // (e.g., never patterns in inappropriate contexts).
1666            if self.diverges.get() != Diverges::Maybe {
1667                self.dcx()
1668                    .struct_span_err(expr.span, "unexpected divergence state in checking array")
1669                    .delay_as_bug();
1670            }
1671
1672            let coerce_to = expected
1673                .to_option(self)
1674                .and_then(|uty| {
1675                    self.try_structurally_resolve_type(expr.span, uty)
1676                        .builtin_index()
1677                        // Avoid using the original type variable as the coerce_to type, as it may resolve
1678                        // during the first coercion instead of being the LUB type.
1679                        .filter(|t| !self.try_structurally_resolve_type(expr.span, *t).is_ty_var())
1680                })
1681                .unwrap_or_else(|| self.next_ty_var(expr.span));
1682            let mut coerce = CoerceMany::with_capacity(coerce_to, args.len());
1683
1684            for e in args {
1685                // FIXME: the element expectation should use
1686                // `try_structurally_resolve_and_adjust_for_branches` just like in `if` and `match`.
1687                // While that fixes nested coercion, it will break [some
1688                // code like this](https://github.com/rust-lang/rust/pull/140283#issuecomment-2958776528).
1689                // If we find a way to support recursive tuple coercion, this break can be avoided.
1690                let e_ty = self.check_expr_with_hint(e, coerce_to);
1691                let cause = self.misc(e.span);
1692                coerce.coerce(self, &cause, e, e_ty);
1693            }
1694            coerce.complete(self)
1695        } else {
1696            self.next_ty_var(expr.span)
1697        };
1698        let array_len = args.len() as u64;
1699        self.suggest_array_len(expr, array_len);
1700        Ty::new_array(self.tcx, element_ty, array_len)
1701    }
1702
1703    fn suggest_array_len(&self, expr: &'tcx hir::Expr<'tcx>, array_len: u64) {
1704        let parent_node = self.tcx.hir_parent_iter(expr.hir_id).find(|(_, node)| {
1705            !#[allow(non_exhaustive_omitted_patterns)] match node {
    hir::Node::Expr(hir::Expr { kind: hir::ExprKind::AddrOf(..), .. }) =>
        true,
    _ => false,
}matches!(node, hir::Node::Expr(hir::Expr { kind: hir::ExprKind::AddrOf(..), .. }))
1706        });
1707        let Some((_, hir::Node::LetStmt(hir::LetStmt { ty: Some(ty), .. }))) = parent_node else {
1708            return;
1709        };
1710        if let hir::TyKind::Array(_, ct) = ty.peel_refs().kind {
1711            let span = ct.span;
1712            self.dcx().try_steal_modify_and_emit_err(
1713                span,
1714                StashKey::UnderscoreForArrayLengths,
1715                |err| {
1716                    err.span_suggestion(
1717                        span,
1718                        "consider specifying the array length",
1719                        array_len,
1720                        Applicability::MaybeIncorrect,
1721                    );
1722                },
1723            );
1724        }
1725    }
1726
1727    pub(super) fn check_expr_const_block(
1728        &self,
1729        block: &'tcx hir::ConstBlock,
1730        expected: Expectation<'tcx>,
1731    ) -> Ty<'tcx> {
1732        let body = self.tcx.hir_body(block.body);
1733
1734        // Create a new function context.
1735        let def_id = block.def_id;
1736        let fcx = FnCtxt::new(self, self.param_env, def_id);
1737
1738        let ty = fcx.check_expr_with_expectation(body.value, expected);
1739        fcx.require_type_is_sized(ty, body.value.span, ObligationCauseCode::SizedConstOrStatic);
1740        fcx.write_ty(block.hir_id, ty);
1741        ty
1742    }
1743
1744    fn check_expr_repeat(
1745        &self,
1746        element: &'tcx hir::Expr<'tcx>,
1747        count: &'tcx hir::ConstArg<'tcx>,
1748        expected: Expectation<'tcx>,
1749        expr: &'tcx hir::Expr<'tcx>,
1750    ) -> Ty<'tcx> {
1751        let tcx = self.tcx;
1752        let count_span = count.span;
1753        let count = self.try_structurally_resolve_const(
1754            count_span,
1755            self.normalize(count_span, self.lower_const_arg(count, tcx.types.usize)),
1756        );
1757
1758        if let Some(count) = count.try_to_target_usize(tcx) {
1759            self.suggest_array_len(expr, count);
1760        }
1761
1762        let uty = match expected {
1763            ExpectHasType(uty) => uty.builtin_index(),
1764            _ => None,
1765        };
1766
1767        let (element_ty, t) = match uty {
1768            Some(uty) => {
1769                self.check_expr_coercible_to_type(element, uty, None);
1770                (uty, uty)
1771            }
1772            None => {
1773                let ty = self.next_ty_var(element.span);
1774                let element_ty = self.check_expr_has_type_or_error(element, ty, |_| {});
1775                (element_ty, ty)
1776            }
1777        };
1778
1779        if let Err(guar) = element_ty.error_reported() {
1780            return Ty::new_error(tcx, guar);
1781        }
1782
1783        // We defer checking whether the element type is `Copy` as it is possible to have
1784        // an inference variable as a repeat count and it seems unlikely that `Copy` would
1785        // have inference side effects required for type checking to succeed.
1786        self.deferred_repeat_expr_checks.borrow_mut().push((element, element_ty, count));
1787
1788        let ty = Ty::new_array_with_const_len(tcx, t, count);
1789        self.register_wf_obligation(ty.into(), expr.span, ObligationCauseCode::WellFormed(None));
1790        ty
1791    }
1792
1793    fn check_expr_tuple(
1794        &self,
1795        elements: &'tcx [hir::Expr<'tcx>],
1796        expected: Expectation<'tcx>,
1797        expr: &'tcx hir::Expr<'tcx>,
1798    ) -> Ty<'tcx> {
1799        let mut expectations = expected
1800            .only_has_type(self)
1801            .and_then(|ty| self.try_structurally_resolve_type(expr.span, ty).opt_tuple_fields())
1802            .unwrap_or_default()
1803            .iter();
1804
1805        let elements = elements.iter().map(|e| {
1806            let ty = expectations.next().unwrap_or_else(|| self.next_ty_var(e.span));
1807            self.check_expr_coercible_to_type(e, ty, None);
1808            ty
1809        });
1810
1811        let tuple = Ty::new_tup_from_iter(self.tcx, elements);
1812
1813        if let Err(guar) = tuple.error_reported() {
1814            Ty::new_error(self.tcx, guar)
1815        } else {
1816            self.require_type_is_sized(
1817                tuple,
1818                expr.span,
1819                ObligationCauseCode::TupleInitializerSized,
1820            );
1821            tuple
1822        }
1823    }
1824
1825    fn check_expr_struct(
1826        &self,
1827        expr: &hir::Expr<'tcx>,
1828        expected: Expectation<'tcx>,
1829        qpath: &'tcx QPath<'tcx>,
1830        fields: &'tcx [hir::ExprField<'tcx>],
1831        base_expr: &'tcx hir::StructTailExpr<'tcx>,
1832    ) -> Ty<'tcx> {
1833        // Find the relevant variant
1834        let (variant, adt_ty) = match self.check_struct_path(qpath, expr.hir_id) {
1835            Ok(data) => data,
1836            Err(guar) => {
1837                self.check_struct_fields_on_error(fields, base_expr);
1838                return Ty::new_error(self.tcx, guar);
1839            }
1840        };
1841
1842        // Prohibit struct expressions when non-exhaustive flag is set.
1843        let adt = adt_ty.ty_adt_def().expect("`check_struct_path` returned non-ADT type");
1844        if variant.field_list_has_applicable_non_exhaustive() {
1845            self.dcx()
1846                .emit_err(StructExprNonExhaustive { span: expr.span, what: adt.variant_descr() });
1847        }
1848
1849        self.check_expr_struct_fields(
1850            adt_ty,
1851            expected,
1852            expr,
1853            qpath.span(),
1854            variant,
1855            fields,
1856            base_expr,
1857        );
1858
1859        self.require_type_is_sized(adt_ty, expr.span, ObligationCauseCode::StructInitializerSized);
1860        adt_ty
1861    }
1862
1863    fn check_expr_struct_fields(
1864        &self,
1865        adt_ty: Ty<'tcx>,
1866        expected: Expectation<'tcx>,
1867        expr: &hir::Expr<'_>,
1868        path_span: Span,
1869        variant: &'tcx ty::VariantDef,
1870        hir_fields: &'tcx [hir::ExprField<'tcx>],
1871        base_expr: &'tcx hir::StructTailExpr<'tcx>,
1872    ) {
1873        let tcx = self.tcx;
1874
1875        let adt_ty = self.try_structurally_resolve_type(path_span, adt_ty);
1876        let adt_ty_hint = expected.only_has_type(self).and_then(|expected| {
1877            self.fudge_inference_if_ok(|| {
1878                let ocx = ObligationCtxt::new(self);
1879                ocx.sup(&self.misc(path_span), self.param_env, expected, adt_ty)?;
1880                if !ocx.try_evaluate_obligations().is_empty() {
1881                    return Err(TypeError::Mismatch);
1882                }
1883                Ok(self.resolve_vars_if_possible(adt_ty))
1884            })
1885            .ok()
1886        });
1887        if let Some(adt_ty_hint) = adt_ty_hint {
1888            // re-link the variables that the fudging above can create.
1889            self.demand_eqtype(path_span, adt_ty_hint, adt_ty);
1890        }
1891
1892        let ty::Adt(adt, args) = adt_ty.kind() else {
1893            ::rustc_middle::util::bug::span_bug_fmt(path_span,
    format_args!("non-ADT passed to check_expr_struct_fields"));span_bug!(path_span, "non-ADT passed to check_expr_struct_fields");
1894        };
1895        let adt_kind = adt.adt_kind();
1896
1897        let mut remaining_fields = variant
1898            .fields
1899            .iter_enumerated()
1900            .map(|(i, field)| (field.ident(tcx).normalize_to_macros_2_0(), (i, field)))
1901            .collect::<UnordMap<_, _>>();
1902
1903        let mut seen_fields = FxHashMap::default();
1904
1905        let mut error_happened = false;
1906
1907        if variant.fields.len() != remaining_fields.len() {
1908            // Some field is defined more than once. Make sure we don't try to
1909            // instantiate this struct in static/const context.
1910            let guar =
1911                self.dcx().span_delayed_bug(expr.span, "struct fields have non-unique names");
1912            self.set_tainted_by_errors(guar);
1913            error_happened = true;
1914        }
1915
1916        // Type-check each field.
1917        for (idx, field) in hir_fields.iter().enumerate() {
1918            let ident = tcx.adjust_ident(field.ident, variant.def_id);
1919            let field_type = if let Some((i, v_field)) = remaining_fields.remove(&ident) {
1920                seen_fields.insert(ident, field.span);
1921                self.write_field_index(field.hir_id, i);
1922
1923                // We don't look at stability attributes on
1924                // struct-like enums (yet...), but it's definitely not
1925                // a bug to have constructed one.
1926                if adt_kind != AdtKind::Enum {
1927                    tcx.check_stability(v_field.did, Some(field.hir_id), field.span, None);
1928                }
1929
1930                self.field_ty(field.span, v_field, args)
1931            } else {
1932                error_happened = true;
1933                let guar = if let Some(prev_span) = seen_fields.get(&ident) {
1934                    self.dcx().emit_err(FieldMultiplySpecifiedInInitializer {
1935                        span: field.ident.span,
1936                        prev_span: *prev_span,
1937                        ident,
1938                    })
1939                } else {
1940                    self.report_unknown_field(
1941                        adt_ty,
1942                        variant,
1943                        expr,
1944                        field,
1945                        hir_fields,
1946                        adt.variant_descr(),
1947                    )
1948                };
1949
1950                Ty::new_error(tcx, guar)
1951            };
1952
1953            // Check that the expected field type is WF. Otherwise, we emit no use-site error
1954            // in the case of coercions for non-WF fields, which leads to incorrect error
1955            // tainting. See issue #126272.
1956            self.register_wf_obligation(
1957                field_type.into(),
1958                field.expr.span,
1959                ObligationCauseCode::WellFormed(None),
1960            );
1961
1962            // Make sure to give a type to the field even if there's
1963            // an error, so we can continue type-checking.
1964            let ty = self.check_expr_with_hint(field.expr, field_type);
1965            let diag = self.demand_coerce_diag(field.expr, ty, field_type, None, AllowTwoPhase::No);
1966
1967            if let Err(diag) = diag {
1968                if idx == hir_fields.len() - 1 {
1969                    if remaining_fields.is_empty() {
1970                        self.suggest_fru_from_range_and_emit(field, variant, args, diag);
1971                    } else {
1972                        diag.stash(field.span, StashKey::MaybeFruTypo);
1973                    }
1974                } else {
1975                    diag.emit();
1976                }
1977            }
1978        }
1979
1980        // Make sure the programmer specified correct number of fields.
1981        if adt_kind == AdtKind::Union && hir_fields.len() != 1 {
1982            {
    self.dcx().struct_span_err(path_span,
            ::alloc::__export::must_use({
                    ::alloc::fmt::format(format_args!("union expressions should have exactly one field"))
                })).with_code(E0784)
}struct_span_code_err!(
1983                self.dcx(),
1984                path_span,
1985                E0784,
1986                "union expressions should have exactly one field",
1987            )
1988            .emit();
1989        }
1990
1991        // If check_expr_struct_fields hit an error, do not attempt to populate
1992        // the fields with the base_expr. This could cause us to hit errors later
1993        // when certain fields are assumed to exist that in fact do not.
1994        if error_happened {
1995            if let hir::StructTailExpr::Base(base_expr) = base_expr {
1996                self.check_expr(base_expr);
1997            }
1998            return;
1999        }
2000
2001        match *base_expr {
2002            hir::StructTailExpr::DefaultFields(span) => {
2003                let mut missing_mandatory_fields = Vec::new();
2004                let mut missing_optional_fields = Vec::new();
2005                for f in &variant.fields {
2006                    let ident = self.tcx.adjust_ident(f.ident(self.tcx), variant.def_id);
2007                    if let Some(_) = remaining_fields.remove(&ident) {
2008                        if f.value.is_none() {
2009                            missing_mandatory_fields.push(ident);
2010                        } else {
2011                            missing_optional_fields.push(ident);
2012                        }
2013                    }
2014                }
2015                if !self.tcx.features().default_field_values() {
2016                    let sugg = self.tcx.crate_level_attribute_injection_span();
2017                    self.dcx().emit_err(BaseExpressionDoubleDot {
2018                        span: span.shrink_to_hi(),
2019                        // We only mention enabling the feature if this is a nightly rustc *and* the
2020                        // expression would make sense with the feature enabled.
2021                        default_field_values_suggestion: if self.tcx.sess.is_nightly_build()
2022                            && missing_mandatory_fields.is_empty()
2023                            && !missing_optional_fields.is_empty()
2024                        {
2025                            Some(sugg)
2026                        } else {
2027                            None
2028                        },
2029                        add_expr: if !missing_mandatory_fields.is_empty()
2030                            || !missing_optional_fields.is_empty()
2031                        {
2032                            Some(BaseExpressionDoubleDotAddExpr { span: span.shrink_to_hi() })
2033                        } else {
2034                            None
2035                        },
2036                        remove_dots: if missing_mandatory_fields.is_empty()
2037                            && missing_optional_fields.is_empty()
2038                        {
2039                            Some(BaseExpressionDoubleDotRemove { span })
2040                        } else {
2041                            None
2042                        },
2043                    });
2044                    return;
2045                }
2046                if variant.fields.is_empty() {
2047                    let mut err = self.dcx().struct_span_err(
2048                        span,
2049                        ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("`{0}` has no fields, `..` needs at least one default field in the struct definition",
                adt_ty))
    })format!(
2050                            "`{adt_ty}` has no fields, `..` needs at least one default field in \
2051                            the struct definition",
2052                        ),
2053                    );
2054                    err.span_label(path_span, "this type has no fields");
2055                    err.emit();
2056                }
2057                if !missing_mandatory_fields.is_empty() {
2058                    let s = if missing_mandatory_fields.len() == 1 { "" } else { "s" }pluralize!(missing_mandatory_fields.len());
2059                    let fields = listify(&missing_mandatory_fields, |f| ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("`{0}`", f))
    })format!("`{f}`")).unwrap();
2060                    self.dcx()
2061                        .struct_span_err(
2062                            span.shrink_to_lo(),
2063                            ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("missing field{0} {1} in initializer",
                s, fields))
    })format!("missing field{s} {fields} in initializer"),
2064                        )
2065                        .with_span_label(
2066                            span.shrink_to_lo(),
2067                            "fields that do not have a defaulted value must be provided explicitly",
2068                        )
2069                        .emit();
2070                    return;
2071                }
2072                let fru_tys = match adt_ty.kind() {
2073                    ty::Adt(adt, args) if adt.is_struct() => variant
2074                        .fields
2075                        .iter()
2076                        .map(|f| self.normalize(span, f.ty(self.tcx, args)))
2077                        .collect(),
2078                    ty::Adt(adt, args) if adt.is_enum() => variant
2079                        .fields
2080                        .iter()
2081                        .map(|f| self.normalize(span, f.ty(self.tcx, args)))
2082                        .collect(),
2083                    _ => {
2084                        self.dcx().emit_err(FunctionalRecordUpdateOnNonStruct { span });
2085                        return;
2086                    }
2087                };
2088                self.typeck_results.borrow_mut().fru_field_types_mut().insert(expr.hir_id, fru_tys);
2089            }
2090            hir::StructTailExpr::Base(base_expr) => {
2091                // FIXME: We are currently creating two branches here in order to maintain
2092                // consistency. But they should be merged as much as possible.
2093                let fru_tys = if self.tcx.features().type_changing_struct_update() {
2094                    if adt.is_struct() {
2095                        // Make some fresh generic parameters for our ADT type.
2096                        let fresh_args = self.fresh_args_for_item(base_expr.span, adt.did());
2097                        // We do subtyping on the FRU fields first, so we can
2098                        // learn exactly what types we expect the base expr
2099                        // needs constrained to be compatible with the struct
2100                        // type we expect from the expectation value.
2101                        let fru_tys = variant
2102                            .fields
2103                            .iter()
2104                            .map(|f| {
2105                                let fru_ty = self.normalize(
2106                                    expr.span,
2107                                    self.field_ty(base_expr.span, f, fresh_args),
2108                                );
2109                                let ident =
2110                                    self.tcx.adjust_ident(f.ident(self.tcx), variant.def_id);
2111                                if let Some(_) = remaining_fields.remove(&ident) {
2112                                    let target_ty = self.field_ty(base_expr.span, f, args);
2113                                    let cause = self.misc(base_expr.span);
2114                                    match self.at(&cause, self.param_env).sup(
2115                                        // We're already using inference variables for any params,
2116                                        // and don't allow converting between different structs,
2117                                        // so there is no way this ever actually defines an opaque
2118                                        // type. Thus choosing `Yes` is fine.
2119                                        DefineOpaqueTypes::Yes,
2120                                        target_ty,
2121                                        fru_ty,
2122                                    ) {
2123                                        Ok(InferOk { obligations, value: () }) => {
2124                                            self.register_predicates(obligations)
2125                                        }
2126                                        Err(_) => {
2127                                            ::rustc_middle::util::bug::span_bug_fmt(cause.span,
    format_args!("subtyping remaining fields of type changing FRU failed: {2} != {3}: {0}::{1}",
        variant.name, ident.name, target_ty, fru_ty));span_bug!(
2128                                                cause.span,
2129                                                "subtyping remaining fields of type changing FRU \
2130                                                failed: {target_ty} != {fru_ty}: {}::{}",
2131                                                variant.name,
2132                                                ident.name,
2133                                            );
2134                                        }
2135                                    }
2136                                }
2137                                self.resolve_vars_if_possible(fru_ty)
2138                            })
2139                            .collect();
2140                        // The use of fresh args that we have subtyped against
2141                        // our base ADT type's fields allows us to guide inference
2142                        // along so that, e.g.
2143                        // ```
2144                        // MyStruct<'a, F1, F2, const C: usize> {
2145                        //     f: F1,
2146                        //     // Other fields that reference `'a`, `F2`, and `C`
2147                        // }
2148                        //
2149                        // let x = MyStruct {
2150                        //    f: 1usize,
2151                        //    ..other_struct
2152                        // };
2153                        // ```
2154                        // will have the `other_struct` expression constrained to
2155                        // `MyStruct<'a, _, F2, C>`, as opposed to just `_`...
2156                        // This is important to allow coercions to happen in
2157                        // `other_struct` itself. See `coerce-in-base-expr.rs`.
2158                        let fresh_base_ty = Ty::new_adt(self.tcx, *adt, fresh_args);
2159                        self.check_expr_has_type_or_error(
2160                            base_expr,
2161                            self.resolve_vars_if_possible(fresh_base_ty),
2162                            |_| {},
2163                        );
2164                        fru_tys
2165                    } else {
2166                        // Check the base_expr, regardless of a bad expected adt_ty, so we can get
2167                        // type errors on that expression, too.
2168                        self.check_expr(base_expr);
2169                        self.dcx()
2170                            .emit_err(FunctionalRecordUpdateOnNonStruct { span: base_expr.span });
2171                        return;
2172                    }
2173                } else {
2174                    self.check_expr_has_type_or_error(base_expr, adt_ty, |_| {
2175                        let base_ty = self.typeck_results.borrow().expr_ty(base_expr);
2176                        let same_adt = #[allow(non_exhaustive_omitted_patterns)] match (adt_ty.kind(),
        base_ty.kind()) {
    (ty::Adt(adt, _), ty::Adt(base_adt, _)) if adt == base_adt => true,
    _ => false,
}matches!((adt_ty.kind(), base_ty.kind()),
2177                            (ty::Adt(adt, _), ty::Adt(base_adt, _)) if adt == base_adt);
2178                        if self.tcx.sess.is_nightly_build() && same_adt {
2179                            feature_err(
2180                                &self.tcx.sess,
2181                                sym::type_changing_struct_update,
2182                                base_expr.span,
2183                                "type changing struct updating is experimental",
2184                            )
2185                            .emit();
2186                        }
2187                    });
2188                    match adt_ty.kind() {
2189                        ty::Adt(adt, args) if adt.is_struct() => variant
2190                            .fields
2191                            .iter()
2192                            .map(|f| self.normalize(expr.span, f.ty(self.tcx, args)))
2193                            .collect(),
2194                        _ => {
2195                            self.dcx().emit_err(FunctionalRecordUpdateOnNonStruct {
2196                                span: base_expr.span,
2197                            });
2198                            return;
2199                        }
2200                    }
2201                };
2202                self.typeck_results.borrow_mut().fru_field_types_mut().insert(expr.hir_id, fru_tys);
2203            }
2204            rustc_hir::StructTailExpr::NoneWithError(guaranteed) => {
2205                // If parsing the struct recovered from a syntax error, do not report missing
2206                // fields. This prevents spurious errors when a field is intended to be present
2207                // but a preceding syntax error caused it not to be parsed. For example, if a
2208                // struct type `StructName` has fields `foo` and `bar`, then
2209                //     StructName { foo(), bar: 2 }
2210                // will not successfully parse a field `foo`, but we will not mention that,
2211                // since the syntax error has already been reported.
2212
2213                // Signal that type checking has failed, even though we haven’t emitted a diagnostic
2214                // about it ourselves.
2215                self.infcx.set_tainted_by_errors(guaranteed);
2216            }
2217            rustc_hir::StructTailExpr::None => {
2218                if adt_kind != AdtKind::Union
2219                    && !remaining_fields.is_empty()
2220                    //~ non_exhaustive already reported, which will only happen for extern modules
2221                    && !variant.field_list_has_applicable_non_exhaustive()
2222                {
2223                    {
    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/expr.rs:2223",
                        "rustc_hir_typeck::expr", ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/expr.rs"),
                        ::tracing_core::__macro_support::Option::Some(2223u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::expr"),
                        ::tracing_core::field::FieldSet::new(&["remaining_fields"],
                            ::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(&remaining_fields)
                                            as &dyn Value))])
            });
    } else { ; }
};debug!(?remaining_fields);
2224
2225                    // Report missing fields.
2226
2227                    let private_fields: Vec<&ty::FieldDef> = variant
2228                        .fields
2229                        .iter()
2230                        .filter(|field| {
2231                            !field.vis.is_accessible_from(tcx.parent_module(expr.hir_id), tcx)
2232                        })
2233                        .collect();
2234
2235                    if !private_fields.is_empty() {
2236                        self.report_private_fields(
2237                            adt_ty,
2238                            path_span,
2239                            expr.span,
2240                            private_fields,
2241                            hir_fields,
2242                        );
2243                    } else {
2244                        self.report_missing_fields(
2245                            adt_ty,
2246                            path_span,
2247                            expr.span,
2248                            remaining_fields,
2249                            variant,
2250                            hir_fields,
2251                            args,
2252                        );
2253                    }
2254                }
2255            }
2256        }
2257    }
2258
2259    fn check_struct_fields_on_error(
2260        &self,
2261        fields: &'tcx [hir::ExprField<'tcx>],
2262        base_expr: &'tcx hir::StructTailExpr<'tcx>,
2263    ) {
2264        for field in fields {
2265            self.check_expr(field.expr);
2266        }
2267        if let hir::StructTailExpr::Base(base) = *base_expr {
2268            self.check_expr(base);
2269        }
2270    }
2271
2272    /// Report an error for a struct field expression when there are fields which aren't provided.
2273    ///
2274    /// ```text
2275    /// error: missing field `you_can_use_this_field` in initializer of `foo::Foo`
2276    ///  --> src/main.rs:8:5
2277    ///   |
2278    /// 8 |     foo::Foo {};
2279    ///   |     ^^^^^^^^ missing `you_can_use_this_field`
2280    ///
2281    /// error: aborting due to 1 previous error
2282    /// ```
2283    fn report_missing_fields(
2284        &self,
2285        adt_ty: Ty<'tcx>,
2286        span: Span,
2287        full_span: Span,
2288        remaining_fields: UnordMap<Ident, (FieldIdx, &ty::FieldDef)>,
2289        variant: &'tcx ty::VariantDef,
2290        hir_fields: &'tcx [hir::ExprField<'tcx>],
2291        args: GenericArgsRef<'tcx>,
2292    ) {
2293        let len = remaining_fields.len();
2294
2295        let displayable_field_names: Vec<&str> =
2296            remaining_fields.items().map(|(ident, _)| ident.as_str()).into_sorted_stable_ord();
2297
2298        let mut truncated_fields_error = String::new();
2299        let remaining_fields_names = match &displayable_field_names[..] {
2300            [field1] => ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("`{0}`", field1))
    })format!("`{field1}`"),
2301            [field1, field2] => ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("`{0}` and `{1}`", field1, field2))
    })format!("`{field1}` and `{field2}`"),
2302            [field1, field2, field3] => ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("`{0}`, `{1}` and `{2}`", field1,
                field2, field3))
    })format!("`{field1}`, `{field2}` and `{field3}`"),
2303            _ => {
2304                truncated_fields_error =
2305                    ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!(" and {0} other field{1}", len - 3,
                if len - 3 == 1 { "" } else { "s" }))
    })format!(" and {} other field{}", len - 3, pluralize!(len - 3));
2306                displayable_field_names
2307                    .iter()
2308                    .take(3)
2309                    .map(|n| ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("`{0}`", n))
    })format!("`{n}`"))
2310                    .collect::<Vec<_>>()
2311                    .join(", ")
2312            }
2313        };
2314
2315        let mut err = {
    self.dcx().struct_span_err(span,
            ::alloc::__export::must_use({
                    ::alloc::fmt::format(format_args!("missing field{0} {1}{2} in initializer of `{3}`",
                            if len == 1 { "" } else { "s" }, remaining_fields_names,
                            truncated_fields_error, adt_ty))
                })).with_code(E0063)
}struct_span_code_err!(
2316            self.dcx(),
2317            span,
2318            E0063,
2319            "missing field{} {}{} in initializer of `{}`",
2320            pluralize!(len),
2321            remaining_fields_names,
2322            truncated_fields_error,
2323            adt_ty
2324        );
2325        err.span_label(span, ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("missing {0}{1}",
                remaining_fields_names, truncated_fields_error))
    })format!("missing {remaining_fields_names}{truncated_fields_error}"));
2326
2327        if remaining_fields.items().all(|(_, (_, field))| field.value.is_some())
2328            && self.tcx.sess.is_nightly_build()
2329        {
2330            let msg = ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("all remaining fields have default values, {0} use those values with `..`",
                if self.tcx.features().default_field_values() {
                    "you can"
                } else {
                    "if you added `#![feature(default_field_values)]` to your crate you could"
                }))
    })format!(
2331                "all remaining fields have default values, {you_can} use those values with `..`",
2332                you_can = if self.tcx.features().default_field_values() {
2333                    "you can"
2334                } else {
2335                    "if you added `#![feature(default_field_values)]` to your crate you could"
2336                },
2337            );
2338            if let Some(hir_field) = hir_fields.last() {
2339                err.span_suggestion_verbose(
2340                    hir_field.span.shrink_to_hi(),
2341                    msg,
2342                    ", ..".to_string(),
2343                    Applicability::MachineApplicable,
2344                );
2345            } else if hir_fields.is_empty() {
2346                err.span_suggestion_verbose(
2347                    span.shrink_to_hi().with_hi(full_span.hi()),
2348                    msg,
2349                    " { .. }".to_string(),
2350                    Applicability::MachineApplicable,
2351                );
2352            }
2353        }
2354
2355        if let Some(hir_field) = hir_fields.last() {
2356            self.suggest_fru_from_range_and_emit(hir_field, variant, args, err);
2357        } else {
2358            err.emit();
2359        }
2360    }
2361
2362    /// If the last field is a range literal, but it isn't supposed to be, then they probably
2363    /// meant to use functional update syntax.
2364    fn suggest_fru_from_range_and_emit(
2365        &self,
2366        last_expr_field: &hir::ExprField<'tcx>,
2367        variant: &ty::VariantDef,
2368        args: GenericArgsRef<'tcx>,
2369        mut err: Diag<'_>,
2370    ) {
2371        if is_range_literal(last_expr_field.expr)
2372            && let ExprKind::Struct(&qpath, [range_start, range_end], _) = last_expr_field.expr.kind
2373            && self.tcx.qpath_is_lang_item(qpath, LangItem::Range)
2374            && let variant_field =
2375                variant.fields.iter().find(|field| field.ident(self.tcx) == last_expr_field.ident)
2376            && let range_def_id = self.tcx.lang_items().range_struct()
2377            && variant_field
2378                .and_then(|field| field.ty(self.tcx, args).ty_adt_def())
2379                .map(|adt| adt.did())
2380                != range_def_id
2381        {
2382            // Use a (somewhat arbitrary) filtering heuristic to avoid printing
2383            // expressions that are either too long, or have control character
2384            // such as newlines in them.
2385            let expr = self
2386                .tcx
2387                .sess
2388                .source_map()
2389                .span_to_snippet(range_end.expr.span)
2390                .ok()
2391                .filter(|s| s.len() < 25 && !s.contains(|c: char| c.is_control()));
2392
2393            let fru_span = self
2394                .tcx
2395                .sess
2396                .source_map()
2397                .span_extend_while_whitespace(range_start.expr.span)
2398                .shrink_to_hi()
2399                .to(range_end.expr.span);
2400
2401            err.subdiagnostic(TypeMismatchFruTypo {
2402                expr_span: range_start.expr.span,
2403                fru_span,
2404                expr,
2405            });
2406
2407            // Suppress any range expr type mismatches
2408            self.dcx().try_steal_replace_and_emit_err(
2409                last_expr_field.span,
2410                StashKey::MaybeFruTypo,
2411                err,
2412            );
2413        } else {
2414            err.emit();
2415        }
2416    }
2417
2418    /// Report an error for a struct field expression when there are invisible fields.
2419    ///
2420    /// ```text
2421    /// error: cannot construct `Foo` with struct literal syntax due to private fields
2422    ///  --> src/main.rs:8:5
2423    ///   |
2424    /// 8 |     foo::Foo {};
2425    ///   |     ^^^^^^^^
2426    ///
2427    /// error: aborting due to 1 previous error
2428    /// ```
2429    fn report_private_fields(
2430        &self,
2431        adt_ty: Ty<'tcx>,
2432        span: Span,
2433        expr_span: Span,
2434        private_fields: Vec<&ty::FieldDef>,
2435        used_fields: &'tcx [hir::ExprField<'tcx>],
2436    ) {
2437        let mut err =
2438            self.dcx().struct_span_err(
2439                span,
2440                ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("cannot construct `{0}` with struct literal syntax due to private fields",
                adt_ty))
    })format!(
2441                    "cannot construct `{adt_ty}` with struct literal syntax due to private fields",
2442                ),
2443            );
2444        let (used_private_fields, remaining_private_fields): (
2445            Vec<(Symbol, Span, bool)>,
2446            Vec<(Symbol, Span, bool)>,
2447        ) = private_fields
2448            .iter()
2449            .map(|field| {
2450                match used_fields.iter().find(|used_field| field.name == used_field.ident.name) {
2451                    Some(used_field) => (field.name, used_field.span, true),
2452                    None => (field.name, self.tcx.def_span(field.did), false),
2453                }
2454            })
2455            .partition(|field| field.2);
2456        err.span_labels(used_private_fields.iter().map(|(_, span, _)| *span), "private field");
2457        if !remaining_private_fields.is_empty() {
2458            let names = if remaining_private_fields.len() > 6 {
2459                String::new()
2460            } else {
2461                ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("{0} ",
                listify(&remaining_private_fields,
                        |(name, _, _)|
                            ::alloc::__export::must_use({
                                    ::alloc::fmt::format(format_args!("`{0}`", name))
                                })).expect("expected at least one private field to report")))
    })format!(
2462                    "{} ",
2463                    listify(&remaining_private_fields, |(name, _, _)| format!("`{name}`"))
2464                        .expect("expected at least one private field to report")
2465                )
2466            };
2467            err.note(::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("{0}private field{1} {3}that {2} not provided",
                if used_fields.is_empty() { "" } else { "...and other " },
                if remaining_private_fields.len() == 1 { "" } else { "s" },
                if remaining_private_fields.len() == 1 {
                    "was"
                } else { "were" }, names))
    })format!(
2468                "{}private field{s} {names}that {were} not provided",
2469                if used_fields.is_empty() { "" } else { "...and other " },
2470                s = pluralize!(remaining_private_fields.len()),
2471                were = pluralize!("was", remaining_private_fields.len()),
2472            ));
2473        }
2474
2475        if let ty::Adt(def, _) = adt_ty.kind() {
2476            let def_id = def.did();
2477            let mut items = self
2478                .tcx
2479                .inherent_impls(def_id)
2480                .into_iter()
2481                .flat_map(|&i| self.tcx.associated_items(i).in_definition_order())
2482                // Only assoc fn with no receivers.
2483                .filter(|item| item.is_fn() && !item.is_method())
2484                .filter_map(|item| {
2485                    // Only assoc fns that return `Self`
2486                    let fn_sig = self
2487                        .tcx
2488                        .fn_sig(item.def_id)
2489                        .instantiate(self.tcx, self.fresh_args_for_item(span, item.def_id));
2490                    let ret_ty = self.tcx.instantiate_bound_regions_with_erased(fn_sig.output());
2491                    if !self.can_eq(self.param_env, ret_ty, adt_ty) {
2492                        return None;
2493                    }
2494                    let input_len = fn_sig.inputs().skip_binder().len();
2495                    let name = item.name();
2496                    let order = !name.as_str().starts_with("new");
2497                    Some((order, name, input_len))
2498                })
2499                .collect::<Vec<_>>();
2500            items.sort_by_key(|(order, _, _)| *order);
2501            let suggestion = |name, args| {
2502                ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("::{1}({0})",
                std::iter::repeat_n("_", args).collect::<Vec<_>>().join(", "),
                name))
    })format!(
2503                    "::{name}({})",
2504                    std::iter::repeat_n("_", args).collect::<Vec<_>>().join(", ")
2505                )
2506            };
2507            match &items[..] {
2508                [] => {}
2509                [(_, name, args)] => {
2510                    err.span_suggestion_verbose(
2511                        span.shrink_to_hi().with_hi(expr_span.hi()),
2512                        ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("you might have meant to use the `{0}` associated function",
                name))
    })format!("you might have meant to use the `{name}` associated function"),
2513                        suggestion(name, *args),
2514                        Applicability::MaybeIncorrect,
2515                    );
2516                }
2517                _ => {
2518                    err.span_suggestions(
2519                        span.shrink_to_hi().with_hi(expr_span.hi()),
2520                        "you might have meant to use an associated function to build this type",
2521                        items.iter().map(|(_, name, args)| suggestion(name, *args)),
2522                        Applicability::MaybeIncorrect,
2523                    );
2524                }
2525            }
2526            if let Some(default_trait) = self.tcx.get_diagnostic_item(sym::Default)
2527                && self
2528                    .infcx
2529                    .type_implements_trait(default_trait, [adt_ty], self.param_env)
2530                    .may_apply()
2531            {
2532                err.multipart_suggestion(
2533                    "consider using the `Default` trait",
2534                    ::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
        [(span.shrink_to_lo(), "<".to_string()),
                (span.shrink_to_hi().with_hi(expr_span.hi()),
                    " as std::default::Default>::default()".to_string())]))vec![
2535                        (span.shrink_to_lo(), "<".to_string()),
2536                        (
2537                            span.shrink_to_hi().with_hi(expr_span.hi()),
2538                            " as std::default::Default>::default()".to_string(),
2539                        ),
2540                    ],
2541                    Applicability::MaybeIncorrect,
2542                );
2543            }
2544        }
2545
2546        err.emit();
2547    }
2548
2549    fn report_unknown_field(
2550        &self,
2551        ty: Ty<'tcx>,
2552        variant: &'tcx ty::VariantDef,
2553        expr: &hir::Expr<'_>,
2554        field: &hir::ExprField<'_>,
2555        skip_fields: &[hir::ExprField<'_>],
2556        kind_name: &str,
2557    ) -> ErrorGuaranteed {
2558        // we don't care to report errors for a struct if the struct itself is tainted
2559        if let Err(guar) = variant.has_errors() {
2560            return guar;
2561        }
2562        let mut err = self.err_ctxt().type_error_struct_with_diag(
2563            field.ident.span,
2564            |actual| match ty.kind() {
2565                ty::Adt(adt, ..) if adt.is_enum() => {
    self.dcx().struct_span_err(field.ident.span,
            ::alloc::__export::must_use({
                    ::alloc::fmt::format(format_args!("{0} `{1}::{2}` has no field named `{3}`",
                            kind_name, actual, variant.name, field.ident))
                })).with_code(E0559)
}struct_span_code_err!(
2566                    self.dcx(),
2567                    field.ident.span,
2568                    E0559,
2569                    "{} `{}::{}` has no field named `{}`",
2570                    kind_name,
2571                    actual,
2572                    variant.name,
2573                    field.ident
2574                ),
2575                _ => {
    self.dcx().struct_span_err(field.ident.span,
            ::alloc::__export::must_use({
                    ::alloc::fmt::format(format_args!("{0} `{1}` has no field named `{2}`",
                            kind_name, actual, field.ident))
                })).with_code(E0560)
}struct_span_code_err!(
2576                    self.dcx(),
2577                    field.ident.span,
2578                    E0560,
2579                    "{} `{}` has no field named `{}`",
2580                    kind_name,
2581                    actual,
2582                    field.ident
2583                ),
2584            },
2585            ty,
2586        );
2587
2588        let variant_ident_span = self.tcx.def_ident_span(variant.def_id).unwrap();
2589        match variant.ctor {
2590            Some((CtorKind::Fn, def_id)) => match ty.kind() {
2591                ty::Adt(adt, ..) if adt.is_enum() => {
2592                    err.span_label(
2593                        variant_ident_span,
2594                        ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("`{0}::{1}` defined here", ty,
                variant.name))
    })format!(
2595                            "`{adt}::{variant}` defined here",
2596                            adt = ty,
2597                            variant = variant.name,
2598                        ),
2599                    );
2600                    err.span_label(field.ident.span, "field does not exist");
2601                    let fn_sig = self.tcx.fn_sig(def_id).instantiate_identity();
2602                    let inputs = fn_sig.inputs().skip_binder();
2603                    let fields = ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("({0})",
                inputs.iter().map(|i|
                                ::alloc::__export::must_use({
                                        ::alloc::fmt::format(format_args!("/* {0} */", i))
                                    })).collect::<Vec<_>>().join(", ")))
    })format!(
2604                        "({})",
2605                        inputs.iter().map(|i| format!("/* {i} */")).collect::<Vec<_>>().join(", ")
2606                    );
2607                    let (replace_span, sugg) = match expr.kind {
2608                        hir::ExprKind::Struct(qpath, ..) => {
2609                            (qpath.span().shrink_to_hi().with_hi(expr.span.hi()), fields)
2610                        }
2611                        _ => {
2612                            (expr.span, ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("{1}::{0}{2}", variant.name, ty,
                fields))
    })format!("{ty}::{variant}{fields}", variant = variant.name))
2613                        }
2614                    };
2615                    err.span_suggestion_verbose(
2616                        replace_span,
2617                        ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("`{0}::{1}` is a tuple {2}, use the appropriate syntax",
                ty, variant.name, kind_name))
    })format!(
2618                            "`{adt}::{variant}` is a tuple {kind_name}, use the appropriate syntax",
2619                            adt = ty,
2620                            variant = variant.name,
2621                        ),
2622                        sugg,
2623                        Applicability::HasPlaceholders,
2624                    );
2625                }
2626                _ => {
2627                    err.span_label(variant_ident_span, ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("`{0}` defined here", ty))
    })format!("`{ty}` defined here"));
2628                    err.span_label(field.ident.span, "field does not exist");
2629                    let fn_sig = self.tcx.fn_sig(def_id).instantiate_identity();
2630                    let inputs = fn_sig.inputs().skip_binder();
2631                    let fields = ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("({0})",
                inputs.iter().map(|i|
                                ::alloc::__export::must_use({
                                        ::alloc::fmt::format(format_args!("/* {0} */", i))
                                    })).collect::<Vec<_>>().join(", ")))
    })format!(
2632                        "({})",
2633                        inputs.iter().map(|i| format!("/* {i} */")).collect::<Vec<_>>().join(", ")
2634                    );
2635                    err.span_suggestion_verbose(
2636                        expr.span,
2637                        ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("`{0}` is a tuple {1}, use the appropriate syntax",
                ty, kind_name))
    })format!("`{ty}` is a tuple {kind_name}, use the appropriate syntax",),
2638                        ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("{0}{1}", ty, fields))
    })format!("{ty}{fields}"),
2639                        Applicability::HasPlaceholders,
2640                    );
2641                }
2642            },
2643            _ => {
2644                // prevent all specified fields from being suggested
2645                let available_field_names = self.available_field_names(variant, expr, skip_fields);
2646                if let Some(field_name) =
2647                    find_best_match_for_name(&available_field_names, field.ident.name, None)
2648                    && !(field.ident.name.as_str().parse::<usize>().is_ok()
2649                        && field_name.as_str().parse::<usize>().is_ok())
2650                {
2651                    err.span_label(field.ident.span, "unknown field");
2652                    err.span_suggestion_verbose(
2653                        field.ident.span,
2654                        "a field with a similar name exists",
2655                        field_name,
2656                        Applicability::MaybeIncorrect,
2657                    );
2658                } else {
2659                    match ty.kind() {
2660                        ty::Adt(adt, ..) => {
2661                            if adt.is_enum() {
2662                                err.span_label(
2663                                    field.ident.span,
2664                                    ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("`{0}::{1}` does not have this field",
                ty, variant.name))
    })format!("`{}::{}` does not have this field", ty, variant.name),
2665                                );
2666                            } else {
2667                                err.span_label(
2668                                    field.ident.span,
2669                                    ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("`{0}` does not have this field",
                ty))
    })format!("`{ty}` does not have this field"),
2670                                );
2671                            }
2672                            if available_field_names.is_empty() {
2673                                err.note("all struct fields are already assigned");
2674                            } else {
2675                                err.note(::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("available fields are: {0}",
                self.name_series_display(available_field_names)))
    })format!(
2676                                    "available fields are: {}",
2677                                    self.name_series_display(available_field_names)
2678                                ));
2679                            }
2680                        }
2681                        _ => ::rustc_middle::util::bug::bug_fmt(format_args!("non-ADT passed to report_unknown_field"))bug!("non-ADT passed to report_unknown_field"),
2682                    }
2683                };
2684            }
2685        }
2686        err.emit()
2687    }
2688
2689    fn available_field_names(
2690        &self,
2691        variant: &'tcx ty::VariantDef,
2692        expr: &hir::Expr<'_>,
2693        skip_fields: &[hir::ExprField<'_>],
2694    ) -> Vec<Symbol> {
2695        variant
2696            .fields
2697            .iter()
2698            .filter(|field| {
2699                skip_fields.iter().all(|&skip| skip.ident.name != field.name)
2700                    && self.is_field_suggestable(field, expr.hir_id, expr.span)
2701            })
2702            .map(|field| field.name)
2703            .collect()
2704    }
2705
2706    fn name_series_display(&self, names: Vec<Symbol>) -> String {
2707        // dynamic limit, to never omit just one field
2708        let limit = if names.len() == 6 { 6 } else { 5 };
2709        let mut display =
2710            names.iter().take(limit).map(|n| ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("`{0}`", n))
    })format!("`{n}`")).collect::<Vec<_>>().join(", ");
2711        if names.len() > limit {
2712            display = ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("{0} ... and {1} others", display,
                names.len() - limit))
    })format!("{} ... and {} others", display, names.len() - limit);
2713        }
2714        display
2715    }
2716
2717    /// Find the position of a field named `ident` in `base_def`, accounting for unnammed fields.
2718    /// Return whether such a field has been found. The path to it is stored in `nested_fields`.
2719    /// `ident` must have been adjusted beforehand.
2720    fn find_adt_field(
2721        &self,
2722        base_def: ty::AdtDef<'tcx>,
2723        ident: Ident,
2724    ) -> Option<(FieldIdx, &'tcx ty::FieldDef)> {
2725        // No way to find a field in an enum.
2726        if base_def.is_enum() {
2727            return None;
2728        }
2729
2730        for (field_idx, field) in base_def.non_enum_variant().fields.iter_enumerated() {
2731            if field.ident(self.tcx).normalize_to_macros_2_0() == ident {
2732                // We found the field we wanted.
2733                return Some((field_idx, field));
2734            }
2735        }
2736
2737        None
2738    }
2739
2740    /// Check field access expressions, this works for both structs and tuples.
2741    /// Returns the Ty of the field.
2742    ///
2743    /// ```ignore (illustrative)
2744    /// base.field
2745    /// ^^^^^^^^^^ expr
2746    /// ^^^^       base
2747    ///      ^^^^^ field
2748    /// ```
2749    fn check_expr_field(
2750        &self,
2751        expr: &'tcx hir::Expr<'tcx>,
2752        base: &'tcx hir::Expr<'tcx>,
2753        field: Ident,
2754        // The expected type hint of the field.
2755        expected: Expectation<'tcx>,
2756    ) -> Ty<'tcx> {
2757        {
    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/expr.rs:2757",
                        "rustc_hir_typeck::expr", ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/expr.rs"),
                        ::tracing_core::__macro_support::Option::Some(2757u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::expr"),
                        ::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!("check_field(expr: {0:?}, base: {1:?}, field: {2:?})",
                                                    expr, base, field) as &dyn Value))])
            });
    } else { ; }
};debug!("check_field(expr: {:?}, base: {:?}, field: {:?})", expr, base, field);
2758        let base_ty = self.check_expr(base);
2759        let base_ty = self.structurally_resolve_type(base.span, base_ty);
2760
2761        // Whether we are trying to access a private field. Used for error reporting.
2762        let mut private_candidate = None;
2763
2764        // Field expressions automatically deref
2765        let mut autoderef = self.autoderef(expr.span, base_ty);
2766        while let Some((deref_base_ty, _)) = autoderef.next() {
2767            {
    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/expr.rs:2767",
                        "rustc_hir_typeck::expr", ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/expr.rs"),
                        ::tracing_core::__macro_support::Option::Some(2767u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::expr"),
                        ::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!("deref_base_ty: {0:?}",
                                                    deref_base_ty) as &dyn Value))])
            });
    } else { ; }
};debug!("deref_base_ty: {:?}", deref_base_ty);
2768            match deref_base_ty.kind() {
2769                ty::Adt(base_def, args) if !base_def.is_enum() => {
2770                    {
    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/expr.rs:2770",
                        "rustc_hir_typeck::expr", ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/expr.rs"),
                        ::tracing_core::__macro_support::Option::Some(2770u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::expr"),
                        ::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!("struct named {0:?}",
                                                    deref_base_ty) as &dyn Value))])
            });
    } else { ; }
};debug!("struct named {:?}", deref_base_ty);
2771                    // we don't care to report errors for a struct if the struct itself is tainted
2772                    if let Err(guar) = base_def.non_enum_variant().has_errors() {
2773                        return Ty::new_error(self.tcx(), guar);
2774                    }
2775
2776                    let fn_body_hir_id = self.tcx.local_def_id_to_hir_id(self.body_id);
2777                    let (ident, def_scope) =
2778                        self.tcx.adjust_ident_and_get_scope(field, base_def.did(), fn_body_hir_id);
2779
2780                    if let Some((idx, field)) = self.find_adt_field(*base_def, ident) {
2781                        self.write_field_index(expr.hir_id, idx);
2782
2783                        let adjustments = self.adjust_steps(&autoderef);
2784                        if field.vis.is_accessible_from(def_scope, self.tcx) {
2785                            self.apply_adjustments(base, adjustments);
2786                            self.register_predicates(autoderef.into_obligations());
2787
2788                            self.tcx.check_stability(field.did, Some(expr.hir_id), expr.span, None);
2789                            return self.field_ty(expr.span, field, args);
2790                        }
2791
2792                        // The field is not accessible, fall through to error reporting.
2793                        private_candidate = Some((adjustments, base_def.did()));
2794                    }
2795                }
2796                ty::Tuple(tys) => {
2797                    if let Ok(index) = field.as_str().parse::<usize>() {
2798                        if field.name == sym::integer(index) {
2799                            if let Some(&field_ty) = tys.get(index) {
2800                                let adjustments = self.adjust_steps(&autoderef);
2801                                self.apply_adjustments(base, adjustments);
2802                                self.register_predicates(autoderef.into_obligations());
2803
2804                                self.write_field_index(expr.hir_id, FieldIdx::from_usize(index));
2805                                return field_ty;
2806                            }
2807                        }
2808                    }
2809                }
2810                _ => {}
2811            }
2812        }
2813        // We failed to check the expression, report an error.
2814
2815        // Emits an error if we deref an infer variable, like calling `.field` on a base type
2816        // of `&_`. We can also use this to suppress unnecessary "missing field" errors that
2817        // will follow ambiguity errors.
2818        let final_ty = self.structurally_resolve_type(autoderef.span(), autoderef.final_ty());
2819        if let ty::Error(_) = final_ty.kind() {
2820            return final_ty;
2821        }
2822
2823        if let Some((adjustments, did)) = private_candidate {
2824            // (#90483) apply adjustments to avoid ExprUseVisitor from
2825            // creating erroneous projection.
2826            self.apply_adjustments(base, adjustments);
2827            let guar = self.ban_private_field_access(
2828                expr,
2829                base_ty,
2830                field,
2831                did,
2832                expected.only_has_type(self),
2833            );
2834            return Ty::new_error(self.tcx(), guar);
2835        }
2836
2837        let guar = if self.method_exists_for_diagnostic(
2838            field,
2839            base_ty,
2840            expr.hir_id,
2841            expected.only_has_type(self),
2842        ) {
2843            // If taking a method instead of calling it
2844            self.ban_take_value_of_method(expr, base_ty, field)
2845        } else if !base_ty.is_primitive_ty() {
2846            self.ban_nonexisting_field(field, base, expr, base_ty)
2847        } else {
2848            let field_name = field.to_string();
2849            let mut err = {
    let mut err =
        {
            self.dcx().struct_span_err(field.span,
                    ::alloc::__export::must_use({
                            ::alloc::fmt::format(format_args!("`{0}` is a primitive type and therefore doesn\'t have fields",
                                    base_ty))
                        })).with_code(E0610)
        };
    if base_ty.references_error() { err.downgrade_to_delayed_bug(); }
    err
}type_error_struct!(
2850                self.dcx(),
2851                field.span,
2852                base_ty,
2853                E0610,
2854                "`{base_ty}` is a primitive type and therefore doesn't have fields",
2855            );
2856            let is_valid_suffix = |field: &str| {
2857                if field == "f32" || field == "f64" {
2858                    return true;
2859                }
2860                let mut chars = field.chars().peekable();
2861                match chars.peek() {
2862                    Some('e') | Some('E') => {
2863                        chars.next();
2864                        if let Some(c) = chars.peek()
2865                            && !c.is_numeric()
2866                            && *c != '-'
2867                            && *c != '+'
2868                        {
2869                            return false;
2870                        }
2871                        while let Some(c) = chars.peek() {
2872                            if !c.is_numeric() {
2873                                break;
2874                            }
2875                            chars.next();
2876                        }
2877                    }
2878                    _ => (),
2879                }
2880                let suffix = chars.collect::<String>();
2881                suffix.is_empty() || suffix == "f32" || suffix == "f64"
2882            };
2883            let maybe_partial_suffix = |field: &str| -> Option<&str> {
2884                let first_chars = ['f', 'l'];
2885                if field.len() >= 1
2886                    && field.to_lowercase().starts_with(first_chars)
2887                    && field[1..].chars().all(|c| c.is_ascii_digit())
2888                {
2889                    if field.to_lowercase().starts_with(['f']) { Some("f32") } else { Some("f64") }
2890                } else {
2891                    None
2892                }
2893            };
2894            if let ty::Infer(ty::IntVar(_)) = base_ty.kind()
2895                && let ExprKind::Lit(Spanned {
2896                    node: ast::LitKind::Int(_, ast::LitIntType::Unsuffixed),
2897                    ..
2898                }) = base.kind
2899                && !base.span.from_expansion()
2900            {
2901                if is_valid_suffix(&field_name) {
2902                    err.span_suggestion_verbose(
2903                        field.span.shrink_to_lo(),
2904                        "if intended to be a floating point literal, consider adding a `0` after the period",
2905                        '0',
2906                        Applicability::MaybeIncorrect,
2907                    );
2908                } else if let Some(correct_suffix) = maybe_partial_suffix(&field_name) {
2909                    err.span_suggestion_verbose(
2910                        field.span,
2911                        ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("if intended to be a floating point literal, consider adding a `0` after the period and a `{0}` suffix",
                correct_suffix))
    })format!("if intended to be a floating point literal, consider adding a `0` after the period and a `{correct_suffix}` suffix"),
2912                        ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("0{0}", correct_suffix))
    })format!("0{correct_suffix}"),
2913                        Applicability::MaybeIncorrect,
2914                    );
2915                }
2916            }
2917            err.emit()
2918        };
2919
2920        Ty::new_error(self.tcx(), guar)
2921    }
2922
2923    fn suggest_await_on_field_access(
2924        &self,
2925        err: &mut Diag<'_>,
2926        field_ident: Ident,
2927        base: &'tcx hir::Expr<'tcx>,
2928        ty: Ty<'tcx>,
2929    ) {
2930        let Some(output_ty) = self.err_ctxt().get_impl_future_output_ty(ty) else {
2931            err.span_label(field_ident.span, "unknown field");
2932            return;
2933        };
2934        let ty::Adt(def, _) = output_ty.kind() else {
2935            err.span_label(field_ident.span, "unknown field");
2936            return;
2937        };
2938        // no field access on enum type
2939        if def.is_enum() {
2940            err.span_label(field_ident.span, "unknown field");
2941            return;
2942        }
2943        if !def.non_enum_variant().fields.iter().any(|field| field.ident(self.tcx) == field_ident) {
2944            err.span_label(field_ident.span, "unknown field");
2945            return;
2946        }
2947        err.span_label(
2948            field_ident.span,
2949            "field not available in `impl Future`, but it is available in its `Output`",
2950        );
2951        match self.tcx.coroutine_kind(self.body_id) {
2952            Some(hir::CoroutineKind::Desugared(hir::CoroutineDesugaring::Async, _)) => {
2953                err.span_suggestion_verbose(
2954                    base.span.shrink_to_hi(),
2955                    "consider `await`ing on the `Future` to access the field",
2956                    ".await",
2957                    Applicability::MaybeIncorrect,
2958                );
2959            }
2960            _ => {
2961                let mut span: MultiSpan = base.span.into();
2962                span.push_span_label(self.tcx.def_span(self.body_id), "this is not `async`");
2963                err.span_note(
2964                    span,
2965                    "this implements `Future` and its output type has the field, \
2966                    but the future cannot be awaited in a synchronous function",
2967                );
2968            }
2969        }
2970    }
2971
2972    fn ban_nonexisting_field(
2973        &self,
2974        ident: Ident,
2975        base: &'tcx hir::Expr<'tcx>,
2976        expr: &'tcx hir::Expr<'tcx>,
2977        base_ty: Ty<'tcx>,
2978    ) -> ErrorGuaranteed {
2979        {
    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/expr.rs:2979",
                        "rustc_hir_typeck::expr", ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/expr.rs"),
                        ::tracing_core::__macro_support::Option::Some(2979u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::expr"),
                        ::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!("ban_nonexisting_field: field={0:?}, base={1:?}, expr={2:?}, base_ty={3:?}",
                                                    ident, base, expr, base_ty) as &dyn Value))])
            });
    } else { ; }
};debug!(
2980            "ban_nonexisting_field: field={:?}, base={:?}, expr={:?}, base_ty={:?}",
2981            ident, base, expr, base_ty
2982        );
2983        let mut err = self.no_such_field_err(ident, base_ty, expr);
2984
2985        match *base_ty.peel_refs().kind() {
2986            ty::Array(_, len) => {
2987                self.maybe_suggest_array_indexing(&mut err, base, ident, len);
2988            }
2989            ty::RawPtr(..) => {
2990                self.suggest_first_deref_field(&mut err, base, ident);
2991            }
2992            ty::Param(param_ty) => {
2993                err.span_label(ident.span, "unknown field");
2994                self.point_at_param_definition(&mut err, param_ty);
2995            }
2996            ty::Alias(ty::Opaque, _) => {
2997                self.suggest_await_on_field_access(&mut err, ident, base, base_ty.peel_refs());
2998            }
2999            _ => {
3000                err.span_label(ident.span, "unknown field");
3001            }
3002        }
3003
3004        self.suggest_fn_call(&mut err, base, base_ty, |output_ty| {
3005            if let ty::Adt(def, _) = output_ty.kind()
3006                && !def.is_enum()
3007            {
3008                def.non_enum_variant().fields.iter().any(|field| {
3009                    field.ident(self.tcx) == ident
3010                        && field.vis.is_accessible_from(expr.hir_id.owner.def_id, self.tcx)
3011                })
3012            } else if let ty::Tuple(tys) = output_ty.kind()
3013                && let Ok(idx) = ident.as_str().parse::<usize>()
3014            {
3015                idx < tys.len()
3016            } else {
3017                false
3018            }
3019        });
3020
3021        if ident.name == kw::Await {
3022            // We know by construction that `<expr>.await` is either on Rust 2015
3023            // or results in `ExprKind::Await`. Suggest switching the edition to 2018.
3024            err.note("to `.await` a `Future`, switch to Rust 2018 or later");
3025            HelpUseLatestEdition::new().add_to_diag(&mut err);
3026        }
3027
3028        err.emit()
3029    }
3030
3031    fn ban_private_field_access(
3032        &self,
3033        expr: &hir::Expr<'tcx>,
3034        expr_t: Ty<'tcx>,
3035        field: Ident,
3036        base_did: DefId,
3037        return_ty: Option<Ty<'tcx>>,
3038    ) -> ErrorGuaranteed {
3039        let mut err = self.private_field_err(field, base_did);
3040
3041        // Also check if an accessible method exists, which is often what is meant.
3042        if self.method_exists_for_diagnostic(field, expr_t, expr.hir_id, return_ty)
3043            && !self.expr_in_place(expr.hir_id)
3044        {
3045            self.suggest_method_call(
3046                &mut err,
3047                ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("a method `{0}` also exists, call it with parentheses",
                field))
    })format!("a method `{field}` also exists, call it with parentheses"),
3048                field,
3049                expr_t,
3050                expr,
3051                None,
3052            );
3053        }
3054        err.emit()
3055    }
3056
3057    fn ban_take_value_of_method(
3058        &self,
3059        expr: &hir::Expr<'tcx>,
3060        expr_t: Ty<'tcx>,
3061        field: Ident,
3062    ) -> ErrorGuaranteed {
3063        let mut err = {
    let mut err =
        {
            self.dcx().struct_span_err(field.span,
                    ::alloc::__export::must_use({
                            ::alloc::fmt::format(format_args!("attempted to take value of method `{0}` on type `{1}`",
                                    field, expr_t))
                        })).with_code(E0615)
        };
    if expr_t.references_error() { err.downgrade_to_delayed_bug(); }
    err
}type_error_struct!(
3064            self.dcx(),
3065            field.span,
3066            expr_t,
3067            E0615,
3068            "attempted to take value of method `{field}` on type `{expr_t}`",
3069        );
3070        err.span_label(field.span, "method, not a field");
3071        let expr_is_call =
3072            if let hir::Node::Expr(hir::Expr { kind: ExprKind::Call(callee, _args), .. }) =
3073                self.tcx.parent_hir_node(expr.hir_id)
3074            {
3075                expr.hir_id == callee.hir_id
3076            } else {
3077                false
3078            };
3079        let expr_snippet =
3080            self.tcx.sess.source_map().span_to_snippet(expr.span).unwrap_or_default();
3081        let is_wrapped = expr_snippet.starts_with('(') && expr_snippet.ends_with(')');
3082        let after_open = expr.span.lo() + rustc_span::BytePos(1);
3083        let before_close = expr.span.hi() - rustc_span::BytePos(1);
3084
3085        if expr_is_call && is_wrapped {
3086            err.multipart_suggestion(
3087                "remove wrapping parentheses to call the method",
3088                ::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
        [(expr.span.with_hi(after_open), String::new()),
                (expr.span.with_lo(before_close), String::new())]))vec![
3089                    (expr.span.with_hi(after_open), String::new()),
3090                    (expr.span.with_lo(before_close), String::new()),
3091                ],
3092                Applicability::MachineApplicable,
3093            );
3094        } else if !self.expr_in_place(expr.hir_id) {
3095            // Suggest call parentheses inside the wrapping parentheses
3096            let span = if is_wrapped {
3097                expr.span.with_lo(after_open).with_hi(before_close)
3098            } else {
3099                expr.span
3100            };
3101            self.suggest_method_call(
3102                &mut err,
3103                "use parentheses to call the method",
3104                field,
3105                expr_t,
3106                expr,
3107                Some(span),
3108            );
3109        } else if let ty::RawPtr(ptr_ty, _) = expr_t.kind()
3110            && let ty::Adt(adt_def, _) = ptr_ty.kind()
3111            && let ExprKind::Field(base_expr, _) = expr.kind
3112            && let [variant] = &adt_def.variants().raw
3113            && variant.fields.iter().any(|f| f.ident(self.tcx) == field)
3114        {
3115            err.multipart_suggestion(
3116                "to access the field, dereference first",
3117                ::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
        [(base_expr.span.shrink_to_lo(), "(*".to_string()),
                (base_expr.span.shrink_to_hi(), ")".to_string())]))vec![
3118                    (base_expr.span.shrink_to_lo(), "(*".to_string()),
3119                    (base_expr.span.shrink_to_hi(), ")".to_string()),
3120                ],
3121                Applicability::MaybeIncorrect,
3122            );
3123        } else {
3124            err.help("methods are immutable and cannot be assigned to");
3125        }
3126
3127        // See `StashKey::GenericInFieldExpr` for more info
3128        self.dcx().try_steal_replace_and_emit_err(field.span, StashKey::GenericInFieldExpr, err)
3129    }
3130
3131    fn point_at_param_definition(&self, err: &mut Diag<'_>, param: ty::ParamTy) {
3132        let generics = self.tcx.generics_of(self.body_id);
3133        let generic_param = generics.type_param(param, self.tcx);
3134        if let ty::GenericParamDefKind::Type { synthetic: true, .. } = generic_param.kind {
3135            return;
3136        }
3137        let param_def_id = generic_param.def_id;
3138        let param_hir_id = match param_def_id.as_local() {
3139            Some(x) => self.tcx.local_def_id_to_hir_id(x),
3140            None => return,
3141        };
3142        let param_span = self.tcx.hir_span(param_hir_id);
3143        let param_name = self.tcx.hir_ty_param_name(param_def_id.expect_local());
3144
3145        err.span_label(param_span, ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("type parameter \'{0}\' declared here",
                param_name))
    })format!("type parameter '{param_name}' declared here"));
3146    }
3147
3148    fn maybe_suggest_array_indexing(
3149        &self,
3150        err: &mut Diag<'_>,
3151        base: &hir::Expr<'_>,
3152        field: Ident,
3153        len: ty::Const<'tcx>,
3154    ) {
3155        err.span_label(field.span, "unknown field");
3156        if let (Some(len), Ok(user_index)) = (
3157            self.try_structurally_resolve_const(base.span, len).try_to_target_usize(self.tcx),
3158            field.as_str().parse::<u64>(),
3159        ) {
3160            let help = "instead of using tuple indexing, use array indexing";
3161            let applicability = if len < user_index {
3162                Applicability::MachineApplicable
3163            } else {
3164                Applicability::MaybeIncorrect
3165            };
3166            err.multipart_suggestion(
3167                help,
3168                ::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
        [(base.span.between(field.span), "[".to_string()),
                (field.span.shrink_to_hi(), "]".to_string())]))vec![
3169                    (base.span.between(field.span), "[".to_string()),
3170                    (field.span.shrink_to_hi(), "]".to_string()),
3171                ],
3172                applicability,
3173            );
3174        }
3175    }
3176
3177    fn suggest_first_deref_field(&self, err: &mut Diag<'_>, base: &hir::Expr<'_>, field: Ident) {
3178        err.span_label(field.span, "unknown field");
3179        let val = if let Ok(base) = self.tcx.sess.source_map().span_to_snippet(base.span)
3180            && base.len() < 20
3181        {
3182            ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("`{0}`", base))
    })format!("`{base}`")
3183        } else {
3184            "the value".to_string()
3185        };
3186        err.multipart_suggestion(
3187            ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("{0} is a raw pointer; try dereferencing it",
                val))
    })format!("{val} is a raw pointer; try dereferencing it"),
3188            ::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
        [(base.span.shrink_to_lo(), "(*".into()),
                (base.span.between(field.span),
                    ::alloc::__export::must_use({
                            ::alloc::fmt::format(format_args!(")."))
                        }))]))vec![
3189                (base.span.shrink_to_lo(), "(*".into()),
3190                (base.span.between(field.span), format!(").")),
3191            ],
3192            Applicability::MaybeIncorrect,
3193        );
3194    }
3195
3196    fn no_such_field_err(
3197        &self,
3198        field: Ident,
3199        base_ty: Ty<'tcx>,
3200        expr: &hir::Expr<'tcx>,
3201    ) -> Diag<'_> {
3202        let span = field.span;
3203        {
    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/expr.rs:3203",
                        "rustc_hir_typeck::expr", ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/expr.rs"),
                        ::tracing_core::__macro_support::Option::Some(3203u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::expr"),
                        ::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!("no_such_field_err(span: {0:?}, field: {1:?}, expr_t: {2:?})",
                                                    span, field, base_ty) as &dyn Value))])
            });
    } else { ; }
};debug!("no_such_field_err(span: {:?}, field: {:?}, expr_t: {:?})", span, field, base_ty);
3204
3205        let mut err = self.dcx().create_err(NoFieldOnType { span, ty: base_ty, field });
3206        if base_ty.references_error() {
3207            err.downgrade_to_delayed_bug();
3208        }
3209
3210        if let Some(within_macro_span) = span.within_macro(expr.span, self.tcx.sess.source_map()) {
3211            err.span_label(within_macro_span, "due to this macro variable");
3212        }
3213
3214        // Check if there is an associated function with the same name.
3215        if let Some(def_id) = base_ty.peel_refs().ty_adt_def().map(|d| d.did()) {
3216            for &impl_def_id in self.tcx.inherent_impls(def_id) {
3217                for item in self.tcx.associated_items(impl_def_id).in_definition_order() {
3218                    if let ExprKind::Field(base_expr, _) = expr.kind
3219                        && item.name() == field.name
3220                        && #[allow(non_exhaustive_omitted_patterns)] match item.kind {
    ty::AssocKind::Fn { has_self: false, .. } => true,
    _ => false,
}matches!(item.kind, ty::AssocKind::Fn { has_self: false, .. })
3221                    {
3222                        err.span_label(field.span, "this is an associated function, not a method");
3223                        err.note("found the following associated function; to be used as method, it must have a `self` parameter");
3224                        let impl_ty = self.tcx.type_of(impl_def_id).instantiate_identity();
3225                        err.span_note(
3226                            self.tcx.def_span(item.def_id),
3227                            ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("the candidate is defined in an impl for the type `{0}`",
                impl_ty))
    })format!("the candidate is defined in an impl for the type `{impl_ty}`"),
3228                        );
3229
3230                        let ty_str = match base_ty.peel_refs().kind() {
3231                            ty::Adt(def, args) => self.tcx.def_path_str_with_args(def.did(), args),
3232                            _ => base_ty.peel_refs().to_string(),
3233                        };
3234                        err.multipart_suggestion(
3235                            "use associated function syntax instead",
3236                            ::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
        [(base_expr.span, ty_str),
                (base_expr.span.between(field.span), "::".to_string())]))vec![
3237                                (base_expr.span, ty_str),
3238                                (base_expr.span.between(field.span), "::".to_string()),
3239                            ],
3240                            Applicability::MaybeIncorrect,
3241                        );
3242                        return err;
3243                    }
3244                }
3245            }
3246        }
3247
3248        // try to add a suggestion in case the field is a nested field of a field of the Adt
3249        let mod_id = self.tcx.parent_module(expr.hir_id).to_def_id();
3250        let (ty, unwrap) = if let ty::Adt(def, args) = base_ty.kind()
3251            && (self.tcx.is_diagnostic_item(sym::Result, def.did())
3252                || self.tcx.is_diagnostic_item(sym::Option, def.did()))
3253            && let Some(arg) = args.get(0)
3254            && let Some(ty) = arg.as_type()
3255        {
3256            (ty, "unwrap().")
3257        } else {
3258            (base_ty, "")
3259        };
3260        for found_fields in
3261            self.get_field_candidates_considering_privacy_for_diag(span, ty, mod_id, expr.hir_id)
3262        {
3263            let field_names = found_fields.iter().map(|field| field.0.name).collect::<Vec<_>>();
3264            let mut candidate_fields: Vec<_> = found_fields
3265                .into_iter()
3266                .filter_map(|candidate_field| {
3267                    self.check_for_nested_field_satisfying_condition_for_diag(
3268                        span,
3269                        &|candidate_field, _| candidate_field == field,
3270                        candidate_field,
3271                        ::alloc::vec::Vec::new()vec![],
3272                        mod_id,
3273                        expr.hir_id,
3274                    )
3275                })
3276                .map(|mut field_path| {
3277                    field_path.pop();
3278                    field_path.iter().map(|id| ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("{0}.", id))
    })format!("{}.", id)).collect::<String>()
3279                })
3280                .collect::<Vec<_>>();
3281            candidate_fields.sort();
3282
3283            let len = candidate_fields.len();
3284            // Don't suggest `.field` if the base expr is from a different
3285            // syntax context than the field.
3286            if len > 0 && expr.span.eq_ctxt(field.span) {
3287                err.span_suggestions(
3288                    field.span.shrink_to_lo(),
3289                    ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("{0} of the expressions\' fields {1} a field of the same name",
                if len > 1 { "some" } else { "one" },
                if len > 1 { "have" } else { "has" }))
    })format!(
3290                        "{} of the expressions' fields {} a field of the same name",
3291                        if len > 1 { "some" } else { "one" },
3292                        if len > 1 { "have" } else { "has" },
3293                    ),
3294                    candidate_fields.iter().map(|path| ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("{0}{1}", unwrap, path))
    })format!("{unwrap}{path}")),
3295                    Applicability::MaybeIncorrect,
3296                );
3297            } else if let Some(field_name) =
3298                find_best_match_for_name(&field_names, field.name, None)
3299                && !(field.name.as_str().parse::<usize>().is_ok()
3300                    && field_name.as_str().parse::<usize>().is_ok())
3301            {
3302                err.span_suggestion_verbose(
3303                    field.span,
3304                    "a field with a similar name exists",
3305                    ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("{1}{0}", field_name, unwrap))
    })format!("{unwrap}{}", field_name),
3306                    Applicability::MaybeIncorrect,
3307                );
3308            } else if !field_names.is_empty() {
3309                let is = if field_names.len() == 1 { " is" } else { "s are" };
3310                err.note(
3311                    ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("available field{1}: {0}",
                self.name_series_display(field_names), is))
    })format!("available field{is}: {}", self.name_series_display(field_names),),
3312                );
3313            }
3314        }
3315        err
3316    }
3317
3318    fn private_field_err(&self, field: Ident, base_did: DefId) -> Diag<'_> {
3319        let struct_path = self.tcx().def_path_str(base_did);
3320        let kind_name = self.tcx().def_descr(base_did);
3321        {
    self.dcx().struct_span_err(field.span,
            ::alloc::__export::must_use({
                    ::alloc::fmt::format(format_args!("field `{0}` of {1} `{2}` is private",
                            field, kind_name, struct_path))
                })).with_code(E0616)
}struct_span_code_err!(
3322            self.dcx(),
3323            field.span,
3324            E0616,
3325            "field `{field}` of {kind_name} `{struct_path}` is private",
3326        )
3327        .with_span_label(field.span, "private field")
3328    }
3329
3330    pub(crate) fn get_field_candidates_considering_privacy_for_diag(
3331        &self,
3332        span: Span,
3333        base_ty: Ty<'tcx>,
3334        mod_id: DefId,
3335        hir_id: HirId,
3336    ) -> Vec<Vec<(Ident, Ty<'tcx>)>> {
3337        {
    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/expr.rs:3337",
                        "rustc_hir_typeck::expr", ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/expr.rs"),
                        ::tracing_core::__macro_support::Option::Some(3337u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::expr"),
                        ::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!("get_field_candidates(span: {0:?}, base_t: {1:?}",
                                                    span, base_ty) as &dyn Value))])
            });
    } else { ; }
};debug!("get_field_candidates(span: {:?}, base_t: {:?}", span, base_ty);
3338
3339        let mut autoderef = self.autoderef(span, base_ty).silence_errors();
3340        let deref_chain: Vec<_> = autoderef.by_ref().collect();
3341
3342        // Don't probe if we hit the recursion limit, since it may result in
3343        // quadratic blowup if we then try to further deref the results of this
3344        // function. This is a best-effort method, after all.
3345        if autoderef.reached_recursion_limit() {
3346            return ::alloc::vec::Vec::new()vec![];
3347        }
3348
3349        deref_chain
3350            .into_iter()
3351            .filter_map(move |(base_t, _)| {
3352                match base_t.kind() {
3353                    ty::Adt(base_def, args) if !base_def.is_enum() => {
3354                        let tcx = self.tcx;
3355                        let fields = &base_def.non_enum_variant().fields;
3356                        // Some struct, e.g. some that impl `Deref`, have all private fields
3357                        // because you're expected to deref them to access the _real_ fields.
3358                        // This, for example, will help us suggest accessing a field through a `Box<T>`.
3359                        if fields.iter().all(|field| !field.vis.is_accessible_from(mod_id, tcx)) {
3360                            return None;
3361                        }
3362                        return Some(
3363                            fields
3364                                .iter()
3365                                .filter(move |field| {
3366                                    field.vis.is_accessible_from(mod_id, tcx)
3367                                        && self.is_field_suggestable(field, hir_id, span)
3368                                })
3369                                // For compile-time reasons put a limit on number of fields we search
3370                                .take(100)
3371                                .map(|field_def| {
3372                                    (
3373                                        field_def.ident(self.tcx).normalize_to_macros_2_0(),
3374                                        field_def.ty(self.tcx, args),
3375                                    )
3376                                })
3377                                .collect::<Vec<_>>(),
3378                        );
3379                    }
3380                    ty::Tuple(types) => {
3381                        return Some(
3382                            types
3383                                .iter()
3384                                .enumerate()
3385                                // For compile-time reasons put a limit on number of fields we search
3386                                .take(100)
3387                                .map(|(i, ty)| (Ident::from_str(&i.to_string()), ty))
3388                                .collect::<Vec<_>>(),
3389                        );
3390                    }
3391                    _ => None,
3392                }
3393            })
3394            .collect()
3395    }
3396
3397    /// This method is called after we have encountered a missing field error to recursively
3398    /// search for the field
3399    #[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_for_nested_field_satisfying_condition_for_diag",
                                    "rustc_hir_typeck::expr", ::tracing::Level::DEBUG,
                                    ::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/expr.rs"),
                                    ::tracing_core::__macro_support::Option::Some(3399u32),
                                    ::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::expr"),
                                    ::tracing_core::field::FieldSet::new(&["span",
                                                    "candidate_name", "candidate_ty", "field_path"],
                                        ::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(&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(&candidate_name)
                                                            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(&candidate_ty)
                                                            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(&field_path)
                                                            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: Option<Vec<Ident>> = loop {};
            return __tracing_attr_fake_return;
        }
        {
            if field_path.len() > 3 { return None; }
            field_path.push(candidate_name);
            if matches(candidate_name, candidate_ty) {
                return Some(field_path);
            }
            for nested_fields in
                self.get_field_candidates_considering_privacy_for_diag(span,
                    candidate_ty, mod_id, hir_id) {
                for field in nested_fields {
                    if let Some(field_path) =
                            self.check_for_nested_field_satisfying_condition_for_diag(span,
                                matches, field, field_path.clone(), mod_id, hir_id) {
                        return Some(field_path);
                    }
                }
            }
            None
        }
    }
}#[instrument(skip(self, matches, mod_id, hir_id), level = "debug")]
3400    pub(crate) fn check_for_nested_field_satisfying_condition_for_diag(
3401        &self,
3402        span: Span,
3403        matches: &impl Fn(Ident, Ty<'tcx>) -> bool,
3404        (candidate_name, candidate_ty): (Ident, Ty<'tcx>),
3405        mut field_path: Vec<Ident>,
3406        mod_id: DefId,
3407        hir_id: HirId,
3408    ) -> Option<Vec<Ident>> {
3409        if field_path.len() > 3 {
3410            // For compile-time reasons and to avoid infinite recursion we only check for fields
3411            // up to a depth of three
3412            return None;
3413        }
3414        field_path.push(candidate_name);
3415        if matches(candidate_name, candidate_ty) {
3416            return Some(field_path);
3417        }
3418        for nested_fields in self.get_field_candidates_considering_privacy_for_diag(
3419            span,
3420            candidate_ty,
3421            mod_id,
3422            hir_id,
3423        ) {
3424            // recursively search fields of `candidate_field` if it's a ty::Adt
3425            for field in nested_fields {
3426                if let Some(field_path) = self.check_for_nested_field_satisfying_condition_for_diag(
3427                    span,
3428                    matches,
3429                    field,
3430                    field_path.clone(),
3431                    mod_id,
3432                    hir_id,
3433                ) {
3434                    return Some(field_path);
3435                }
3436            }
3437        }
3438        None
3439    }
3440
3441    fn check_expr_index(
3442        &self,
3443        base: &'tcx hir::Expr<'tcx>,
3444        idx: &'tcx hir::Expr<'tcx>,
3445        expr: &'tcx hir::Expr<'tcx>,
3446        brackets_span: Span,
3447    ) -> Ty<'tcx> {
3448        let base_t = self.check_expr(base);
3449        let idx_t = self.check_expr(idx);
3450
3451        if base_t.references_error() {
3452            base_t
3453        } else if idx_t.references_error() {
3454            idx_t
3455        } else {
3456            let base_t = self.structurally_resolve_type(base.span, base_t);
3457            match self.lookup_indexing(expr, base, base_t, idx, idx_t) {
3458                Some((index_ty, element_ty)) => {
3459                    // two-phase not needed because index_ty is never mutable
3460                    self.demand_coerce(idx, idx_t, index_ty, None, AllowTwoPhase::No);
3461                    self.select_obligations_where_possible(|errors| {
3462                        self.point_at_index(errors, idx.span);
3463                    });
3464                    element_ty
3465                }
3466                None => {
3467                    // Attempt to *shallowly* search for an impl which matches,
3468                    // but has nested obligations which are unsatisfied.
3469                    for (base_t, _) in self.autoderef(base.span, base_t).silence_errors() {
3470                        if let Some((_, index_ty, element_ty)) =
3471                            self.find_and_report_unsatisfied_index_impl(base, base_t)
3472                        {
3473                            self.demand_coerce(idx, idx_t, index_ty, None, AllowTwoPhase::No);
3474                            return element_ty;
3475                        }
3476                    }
3477
3478                    let mut err = {
    let mut err =
        {
            self.dcx().struct_span_err(brackets_span,
                    ::alloc::__export::must_use({
                            ::alloc::fmt::format(format_args!("cannot index into a value of type `{0}`",
                                    base_t))
                        })).with_code(E0608)
        };
    if base_t.references_error() { err.downgrade_to_delayed_bug(); }
    err
}type_error_struct!(
3479                        self.dcx(),
3480                        brackets_span,
3481                        base_t,
3482                        E0608,
3483                        "cannot index into a value of type `{base_t}`",
3484                    );
3485                    // Try to give some advice about indexing tuples.
3486                    if let ty::Tuple(types) = base_t.kind() {
3487                        err.help(
3488                            "tuples are indexed with a dot and a literal index: `tuple.0`, `tuple.1`, etc.",
3489                        );
3490                        // If index is an unsuffixed integer, show the fixed expression:
3491                        if let ExprKind::Lit(lit) = idx.kind
3492                            && let ast::LitKind::Int(i, ast::LitIntType::Unsuffixed) = lit.node
3493                            && i.get() < types.len().try_into().expect("tuple length fits in u128")
3494                        {
3495                            err.span_suggestion(
3496                                brackets_span,
3497                                ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("to access tuple element `{0}`, use",
                i))
    })format!("to access tuple element `{i}`, use"),
3498                                ::alloc::__export::must_use({ ::alloc::fmt::format(format_args!(".{0}", i)) })format!(".{i}"),
3499                                Applicability::MachineApplicable,
3500                            );
3501                        }
3502                    }
3503
3504                    if base_t.is_raw_ptr() && idx_t.is_integral() {
3505                        err.multipart_suggestion(
3506                            "consider using `wrapping_add` or `add` for indexing into raw pointer",
3507                            ::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
        [(base.span.between(idx.span), ".wrapping_add(".to_owned()),
                (idx.span.shrink_to_hi().until(expr.span.shrink_to_hi()),
                    ")".to_owned())]))vec![
3508                                (base.span.between(idx.span), ".wrapping_add(".to_owned()),
3509                                (
3510                                    idx.span.shrink_to_hi().until(expr.span.shrink_to_hi()),
3511                                    ")".to_owned(),
3512                                ),
3513                            ],
3514                            Applicability::MaybeIncorrect,
3515                        );
3516                    }
3517
3518                    let reported = err.emit();
3519                    Ty::new_error(self.tcx, reported)
3520                }
3521            }
3522        }
3523    }
3524
3525    /// Try to match an implementation of `Index` against a self type, and report
3526    /// the unsatisfied predicates that result from confirming this impl.
3527    ///
3528    /// Given an index expression, sometimes the `Self` type shallowly but does not
3529    /// deeply satisfy an impl predicate. Instead of simply saying that the type
3530    /// does not support being indexed, we want to point out exactly what nested
3531    /// predicates cause this to be, so that the user can add them to fix their code.
3532    fn find_and_report_unsatisfied_index_impl(
3533        &self,
3534        base_expr: &hir::Expr<'_>,
3535        base_ty: Ty<'tcx>,
3536    ) -> Option<(ErrorGuaranteed, Ty<'tcx>, Ty<'tcx>)> {
3537        let index_trait_def_id = self.tcx.lang_items().index_trait()?;
3538        let index_trait_output_def_id = self.tcx.get_diagnostic_item(sym::IndexOutput)?;
3539
3540        let mut relevant_impls = ::alloc::vec::Vec::new()vec![];
3541        self.tcx.for_each_relevant_impl(index_trait_def_id, base_ty, |impl_def_id| {
3542            relevant_impls.push(impl_def_id);
3543        });
3544        let [impl_def_id] = relevant_impls[..] else {
3545            // Only report unsatisfied impl predicates if there's one impl
3546            return None;
3547        };
3548
3549        self.commit_if_ok(|snapshot| {
3550            let outer_universe = self.universe();
3551
3552            let ocx = ObligationCtxt::new_with_diagnostics(self);
3553            let impl_args = self.fresh_args_for_item(base_expr.span, impl_def_id);
3554            let impl_trait_ref =
3555                self.tcx.impl_trait_ref(impl_def_id).instantiate(self.tcx, impl_args);
3556            let cause = self.misc(base_expr.span);
3557
3558            // Match the impl self type against the base ty. If this fails,
3559            // we just skip this impl, since it's not particularly useful.
3560            let impl_trait_ref = ocx.normalize(&cause, self.param_env, impl_trait_ref);
3561            ocx.eq(&cause, self.param_env, base_ty, impl_trait_ref.self_ty())?;
3562
3563            // Register the impl's predicates. One of these predicates
3564            // must be unsatisfied, or else we wouldn't have gotten here
3565            // in the first place.
3566            ocx.register_obligations(traits::predicates_for_generics(
3567                |idx, span| {
3568                    cause.clone().derived_cause(
3569                        ty::Binder::dummy(ty::TraitPredicate {
3570                            trait_ref: impl_trait_ref,
3571                            polarity: ty::PredicatePolarity::Positive,
3572                        }),
3573                        |derived| {
3574                            ObligationCauseCode::ImplDerived(Box::new(traits::ImplDerivedCause {
3575                                derived,
3576                                impl_or_alias_def_id: impl_def_id,
3577                                impl_def_predicate_index: Some(idx),
3578                                span,
3579                            }))
3580                        },
3581                    )
3582                },
3583                self.param_env,
3584                self.tcx.predicates_of(impl_def_id).instantiate(self.tcx, impl_args),
3585            ));
3586
3587            // Normalize the output type, which we can use later on as the
3588            // return type of the index expression...
3589            let element_ty = ocx.normalize(
3590                &cause,
3591                self.param_env,
3592                Ty::new_projection_from_args(
3593                    self.tcx,
3594                    index_trait_output_def_id,
3595                    impl_trait_ref.args,
3596                ),
3597            );
3598
3599            let true_errors = ocx.try_evaluate_obligations();
3600
3601            // Do a leak check -- we can't really report a useful error here,
3602            // but it at least avoids an ICE when the error has to do with higher-ranked
3603            // lifetimes.
3604            self.leak_check(outer_universe, Some(snapshot))?;
3605
3606            // Bail if we have ambiguity errors, which we can't report in a useful way.
3607            let ambiguity_errors = ocx.evaluate_obligations_error_on_ambiguity();
3608            if true_errors.is_empty() && !ambiguity_errors.is_empty() {
3609                return Err(NoSolution);
3610            }
3611
3612            // There should be at least one error reported. If not, we
3613            // will still delay a span bug in `report_fulfillment_errors`.
3614            Ok::<_, NoSolution>((
3615                self.err_ctxt().report_fulfillment_errors(true_errors),
3616                impl_trait_ref.args.type_at(1),
3617                element_ty,
3618            ))
3619        })
3620        .ok()
3621    }
3622
3623    fn point_at_index(&self, errors: &mut Vec<traits::FulfillmentError<'tcx>>, span: Span) {
3624        let mut seen_preds = FxHashSet::default();
3625        // We re-sort here so that the outer most root obligations comes first, as we have the
3626        // subsequent weird logic to identify *every* relevant obligation for proper deduplication
3627        // of diagnostics.
3628        errors.sort_by_key(|error| error.root_obligation.recursion_depth);
3629        for error in errors {
3630            match (
3631                error.root_obligation.predicate.kind().skip_binder(),
3632                error.obligation.predicate.kind().skip_binder(),
3633            ) {
3634                (ty::PredicateKind::Clause(ty::ClauseKind::Trait(predicate)), _)
3635                    if self.tcx.is_lang_item(predicate.trait_ref.def_id, LangItem::Index) =>
3636                {
3637                    seen_preds.insert(error.obligation.predicate.kind().skip_binder());
3638                }
3639                (_, ty::PredicateKind::Clause(ty::ClauseKind::Trait(predicate)))
3640                    if self.tcx.is_diagnostic_item(sym::SliceIndex, predicate.trait_ref.def_id) =>
3641                {
3642                    seen_preds.insert(error.obligation.predicate.kind().skip_binder());
3643                }
3644                (root, pred) if seen_preds.contains(&pred) || seen_preds.contains(&root) => {}
3645                _ => continue,
3646            }
3647            error.obligation.cause.span = span;
3648        }
3649    }
3650
3651    fn check_expr_yield(
3652        &self,
3653        value: &'tcx hir::Expr<'tcx>,
3654        expr: &'tcx hir::Expr<'tcx>,
3655    ) -> Ty<'tcx> {
3656        match self.coroutine_types {
3657            Some(CoroutineTypes { resume_ty, yield_ty }) => {
3658                self.check_expr_coercible_to_type(value, yield_ty, None);
3659
3660                resume_ty
3661            }
3662            _ => {
3663                self.dcx().emit_err(YieldExprOutsideOfCoroutine { span: expr.span });
3664                // Avoid expressions without types during writeback (#78653).
3665                self.check_expr(value);
3666                self.tcx.types.unit
3667            }
3668        }
3669    }
3670
3671    fn check_expr_asm_operand(&self, expr: &'tcx hir::Expr<'tcx>, is_input: bool) {
3672        let needs = if is_input { Needs::None } else { Needs::MutPlace };
3673        let ty = self.check_expr_with_needs(expr, needs);
3674        self.require_type_is_sized(ty, expr.span, ObligationCauseCode::InlineAsmSized);
3675
3676        if !is_input && !expr.is_syntactic_place_expr() {
3677            self.dcx()
3678                .struct_span_err(expr.span, "invalid asm output")
3679                .with_span_label(expr.span, "cannot assign to this expression")
3680                .emit();
3681        }
3682
3683        // If this is an input value, we require its type to be fully resolved
3684        // at this point. This allows us to provide helpful coercions which help
3685        // pass the type candidate list in a later pass.
3686        //
3687        // We don't require output types to be resolved at this point, which
3688        // allows them to be inferred based on how they are used later in the
3689        // function.
3690        if is_input {
3691            let ty = self.structurally_resolve_type(expr.span, ty);
3692            match *ty.kind() {
3693                ty::FnDef(..) => {
3694                    let fnptr_ty = Ty::new_fn_ptr(self.tcx, ty.fn_sig(self.tcx));
3695                    self.demand_coerce(expr, ty, fnptr_ty, None, AllowTwoPhase::No);
3696                }
3697                ty::Ref(_, base_ty, mutbl) => {
3698                    let ptr_ty = Ty::new_ptr(self.tcx, base_ty, mutbl);
3699                    self.demand_coerce(expr, ty, ptr_ty, None, AllowTwoPhase::No);
3700                }
3701                _ => {}
3702            }
3703        }
3704    }
3705
3706    fn check_expr_asm(&self, asm: &'tcx hir::InlineAsm<'tcx>, span: Span) -> Ty<'tcx> {
3707        if let rustc_ast::AsmMacro::NakedAsm = asm.asm_macro {
3708            if !{

        #[allow(deprecated)]
        {
            {
                'done:
                    {
                    for i in self.tcx.get_all_attrs(self.body_id) {
                        #[allow(unused_imports)]
                        use rustc_hir::attrs::AttributeKind::*;
                        let i: &rustc_hir::Attribute = i;
                        match i {
                            rustc_hir::Attribute::Parsed(Naked(..)) => {
                                break 'done Some(());
                            }
                            rustc_hir::Attribute::Unparsed(..) =>
                                {}
                                #[deny(unreachable_patterns)]
                                _ => {}
                        }
                    }
                    None
                }
            }
        }
    }.is_some()find_attr!(self.tcx, self.body_id, Naked(..)) {
3709                self.tcx.dcx().emit_err(NakedAsmOutsideNakedFn { span });
3710            }
3711        }
3712
3713        let mut diverge = asm.asm_macro.diverges(asm.options);
3714
3715        for (op, _op_sp) in asm.operands {
3716            match *op {
3717                hir::InlineAsmOperand::In { expr, .. } => {
3718                    self.check_expr_asm_operand(expr, true);
3719                }
3720                hir::InlineAsmOperand::Out { expr: Some(expr), .. }
3721                | hir::InlineAsmOperand::InOut { expr, .. } => {
3722                    self.check_expr_asm_operand(expr, false);
3723                }
3724                hir::InlineAsmOperand::Out { expr: None, .. } => {}
3725                hir::InlineAsmOperand::SplitInOut { in_expr, out_expr, .. } => {
3726                    self.check_expr_asm_operand(in_expr, true);
3727                    if let Some(out_expr) = out_expr {
3728                        self.check_expr_asm_operand(out_expr, false);
3729                    }
3730                }
3731                hir::InlineAsmOperand::Const { ref anon_const } => {
3732                    self.check_expr_const_block(anon_const, Expectation::NoExpectation);
3733                }
3734                hir::InlineAsmOperand::SymFn { expr } => {
3735                    self.check_expr(expr);
3736                }
3737                hir::InlineAsmOperand::SymStatic { .. } => {}
3738                hir::InlineAsmOperand::Label { block } => {
3739                    let previous_diverges = self.diverges.get();
3740
3741                    // The label blocks should have unit return value or diverge.
3742                    let ty = self.check_expr_block(block, ExpectHasType(self.tcx.types.unit));
3743                    if !ty.is_never() {
3744                        self.demand_suptype(block.span, self.tcx.types.unit, ty);
3745                        diverge = false;
3746                    }
3747
3748                    // We need this to avoid false unreachable warning when a label diverges.
3749                    self.diverges.set(previous_diverges);
3750                }
3751            }
3752        }
3753
3754        if diverge { self.tcx.types.never } else { self.tcx.types.unit }
3755    }
3756
3757    fn check_expr_offset_of(
3758        &self,
3759        container: &'tcx hir::Ty<'tcx>,
3760        fields: &[Ident],
3761        expr: &'tcx hir::Expr<'tcx>,
3762    ) -> Ty<'tcx> {
3763        let mut current_container = self.lower_ty(container).normalized;
3764        let mut field_indices = Vec::with_capacity(fields.len());
3765        let mut fields = fields.into_iter();
3766
3767        while let Some(&field) = fields.next() {
3768            let container = self.structurally_resolve_type(expr.span, current_container);
3769
3770            match container.kind() {
3771                ty::Adt(container_def, args) if container_def.is_enum() => {
3772                    let block = self.tcx.local_def_id_to_hir_id(self.body_id);
3773                    let (ident, _def_scope) =
3774                        self.tcx.adjust_ident_and_get_scope(field, container_def.did(), block);
3775
3776                    if !self.tcx.features().offset_of_enum() {
3777                        rustc_session::parse::feature_err(
3778                            &self.tcx.sess,
3779                            sym::offset_of_enum,
3780                            ident.span,
3781                            "using enums in offset_of is experimental",
3782                        )
3783                        .emit();
3784                    }
3785
3786                    let Some((index, variant)) = container_def
3787                        .variants()
3788                        .iter_enumerated()
3789                        .find(|(_, v)| v.ident(self.tcx).normalize_to_macros_2_0() == ident)
3790                    else {
3791                        self.dcx()
3792                            .create_err(NoVariantNamed { span: ident.span, ident, ty: container })
3793                            .with_span_label(field.span, "variant not found")
3794                            .emit_unless_delay(container.references_error());
3795                        break;
3796                    };
3797                    let Some(&subfield) = fields.next() else {
3798                        {
    let mut err =
        {
            self.dcx().struct_span_err(ident.span,
                    ::alloc::__export::must_use({
                            ::alloc::fmt::format(format_args!("`{0}` is an enum variant; expected field at end of `offset_of`",
                                    ident))
                        })).with_code(E0795)
        };
    if container.references_error() { err.downgrade_to_delayed_bug(); }
    err
}type_error_struct!(
3799                            self.dcx(),
3800                            ident.span,
3801                            container,
3802                            E0795,
3803                            "`{ident}` is an enum variant; expected field at end of `offset_of`",
3804                        )
3805                        .with_span_label(field.span, "enum variant")
3806                        .emit();
3807                        break;
3808                    };
3809                    let (subident, sub_def_scope) =
3810                        self.tcx.adjust_ident_and_get_scope(subfield, variant.def_id, block);
3811
3812                    let Some((subindex, field)) = variant
3813                        .fields
3814                        .iter_enumerated()
3815                        .find(|(_, f)| f.ident(self.tcx).normalize_to_macros_2_0() == subident)
3816                    else {
3817                        self.dcx()
3818                            .create_err(NoFieldOnVariant {
3819                                span: ident.span,
3820                                container,
3821                                ident,
3822                                field: subfield,
3823                                enum_span: field.span,
3824                                field_span: subident.span,
3825                            })
3826                            .emit_unless_delay(container.references_error());
3827                        break;
3828                    };
3829
3830                    let field_ty = self.field_ty(expr.span, field, args);
3831
3832                    // Enums are anyway always sized. But just to safeguard against future
3833                    // language extensions, let's double-check.
3834                    self.require_type_is_sized(
3835                        field_ty,
3836                        expr.span,
3837                        ObligationCauseCode::FieldSized {
3838                            adt_kind: AdtKind::Enum,
3839                            span: self.tcx.def_span(field.did),
3840                            last: false,
3841                        },
3842                    );
3843
3844                    if field.vis.is_accessible_from(sub_def_scope, self.tcx) {
3845                        self.tcx.check_stability(field.did, Some(expr.hir_id), expr.span, None);
3846                    } else {
3847                        self.private_field_err(ident, container_def.did()).emit();
3848                    }
3849
3850                    // Save the index of all fields regardless of their visibility in case
3851                    // of error recovery.
3852                    field_indices.push((current_container, index, subindex));
3853                    current_container = field_ty;
3854
3855                    continue;
3856                }
3857                ty::Adt(container_def, args) => {
3858                    let block = self.tcx.local_def_id_to_hir_id(self.body_id);
3859                    let (ident, def_scope) =
3860                        self.tcx.adjust_ident_and_get_scope(field, container_def.did(), block);
3861
3862                    let fields = &container_def.non_enum_variant().fields;
3863                    if let Some((index, field)) = fields
3864                        .iter_enumerated()
3865                        .find(|(_, f)| f.ident(self.tcx).normalize_to_macros_2_0() == ident)
3866                    {
3867                        let field_ty = self.field_ty(expr.span, field, args);
3868
3869                        if self.tcx.features().offset_of_slice() {
3870                            self.require_type_has_static_alignment(field_ty, expr.span);
3871                        } else {
3872                            self.require_type_is_sized(
3873                                field_ty,
3874                                expr.span,
3875                                ObligationCauseCode::Misc,
3876                            );
3877                        }
3878
3879                        if field.vis.is_accessible_from(def_scope, self.tcx) {
3880                            self.tcx.check_stability(field.did, Some(expr.hir_id), expr.span, None);
3881                        } else {
3882                            self.private_field_err(ident, container_def.did()).emit();
3883                        }
3884
3885                        // Save the index of all fields regardless of their visibility in case
3886                        // of error recovery.
3887                        field_indices.push((current_container, FIRST_VARIANT, index));
3888                        current_container = field_ty;
3889
3890                        continue;
3891                    }
3892                }
3893                ty::Tuple(tys) => {
3894                    if let Ok(index) = field.as_str().parse::<usize>()
3895                        && field.name == sym::integer(index)
3896                    {
3897                        if let Some(&field_ty) = tys.get(index) {
3898                            if self.tcx.features().offset_of_slice() {
3899                                self.require_type_has_static_alignment(field_ty, expr.span);
3900                            } else {
3901                                self.require_type_is_sized(
3902                                    field_ty,
3903                                    expr.span,
3904                                    ObligationCauseCode::Misc,
3905                                );
3906                            }
3907
3908                            field_indices.push((current_container, FIRST_VARIANT, index.into()));
3909                            current_container = field_ty;
3910
3911                            continue;
3912                        }
3913                    }
3914                }
3915                _ => (),
3916            };
3917
3918            self.no_such_field_err(field, container, expr).emit();
3919
3920            break;
3921        }
3922
3923        self.typeck_results.borrow_mut().offset_of_data_mut().insert(expr.hir_id, field_indices);
3924
3925        self.tcx.types.usize
3926    }
3927}