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