1use rustc_abi::{FIRST_VARIANT, FieldIdx};
9use rustc_ast as ast;
10use rustc_ast::util::parser::ExprPrecedence;
11use rustc_data_structures::fx::{FxHashMap, FxHashSet};
12use rustc_data_structures::stack::ensure_sufficient_stack;
13use rustc_data_structures::unord::UnordMap;
14use rustc_errors::codes::*;
15use rustc_errors::{
16 Applicability, Diag, ErrorGuaranteed, MultiSpan, StashKey, Subdiagnostic, listify, pluralize,
17 struct_span_code_err,
18};
19use rustc_hir as hir;
20use rustc_hir::def::{CtorKind, DefKind, Res};
21use rustc_hir::def_id::DefId;
22use rustc_hir::lang_items::LangItem;
23use rustc_hir::{ExprKind, HirId, QPath, find_attr, is_range_literal};
24use rustc_hir_analysis::NoVariantNamed;
25use rustc_hir_analysis::errors::NoFieldOnType;
26use rustc_hir_analysis::hir_ty_lowering::HirTyLowerer as _;
27use rustc_infer::infer::{self, DefineOpaqueTypes, InferOk, RegionVariableOrigin};
28use rustc_infer::traits::query::NoSolution;
29use rustc_middle::ty::adjustment::{Adjust, Adjustment, AllowTwoPhase};
30use rustc_middle::ty::error::{ExpectedFound, TypeError};
31use rustc_middle::ty::{self, AdtKind, GenericArgsRef, Ty, TypeVisitableExt};
32use rustc_middle::{bug, span_bug};
33use rustc_session::errors::ExprParenthesesNeeded;
34use rustc_session::parse::feature_err;
35use rustc_span::edit_distance::find_best_match_for_name;
36use rustc_span::hygiene::DesugaringKind;
37use rustc_span::source_map::Spanned;
38use rustc_span::{Ident, Span, Symbol, kw, sym};
39use rustc_trait_selection::infer::InferCtxtExt;
40use rustc_trait_selection::traits::{self, ObligationCauseCode, ObligationCtxt};
41use tracing::{debug, instrument, trace};
42
43use crate::Expectation::{self, ExpectCastableToType, ExpectHasType, NoExpectation};
44use crate::coercion::CoerceMany;
45use crate::errors::{
46 AddressOfTemporaryTaken, BaseExpressionDoubleDot, BaseExpressionDoubleDotAddExpr,
47 BaseExpressionDoubleDotRemove, CantDereference, FieldMultiplySpecifiedInInitializer,
48 FunctionalRecordUpdateOnNonStruct, HelpUseLatestEdition, NakedAsmOutsideNakedFn,
49 NoFieldOnVariant, ReturnLikeStatementKind, ReturnStmtOutsideOfFnBody, StructExprNonExhaustive,
50 TypeMismatchFruTypo, YieldExprOutsideOfCoroutine,
51};
52use crate::op::contains_let_in_chain;
53use crate::{
54 BreakableCtxt, CoroutineTypes, Diverges, FnCtxt, GatherLocalsVisitor, Needs,
55 TupleArgumentsFlag, cast, fatally_break_rust, report_unexpected_variant_res, type_error_struct,
56};
57
58impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
59 pub(crate) fn precedence(&self, expr: &hir::Expr<'_>) -> ExprPrecedence {
60 let has_attr = |id: HirId| -> bool {
61 for attr in self.tcx.hir_attrs(id) {
62 if attr.span().desugaring_kind().is_none() {
76 return true;
77 }
78 }
79 false
80 };
81
82 if is_range_literal(expr) {
86 return ExprPrecedence::Range;
87 }
88
89 expr.precedence(&has_attr)
90 }
91
92 pub(crate) fn check_expr_has_type_or_error(
96 &self,
97 expr: &'tcx hir::Expr<'tcx>,
98 expected_ty: Ty<'tcx>,
99 extend_err: impl FnOnce(&mut Diag<'_>),
100 ) -> Ty<'tcx> {
101 let mut ty = self.check_expr_with_expectation(expr, ExpectHasType(expected_ty));
102
103 if self.try_structurally_resolve_type(expr.span, ty).is_never()
106 && self.tcx.expr_guaranteed_to_constitute_read_for_never(expr)
107 {
108 if let Some(adjustments) = self.typeck_results.borrow().adjustments().get(expr.hir_id) {
109 let reported = self.dcx().span_delayed_bug(
110 expr.span,
111 "expression with never type wound up being adjusted",
112 );
113
114 return if let [Adjustment { kind: Adjust::NeverToAny, target }] = &adjustments[..] {
115 target.to_owned()
116 } else {
117 Ty::new_error(self.tcx(), reported)
118 };
119 }
120
121 let adj_ty = self.next_ty_var(expr.span);
122 self.apply_adjustments(
123 expr,
124 ::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[Adjustment { kind: Adjust::NeverToAny, target: adj_ty }]))vec![Adjustment { kind: Adjust::NeverToAny, target: adj_ty }],
125 );
126 ty = adj_ty;
127 }
128
129 if let Err(mut err) = self.demand_suptype_diag(expr.span, expected_ty, ty) {
130 let _ = self.emit_type_mismatch_suggestions(
131 &mut err,
132 expr.peel_drop_temps(),
133 ty,
134 expected_ty,
135 None,
136 None,
137 );
138 extend_err(&mut err);
139 err.emit();
140 }
141 ty
142 }
143
144 pub(super) fn check_expr_coercible_to_type(
148 &self,
149 expr: &'tcx hir::Expr<'tcx>,
150 expected: Ty<'tcx>,
151 expected_ty_expr: Option<&'tcx hir::Expr<'tcx>>,
152 ) -> Ty<'tcx> {
153 self.check_expr_coercible_to_type_or_error(expr, expected, expected_ty_expr, |_, _| {})
154 }
155
156 pub(crate) fn check_expr_coercible_to_type_or_error(
157 &self,
158 expr: &'tcx hir::Expr<'tcx>,
159 expected: Ty<'tcx>,
160 expected_ty_expr: Option<&'tcx hir::Expr<'tcx>>,
161 extend_err: impl FnOnce(&mut Diag<'_>, Ty<'tcx>),
162 ) -> Ty<'tcx> {
163 let ty = self.check_expr_with_hint(expr, expected);
164 match self.demand_coerce_diag(expr, ty, expected, expected_ty_expr, AllowTwoPhase::No) {
166 Ok(ty) => ty,
167 Err(mut err) => {
168 extend_err(&mut err, ty);
169 err.emit();
170 expected
174 }
175 }
176 }
177
178 pub(super) fn check_expr_with_hint(
183 &self,
184 expr: &'tcx hir::Expr<'tcx>,
185 expected: Ty<'tcx>,
186 ) -> Ty<'tcx> {
187 self.check_expr_with_expectation(expr, ExpectHasType(expected))
188 }
189
190 fn check_expr_with_expectation_and_needs(
193 &self,
194 expr: &'tcx hir::Expr<'tcx>,
195 expected: Expectation<'tcx>,
196 needs: Needs,
197 ) -> Ty<'tcx> {
198 let ty = self.check_expr_with_expectation(expr, expected);
199
200 if let Needs::MutPlace = needs {
203 self.convert_place_derefs_to_mutable(expr);
204 }
205
206 ty
207 }
208
209 pub(super) fn check_expr(&self, expr: &'tcx hir::Expr<'tcx>) -> Ty<'tcx> {
211 self.check_expr_with_expectation(expr, NoExpectation)
212 }
213
214 pub(super) fn check_expr_with_needs(
217 &self,
218 expr: &'tcx hir::Expr<'tcx>,
219 needs: Needs,
220 ) -> Ty<'tcx> {
221 self.check_expr_with_expectation_and_needs(expr, NoExpectation, needs)
222 }
223
224 #[allow(clippy :: suspicious_else_formatting)]
{
let __tracing_attr_span;
let __tracing_attr_guard;
if ::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
&&
::tracing::Level::DEBUG <=
::tracing::level_filters::LevelFilter::current() ||
{ false } {
__tracing_attr_span =
{
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("check_expr_with_expectation",
"rustc_hir_typeck::expr", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/expr.rs"),
::tracing_core::__macro_support::Option::Some(226u32),
::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::expr"),
::tracing_core::field::FieldSet::new(&["expected"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::SPAN)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let mut interest = ::tracing::subscriber::Interest::never();
if ::tracing::Level::DEBUG <=
::tracing::level_filters::STATIC_MAX_LEVEL &&
::tracing::Level::DEBUG <=
::tracing::level_filters::LevelFilter::current() &&
{ interest = __CALLSITE.interest(); !interest.is_never() }
&&
::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
interest) {
let meta = __CALLSITE.metadata();
::tracing::Span::new(meta,
&{
#[allow(unused_imports)]
use ::tracing::field::{debug, display, Value};
let mut iter = meta.fields().iter();
meta.fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&::tracing::field::debug(&expected)
as &dyn Value))])
})
} else {
let span =
::tracing::__macro_support::__disabled_span(__CALLSITE.metadata());
{};
span
}
};
__tracing_attr_guard = __tracing_attr_span.enter();
}
#[warn(clippy :: suspicious_else_formatting)]
{
#[allow(unknown_lints, unreachable_code, clippy ::
diverging_sub_expression, clippy :: empty_loop, clippy ::
let_unit_value, clippy :: let_with_type_underscore, clippy ::
needless_return, clippy :: unreachable)]
if false {
let __tracing_attr_fake_return: Ty<'tcx> = loop {};
return __tracing_attr_fake_return;
}
{ self.check_expr_with_expectation_and_args(expr, expected, None) }
}
}#[instrument(skip(self, expr), level = "debug")]
227 pub(super) fn check_expr_with_expectation(
228 &self,
229 expr: &'tcx hir::Expr<'tcx>,
230 expected: Expectation<'tcx>,
231 ) -> Ty<'tcx> {
232 self.check_expr_with_expectation_and_args(expr, expected, None)
233 }
234
235 pub(super) fn check_expr_with_expectation_and_args(
240 &self,
241 expr: &'tcx hir::Expr<'tcx>,
242 expected: Expectation<'tcx>,
243 call_expr_and_args: Option<(&'tcx hir::Expr<'tcx>, &'tcx [hir::Expr<'tcx>])>,
244 ) -> Ty<'tcx> {
245 if self.tcx().sess.verbose_internals() {
246 if let Ok(lint_str) = self.tcx.sess.source_map().span_to_snippet(expr.span) {
248 if !lint_str.contains('\n') {
249 {
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_hir_typeck/src/expr.rs:249",
"rustc_hir_typeck::expr", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/expr.rs"),
::tracing_core::__macro_support::Option::Some(249u32),
::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::expr"),
::tracing_core::field::FieldSet::new(&["message"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::EVENT)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let enabled =
::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
&&
::tracing::Level::DEBUG <=
::tracing::level_filters::LevelFilter::current() &&
{
let interest = __CALLSITE.interest();
!interest.is_never() &&
::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
interest)
};
if enabled {
(|value_set: ::tracing::field::ValueSet|
{
let meta = __CALLSITE.metadata();
::tracing::Event::dispatch(meta, &value_set);
;
})({
#[allow(unused_imports)]
use ::tracing::field::{debug, display, Value};
let mut iter = __CALLSITE.metadata().fields().iter();
__CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&format_args!("expr text: {0}",
lint_str) as &dyn Value))])
});
} else { ; }
};debug!("expr text: {lint_str}");
250 } else {
251 let mut lines = lint_str.lines();
252 if let Some(line0) = lines.next() {
253 let remaining_lines = lines.count();
254 {
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_hir_typeck/src/expr.rs:254",
"rustc_hir_typeck::expr", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/expr.rs"),
::tracing_core::__macro_support::Option::Some(254u32),
::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::expr"),
::tracing_core::field::FieldSet::new(&["message"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::EVENT)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let enabled =
::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
&&
::tracing::Level::DEBUG <=
::tracing::level_filters::LevelFilter::current() &&
{
let interest = __CALLSITE.interest();
!interest.is_never() &&
::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
interest)
};
if enabled {
(|value_set: ::tracing::field::ValueSet|
{
let meta = __CALLSITE.metadata();
::tracing::Event::dispatch(meta, &value_set);
;
})({
#[allow(unused_imports)]
use ::tracing::field::{debug, display, Value};
let mut iter = __CALLSITE.metadata().fields().iter();
__CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&format_args!("expr text: {0}",
line0) as &dyn Value))])
});
} else { ; }
};debug!("expr text: {line0}");
255 {
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_hir_typeck/src/expr.rs:255",
"rustc_hir_typeck::expr", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/expr.rs"),
::tracing_core::__macro_support::Option::Some(255u32),
::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::expr"),
::tracing_core::field::FieldSet::new(&["message"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::EVENT)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let enabled =
::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
&&
::tracing::Level::DEBUG <=
::tracing::level_filters::LevelFilter::current() &&
{
let interest = __CALLSITE.interest();
!interest.is_never() &&
::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
interest)
};
if enabled {
(|value_set: ::tracing::field::ValueSet|
{
let meta = __CALLSITE.metadata();
::tracing::Event::dispatch(meta, &value_set);
;
})({
#[allow(unused_imports)]
use ::tracing::field::{debug, display, Value};
let mut iter = __CALLSITE.metadata().fields().iter();
__CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&format_args!("expr text: ...(and {0} more lines)",
remaining_lines) as &dyn Value))])
});
} else { ; }
};debug!("expr text: ...(and {remaining_lines} more lines)");
256 }
257 }
258 }
259 }
260
261 let is_try_block_generated_unit_expr = match expr.kind {
265 ExprKind::Call(_, [arg]) => {
266 expr.span.is_desugaring(DesugaringKind::TryBlock)
267 && arg.span.is_desugaring(DesugaringKind::TryBlock)
268 }
269 _ => false,
270 };
271
272 if !is_try_block_generated_unit_expr {
274 self.warn_if_unreachable(expr.hir_id, expr.span, "expression");
275 }
276
277 let old_diverges = self.diverges.replace(Diverges::Maybe);
280
281 if self.is_whole_body.replace(false) {
282 self.diverges.set(self.function_diverges_because_of_empty_arguments.get())
285 };
286
287 let ty = ensure_sufficient_stack(|| match &expr.kind {
288 hir::ExprKind::Path(
290 qpath @ (hir::QPath::Resolved(..) | hir::QPath::TypeRelative(..)),
291 ) => self.check_expr_path(qpath, expr, call_expr_and_args),
292 _ => self.check_expr_kind(expr, expected),
293 });
294 let ty = self.resolve_vars_if_possible(ty);
295
296 match expr.kind {
298 ExprKind::Block(..)
299 | ExprKind::If(..)
300 | ExprKind::Let(..)
301 | ExprKind::Loop(..)
302 | ExprKind::Match(..) => {}
303 ExprKind::Cast(_, _) => {}
306 ExprKind::Call(..) if expr.span.is_desugaring(DesugaringKind::TryBlock) => {}
310 ExprKind::Call(..) if expr.span.is_desugaring(DesugaringKind::Contract) => {}
312 ExprKind::Call(callee, _) => self.warn_if_unreachable(expr.hir_id, callee.span, "call"),
313 ExprKind::MethodCall(segment, ..) => {
314 self.warn_if_unreachable(expr.hir_id, segment.ident.span, "call")
315 }
316 _ => self.warn_if_unreachable(expr.hir_id, expr.span, "expression"),
317 }
318
319 if self.try_structurally_resolve_type(expr.span, ty).is_never()
324 && self.tcx.expr_guaranteed_to_constitute_read_for_never(expr)
325 {
326 self.diverges.set(self.diverges.get() | Diverges::always(expr.span));
327 }
328
329 self.write_ty(expr.hir_id, ty);
333
334 self.diverges.set(self.diverges.get() | old_diverges);
336
337 {
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_hir_typeck/src/expr.rs:337",
"rustc_hir_typeck::expr", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/expr.rs"),
::tracing_core::__macro_support::Option::Some(337u32),
::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::expr"),
::tracing_core::field::FieldSet::new(&["message"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::EVENT)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let enabled =
::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
&&
::tracing::Level::DEBUG <=
::tracing::level_filters::LevelFilter::current() &&
{
let interest = __CALLSITE.interest();
!interest.is_never() &&
::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
interest)
};
if enabled {
(|value_set: ::tracing::field::ValueSet|
{
let meta = __CALLSITE.metadata();
::tracing::Event::dispatch(meta, &value_set);
;
})({
#[allow(unused_imports)]
use ::tracing::field::{debug, display, Value};
let mut iter = __CALLSITE.metadata().fields().iter();
__CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&format_args!("type of {0} is...",
self.tcx.hir_id_to_string(expr.hir_id)) as &dyn Value))])
});
} else { ; }
};debug!("type of {} is...", self.tcx.hir_id_to_string(expr.hir_id));
338 {
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_hir_typeck/src/expr.rs:338",
"rustc_hir_typeck::expr", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/expr.rs"),
::tracing_core::__macro_support::Option::Some(338u32),
::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::expr"),
::tracing_core::field::FieldSet::new(&["message"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::EVENT)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let enabled =
::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
&&
::tracing::Level::DEBUG <=
::tracing::level_filters::LevelFilter::current() &&
{
let interest = __CALLSITE.interest();
!interest.is_never() &&
::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
interest)
};
if enabled {
(|value_set: ::tracing::field::ValueSet|
{
let meta = __CALLSITE.metadata();
::tracing::Event::dispatch(meta, &value_set);
;
})({
#[allow(unused_imports)]
use ::tracing::field::{debug, display, Value};
let mut iter = __CALLSITE.metadata().fields().iter();
__CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&format_args!("... {0:?}, expected is {1:?}",
ty, expected) as &dyn Value))])
});
} else { ; }
};debug!("... {:?}, expected is {:?}", ty, expected);
339
340 ty
341 }
342
343 #[allow(clippy :: suspicious_else_formatting)]
{
let __tracing_attr_span;
let __tracing_attr_guard;
if ::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
&&
::tracing::Level::DEBUG <=
::tracing::level_filters::LevelFilter::current() ||
{ false } {
__tracing_attr_span =
{
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("check_expr_kind",
"rustc_hir_typeck::expr", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/expr.rs"),
::tracing_core::__macro_support::Option::Some(343u32),
::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::expr"),
::tracing_core::field::FieldSet::new(&["expected"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::SPAN)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let mut interest = ::tracing::subscriber::Interest::never();
if ::tracing::Level::DEBUG <=
::tracing::level_filters::STATIC_MAX_LEVEL &&
::tracing::Level::DEBUG <=
::tracing::level_filters::LevelFilter::current() &&
{ interest = __CALLSITE.interest(); !interest.is_never() }
&&
::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
interest) {
let meta = __CALLSITE.metadata();
::tracing::Span::new(meta,
&{
#[allow(unused_imports)]
use ::tracing::field::{debug, display, Value};
let mut iter = meta.fields().iter();
meta.fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&::tracing::field::debug(&expected)
as &dyn Value))])
})
} else {
let span =
::tracing::__macro_support::__disabled_span(__CALLSITE.metadata());
{};
span
}
};
__tracing_attr_guard = __tracing_attr_span.enter();
}
#[warn(clippy :: suspicious_else_formatting)]
{
#[allow(unknown_lints, unreachable_code, clippy ::
diverging_sub_expression, clippy :: empty_loop, clippy ::
let_unit_value, clippy :: let_with_type_underscore, clippy ::
needless_return, clippy :: unreachable)]
if false {
let __tracing_attr_fake_return: Ty<'tcx> = loop {};
return __tracing_attr_fake_return;
}
{
{
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_hir_typeck/src/expr.rs:349",
"rustc_hir_typeck::expr", ::tracing::Level::TRACE,
::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/expr.rs"),
::tracing_core::__macro_support::Option::Some(349u32),
::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::expr"),
::tracing_core::field::FieldSet::new(&["message"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::EVENT)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let enabled =
::tracing::Level::TRACE <=
::tracing::level_filters::STATIC_MAX_LEVEL &&
::tracing::Level::TRACE <=
::tracing::level_filters::LevelFilter::current() &&
{
let interest = __CALLSITE.interest();
!interest.is_never() &&
::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
interest)
};
if enabled {
(|value_set: ::tracing::field::ValueSet|
{
let meta = __CALLSITE.metadata();
::tracing::Event::dispatch(meta, &value_set);
;
})({
#[allow(unused_imports)]
use ::tracing::field::{debug, display, Value};
let mut iter = __CALLSITE.metadata().fields().iter();
__CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&format_args!("expr={0:#?}",
expr) as &dyn Value))])
});
} else { ; }
};
let tcx = self.tcx;
match expr.kind {
ExprKind::Lit(ref lit) => self.check_expr_lit(lit, expected),
ExprKind::Binary(op, lhs, rhs) =>
self.check_expr_binop(expr, op, lhs, rhs, expected),
ExprKind::Assign(lhs, rhs, span) => {
self.check_expr_assign(expr, expected, lhs, rhs, span)
}
ExprKind::AssignOp(op, lhs, rhs) => {
self.check_expr_assign_op(expr, op, lhs, rhs, expected)
}
ExprKind::Unary(unop, oprnd) =>
self.check_expr_unop(unop, oprnd, expected, expr),
ExprKind::AddrOf(kind, mutbl, oprnd) => {
self.check_expr_addr_of(kind, mutbl, oprnd, expected, expr)
}
ExprKind::Path(ref qpath) =>
self.check_expr_path(qpath, expr, None),
ExprKind::InlineAsm(asm) => {
self.deferred_asm_checks.borrow_mut().push((asm,
expr.hir_id));
self.check_expr_asm(asm, expr.span)
}
ExprKind::OffsetOf(container, fields) => {
self.check_expr_offset_of(container, fields, expr)
}
ExprKind::Break(destination, ref expr_opt) => {
self.check_expr_break(destination, expr_opt.as_deref(),
expr)
}
ExprKind::Continue(destination) =>
self.check_expr_continue(destination, expr),
ExprKind::Ret(ref expr_opt) =>
self.check_expr_return(expr_opt.as_deref(), expr),
ExprKind::Become(call) => self.check_expr_become(call, expr),
ExprKind::Let(let_expr) =>
self.check_expr_let(let_expr, expr.hir_id),
ExprKind::Loop(body, _, source, _) => {
self.check_expr_loop(body, source, expected, expr)
}
ExprKind::Match(discrim, arms, match_src) => {
self.check_expr_match(expr, discrim, arms, expected,
match_src)
}
ExprKind::Closure(closure) =>
self.check_expr_closure(closure, expr.span, expected),
ExprKind::Block(body, _) =>
self.check_expr_block(body, expected),
ExprKind::Call(callee, args) =>
self.check_expr_call(expr, callee, args, expected),
ExprKind::Use(used_expr, _) =>
self.check_expr_use(used_expr, expected),
ExprKind::MethodCall(segment, receiver, args, _) => {
self.check_expr_method_call(expr, segment, receiver, args,
expected)
}
ExprKind::Cast(e, t) => self.check_expr_cast(e, t, expr),
ExprKind::Type(e, t) => {
let ascribed_ty = self.lower_ty_saving_user_provided_ty(t);
let ty = self.check_expr_with_hint(e, ascribed_ty);
self.demand_eqtype(e.span, ascribed_ty, ty);
ascribed_ty
}
ExprKind::If(cond, then_expr, opt_else_expr) => {
self.check_expr_if(expr.hir_id, cond, then_expr,
opt_else_expr, expr.span, expected)
}
ExprKind::DropTemps(e) =>
self.check_expr_with_expectation(e, expected),
ExprKind::Array(args) =>
self.check_expr_array(args, expected, expr),
ExprKind::ConstBlock(ref block) =>
self.check_expr_const_block(block, expected),
ExprKind::Repeat(element, ref count) => {
self.check_expr_repeat(element, count, expected, expr)
}
ExprKind::Tup(elts) =>
self.check_expr_tuple(elts, expected, expr),
ExprKind::Struct(qpath, fields, ref base_expr) => {
self.check_expr_struct(expr, expected, qpath, fields,
base_expr)
}
ExprKind::Field(base, field) =>
self.check_expr_field(expr, base, field, expected),
ExprKind::Index(base, idx, brackets_span) => {
self.check_expr_index(base, idx, expr, brackets_span)
}
ExprKind::Yield(value, _) =>
self.check_expr_yield(value, expr),
ExprKind::UnsafeBinderCast(kind, inner_expr, ty) => {
self.check_expr_unsafe_binder_cast(expr.span, kind,
inner_expr, ty, expected)
}
ExprKind::Err(guar) => Ty::new_error(tcx, guar),
}
}
}
}#[instrument(skip(self, expr), level = "debug")]
344 fn check_expr_kind(
345 &self,
346 expr: &'tcx hir::Expr<'tcx>,
347 expected: Expectation<'tcx>,
348 ) -> Ty<'tcx> {
349 trace!("expr={:#?}", expr);
350
351 let tcx = self.tcx;
352 match expr.kind {
353 ExprKind::Lit(ref lit) => self.check_expr_lit(lit, expected),
354 ExprKind::Binary(op, lhs, rhs) => self.check_expr_binop(expr, op, lhs, rhs, expected),
355 ExprKind::Assign(lhs, rhs, span) => {
356 self.check_expr_assign(expr, expected, lhs, rhs, span)
357 }
358 ExprKind::AssignOp(op, lhs, rhs) => {
359 self.check_expr_assign_op(expr, op, lhs, rhs, expected)
360 }
361 ExprKind::Unary(unop, oprnd) => self.check_expr_unop(unop, oprnd, expected, expr),
362 ExprKind::AddrOf(kind, mutbl, oprnd) => {
363 self.check_expr_addr_of(kind, mutbl, oprnd, expected, expr)
364 }
365 ExprKind::Path(ref qpath) => self.check_expr_path(qpath, expr, None),
366 ExprKind::InlineAsm(asm) => {
367 self.deferred_asm_checks.borrow_mut().push((asm, expr.hir_id));
369 self.check_expr_asm(asm, expr.span)
370 }
371 ExprKind::OffsetOf(container, fields) => {
372 self.check_expr_offset_of(container, fields, expr)
373 }
374 ExprKind::Break(destination, ref expr_opt) => {
375 self.check_expr_break(destination, expr_opt.as_deref(), expr)
376 }
377 ExprKind::Continue(destination) => self.check_expr_continue(destination, expr),
378 ExprKind::Ret(ref expr_opt) => self.check_expr_return(expr_opt.as_deref(), expr),
379 ExprKind::Become(call) => self.check_expr_become(call, expr),
380 ExprKind::Let(let_expr) => self.check_expr_let(let_expr, expr.hir_id),
381 ExprKind::Loop(body, _, source, _) => {
382 self.check_expr_loop(body, source, expected, expr)
383 }
384 ExprKind::Match(discrim, arms, match_src) => {
385 self.check_expr_match(expr, discrim, arms, expected, match_src)
386 }
387 ExprKind::Closure(closure) => self.check_expr_closure(closure, expr.span, expected),
388 ExprKind::Block(body, _) => self.check_expr_block(body, expected),
389 ExprKind::Call(callee, args) => self.check_expr_call(expr, callee, args, expected),
390 ExprKind::Use(used_expr, _) => self.check_expr_use(used_expr, expected),
391 ExprKind::MethodCall(segment, receiver, args, _) => {
392 self.check_expr_method_call(expr, segment, receiver, args, expected)
393 }
394 ExprKind::Cast(e, t) => self.check_expr_cast(e, t, expr),
395 ExprKind::Type(e, t) => {
396 let ascribed_ty = self.lower_ty_saving_user_provided_ty(t);
397 let ty = self.check_expr_with_hint(e, ascribed_ty);
398 self.demand_eqtype(e.span, ascribed_ty, ty);
399 ascribed_ty
400 }
401 ExprKind::If(cond, then_expr, opt_else_expr) => {
402 self.check_expr_if(expr.hir_id, cond, then_expr, opt_else_expr, expr.span, expected)
403 }
404 ExprKind::DropTemps(e) => self.check_expr_with_expectation(e, expected),
405 ExprKind::Array(args) => self.check_expr_array(args, expected, expr),
406 ExprKind::ConstBlock(ref block) => self.check_expr_const_block(block, expected),
407 ExprKind::Repeat(element, ref count) => {
408 self.check_expr_repeat(element, count, expected, expr)
409 }
410 ExprKind::Tup(elts) => self.check_expr_tuple(elts, expected, expr),
411 ExprKind::Struct(qpath, fields, ref base_expr) => {
412 self.check_expr_struct(expr, expected, qpath, fields, base_expr)
413 }
414 ExprKind::Field(base, field) => self.check_expr_field(expr, base, field, expected),
415 ExprKind::Index(base, idx, brackets_span) => {
416 self.check_expr_index(base, idx, expr, brackets_span)
417 }
418 ExprKind::Yield(value, _) => self.check_expr_yield(value, expr),
419 ExprKind::UnsafeBinderCast(kind, inner_expr, ty) => {
420 self.check_expr_unsafe_binder_cast(expr.span, kind, inner_expr, ty, expected)
421 }
422 ExprKind::Err(guar) => Ty::new_error(tcx, guar),
423 }
424 }
425
426 fn check_expr_unop(
427 &self,
428 unop: hir::UnOp,
429 oprnd: &'tcx hir::Expr<'tcx>,
430 expected: Expectation<'tcx>,
431 expr: &'tcx hir::Expr<'tcx>,
432 ) -> Ty<'tcx> {
433 let tcx = self.tcx;
434 let expected_inner = match unop {
435 hir::UnOp::Not | hir::UnOp::Neg => expected,
436 hir::UnOp::Deref => NoExpectation,
437 };
438 let oprnd_t = self.check_expr_with_expectation(oprnd, expected_inner);
439
440 if let Err(guar) = oprnd_t.error_reported() {
441 return Ty::new_error(tcx, guar);
442 }
443
444 let oprnd_t = self.structurally_resolve_type(expr.span, oprnd_t);
445 match unop {
446 hir::UnOp::Deref => self.lookup_derefing(expr, oprnd, oprnd_t).unwrap_or_else(|| {
447 let mut err =
448 self.dcx().create_err(CantDereference { span: expr.span, ty: oprnd_t });
449 let sp = tcx.sess.source_map().start_point(expr.span).with_parent(None);
450 if let Some(sp) = tcx.sess.psess.ambiguous_block_expr_parse.borrow().get(&sp) {
451 err.subdiagnostic(ExprParenthesesNeeded::surrounding(*sp));
452 }
453 Ty::new_error(tcx, err.emit())
454 }),
455 hir::UnOp::Not => {
456 let result = self.check_user_unop(expr, oprnd_t, unop, expected_inner);
457 if oprnd_t.is_integral() || *oprnd_t.kind() == ty::Bool { oprnd_t } else { result }
459 }
460 hir::UnOp::Neg => {
461 let result = self.check_user_unop(expr, oprnd_t, unop, expected_inner);
462 if oprnd_t.is_numeric() { oprnd_t } else { result }
464 }
465 }
466 }
467
468 fn check_expr_addr_of(
469 &self,
470 kind: hir::BorrowKind,
471 mutbl: hir::Mutability,
472 oprnd: &'tcx hir::Expr<'tcx>,
473 expected: Expectation<'tcx>,
474 expr: &'tcx hir::Expr<'tcx>,
475 ) -> Ty<'tcx> {
476 let hint = expected.only_has_type(self).map_or(NoExpectation, |ty| {
477 match self.try_structurally_resolve_type(expr.span, ty).kind() {
478 ty::Ref(_, ty, _) | ty::RawPtr(ty, _) => {
479 if oprnd.is_syntactic_place_expr() {
480 ExpectHasType(*ty)
484 } else {
485 Expectation::rvalue_hint(self, *ty)
486 }
487 }
488 _ => NoExpectation,
489 }
490 });
491 let ty =
492 self.check_expr_with_expectation_and_needs(oprnd, hint, Needs::maybe_mut_place(mutbl));
493 if let Err(guar) = ty.error_reported() {
494 return Ty::new_error(self.tcx, guar);
495 }
496
497 match kind {
498 hir::BorrowKind::Raw => {
499 self.check_named_place_expr(oprnd);
500 Ty::new_ptr(self.tcx, ty, mutbl)
501 }
502 hir::BorrowKind::Ref | hir::BorrowKind::Pin => {
503 let region = self.next_region_var(RegionVariableOrigin::BorrowRegion(expr.span));
518 match kind {
519 hir::BorrowKind::Ref => Ty::new_ref(self.tcx, region, ty, mutbl),
520 hir::BorrowKind::Pin => Ty::new_pinned_ref(self.tcx, region, ty, mutbl),
521 _ => ::core::panicking::panic("internal error: entered unreachable code")unreachable!(),
522 }
523 }
524 }
525 }
526
527 fn check_named_place_expr(&self, oprnd: &'tcx hir::Expr<'tcx>) {
533 let is_named = oprnd.is_place_expr(|base| {
534 self.typeck_results
546 .borrow()
547 .adjustments()
548 .get(base.hir_id)
549 .is_some_and(|x| x.iter().any(|adj| #[allow(non_exhaustive_omitted_patterns)] match adj.kind {
Adjust::Deref(_) => true,
_ => false,
}matches!(adj.kind, Adjust::Deref(_))))
550 });
551 if !is_named {
552 self.dcx().emit_err(AddressOfTemporaryTaken { span: oprnd.span });
553 }
554 }
555
556 pub(crate) fn check_expr_path(
557 &self,
558 qpath: &'tcx hir::QPath<'tcx>,
559 expr: &'tcx hir::Expr<'tcx>,
560 call_expr_and_args: Option<(&'tcx hir::Expr<'tcx>, &'tcx [hir::Expr<'tcx>])>,
561 ) -> Ty<'tcx> {
562 let tcx = self.tcx;
563
564 if let Some((_, [arg])) = call_expr_and_args
565 && let QPath::Resolved(_, path) = qpath
566 && let Res::Def(_, def_id) = path.res
567 && let Some(lang_item) = tcx.lang_items().from_def_id(def_id)
568 {
569 let code = match lang_item {
570 LangItem::IntoFutureIntoFuture
571 if expr.span.is_desugaring(DesugaringKind::Await) =>
572 {
573 Some(ObligationCauseCode::AwaitableExpr(arg.hir_id))
574 }
575 LangItem::IntoIterIntoIter | LangItem::IteratorNext
576 if expr.span.is_desugaring(DesugaringKind::ForLoop) =>
577 {
578 Some(ObligationCauseCode::ForLoopIterator)
579 }
580 LangItem::TryTraitFromOutput
581 if expr.span.is_desugaring(DesugaringKind::TryBlock) =>
582 {
583 Some(ObligationCauseCode::QuestionMark)
585 }
586 LangItem::TryTraitBranch | LangItem::TryTraitFromResidual
587 if expr.span.is_desugaring(DesugaringKind::QuestionMark) =>
588 {
589 Some(ObligationCauseCode::QuestionMark)
590 }
591 _ => None,
592 };
593 if let Some(code) = code {
594 let args = self.fresh_args_for_item(expr.span, def_id);
595 self.add_required_obligations_with_code(expr.span, def_id, args, |_, _| {
596 code.clone()
597 });
598 return tcx.type_of(def_id).instantiate(tcx, args);
599 }
600 }
601
602 let (res, opt_ty, segs) =
603 self.resolve_ty_and_res_fully_qualified_call(qpath, expr.hir_id, expr.span);
604 let ty = match res {
605 Res::Err => {
606 self.suggest_assoc_method_call(segs);
607 let e =
608 self.dcx().span_delayed_bug(qpath.span(), "`Res::Err` but no error emitted");
609 Ty::new_error(tcx, e)
610 }
611 Res::Def(DefKind::Variant, _) => {
612 let e = report_unexpected_variant_res(
613 tcx,
614 res,
615 Some(expr),
616 qpath,
617 expr.span,
618 E0533,
619 "value",
620 );
621 Ty::new_error(tcx, e)
622 }
623 _ => {
624 self.instantiate_value_path(
625 segs,
626 opt_ty,
627 res,
628 call_expr_and_args.map_or(expr.span, |(e, _)| e.span),
629 expr.span,
630 expr.hir_id,
631 )
632 .0
633 }
634 };
635
636 if let ty::FnDef(did, _) = *ty.kind() {
637 let fn_sig = ty.fn_sig(tcx);
638
639 if tcx.is_intrinsic(did, sym::transmute) {
640 let Some(from) = fn_sig.inputs().skip_binder().get(0) else {
641 ::rustc_middle::util::bug::span_bug_fmt(tcx.def_span(did),
format_args!("intrinsic fn `transmute` defined with no parameters"));span_bug!(
642 tcx.def_span(did),
643 "intrinsic fn `transmute` defined with no parameters"
644 );
645 };
646 let to = fn_sig.output().skip_binder();
647 self.deferred_transmute_checks.borrow_mut().push((*from, to, expr.hir_id));
652 }
653 if !tcx.features().unsized_fn_params() {
654 for i in 0..fn_sig.inputs().skip_binder().len() {
664 let span = call_expr_and_args
668 .and_then(|(_, args)| args.get(i))
669 .map_or(expr.span, |arg| arg.span);
670 let input = self.instantiate_binder_with_fresh_vars(
671 span,
672 infer::BoundRegionConversionTime::FnCall,
673 fn_sig.input(i),
674 );
675 self.require_type_is_sized_deferred(
676 input,
677 span,
678 ObligationCauseCode::SizedArgumentType(None),
679 );
680 }
681 }
682 let output = self.instantiate_binder_with_fresh_vars(
688 expr.span,
689 infer::BoundRegionConversionTime::FnCall,
690 fn_sig.output(),
691 );
692 self.require_type_is_sized_deferred(
693 output,
694 call_expr_and_args.map_or(expr.span, |(e, _)| e.span),
695 ObligationCauseCode::SizedCallReturnType,
696 );
697 }
698
699 let args = self.typeck_results.borrow().node_args(expr.hir_id);
702 self.add_wf_bounds(args, expr.span);
703
704 ty
705 }
706
707 fn check_expr_break(
708 &self,
709 destination: hir::Destination,
710 expr_opt: Option<&'tcx hir::Expr<'tcx>>,
711 expr: &'tcx hir::Expr<'tcx>,
712 ) -> Ty<'tcx> {
713 let tcx = self.tcx;
714 if let Ok(target_id) = destination.target_id {
715 let (e_ty, cause);
716 if let Some(e) = expr_opt {
717 let opt_coerce_to = {
720 let mut enclosing_breakables = self.enclosing_breakables.borrow_mut();
724 match enclosing_breakables.opt_find_breakable(target_id) {
725 Some(ctxt) => ctxt.coerce.as_ref().map(|coerce| coerce.expected_ty()),
726 None => {
727 return Ty::new_error_with_message(
729 tcx,
730 expr.span,
731 "break was outside loop, but no error was emitted",
732 );
733 }
734 }
735 };
736
737 let coerce_to = opt_coerce_to.unwrap_or_else(|| {
742 let guar = self.dcx().span_delayed_bug(
743 expr.span,
744 "illegal break with value found but no error reported",
745 );
746 self.set_tainted_by_errors(guar);
747 Ty::new_error(tcx, guar)
748 });
749
750 e_ty = self.check_expr_with_hint(e, coerce_to);
752 cause = self.misc(e.span);
753 } else {
754 e_ty = tcx.types.unit;
757 cause = self.misc(expr.span);
758 }
759
760 let mut enclosing_breakables = self.enclosing_breakables.borrow_mut();
764 let Some(ctxt) = enclosing_breakables.opt_find_breakable(target_id) else {
765 return Ty::new_error_with_message(
767 tcx,
768 expr.span,
769 "break was outside loop, but no error was emitted",
770 );
771 };
772
773 if let Some(ref mut coerce) = ctxt.coerce {
774 if let Some(e) = expr_opt {
775 coerce.coerce(self, &cause, e, e_ty);
776 } else {
777 if !e_ty.is_unit() {
::core::panicking::panic("assertion failed: e_ty.is_unit()")
};assert!(e_ty.is_unit());
778 let ty = coerce.expected_ty();
779 coerce.coerce_forced_unit(
780 self,
781 &cause,
782 |mut err| {
783 self.suggest_missing_semicolon(&mut err, expr, e_ty, false, false);
784 self.suggest_mismatched_types_on_tail(
785 &mut err, expr, ty, e_ty, target_id,
786 );
787 let error =
788 Some(TypeError::Sorts(ExpectedFound { expected: ty, found: e_ty }));
789 self.annotate_loop_expected_due_to_inference(err, expr, error);
790 if let Some(val) =
791 self.err_ctxt().ty_kind_suggestion(self.param_env, ty)
792 {
793 err.span_suggestion_verbose(
794 expr.span.shrink_to_hi(),
795 "give the `break` a value of the expected type",
796 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!(" {0}", val))
})format!(" {val}"),
797 Applicability::HasPlaceholders,
798 );
799 }
800 },
801 false,
802 );
803 }
804 } else {
805 if !(expr_opt.is_none() || self.tainted_by_errors().is_some()) {
::core::panicking::panic("assertion failed: expr_opt.is_none() || self.tainted_by_errors().is_some()")
};assert!(expr_opt.is_none() || self.tainted_by_errors().is_some());
813 }
814
815 ctxt.may_break |= !self.diverges.get().is_always();
819
820 tcx.types.never
822 } else {
823 let err = Ty::new_error_with_message(
828 self.tcx,
829 expr.span,
830 "break was outside loop, but no error was emitted",
831 );
832
833 if let Some(e) = expr_opt {
836 self.check_expr_with_hint(e, err);
837
838 if let ExprKind::Path(QPath::Resolved(_, path)) = e.kind {
841 if let [segment] = path.segments
842 && segment.ident.name == sym::rust
843 {
844 fatally_break_rust(self.tcx, expr.span);
845 }
846 }
847 }
848
849 err
851 }
852 }
853
854 fn check_expr_continue(
855 &self,
856 destination: hir::Destination,
857 expr: &'tcx hir::Expr<'tcx>,
858 ) -> Ty<'tcx> {
859 if let Ok(target_id) = destination.target_id {
860 if let hir::Node::Expr(hir::Expr { kind: ExprKind::Loop(..), .. }) =
861 self.tcx.hir_node(target_id)
862 {
863 self.tcx.types.never
864 } else {
865 let guar = self.dcx().span_delayed_bug(
868 expr.span,
869 "found `continue` not pointing to loop, but no error reported",
870 );
871 Ty::new_error(self.tcx, guar)
872 }
873 } else {
874 Ty::new_misc_error(self.tcx)
876 }
877 }
878
879 fn check_expr_return(
880 &self,
881 expr_opt: Option<&'tcx hir::Expr<'tcx>>,
882 expr: &'tcx hir::Expr<'tcx>,
883 ) -> Ty<'tcx> {
884 if self.ret_coercion.is_none() {
885 self.emit_return_outside_of_fn_body(expr, ReturnLikeStatementKind::Return);
886
887 if let Some(e) = expr_opt {
888 self.check_expr(e);
891 }
892 } else if let Some(e) = expr_opt {
893 if self.ret_coercion_span.get().is_none() {
894 self.ret_coercion_span.set(Some(e.span));
895 }
896 self.check_return_or_body_tail(e, true);
897 } else {
898 let mut coercion = self.ret_coercion.as_ref().unwrap().borrow_mut();
899 if self.ret_coercion_span.get().is_none() {
900 self.ret_coercion_span.set(Some(expr.span));
901 }
902 let cause = self.cause(expr.span, ObligationCauseCode::ReturnNoExpression);
903 if let Some((_, fn_decl)) = self.get_fn_decl(expr.hir_id) {
904 coercion.coerce_forced_unit(
905 self,
906 &cause,
907 |db| {
908 let span = fn_decl.output.span();
909 if let Ok(snippet) = self.tcx.sess.source_map().span_to_snippet(span) {
910 db.span_label(
911 span,
912 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("expected `{0}` because of this return type",
snippet))
})format!("expected `{snippet}` because of this return type"),
913 );
914 }
915 },
916 true,
917 );
918 } else {
919 coercion.coerce_forced_unit(self, &cause, |_| (), true);
920 }
921 }
922 self.tcx.types.never
923 }
924
925 fn check_expr_become(
926 &self,
927 call: &'tcx hir::Expr<'tcx>,
928 expr: &'tcx hir::Expr<'tcx>,
929 ) -> Ty<'tcx> {
930 match &self.ret_coercion {
931 Some(ret_coercion) => {
932 let ret_ty = ret_coercion.borrow().expected_ty();
933 let call_expr_ty = self.check_expr_with_hint(call, ret_ty);
934
935 self.demand_suptype(expr.span, ret_ty, call_expr_ty);
938 }
939 None => {
940 self.emit_return_outside_of_fn_body(expr, ReturnLikeStatementKind::Become);
941
942 self.check_expr(call);
945 }
946 }
947
948 self.tcx.types.never
949 }
950
951 pub(super) fn check_return_or_body_tail(
960 &self,
961 return_expr: &'tcx hir::Expr<'tcx>,
962 explicit_return: bool,
963 ) {
964 let ret_coercion = self.ret_coercion.as_ref().unwrap_or_else(|| {
965 ::rustc_middle::util::bug::span_bug_fmt(return_expr.span,
format_args!("check_return_expr called outside fn body"))span_bug!(return_expr.span, "check_return_expr called outside fn body")
966 });
967
968 let ret_ty = ret_coercion.borrow().expected_ty();
969 let return_expr_ty = self.check_expr_with_hint(return_expr, ret_ty);
970 let mut span = return_expr.span;
971 let mut hir_id = return_expr.hir_id;
972 if !explicit_return
975 && let ExprKind::Block(body, _) = return_expr.kind
976 && let Some(last_expr) = body.expr
977 {
978 span = last_expr.span;
979 hir_id = last_expr.hir_id;
980 }
981 ret_coercion.borrow_mut().coerce(
982 self,
983 &self.cause(span, ObligationCauseCode::ReturnValue(return_expr.hir_id)),
984 return_expr,
985 return_expr_ty,
986 );
987
988 if let Some(fn_sig) = self.body_fn_sig()
989 && fn_sig.output().has_opaque_types()
990 {
991 self.select_obligations_where_possible(|errors| {
994 self.point_at_return_for_opaque_ty_error(
995 errors,
996 hir_id,
997 span,
998 return_expr_ty,
999 return_expr.span,
1000 );
1001 });
1002 }
1003 }
1004
1005 fn emit_return_outside_of_fn_body(&self, expr: &hir::Expr<'_>, kind: ReturnLikeStatementKind) {
1010 let mut err = ReturnStmtOutsideOfFnBody {
1011 span: expr.span,
1012 encl_body_span: None,
1013 encl_fn_span: None,
1014 statement_kind: kind,
1015 };
1016
1017 let encl_item_id = self.tcx.hir_get_parent_item(expr.hir_id);
1018
1019 if let hir::Node::Item(hir::Item {
1020 kind: hir::ItemKind::Fn { .. },
1021 span: encl_fn_span,
1022 ..
1023 })
1024 | hir::Node::TraitItem(hir::TraitItem {
1025 kind: hir::TraitItemKind::Fn(_, hir::TraitFn::Provided(_)),
1026 span: encl_fn_span,
1027 ..
1028 })
1029 | hir::Node::ImplItem(hir::ImplItem {
1030 kind: hir::ImplItemKind::Fn(..),
1031 span: encl_fn_span,
1032 ..
1033 }) = self.tcx.hir_node_by_def_id(encl_item_id.def_id)
1034 {
1035 let encl_body_owner_id = self.tcx.hir_enclosing_body_owner(expr.hir_id);
1039
1040 match (&encl_item_id.def_id, &encl_body_owner_id) {
(left_val, right_val) => {
if *left_val == *right_val {
let kind = ::core::panicking::AssertKind::Ne;
::core::panicking::assert_failed(kind, &*left_val, &*right_val,
::core::option::Option::None);
}
}
};assert_ne!(encl_item_id.def_id, encl_body_owner_id);
1043
1044 let encl_body = self.tcx.hir_body_owned_by(encl_body_owner_id);
1045
1046 err.encl_body_span = Some(encl_body.value.span);
1047 err.encl_fn_span = Some(*encl_fn_span);
1048 }
1049
1050 self.dcx().emit_err(err);
1051 }
1052
1053 fn point_at_return_for_opaque_ty_error(
1054 &self,
1055 errors: &mut Vec<traits::FulfillmentError<'tcx>>,
1056 hir_id: HirId,
1057 span: Span,
1058 return_expr_ty: Ty<'tcx>,
1059 return_span: Span,
1060 ) {
1061 if span == return_span {
1063 return;
1064 }
1065 for err in errors {
1066 let cause = &mut err.obligation.cause;
1067 if let ObligationCauseCode::OpaqueReturnType(None) = cause.code() {
1068 let new_cause = self.cause(
1069 cause.span,
1070 ObligationCauseCode::OpaqueReturnType(Some((return_expr_ty, hir_id))),
1071 );
1072 *cause = new_cause;
1073 }
1074 }
1075 }
1076
1077 pub(crate) fn check_lhs_assignable(
1078 &self,
1079 lhs: &'tcx hir::Expr<'tcx>,
1080 code: ErrCode,
1081 op_span: Span,
1082 adjust_err: impl FnOnce(&mut Diag<'_>),
1083 ) {
1084 if lhs.is_syntactic_place_expr() {
1085 return;
1086 }
1087
1088 if contains_let_in_chain(lhs) {
1091 return;
1092 }
1093
1094 let mut err = self.dcx().struct_span_err(op_span, "invalid left-hand side of assignment");
1095 err.code(code);
1096 err.span_label(lhs.span, "cannot assign to this expression");
1097
1098 self.comes_from_while_condition(lhs.hir_id, |expr| {
1099 err.span_suggestion_verbose(
1100 expr.span.shrink_to_lo(),
1101 "you might have meant to use pattern destructuring",
1102 "let ",
1103 Applicability::MachineApplicable,
1104 );
1105 });
1106 self.check_for_missing_semi(lhs, &mut err);
1107
1108 adjust_err(&mut err);
1109
1110 err.emit();
1111 }
1112
1113 pub(crate) fn check_for_missing_semi(
1115 &self,
1116 expr: &'tcx hir::Expr<'tcx>,
1117 err: &mut Diag<'_>,
1118 ) -> bool {
1119 if let hir::ExprKind::Binary(binop, lhs, rhs) = expr.kind
1120 && let hir::BinOpKind::Mul = binop.node
1121 && self.tcx.sess.source_map().is_multiline(lhs.span.between(rhs.span))
1122 && rhs.is_syntactic_place_expr()
1123 {
1124 err.span_suggestion_verbose(
1129 lhs.span.shrink_to_hi(),
1130 "you might have meant to write a semicolon here",
1131 ";",
1132 Applicability::MachineApplicable,
1133 );
1134 return true;
1135 }
1136 false
1137 }
1138
1139 pub(super) fn comes_from_while_condition(
1143 &self,
1144 original_expr_id: HirId,
1145 then: impl FnOnce(&hir::Expr<'_>),
1146 ) {
1147 let mut parent = self.tcx.parent_hir_id(original_expr_id);
1148 loop {
1149 let node = self.tcx.hir_node(parent);
1150 match node {
1151 hir::Node::Expr(hir::Expr {
1152 kind:
1153 hir::ExprKind::Loop(
1154 hir::Block {
1155 expr:
1156 Some(hir::Expr {
1157 kind:
1158 hir::ExprKind::Match(expr, ..) | hir::ExprKind::If(expr, ..),
1159 ..
1160 }),
1161 ..
1162 },
1163 _,
1164 hir::LoopSource::While,
1165 _,
1166 ),
1167 ..
1168 }) => {
1169 if self.tcx.hir_parent_id_iter(original_expr_id).any(|id| id == expr.hir_id) {
1173 then(expr);
1174 }
1175 break;
1176 }
1177 hir::Node::Item(_)
1178 | hir::Node::ImplItem(_)
1179 | hir::Node::TraitItem(_)
1180 | hir::Node::Crate(_) => break,
1181 _ => {
1182 parent = self.tcx.parent_hir_id(parent);
1183 }
1184 }
1185 }
1186 }
1187
1188 fn check_expr_if(
1191 &self,
1192 expr_id: HirId,
1193 cond_expr: &'tcx hir::Expr<'tcx>,
1194 then_expr: &'tcx hir::Expr<'tcx>,
1195 opt_else_expr: Option<&'tcx hir::Expr<'tcx>>,
1196 sp: Span,
1197 orig_expected: Expectation<'tcx>,
1198 ) -> Ty<'tcx> {
1199 let cond_ty = self.check_expr_has_type_or_error(cond_expr, self.tcx.types.bool, |_| {});
1200
1201 self.warn_if_unreachable(
1202 cond_expr.hir_id,
1203 then_expr.span,
1204 "block in `if` or `while` expression",
1205 );
1206
1207 let cond_diverges = self.diverges.get();
1208 self.diverges.set(Diverges::Maybe);
1209
1210 let expected = orig_expected.try_structurally_resolve_and_adjust_for_branches(self, sp);
1211 let then_ty = self.check_expr_with_expectation(then_expr, expected);
1212 let then_diverges = self.diverges.get();
1213 self.diverges.set(Diverges::Maybe);
1214
1215 let coerce_to_ty = expected.coercion_target_type(self, sp);
1222 let mut coerce = CoerceMany::with_capacity(coerce_to_ty, 2);
1223
1224 coerce.coerce(self, &self.misc(sp), then_expr, then_ty);
1225
1226 if let Some(else_expr) = opt_else_expr {
1227 let else_ty = self.check_expr_with_expectation(else_expr, expected);
1228 let else_diverges = self.diverges.get();
1229
1230 let tail_defines_return_position_impl_trait =
1231 self.return_position_impl_trait_from_match_expectation(orig_expected);
1232 let if_cause =
1233 self.if_cause(expr_id, else_expr, tail_defines_return_position_impl_trait);
1234
1235 coerce.coerce(self, &if_cause, else_expr, else_ty);
1236
1237 self.diverges.set(cond_diverges | then_diverges & else_diverges);
1239 } else {
1240 self.if_fallback_coercion(sp, cond_expr, then_expr, &mut coerce);
1241
1242 self.diverges.set(cond_diverges);
1244 }
1245
1246 let result_ty = coerce.complete(self);
1247 if let Err(guar) = cond_ty.error_reported() {
1248 Ty::new_error(self.tcx, guar)
1249 } else {
1250 result_ty
1251 }
1252 }
1253
1254 fn check_expr_assign(
1257 &self,
1258 expr: &'tcx hir::Expr<'tcx>,
1259 expected: Expectation<'tcx>,
1260 lhs: &'tcx hir::Expr<'tcx>,
1261 rhs: &'tcx hir::Expr<'tcx>,
1262 span: Span,
1263 ) -> Ty<'tcx> {
1264 let expected_ty = expected.only_has_type(self);
1265 if expected_ty == Some(self.tcx.types.bool) {
1266 let guar = self.expr_assign_expected_bool_error(expr, lhs, rhs, span);
1267 return Ty::new_error(self.tcx, guar);
1268 }
1269
1270 let lhs_ty = self.check_expr_with_needs(lhs, Needs::MutPlace);
1271
1272 let suggest_deref_binop = |err: &mut Diag<'_>, rhs_ty: Ty<'tcx>| {
1273 if let Some(lhs_deref_ty) = self.deref_once_mutably_for_diagnostic(lhs_ty) {
1274 let lhs_deref_ty_is_sized = self
1277 .infcx
1278 .type_implements_trait(
1279 self.tcx.require_lang_item(LangItem::Sized, span),
1280 [lhs_deref_ty],
1281 self.param_env,
1282 )
1283 .may_apply();
1284 if lhs_deref_ty_is_sized && self.may_coerce(rhs_ty, lhs_deref_ty) {
1285 err.span_suggestion_verbose(
1286 lhs.span.shrink_to_lo(),
1287 "consider dereferencing here to assign to the mutably borrowed value",
1288 "*",
1289 Applicability::MachineApplicable,
1290 );
1291 }
1292 }
1293 };
1294
1295 let rhs_ty = self.check_expr_with_hint(rhs, lhs_ty);
1298 if let Err(mut diag) =
1299 self.demand_coerce_diag(rhs, rhs_ty, lhs_ty, Some(lhs), AllowTwoPhase::No)
1300 {
1301 suggest_deref_binop(&mut diag, rhs_ty);
1302 diag.emit();
1303 }
1304
1305 self.check_lhs_assignable(lhs, E0070, span, |err| {
1306 if let Some(rhs_ty) = self.typeck_results.borrow().expr_ty_opt(rhs) {
1307 suggest_deref_binop(err, rhs_ty);
1308 }
1309 });
1310
1311 self.require_type_is_sized(lhs_ty, lhs.span, ObligationCauseCode::AssignmentLhsSized);
1312
1313 if let Err(guar) = (lhs_ty, rhs_ty).error_reported() {
1314 Ty::new_error(self.tcx, guar)
1315 } else {
1316 self.tcx.types.unit
1317 }
1318 }
1319
1320 fn expr_assign_expected_bool_error(
1324 &self,
1325 expr: &'tcx hir::Expr<'tcx>,
1326 lhs: &'tcx hir::Expr<'tcx>,
1327 rhs: &'tcx hir::Expr<'tcx>,
1328 span: Span,
1329 ) -> ErrorGuaranteed {
1330 let actual_ty = self.tcx.types.unit;
1331 let expected_ty = self.tcx.types.bool;
1332 let mut err = self.demand_suptype_diag(expr.span, expected_ty, actual_ty).unwrap_err();
1333 let lhs_ty = self.check_expr(lhs);
1334 let rhs_ty = self.check_expr(rhs);
1335 let refs_can_coerce = |lhs: Ty<'tcx>, rhs: Ty<'tcx>| {
1336 let lhs = Ty::new_imm_ref(self.tcx, self.tcx.lifetimes.re_erased, lhs.peel_refs());
1337 let rhs = Ty::new_imm_ref(self.tcx, self.tcx.lifetimes.re_erased, rhs.peel_refs());
1338 self.may_coerce(rhs, lhs)
1339 };
1340 let (applicability, eq) = if self.may_coerce(rhs_ty, lhs_ty) {
1341 (Applicability::MachineApplicable, true)
1342 } else if refs_can_coerce(rhs_ty, lhs_ty) {
1343 (Applicability::MaybeIncorrect, true)
1346 } else if let ExprKind::Binary(
1347 Spanned { node: hir::BinOpKind::And | hir::BinOpKind::Or, .. },
1348 _,
1349 rhs_expr,
1350 ) = lhs.kind
1351 {
1352 let actual_lhs = self.check_expr(rhs_expr);
1355 let may_eq = self.may_coerce(rhs_ty, actual_lhs) || refs_can_coerce(rhs_ty, actual_lhs);
1356 (Applicability::MaybeIncorrect, may_eq)
1357 } else if let ExprKind::Binary(
1358 Spanned { node: hir::BinOpKind::And | hir::BinOpKind::Or, .. },
1359 lhs_expr,
1360 _,
1361 ) = rhs.kind
1362 {
1363 let actual_rhs = self.check_expr(lhs_expr);
1366 let may_eq = self.may_coerce(actual_rhs, lhs_ty) || refs_can_coerce(actual_rhs, lhs_ty);
1367 (Applicability::MaybeIncorrect, may_eq)
1368 } else {
1369 (Applicability::MaybeIncorrect, false)
1370 };
1371
1372 if !lhs.is_syntactic_place_expr()
1373 && lhs.is_approximately_pattern()
1374 && !#[allow(non_exhaustive_omitted_patterns)] match lhs.kind {
hir::ExprKind::Lit(_) => true,
_ => false,
}matches!(lhs.kind, hir::ExprKind::Lit(_))
1375 {
1376 if let hir::Node::Expr(hir::Expr { kind: ExprKind::If { .. }, .. }) =
1378 self.tcx.parent_hir_node(expr.hir_id)
1379 {
1380 err.span_suggestion_verbose(
1381 expr.span.shrink_to_lo(),
1382 "you might have meant to use pattern matching",
1383 "let ",
1384 applicability,
1385 );
1386 };
1387 }
1388 if eq {
1389 err.span_suggestion_verbose(
1390 span.shrink_to_hi(),
1391 "you might have meant to compare for equality",
1392 '=',
1393 applicability,
1394 );
1395 }
1396
1397 err.emit_unless_delay(lhs_ty.references_error() || rhs_ty.references_error())
1400 }
1401
1402 pub(super) fn check_expr_let(
1403 &self,
1404 let_expr: &'tcx hir::LetExpr<'tcx>,
1405 hir_id: HirId,
1406 ) -> Ty<'tcx> {
1407 GatherLocalsVisitor::gather_from_let_expr(self, let_expr, hir_id);
1408
1409 let init = let_expr.init;
1411 self.warn_if_unreachable(init.hir_id, init.span, "block in `let` expression");
1412
1413 self.check_decl((let_expr, hir_id).into());
1415
1416 if let ast::Recovered::Yes(error_guaranteed) = let_expr.recovered {
1418 self.set_tainted_by_errors(error_guaranteed);
1419 Ty::new_error(self.tcx, error_guaranteed)
1420 } else {
1421 self.tcx.types.bool
1422 }
1423 }
1424
1425 fn check_expr_loop(
1426 &self,
1427 body: &'tcx hir::Block<'tcx>,
1428 source: hir::LoopSource,
1429 expected: Expectation<'tcx>,
1430 expr: &'tcx hir::Expr<'tcx>,
1431 ) -> Ty<'tcx> {
1432 let coerce = match source {
1433 hir::LoopSource::Loop => {
1435 let coerce_to = expected.coercion_target_type(self, body.span);
1436 Some(CoerceMany::new(coerce_to))
1437 }
1438
1439 hir::LoopSource::While | hir::LoopSource::ForLoop => None,
1440 };
1441
1442 let ctxt = BreakableCtxt {
1443 coerce,
1444 may_break: false, };
1446
1447 let (ctxt, ()) = self.with_breakable_ctxt(expr.hir_id, ctxt, || {
1448 self.check_block_no_value(body);
1449 });
1450
1451 if ctxt.may_break {
1452 self.diverges.set(Diverges::Maybe);
1455 } else {
1456 self.diverges.set(self.diverges.get() | Diverges::always(expr.span));
1457 }
1458
1459 if ctxt.coerce.is_none() && !ctxt.may_break {
1465 self.dcx().span_bug(body.span, "no coercion, but loop may not break");
1466 }
1467 ctxt.coerce.map(|c| c.complete(self)).unwrap_or_else(|| self.tcx.types.unit)
1468 }
1469
1470 fn check_expr_method_call(
1472 &self,
1473 expr: &'tcx hir::Expr<'tcx>,
1474 segment: &'tcx hir::PathSegment<'tcx>,
1475 rcvr: &'tcx hir::Expr<'tcx>,
1476 args: &'tcx [hir::Expr<'tcx>],
1477 expected: Expectation<'tcx>,
1478 ) -> Ty<'tcx> {
1479 let rcvr_t = self.check_expr(rcvr);
1480 let rcvr_t = self.try_structurally_resolve_type(rcvr.span, rcvr_t);
1481
1482 match self.lookup_method(rcvr_t, segment, segment.ident.span, expr, rcvr, args) {
1483 Ok(method) => {
1484 self.write_method_call_and_enforce_effects(expr.hir_id, expr.span, method);
1485
1486 self.check_argument_types(
1487 segment.ident.span,
1488 expr,
1489 &method.sig.inputs()[1..],
1490 method.sig.output(),
1491 expected,
1492 args,
1493 method.sig.c_variadic,
1494 TupleArgumentsFlag::DontTupleArguments,
1495 Some(method.def_id),
1496 );
1497
1498 self.check_call_abi(method.sig.abi, expr.span);
1499
1500 method.sig.output()
1501 }
1502 Err(error) => {
1503 let guar = self.report_method_error(expr.hir_id, rcvr_t, error, expected, false);
1504
1505 let err_inputs = self.err_args(args.len(), guar);
1506 let err_output = Ty::new_error(self.tcx, guar);
1507
1508 self.check_argument_types(
1509 segment.ident.span,
1510 expr,
1511 &err_inputs,
1512 err_output,
1513 NoExpectation,
1514 args,
1515 false,
1516 TupleArgumentsFlag::DontTupleArguments,
1517 None,
1518 );
1519
1520 err_output
1521 }
1522 }
1523 }
1524
1525 fn check_expr_use(
1527 &self,
1528 used_expr: &'tcx hir::Expr<'tcx>,
1529 expected: Expectation<'tcx>,
1530 ) -> Ty<'tcx> {
1531 self.check_expr_with_expectation(used_expr, expected)
1532 }
1533
1534 fn check_expr_cast(
1535 &self,
1536 e: &'tcx hir::Expr<'tcx>,
1537 t: &'tcx hir::Ty<'tcx>,
1538 expr: &'tcx hir::Expr<'tcx>,
1539 ) -> Ty<'tcx> {
1540 let t_cast = self.lower_ty_saving_user_provided_ty(t);
1543 let t_cast = self.resolve_vars_if_possible(t_cast);
1544 let t_expr = self.check_expr_with_expectation(e, ExpectCastableToType(t_cast));
1545 let t_expr = self.resolve_vars_if_possible(t_expr);
1546
1547 if let Err(guar) = (t_expr, t_cast).error_reported() {
1549 Ty::new_error(self.tcx, guar)
1550 } else {
1551 let mut deferred_cast_checks = self.deferred_cast_checks.borrow_mut();
1553 match cast::CastCheck::new(self, e, t_expr, t_cast, t.span, expr.span) {
1554 Ok(cast_check) => {
1555 {
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_hir_typeck/src/expr.rs:1555",
"rustc_hir_typeck::expr", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/expr.rs"),
::tracing_core::__macro_support::Option::Some(1555u32),
::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::expr"),
::tracing_core::field::FieldSet::new(&["message"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::EVENT)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let enabled =
::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
&&
::tracing::Level::DEBUG <=
::tracing::level_filters::LevelFilter::current() &&
{
let interest = __CALLSITE.interest();
!interest.is_never() &&
::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
interest)
};
if enabled {
(|value_set: ::tracing::field::ValueSet|
{
let meta = __CALLSITE.metadata();
::tracing::Event::dispatch(meta, &value_set);
;
})({
#[allow(unused_imports)]
use ::tracing::field::{debug, display, Value};
let mut iter = __CALLSITE.metadata().fields().iter();
__CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&format_args!("check_expr_cast: deferring cast from {0:?} to {1:?}: {2:?}",
t_cast, t_expr, cast_check) as &dyn Value))])
});
} else { ; }
};debug!(
1556 "check_expr_cast: deferring cast from {:?} to {:?}: {:?}",
1557 t_cast, t_expr, cast_check,
1558 );
1559 deferred_cast_checks.push(cast_check);
1560 t_cast
1561 }
1562 Err(guar) => Ty::new_error(self.tcx, guar),
1563 }
1564 }
1565 }
1566
1567 fn check_expr_unsafe_binder_cast(
1568 &self,
1569 span: Span,
1570 kind: ast::UnsafeBinderCastKind,
1571 inner_expr: &'tcx hir::Expr<'tcx>,
1572 hir_ty: Option<&'tcx hir::Ty<'tcx>>,
1573 expected: Expectation<'tcx>,
1574 ) -> Ty<'tcx> {
1575 match kind {
1576 ast::UnsafeBinderCastKind::Wrap => {
1577 let ascribed_ty =
1578 hir_ty.map(|hir_ty| self.lower_ty_saving_user_provided_ty(hir_ty));
1579 let expected_ty = expected.only_has_type(self);
1580 let binder_ty = match (ascribed_ty, expected_ty) {
1581 (Some(ascribed_ty), Some(expected_ty)) => {
1582 self.demand_eqtype(inner_expr.span, expected_ty, ascribed_ty);
1583 expected_ty
1584 }
1585 (Some(ty), None) | (None, Some(ty)) => ty,
1586 (None, None) => self.next_ty_var(inner_expr.span),
1590 };
1591
1592 let binder_ty = self.structurally_resolve_type(inner_expr.span, binder_ty);
1593 let hint_ty = match *binder_ty.kind() {
1594 ty::UnsafeBinder(binder) => self.instantiate_binder_with_fresh_vars(
1595 inner_expr.span,
1596 infer::BoundRegionConversionTime::HigherRankedType,
1597 binder.into(),
1598 ),
1599 ty::Error(e) => Ty::new_error(self.tcx, e),
1600 _ => {
1601 let guar = self
1602 .dcx()
1603 .struct_span_err(
1604 hir_ty.map_or(span, |hir_ty| hir_ty.span),
1605 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("`wrap_binder!()` can only wrap into unsafe binder, not {0}",
binder_ty.sort_string(self.tcx)))
})format!(
1606 "`wrap_binder!()` can only wrap into unsafe binder, not {}",
1607 binder_ty.sort_string(self.tcx)
1608 ),
1609 )
1610 .with_note("unsafe binders are the only valid output of wrap")
1611 .emit();
1612 Ty::new_error(self.tcx, guar)
1613 }
1614 };
1615
1616 self.check_expr_has_type_or_error(inner_expr, hint_ty, |_| {});
1617
1618 binder_ty
1619 }
1620 ast::UnsafeBinderCastKind::Unwrap => {
1621 let ascribed_ty =
1622 hir_ty.map(|hir_ty| self.lower_ty_saving_user_provided_ty(hir_ty));
1623 let hint_ty = ascribed_ty.unwrap_or_else(|| self.next_ty_var(inner_expr.span));
1624 let binder_ty = self.check_expr_has_type_or_error(inner_expr, hint_ty, |_| {});
1626
1627 let binder_ty = self.structurally_resolve_type(inner_expr.span, binder_ty);
1630 match *binder_ty.kind() {
1631 ty::UnsafeBinder(binder) => self.instantiate_binder_with_fresh_vars(
1632 inner_expr.span,
1633 infer::BoundRegionConversionTime::HigherRankedType,
1634 binder.into(),
1635 ),
1636 ty::Error(e) => Ty::new_error(self.tcx, e),
1637 _ => {
1638 let guar = self
1639 .dcx()
1640 .struct_span_err(
1641 hir_ty.map_or(inner_expr.span, |hir_ty| hir_ty.span),
1642 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("expected unsafe binder, found {0} as input of `unwrap_binder!()`",
binder_ty.sort_string(self.tcx)))
})format!(
1643 "expected unsafe binder, found {} as input of \
1644 `unwrap_binder!()`",
1645 binder_ty.sort_string(self.tcx)
1646 ),
1647 )
1648 .with_note("only an unsafe binder type can be unwrapped")
1649 .emit();
1650 Ty::new_error(self.tcx, guar)
1651 }
1652 }
1653 }
1654 }
1655 }
1656
1657 fn check_expr_array(
1658 &self,
1659 args: &'tcx [hir::Expr<'tcx>],
1660 expected: Expectation<'tcx>,
1661 expr: &'tcx hir::Expr<'tcx>,
1662 ) -> Ty<'tcx> {
1663 let element_ty = if !args.is_empty() {
1664 if self.diverges.get() != Diverges::Maybe {
1667 self.dcx()
1668 .struct_span_err(expr.span, "unexpected divergence state in checking array")
1669 .delay_as_bug();
1670 }
1671
1672 let coerce_to = expected
1673 .to_option(self)
1674 .and_then(|uty| {
1675 self.try_structurally_resolve_type(expr.span, uty)
1676 .builtin_index()
1677 .filter(|t| !self.try_structurally_resolve_type(expr.span, *t).is_ty_var())
1680 })
1681 .unwrap_or_else(|| self.next_ty_var(expr.span));
1682 let mut coerce = CoerceMany::with_capacity(coerce_to, args.len());
1683
1684 for e in args {
1685 let e_ty = self.check_expr_with_hint(e, coerce_to);
1691 let cause = self.misc(e.span);
1692 coerce.coerce(self, &cause, e, e_ty);
1693 }
1694 coerce.complete(self)
1695 } else {
1696 self.next_ty_var(expr.span)
1697 };
1698 let array_len = args.len() as u64;
1699 self.suggest_array_len(expr, array_len);
1700 Ty::new_array(self.tcx, element_ty, array_len)
1701 }
1702
1703 fn suggest_array_len(&self, expr: &'tcx hir::Expr<'tcx>, array_len: u64) {
1704 let parent_node = self.tcx.hir_parent_iter(expr.hir_id).find(|(_, node)| {
1705 !#[allow(non_exhaustive_omitted_patterns)] match node {
hir::Node::Expr(hir::Expr { kind: hir::ExprKind::AddrOf(..), .. }) =>
true,
_ => false,
}matches!(node, hir::Node::Expr(hir::Expr { kind: hir::ExprKind::AddrOf(..), .. }))
1706 });
1707 let Some((_, hir::Node::LetStmt(hir::LetStmt { ty: Some(ty), .. }))) = parent_node else {
1708 return;
1709 };
1710 if let hir::TyKind::Array(_, ct) = ty.peel_refs().kind {
1711 let span = ct.span;
1712 self.dcx().try_steal_modify_and_emit_err(
1713 span,
1714 StashKey::UnderscoreForArrayLengths,
1715 |err| {
1716 err.span_suggestion(
1717 span,
1718 "consider specifying the array length",
1719 array_len,
1720 Applicability::MaybeIncorrect,
1721 );
1722 },
1723 );
1724 }
1725 }
1726
1727 pub(super) fn check_expr_const_block(
1728 &self,
1729 block: &'tcx hir::ConstBlock,
1730 expected: Expectation<'tcx>,
1731 ) -> Ty<'tcx> {
1732 let body = self.tcx.hir_body(block.body);
1733
1734 let def_id = block.def_id;
1736 let fcx = FnCtxt::new(self, self.param_env, def_id);
1737
1738 let ty = fcx.check_expr_with_expectation(body.value, expected);
1739 fcx.require_type_is_sized(ty, body.value.span, ObligationCauseCode::SizedConstOrStatic);
1740 fcx.write_ty(block.hir_id, ty);
1741 ty
1742 }
1743
1744 fn check_expr_repeat(
1745 &self,
1746 element: &'tcx hir::Expr<'tcx>,
1747 count: &'tcx hir::ConstArg<'tcx>,
1748 expected: Expectation<'tcx>,
1749 expr: &'tcx hir::Expr<'tcx>,
1750 ) -> Ty<'tcx> {
1751 let tcx = self.tcx;
1752 let count_span = count.span;
1753 let count = self.try_structurally_resolve_const(
1754 count_span,
1755 self.normalize(count_span, self.lower_const_arg(count, tcx.types.usize)),
1756 );
1757
1758 if let Some(count) = count.try_to_target_usize(tcx) {
1759 self.suggest_array_len(expr, count);
1760 }
1761
1762 let uty = match expected {
1763 ExpectHasType(uty) => uty.builtin_index(),
1764 _ => None,
1765 };
1766
1767 let (element_ty, t) = match uty {
1768 Some(uty) => {
1769 self.check_expr_coercible_to_type(element, uty, None);
1770 (uty, uty)
1771 }
1772 None => {
1773 let ty = self.next_ty_var(element.span);
1774 let element_ty = self.check_expr_has_type_or_error(element, ty, |_| {});
1775 (element_ty, ty)
1776 }
1777 };
1778
1779 if let Err(guar) = element_ty.error_reported() {
1780 return Ty::new_error(tcx, guar);
1781 }
1782
1783 self.deferred_repeat_expr_checks.borrow_mut().push((element, element_ty, count));
1787
1788 let ty = Ty::new_array_with_const_len(tcx, t, count);
1789 self.register_wf_obligation(ty.into(), expr.span, ObligationCauseCode::WellFormed(None));
1790 ty
1791 }
1792
1793 fn check_expr_tuple(
1794 &self,
1795 elements: &'tcx [hir::Expr<'tcx>],
1796 expected: Expectation<'tcx>,
1797 expr: &'tcx hir::Expr<'tcx>,
1798 ) -> Ty<'tcx> {
1799 let mut expectations = expected
1800 .only_has_type(self)
1801 .and_then(|ty| self.try_structurally_resolve_type(expr.span, ty).opt_tuple_fields())
1802 .unwrap_or_default()
1803 .iter();
1804
1805 let elements = elements.iter().map(|e| {
1806 let ty = expectations.next().unwrap_or_else(|| self.next_ty_var(e.span));
1807 self.check_expr_coercible_to_type(e, ty, None);
1808 ty
1809 });
1810
1811 let tuple = Ty::new_tup_from_iter(self.tcx, elements);
1812
1813 if let Err(guar) = tuple.error_reported() {
1814 Ty::new_error(self.tcx, guar)
1815 } else {
1816 self.require_type_is_sized(
1817 tuple,
1818 expr.span,
1819 ObligationCauseCode::TupleInitializerSized,
1820 );
1821 tuple
1822 }
1823 }
1824
1825 fn check_expr_struct(
1826 &self,
1827 expr: &hir::Expr<'tcx>,
1828 expected: Expectation<'tcx>,
1829 qpath: &'tcx QPath<'tcx>,
1830 fields: &'tcx [hir::ExprField<'tcx>],
1831 base_expr: &'tcx hir::StructTailExpr<'tcx>,
1832 ) -> Ty<'tcx> {
1833 let (variant, adt_ty) = match self.check_struct_path(qpath, expr.hir_id) {
1835 Ok(data) => data,
1836 Err(guar) => {
1837 self.check_struct_fields_on_error(fields, base_expr);
1838 return Ty::new_error(self.tcx, guar);
1839 }
1840 };
1841
1842 let adt = adt_ty.ty_adt_def().expect("`check_struct_path` returned non-ADT type");
1844 if variant.field_list_has_applicable_non_exhaustive() {
1845 self.dcx()
1846 .emit_err(StructExprNonExhaustive { span: expr.span, what: adt.variant_descr() });
1847 }
1848
1849 self.check_expr_struct_fields(
1850 adt_ty,
1851 expected,
1852 expr,
1853 qpath.span(),
1854 variant,
1855 fields,
1856 base_expr,
1857 );
1858
1859 self.require_type_is_sized(adt_ty, expr.span, ObligationCauseCode::StructInitializerSized);
1860 adt_ty
1861 }
1862
1863 fn check_expr_struct_fields(
1864 &self,
1865 adt_ty: Ty<'tcx>,
1866 expected: Expectation<'tcx>,
1867 expr: &hir::Expr<'_>,
1868 path_span: Span,
1869 variant: &'tcx ty::VariantDef,
1870 hir_fields: &'tcx [hir::ExprField<'tcx>],
1871 base_expr: &'tcx hir::StructTailExpr<'tcx>,
1872 ) {
1873 let tcx = self.tcx;
1874
1875 let adt_ty = self.try_structurally_resolve_type(path_span, adt_ty);
1876 let adt_ty_hint = expected.only_has_type(self).and_then(|expected| {
1877 self.fudge_inference_if_ok(|| {
1878 let ocx = ObligationCtxt::new(self);
1879 ocx.sup(&self.misc(path_span), self.param_env, expected, adt_ty)?;
1880 if !ocx.try_evaluate_obligations().is_empty() {
1881 return Err(TypeError::Mismatch);
1882 }
1883 Ok(self.resolve_vars_if_possible(adt_ty))
1884 })
1885 .ok()
1886 });
1887 if let Some(adt_ty_hint) = adt_ty_hint {
1888 self.demand_eqtype(path_span, adt_ty_hint, adt_ty);
1890 }
1891
1892 let ty::Adt(adt, args) = adt_ty.kind() else {
1893 ::rustc_middle::util::bug::span_bug_fmt(path_span,
format_args!("non-ADT passed to check_expr_struct_fields"));span_bug!(path_span, "non-ADT passed to check_expr_struct_fields");
1894 };
1895 let adt_kind = adt.adt_kind();
1896
1897 let mut remaining_fields = variant
1898 .fields
1899 .iter_enumerated()
1900 .map(|(i, field)| (field.ident(tcx).normalize_to_macros_2_0(), (i, field)))
1901 .collect::<UnordMap<_, _>>();
1902
1903 let mut seen_fields = FxHashMap::default();
1904
1905 let mut error_happened = false;
1906
1907 if variant.fields.len() != remaining_fields.len() {
1908 let guar =
1911 self.dcx().span_delayed_bug(expr.span, "struct fields have non-unique names");
1912 self.set_tainted_by_errors(guar);
1913 error_happened = true;
1914 }
1915
1916 for (idx, field) in hir_fields.iter().enumerate() {
1918 let ident = tcx.adjust_ident(field.ident, variant.def_id);
1919 let field_type = if let Some((i, v_field)) = remaining_fields.remove(&ident) {
1920 seen_fields.insert(ident, field.span);
1921 self.write_field_index(field.hir_id, i);
1922
1923 if adt_kind != AdtKind::Enum {
1927 tcx.check_stability(v_field.did, Some(field.hir_id), field.span, None);
1928 }
1929
1930 self.field_ty(field.span, v_field, args)
1931 } else {
1932 error_happened = true;
1933 let guar = if let Some(prev_span) = seen_fields.get(&ident) {
1934 self.dcx().emit_err(FieldMultiplySpecifiedInInitializer {
1935 span: field.ident.span,
1936 prev_span: *prev_span,
1937 ident,
1938 })
1939 } else {
1940 self.report_unknown_field(
1941 adt_ty,
1942 variant,
1943 expr,
1944 field,
1945 hir_fields,
1946 adt.variant_descr(),
1947 )
1948 };
1949
1950 Ty::new_error(tcx, guar)
1951 };
1952
1953 self.register_wf_obligation(
1957 field_type.into(),
1958 field.expr.span,
1959 ObligationCauseCode::WellFormed(None),
1960 );
1961
1962 let ty = self.check_expr_with_hint(field.expr, field_type);
1965 let diag = self.demand_coerce_diag(field.expr, ty, field_type, None, AllowTwoPhase::No);
1966
1967 if let Err(diag) = diag {
1968 if idx == hir_fields.len() - 1 {
1969 if remaining_fields.is_empty() {
1970 self.suggest_fru_from_range_and_emit(field, variant, args, diag);
1971 } else {
1972 diag.stash(field.span, StashKey::MaybeFruTypo);
1973 }
1974 } else {
1975 diag.emit();
1976 }
1977 }
1978 }
1979
1980 if adt_kind == AdtKind::Union && hir_fields.len() != 1 {
1982 {
self.dcx().struct_span_err(path_span,
::alloc::__export::must_use({
::alloc::fmt::format(format_args!("union expressions should have exactly one field"))
})).with_code(E0784)
}struct_span_code_err!(
1983 self.dcx(),
1984 path_span,
1985 E0784,
1986 "union expressions should have exactly one field",
1987 )
1988 .emit();
1989 }
1990
1991 if error_happened {
1995 if let hir::StructTailExpr::Base(base_expr) = base_expr {
1996 self.check_expr(base_expr);
1997 }
1998 return;
1999 }
2000
2001 match *base_expr {
2002 hir::StructTailExpr::DefaultFields(span) => {
2003 let mut missing_mandatory_fields = Vec::new();
2004 let mut missing_optional_fields = Vec::new();
2005 for f in &variant.fields {
2006 let ident = self.tcx.adjust_ident(f.ident(self.tcx), variant.def_id);
2007 if let Some(_) = remaining_fields.remove(&ident) {
2008 if f.value.is_none() {
2009 missing_mandatory_fields.push(ident);
2010 } else {
2011 missing_optional_fields.push(ident);
2012 }
2013 }
2014 }
2015 if !self.tcx.features().default_field_values() {
2016 let sugg = self.tcx.crate_level_attribute_injection_span();
2017 self.dcx().emit_err(BaseExpressionDoubleDot {
2018 span: span.shrink_to_hi(),
2019 default_field_values_suggestion: if self.tcx.sess.is_nightly_build()
2022 && missing_mandatory_fields.is_empty()
2023 && !missing_optional_fields.is_empty()
2024 {
2025 Some(sugg)
2026 } else {
2027 None
2028 },
2029 add_expr: if !missing_mandatory_fields.is_empty()
2030 || !missing_optional_fields.is_empty()
2031 {
2032 Some(BaseExpressionDoubleDotAddExpr { span: span.shrink_to_hi() })
2033 } else {
2034 None
2035 },
2036 remove_dots: if missing_mandatory_fields.is_empty()
2037 && missing_optional_fields.is_empty()
2038 {
2039 Some(BaseExpressionDoubleDotRemove { span })
2040 } else {
2041 None
2042 },
2043 });
2044 return;
2045 }
2046 if variant.fields.is_empty() {
2047 let mut err = self.dcx().struct_span_err(
2048 span,
2049 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("`{0}` has no fields, `..` needs at least one default field in the struct definition",
adt_ty))
})format!(
2050 "`{adt_ty}` has no fields, `..` needs at least one default field in \
2051 the struct definition",
2052 ),
2053 );
2054 err.span_label(path_span, "this type has no fields");
2055 err.emit();
2056 }
2057 if !missing_mandatory_fields.is_empty() {
2058 let s = if missing_mandatory_fields.len() == 1 { "" } else { "s" }pluralize!(missing_mandatory_fields.len());
2059 let fields = listify(&missing_mandatory_fields, |f| ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("`{0}`", f))
})format!("`{f}`")).unwrap();
2060 self.dcx()
2061 .struct_span_err(
2062 span.shrink_to_lo(),
2063 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("missing field{0} {1} in initializer",
s, fields))
})format!("missing field{s} {fields} in initializer"),
2064 )
2065 .with_span_label(
2066 span.shrink_to_lo(),
2067 "fields that do not have a defaulted value must be provided explicitly",
2068 )
2069 .emit();
2070 return;
2071 }
2072 let fru_tys = match adt_ty.kind() {
2073 ty::Adt(adt, args) if adt.is_struct() => variant
2074 .fields
2075 .iter()
2076 .map(|f| self.normalize(span, f.ty(self.tcx, args)))
2077 .collect(),
2078 ty::Adt(adt, args) if adt.is_enum() => variant
2079 .fields
2080 .iter()
2081 .map(|f| self.normalize(span, f.ty(self.tcx, args)))
2082 .collect(),
2083 _ => {
2084 self.dcx().emit_err(FunctionalRecordUpdateOnNonStruct { span });
2085 return;
2086 }
2087 };
2088 self.typeck_results.borrow_mut().fru_field_types_mut().insert(expr.hir_id, fru_tys);
2089 }
2090 hir::StructTailExpr::Base(base_expr) => {
2091 let fru_tys = if self.tcx.features().type_changing_struct_update() {
2094 if adt.is_struct() {
2095 let fresh_args = self.fresh_args_for_item(base_expr.span, adt.did());
2097 let fru_tys = variant
2102 .fields
2103 .iter()
2104 .map(|f| {
2105 let fru_ty = self.normalize(
2106 expr.span,
2107 self.field_ty(base_expr.span, f, fresh_args),
2108 );
2109 let ident =
2110 self.tcx.adjust_ident(f.ident(self.tcx), variant.def_id);
2111 if let Some(_) = remaining_fields.remove(&ident) {
2112 let target_ty = self.field_ty(base_expr.span, f, args);
2113 let cause = self.misc(base_expr.span);
2114 match self.at(&cause, self.param_env).sup(
2115 DefineOpaqueTypes::Yes,
2120 target_ty,
2121 fru_ty,
2122 ) {
2123 Ok(InferOk { obligations, value: () }) => {
2124 self.register_predicates(obligations)
2125 }
2126 Err(_) => {
2127 ::rustc_middle::util::bug::span_bug_fmt(cause.span,
format_args!("subtyping remaining fields of type changing FRU failed: {2} != {3}: {0}::{1}",
variant.name, ident.name, target_ty, fru_ty));span_bug!(
2128 cause.span,
2129 "subtyping remaining fields of type changing FRU \
2130 failed: {target_ty} != {fru_ty}: {}::{}",
2131 variant.name,
2132 ident.name,
2133 );
2134 }
2135 }
2136 }
2137 self.resolve_vars_if_possible(fru_ty)
2138 })
2139 .collect();
2140 let fresh_base_ty = Ty::new_adt(self.tcx, *adt, fresh_args);
2159 self.check_expr_has_type_or_error(
2160 base_expr,
2161 self.resolve_vars_if_possible(fresh_base_ty),
2162 |_| {},
2163 );
2164 fru_tys
2165 } else {
2166 self.check_expr(base_expr);
2169 self.dcx()
2170 .emit_err(FunctionalRecordUpdateOnNonStruct { span: base_expr.span });
2171 return;
2172 }
2173 } else {
2174 self.check_expr_has_type_or_error(base_expr, adt_ty, |_| {
2175 let base_ty = self.typeck_results.borrow().expr_ty(base_expr);
2176 let same_adt = #[allow(non_exhaustive_omitted_patterns)] match (adt_ty.kind(),
base_ty.kind()) {
(ty::Adt(adt, _), ty::Adt(base_adt, _)) if adt == base_adt => true,
_ => false,
}matches!((adt_ty.kind(), base_ty.kind()),
2177 (ty::Adt(adt, _), ty::Adt(base_adt, _)) if adt == base_adt);
2178 if self.tcx.sess.is_nightly_build() && same_adt {
2179 feature_err(
2180 &self.tcx.sess,
2181 sym::type_changing_struct_update,
2182 base_expr.span,
2183 "type changing struct updating is experimental",
2184 )
2185 .emit();
2186 }
2187 });
2188 match adt_ty.kind() {
2189 ty::Adt(adt, args) if adt.is_struct() => variant
2190 .fields
2191 .iter()
2192 .map(|f| self.normalize(expr.span, f.ty(self.tcx, args)))
2193 .collect(),
2194 _ => {
2195 self.dcx().emit_err(FunctionalRecordUpdateOnNonStruct {
2196 span: base_expr.span,
2197 });
2198 return;
2199 }
2200 }
2201 };
2202 self.typeck_results.borrow_mut().fru_field_types_mut().insert(expr.hir_id, fru_tys);
2203 }
2204 rustc_hir::StructTailExpr::NoneWithError(guaranteed) => {
2205 self.infcx.set_tainted_by_errors(guaranteed);
2216 }
2217 rustc_hir::StructTailExpr::None => {
2218 if adt_kind != AdtKind::Union
2219 && !remaining_fields.is_empty()
2220 && !variant.field_list_has_applicable_non_exhaustive()
2222 {
2223 {
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_hir_typeck/src/expr.rs:2223",
"rustc_hir_typeck::expr", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/expr.rs"),
::tracing_core::__macro_support::Option::Some(2223u32),
::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::expr"),
::tracing_core::field::FieldSet::new(&["remaining_fields"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::EVENT)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let enabled =
::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
&&
::tracing::Level::DEBUG <=
::tracing::level_filters::LevelFilter::current() &&
{
let interest = __CALLSITE.interest();
!interest.is_never() &&
::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
interest)
};
if enabled {
(|value_set: ::tracing::field::ValueSet|
{
let meta = __CALLSITE.metadata();
::tracing::Event::dispatch(meta, &value_set);
;
})({
#[allow(unused_imports)]
use ::tracing::field::{debug, display, Value};
let mut iter = __CALLSITE.metadata().fields().iter();
__CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&debug(&remaining_fields)
as &dyn Value))])
});
} else { ; }
};debug!(?remaining_fields);
2224
2225 let private_fields: Vec<&ty::FieldDef> = variant
2228 .fields
2229 .iter()
2230 .filter(|field| {
2231 !field.vis.is_accessible_from(tcx.parent_module(expr.hir_id), tcx)
2232 })
2233 .collect();
2234
2235 if !private_fields.is_empty() {
2236 self.report_private_fields(
2237 adt_ty,
2238 path_span,
2239 expr.span,
2240 private_fields,
2241 hir_fields,
2242 );
2243 } else {
2244 self.report_missing_fields(
2245 adt_ty,
2246 path_span,
2247 expr.span,
2248 remaining_fields,
2249 variant,
2250 hir_fields,
2251 args,
2252 );
2253 }
2254 }
2255 }
2256 }
2257 }
2258
2259 fn check_struct_fields_on_error(
2260 &self,
2261 fields: &'tcx [hir::ExprField<'tcx>],
2262 base_expr: &'tcx hir::StructTailExpr<'tcx>,
2263 ) {
2264 for field in fields {
2265 self.check_expr(field.expr);
2266 }
2267 if let hir::StructTailExpr::Base(base) = *base_expr {
2268 self.check_expr(base);
2269 }
2270 }
2271
2272 fn report_missing_fields(
2284 &self,
2285 adt_ty: Ty<'tcx>,
2286 span: Span,
2287 full_span: Span,
2288 remaining_fields: UnordMap<Ident, (FieldIdx, &ty::FieldDef)>,
2289 variant: &'tcx ty::VariantDef,
2290 hir_fields: &'tcx [hir::ExprField<'tcx>],
2291 args: GenericArgsRef<'tcx>,
2292 ) {
2293 let len = remaining_fields.len();
2294
2295 let displayable_field_names: Vec<&str> =
2296 remaining_fields.items().map(|(ident, _)| ident.as_str()).into_sorted_stable_ord();
2297
2298 let mut truncated_fields_error = String::new();
2299 let remaining_fields_names = match &displayable_field_names[..] {
2300 [field1] => ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("`{0}`", field1))
})format!("`{field1}`"),
2301 [field1, field2] => ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("`{0}` and `{1}`", field1, field2))
})format!("`{field1}` and `{field2}`"),
2302 [field1, field2, field3] => ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("`{0}`, `{1}` and `{2}`", field1,
field2, field3))
})format!("`{field1}`, `{field2}` and `{field3}`"),
2303 _ => {
2304 truncated_fields_error =
2305 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!(" and {0} other field{1}", len - 3,
if len - 3 == 1 { "" } else { "s" }))
})format!(" and {} other field{}", len - 3, pluralize!(len - 3));
2306 displayable_field_names
2307 .iter()
2308 .take(3)
2309 .map(|n| ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("`{0}`", n))
})format!("`{n}`"))
2310 .collect::<Vec<_>>()
2311 .join(", ")
2312 }
2313 };
2314
2315 let mut err = {
self.dcx().struct_span_err(span,
::alloc::__export::must_use({
::alloc::fmt::format(format_args!("missing field{0} {1}{2} in initializer of `{3}`",
if len == 1 { "" } else { "s" }, remaining_fields_names,
truncated_fields_error, adt_ty))
})).with_code(E0063)
}struct_span_code_err!(
2316 self.dcx(),
2317 span,
2318 E0063,
2319 "missing field{} {}{} in initializer of `{}`",
2320 pluralize!(len),
2321 remaining_fields_names,
2322 truncated_fields_error,
2323 adt_ty
2324 );
2325 err.span_label(span, ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("missing {0}{1}",
remaining_fields_names, truncated_fields_error))
})format!("missing {remaining_fields_names}{truncated_fields_error}"));
2326
2327 if remaining_fields.items().all(|(_, (_, field))| field.value.is_some())
2328 && self.tcx.sess.is_nightly_build()
2329 {
2330 let msg = ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("all remaining fields have default values, {0} use those values with `..`",
if self.tcx.features().default_field_values() {
"you can"
} else {
"if you added `#![feature(default_field_values)]` to your crate you could"
}))
})format!(
2331 "all remaining fields have default values, {you_can} use those values with `..`",
2332 you_can = if self.tcx.features().default_field_values() {
2333 "you can"
2334 } else {
2335 "if you added `#![feature(default_field_values)]` to your crate you could"
2336 },
2337 );
2338 if let Some(hir_field) = hir_fields.last() {
2339 err.span_suggestion_verbose(
2340 hir_field.span.shrink_to_hi(),
2341 msg,
2342 ", ..".to_string(),
2343 Applicability::MachineApplicable,
2344 );
2345 } else if hir_fields.is_empty() {
2346 err.span_suggestion_verbose(
2347 span.shrink_to_hi().with_hi(full_span.hi()),
2348 msg,
2349 " { .. }".to_string(),
2350 Applicability::MachineApplicable,
2351 );
2352 }
2353 }
2354
2355 if let Some(hir_field) = hir_fields.last() {
2356 self.suggest_fru_from_range_and_emit(hir_field, variant, args, err);
2357 } else {
2358 err.emit();
2359 }
2360 }
2361
2362 fn suggest_fru_from_range_and_emit(
2365 &self,
2366 last_expr_field: &hir::ExprField<'tcx>,
2367 variant: &ty::VariantDef,
2368 args: GenericArgsRef<'tcx>,
2369 mut err: Diag<'_>,
2370 ) {
2371 if is_range_literal(last_expr_field.expr)
2372 && let ExprKind::Struct(&qpath, [range_start, range_end], _) = last_expr_field.expr.kind
2373 && self.tcx.qpath_is_lang_item(qpath, LangItem::Range)
2374 && let variant_field =
2375 variant.fields.iter().find(|field| field.ident(self.tcx) == last_expr_field.ident)
2376 && let range_def_id = self.tcx.lang_items().range_struct()
2377 && variant_field
2378 .and_then(|field| field.ty(self.tcx, args).ty_adt_def())
2379 .map(|adt| adt.did())
2380 != range_def_id
2381 {
2382 let expr = self
2386 .tcx
2387 .sess
2388 .source_map()
2389 .span_to_snippet(range_end.expr.span)
2390 .ok()
2391 .filter(|s| s.len() < 25 && !s.contains(|c: char| c.is_control()));
2392
2393 let fru_span = self
2394 .tcx
2395 .sess
2396 .source_map()
2397 .span_extend_while_whitespace(range_start.expr.span)
2398 .shrink_to_hi()
2399 .to(range_end.expr.span);
2400
2401 err.subdiagnostic(TypeMismatchFruTypo {
2402 expr_span: range_start.expr.span,
2403 fru_span,
2404 expr,
2405 });
2406
2407 self.dcx().try_steal_replace_and_emit_err(
2409 last_expr_field.span,
2410 StashKey::MaybeFruTypo,
2411 err,
2412 );
2413 } else {
2414 err.emit();
2415 }
2416 }
2417
2418 fn report_private_fields(
2430 &self,
2431 adt_ty: Ty<'tcx>,
2432 span: Span,
2433 expr_span: Span,
2434 private_fields: Vec<&ty::FieldDef>,
2435 used_fields: &'tcx [hir::ExprField<'tcx>],
2436 ) {
2437 let mut err =
2438 self.dcx().struct_span_err(
2439 span,
2440 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("cannot construct `{0}` with struct literal syntax due to private fields",
adt_ty))
})format!(
2441 "cannot construct `{adt_ty}` with struct literal syntax due to private fields",
2442 ),
2443 );
2444 let (used_private_fields, remaining_private_fields): (
2445 Vec<(Symbol, Span, bool)>,
2446 Vec<(Symbol, Span, bool)>,
2447 ) = private_fields
2448 .iter()
2449 .map(|field| {
2450 match used_fields.iter().find(|used_field| field.name == used_field.ident.name) {
2451 Some(used_field) => (field.name, used_field.span, true),
2452 None => (field.name, self.tcx.def_span(field.did), false),
2453 }
2454 })
2455 .partition(|field| field.2);
2456 err.span_labels(used_private_fields.iter().map(|(_, span, _)| *span), "private field");
2457 if !remaining_private_fields.is_empty() {
2458 let names = if remaining_private_fields.len() > 6 {
2459 String::new()
2460 } else {
2461 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{0} ",
listify(&remaining_private_fields,
|(name, _, _)|
::alloc::__export::must_use({
::alloc::fmt::format(format_args!("`{0}`", name))
})).expect("expected at least one private field to report")))
})format!(
2462 "{} ",
2463 listify(&remaining_private_fields, |(name, _, _)| format!("`{name}`"))
2464 .expect("expected at least one private field to report")
2465 )
2466 };
2467 err.note(::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{0}private field{1} {3}that {2} not provided",
if used_fields.is_empty() { "" } else { "...and other " },
if remaining_private_fields.len() == 1 { "" } else { "s" },
if remaining_private_fields.len() == 1 {
"was"
} else { "were" }, names))
})format!(
2468 "{}private field{s} {names}that {were} not provided",
2469 if used_fields.is_empty() { "" } else { "...and other " },
2470 s = pluralize!(remaining_private_fields.len()),
2471 were = pluralize!("was", remaining_private_fields.len()),
2472 ));
2473 }
2474
2475 if let ty::Adt(def, _) = adt_ty.kind() {
2476 let def_id = def.did();
2477 let mut items = self
2478 .tcx
2479 .inherent_impls(def_id)
2480 .into_iter()
2481 .flat_map(|&i| self.tcx.associated_items(i).in_definition_order())
2482 .filter(|item| item.is_fn() && !item.is_method())
2484 .filter_map(|item| {
2485 let fn_sig = self
2487 .tcx
2488 .fn_sig(item.def_id)
2489 .instantiate(self.tcx, self.fresh_args_for_item(span, item.def_id));
2490 let ret_ty = self.tcx.instantiate_bound_regions_with_erased(fn_sig.output());
2491 if !self.can_eq(self.param_env, ret_ty, adt_ty) {
2492 return None;
2493 }
2494 let input_len = fn_sig.inputs().skip_binder().len();
2495 let name = item.name();
2496 let order = !name.as_str().starts_with("new");
2497 Some((order, name, input_len))
2498 })
2499 .collect::<Vec<_>>();
2500 items.sort_by_key(|(order, _, _)| *order);
2501 let suggestion = |name, args| {
2502 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("::{1}({0})",
std::iter::repeat_n("_", args).collect::<Vec<_>>().join(", "),
name))
})format!(
2503 "::{name}({})",
2504 std::iter::repeat_n("_", args).collect::<Vec<_>>().join(", ")
2505 )
2506 };
2507 match &items[..] {
2508 [] => {}
2509 [(_, name, args)] => {
2510 err.span_suggestion_verbose(
2511 span.shrink_to_hi().with_hi(expr_span.hi()),
2512 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("you might have meant to use the `{0}` associated function",
name))
})format!("you might have meant to use the `{name}` associated function"),
2513 suggestion(name, *args),
2514 Applicability::MaybeIncorrect,
2515 );
2516 }
2517 _ => {
2518 err.span_suggestions(
2519 span.shrink_to_hi().with_hi(expr_span.hi()),
2520 "you might have meant to use an associated function to build this type",
2521 items.iter().map(|(_, name, args)| suggestion(name, *args)),
2522 Applicability::MaybeIncorrect,
2523 );
2524 }
2525 }
2526 if let Some(default_trait) = self.tcx.get_diagnostic_item(sym::Default)
2527 && self
2528 .infcx
2529 .type_implements_trait(default_trait, [adt_ty], self.param_env)
2530 .may_apply()
2531 {
2532 err.multipart_suggestion(
2533 "consider using the `Default` trait",
2534 ::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[(span.shrink_to_lo(), "<".to_string()),
(span.shrink_to_hi().with_hi(expr_span.hi()),
" as std::default::Default>::default()".to_string())]))vec![
2535 (span.shrink_to_lo(), "<".to_string()),
2536 (
2537 span.shrink_to_hi().with_hi(expr_span.hi()),
2538 " as std::default::Default>::default()".to_string(),
2539 ),
2540 ],
2541 Applicability::MaybeIncorrect,
2542 );
2543 }
2544 }
2545
2546 err.emit();
2547 }
2548
2549 fn report_unknown_field(
2550 &self,
2551 ty: Ty<'tcx>,
2552 variant: &'tcx ty::VariantDef,
2553 expr: &hir::Expr<'_>,
2554 field: &hir::ExprField<'_>,
2555 skip_fields: &[hir::ExprField<'_>],
2556 kind_name: &str,
2557 ) -> ErrorGuaranteed {
2558 if let Err(guar) = variant.has_errors() {
2560 return guar;
2561 }
2562 let mut err = self.err_ctxt().type_error_struct_with_diag(
2563 field.ident.span,
2564 |actual| match ty.kind() {
2565 ty::Adt(adt, ..) if adt.is_enum() => {
self.dcx().struct_span_err(field.ident.span,
::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{0} `{1}::{2}` has no field named `{3}`",
kind_name, actual, variant.name, field.ident))
})).with_code(E0559)
}struct_span_code_err!(
2566 self.dcx(),
2567 field.ident.span,
2568 E0559,
2569 "{} `{}::{}` has no field named `{}`",
2570 kind_name,
2571 actual,
2572 variant.name,
2573 field.ident
2574 ),
2575 _ => {
self.dcx().struct_span_err(field.ident.span,
::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{0} `{1}` has no field named `{2}`",
kind_name, actual, field.ident))
})).with_code(E0560)
}struct_span_code_err!(
2576 self.dcx(),
2577 field.ident.span,
2578 E0560,
2579 "{} `{}` has no field named `{}`",
2580 kind_name,
2581 actual,
2582 field.ident
2583 ),
2584 },
2585 ty,
2586 );
2587
2588 let variant_ident_span = self.tcx.def_ident_span(variant.def_id).unwrap();
2589 match variant.ctor {
2590 Some((CtorKind::Fn, def_id)) => match ty.kind() {
2591 ty::Adt(adt, ..) if adt.is_enum() => {
2592 err.span_label(
2593 variant_ident_span,
2594 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("`{0}::{1}` defined here", ty,
variant.name))
})format!(
2595 "`{adt}::{variant}` defined here",
2596 adt = ty,
2597 variant = variant.name,
2598 ),
2599 );
2600 err.span_label(field.ident.span, "field does not exist");
2601 let fn_sig = self.tcx.fn_sig(def_id).instantiate_identity();
2602 let inputs = fn_sig.inputs().skip_binder();
2603 let fields = ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("({0})",
inputs.iter().map(|i|
::alloc::__export::must_use({
::alloc::fmt::format(format_args!("/* {0} */", i))
})).collect::<Vec<_>>().join(", ")))
})format!(
2604 "({})",
2605 inputs.iter().map(|i| format!("/* {i} */")).collect::<Vec<_>>().join(", ")
2606 );
2607 let (replace_span, sugg) = match expr.kind {
2608 hir::ExprKind::Struct(qpath, ..) => {
2609 (qpath.span().shrink_to_hi().with_hi(expr.span.hi()), fields)
2610 }
2611 _ => {
2612 (expr.span, ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{1}::{0}{2}", variant.name, ty,
fields))
})format!("{ty}::{variant}{fields}", variant = variant.name))
2613 }
2614 };
2615 err.span_suggestion_verbose(
2616 replace_span,
2617 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("`{0}::{1}` is a tuple {2}, use the appropriate syntax",
ty, variant.name, kind_name))
})format!(
2618 "`{adt}::{variant}` is a tuple {kind_name}, use the appropriate syntax",
2619 adt = ty,
2620 variant = variant.name,
2621 ),
2622 sugg,
2623 Applicability::HasPlaceholders,
2624 );
2625 }
2626 _ => {
2627 err.span_label(variant_ident_span, ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("`{0}` defined here", ty))
})format!("`{ty}` defined here"));
2628 err.span_label(field.ident.span, "field does not exist");
2629 let fn_sig = self.tcx.fn_sig(def_id).instantiate_identity();
2630 let inputs = fn_sig.inputs().skip_binder();
2631 let fields = ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("({0})",
inputs.iter().map(|i|
::alloc::__export::must_use({
::alloc::fmt::format(format_args!("/* {0} */", i))
})).collect::<Vec<_>>().join(", ")))
})format!(
2632 "({})",
2633 inputs.iter().map(|i| format!("/* {i} */")).collect::<Vec<_>>().join(", ")
2634 );
2635 err.span_suggestion_verbose(
2636 expr.span,
2637 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("`{0}` is a tuple {1}, use the appropriate syntax",
ty, kind_name))
})format!("`{ty}` is a tuple {kind_name}, use the appropriate syntax",),
2638 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{0}{1}", ty, fields))
})format!("{ty}{fields}"),
2639 Applicability::HasPlaceholders,
2640 );
2641 }
2642 },
2643 _ => {
2644 let available_field_names = self.available_field_names(variant, expr, skip_fields);
2646 if let Some(field_name) =
2647 find_best_match_for_name(&available_field_names, field.ident.name, None)
2648 && !(field.ident.name.as_str().parse::<usize>().is_ok()
2649 && field_name.as_str().parse::<usize>().is_ok())
2650 {
2651 err.span_label(field.ident.span, "unknown field");
2652 err.span_suggestion_verbose(
2653 field.ident.span,
2654 "a field with a similar name exists",
2655 field_name,
2656 Applicability::MaybeIncorrect,
2657 );
2658 } else {
2659 match ty.kind() {
2660 ty::Adt(adt, ..) => {
2661 if adt.is_enum() {
2662 err.span_label(
2663 field.ident.span,
2664 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("`{0}::{1}` does not have this field",
ty, variant.name))
})format!("`{}::{}` does not have this field", ty, variant.name),
2665 );
2666 } else {
2667 err.span_label(
2668 field.ident.span,
2669 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("`{0}` does not have this field",
ty))
})format!("`{ty}` does not have this field"),
2670 );
2671 }
2672 if available_field_names.is_empty() {
2673 err.note("all struct fields are already assigned");
2674 } else {
2675 err.note(::alloc::__export::must_use({
::alloc::fmt::format(format_args!("available fields are: {0}",
self.name_series_display(available_field_names)))
})format!(
2676 "available fields are: {}",
2677 self.name_series_display(available_field_names)
2678 ));
2679 }
2680 }
2681 _ => ::rustc_middle::util::bug::bug_fmt(format_args!("non-ADT passed to report_unknown_field"))bug!("non-ADT passed to report_unknown_field"),
2682 }
2683 };
2684 }
2685 }
2686 err.emit()
2687 }
2688
2689 fn available_field_names(
2690 &self,
2691 variant: &'tcx ty::VariantDef,
2692 expr: &hir::Expr<'_>,
2693 skip_fields: &[hir::ExprField<'_>],
2694 ) -> Vec<Symbol> {
2695 variant
2696 .fields
2697 .iter()
2698 .filter(|field| {
2699 skip_fields.iter().all(|&skip| skip.ident.name != field.name)
2700 && self.is_field_suggestable(field, expr.hir_id, expr.span)
2701 })
2702 .map(|field| field.name)
2703 .collect()
2704 }
2705
2706 fn name_series_display(&self, names: Vec<Symbol>) -> String {
2707 let limit = if names.len() == 6 { 6 } else { 5 };
2709 let mut display =
2710 names.iter().take(limit).map(|n| ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("`{0}`", n))
})format!("`{n}`")).collect::<Vec<_>>().join(", ");
2711 if names.len() > limit {
2712 display = ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{0} ... and {1} others", display,
names.len() - limit))
})format!("{} ... and {} others", display, names.len() - limit);
2713 }
2714 display
2715 }
2716
2717 fn find_adt_field(
2721 &self,
2722 base_def: ty::AdtDef<'tcx>,
2723 ident: Ident,
2724 ) -> Option<(FieldIdx, &'tcx ty::FieldDef)> {
2725 if base_def.is_enum() {
2727 return None;
2728 }
2729
2730 for (field_idx, field) in base_def.non_enum_variant().fields.iter_enumerated() {
2731 if field.ident(self.tcx).normalize_to_macros_2_0() == ident {
2732 return Some((field_idx, field));
2734 }
2735 }
2736
2737 None
2738 }
2739
2740 fn check_expr_field(
2750 &self,
2751 expr: &'tcx hir::Expr<'tcx>,
2752 base: &'tcx hir::Expr<'tcx>,
2753 field: Ident,
2754 expected: Expectation<'tcx>,
2756 ) -> Ty<'tcx> {
2757 {
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_hir_typeck/src/expr.rs:2757",
"rustc_hir_typeck::expr", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/expr.rs"),
::tracing_core::__macro_support::Option::Some(2757u32),
::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::expr"),
::tracing_core::field::FieldSet::new(&["message"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::EVENT)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let enabled =
::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
&&
::tracing::Level::DEBUG <=
::tracing::level_filters::LevelFilter::current() &&
{
let interest = __CALLSITE.interest();
!interest.is_never() &&
::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
interest)
};
if enabled {
(|value_set: ::tracing::field::ValueSet|
{
let meta = __CALLSITE.metadata();
::tracing::Event::dispatch(meta, &value_set);
;
})({
#[allow(unused_imports)]
use ::tracing::field::{debug, display, Value};
let mut iter = __CALLSITE.metadata().fields().iter();
__CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&format_args!("check_field(expr: {0:?}, base: {1:?}, field: {2:?})",
expr, base, field) as &dyn Value))])
});
} else { ; }
};debug!("check_field(expr: {:?}, base: {:?}, field: {:?})", expr, base, field);
2758 let base_ty = self.check_expr(base);
2759 let base_ty = self.structurally_resolve_type(base.span, base_ty);
2760
2761 let mut private_candidate = None;
2763
2764 let mut autoderef = self.autoderef(expr.span, base_ty);
2766 while let Some((deref_base_ty, _)) = autoderef.next() {
2767 {
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_hir_typeck/src/expr.rs:2767",
"rustc_hir_typeck::expr", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/expr.rs"),
::tracing_core::__macro_support::Option::Some(2767u32),
::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::expr"),
::tracing_core::field::FieldSet::new(&["message"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::EVENT)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let enabled =
::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
&&
::tracing::Level::DEBUG <=
::tracing::level_filters::LevelFilter::current() &&
{
let interest = __CALLSITE.interest();
!interest.is_never() &&
::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
interest)
};
if enabled {
(|value_set: ::tracing::field::ValueSet|
{
let meta = __CALLSITE.metadata();
::tracing::Event::dispatch(meta, &value_set);
;
})({
#[allow(unused_imports)]
use ::tracing::field::{debug, display, Value};
let mut iter = __CALLSITE.metadata().fields().iter();
__CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&format_args!("deref_base_ty: {0:?}",
deref_base_ty) as &dyn Value))])
});
} else { ; }
};debug!("deref_base_ty: {:?}", deref_base_ty);
2768 match deref_base_ty.kind() {
2769 ty::Adt(base_def, args) if !base_def.is_enum() => {
2770 {
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_hir_typeck/src/expr.rs:2770",
"rustc_hir_typeck::expr", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/expr.rs"),
::tracing_core::__macro_support::Option::Some(2770u32),
::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::expr"),
::tracing_core::field::FieldSet::new(&["message"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::EVENT)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let enabled =
::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
&&
::tracing::Level::DEBUG <=
::tracing::level_filters::LevelFilter::current() &&
{
let interest = __CALLSITE.interest();
!interest.is_never() &&
::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
interest)
};
if enabled {
(|value_set: ::tracing::field::ValueSet|
{
let meta = __CALLSITE.metadata();
::tracing::Event::dispatch(meta, &value_set);
;
})({
#[allow(unused_imports)]
use ::tracing::field::{debug, display, Value};
let mut iter = __CALLSITE.metadata().fields().iter();
__CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&format_args!("struct named {0:?}",
deref_base_ty) as &dyn Value))])
});
} else { ; }
};debug!("struct named {:?}", deref_base_ty);
2771 if let Err(guar) = base_def.non_enum_variant().has_errors() {
2773 return Ty::new_error(self.tcx(), guar);
2774 }
2775
2776 let fn_body_hir_id = self.tcx.local_def_id_to_hir_id(self.body_id);
2777 let (ident, def_scope) =
2778 self.tcx.adjust_ident_and_get_scope(field, base_def.did(), fn_body_hir_id);
2779
2780 if let Some((idx, field)) = self.find_adt_field(*base_def, ident) {
2781 self.write_field_index(expr.hir_id, idx);
2782
2783 let adjustments = self.adjust_steps(&autoderef);
2784 if field.vis.is_accessible_from(def_scope, self.tcx) {
2785 self.apply_adjustments(base, adjustments);
2786 self.register_predicates(autoderef.into_obligations());
2787
2788 self.tcx.check_stability(field.did, Some(expr.hir_id), expr.span, None);
2789 return self.field_ty(expr.span, field, args);
2790 }
2791
2792 private_candidate = Some((adjustments, base_def.did()));
2794 }
2795 }
2796 ty::Tuple(tys) => {
2797 if let Ok(index) = field.as_str().parse::<usize>() {
2798 if field.name == sym::integer(index) {
2799 if let Some(&field_ty) = tys.get(index) {
2800 let adjustments = self.adjust_steps(&autoderef);
2801 self.apply_adjustments(base, adjustments);
2802 self.register_predicates(autoderef.into_obligations());
2803
2804 self.write_field_index(expr.hir_id, FieldIdx::from_usize(index));
2805 return field_ty;
2806 }
2807 }
2808 }
2809 }
2810 _ => {}
2811 }
2812 }
2813 let final_ty = self.structurally_resolve_type(autoderef.span(), autoderef.final_ty());
2819 if let ty::Error(_) = final_ty.kind() {
2820 return final_ty;
2821 }
2822
2823 if let Some((adjustments, did)) = private_candidate {
2824 self.apply_adjustments(base, adjustments);
2827 let guar = self.ban_private_field_access(
2828 expr,
2829 base_ty,
2830 field,
2831 did,
2832 expected.only_has_type(self),
2833 );
2834 return Ty::new_error(self.tcx(), guar);
2835 }
2836
2837 let guar = if self.method_exists_for_diagnostic(
2838 field,
2839 base_ty,
2840 expr.hir_id,
2841 expected.only_has_type(self),
2842 ) {
2843 self.ban_take_value_of_method(expr, base_ty, field)
2845 } else if !base_ty.is_primitive_ty() {
2846 self.ban_nonexisting_field(field, base, expr, base_ty)
2847 } else {
2848 let field_name = field.to_string();
2849 let mut err = {
let mut err =
{
self.dcx().struct_span_err(field.span,
::alloc::__export::must_use({
::alloc::fmt::format(format_args!("`{0}` is a primitive type and therefore doesn\'t have fields",
base_ty))
})).with_code(E0610)
};
if base_ty.references_error() { err.downgrade_to_delayed_bug(); }
err
}type_error_struct!(
2850 self.dcx(),
2851 field.span,
2852 base_ty,
2853 E0610,
2854 "`{base_ty}` is a primitive type and therefore doesn't have fields",
2855 );
2856 let is_valid_suffix = |field: &str| {
2857 if field == "f32" || field == "f64" {
2858 return true;
2859 }
2860 let mut chars = field.chars().peekable();
2861 match chars.peek() {
2862 Some('e') | Some('E') => {
2863 chars.next();
2864 if let Some(c) = chars.peek()
2865 && !c.is_numeric()
2866 && *c != '-'
2867 && *c != '+'
2868 {
2869 return false;
2870 }
2871 while let Some(c) = chars.peek() {
2872 if !c.is_numeric() {
2873 break;
2874 }
2875 chars.next();
2876 }
2877 }
2878 _ => (),
2879 }
2880 let suffix = chars.collect::<String>();
2881 suffix.is_empty() || suffix == "f32" || suffix == "f64"
2882 };
2883 let maybe_partial_suffix = |field: &str| -> Option<&str> {
2884 let first_chars = ['f', 'l'];
2885 if field.len() >= 1
2886 && field.to_lowercase().starts_with(first_chars)
2887 && field[1..].chars().all(|c| c.is_ascii_digit())
2888 {
2889 if field.to_lowercase().starts_with(['f']) { Some("f32") } else { Some("f64") }
2890 } else {
2891 None
2892 }
2893 };
2894 if let ty::Infer(ty::IntVar(_)) = base_ty.kind()
2895 && let ExprKind::Lit(Spanned {
2896 node: ast::LitKind::Int(_, ast::LitIntType::Unsuffixed),
2897 ..
2898 }) = base.kind
2899 && !base.span.from_expansion()
2900 {
2901 if is_valid_suffix(&field_name) {
2902 err.span_suggestion_verbose(
2903 field.span.shrink_to_lo(),
2904 "if intended to be a floating point literal, consider adding a `0` after the period",
2905 '0',
2906 Applicability::MaybeIncorrect,
2907 );
2908 } else if let Some(correct_suffix) = maybe_partial_suffix(&field_name) {
2909 err.span_suggestion_verbose(
2910 field.span,
2911 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("if intended to be a floating point literal, consider adding a `0` after the period and a `{0}` suffix",
correct_suffix))
})format!("if intended to be a floating point literal, consider adding a `0` after the period and a `{correct_suffix}` suffix"),
2912 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("0{0}", correct_suffix))
})format!("0{correct_suffix}"),
2913 Applicability::MaybeIncorrect,
2914 );
2915 }
2916 }
2917 err.emit()
2918 };
2919
2920 Ty::new_error(self.tcx(), guar)
2921 }
2922
2923 fn suggest_await_on_field_access(
2924 &self,
2925 err: &mut Diag<'_>,
2926 field_ident: Ident,
2927 base: &'tcx hir::Expr<'tcx>,
2928 ty: Ty<'tcx>,
2929 ) {
2930 let Some(output_ty) = self.err_ctxt().get_impl_future_output_ty(ty) else {
2931 err.span_label(field_ident.span, "unknown field");
2932 return;
2933 };
2934 let ty::Adt(def, _) = output_ty.kind() else {
2935 err.span_label(field_ident.span, "unknown field");
2936 return;
2937 };
2938 if def.is_enum() {
2940 err.span_label(field_ident.span, "unknown field");
2941 return;
2942 }
2943 if !def.non_enum_variant().fields.iter().any(|field| field.ident(self.tcx) == field_ident) {
2944 err.span_label(field_ident.span, "unknown field");
2945 return;
2946 }
2947 err.span_label(
2948 field_ident.span,
2949 "field not available in `impl Future`, but it is available in its `Output`",
2950 );
2951 match self.tcx.coroutine_kind(self.body_id) {
2952 Some(hir::CoroutineKind::Desugared(hir::CoroutineDesugaring::Async, _)) => {
2953 err.span_suggestion_verbose(
2954 base.span.shrink_to_hi(),
2955 "consider `await`ing on the `Future` to access the field",
2956 ".await",
2957 Applicability::MaybeIncorrect,
2958 );
2959 }
2960 _ => {
2961 let mut span: MultiSpan = base.span.into();
2962 span.push_span_label(self.tcx.def_span(self.body_id), "this is not `async`");
2963 err.span_note(
2964 span,
2965 "this implements `Future` and its output type has the field, \
2966 but the future cannot be awaited in a synchronous function",
2967 );
2968 }
2969 }
2970 }
2971
2972 fn ban_nonexisting_field(
2973 &self,
2974 ident: Ident,
2975 base: &'tcx hir::Expr<'tcx>,
2976 expr: &'tcx hir::Expr<'tcx>,
2977 base_ty: Ty<'tcx>,
2978 ) -> ErrorGuaranteed {
2979 {
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_hir_typeck/src/expr.rs:2979",
"rustc_hir_typeck::expr", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/expr.rs"),
::tracing_core::__macro_support::Option::Some(2979u32),
::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::expr"),
::tracing_core::field::FieldSet::new(&["message"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::EVENT)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let enabled =
::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
&&
::tracing::Level::DEBUG <=
::tracing::level_filters::LevelFilter::current() &&
{
let interest = __CALLSITE.interest();
!interest.is_never() &&
::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
interest)
};
if enabled {
(|value_set: ::tracing::field::ValueSet|
{
let meta = __CALLSITE.metadata();
::tracing::Event::dispatch(meta, &value_set);
;
})({
#[allow(unused_imports)]
use ::tracing::field::{debug, display, Value};
let mut iter = __CALLSITE.metadata().fields().iter();
__CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&format_args!("ban_nonexisting_field: field={0:?}, base={1:?}, expr={2:?}, base_ty={3:?}",
ident, base, expr, base_ty) as &dyn Value))])
});
} else { ; }
};debug!(
2980 "ban_nonexisting_field: field={:?}, base={:?}, expr={:?}, base_ty={:?}",
2981 ident, base, expr, base_ty
2982 );
2983 let mut err = self.no_such_field_err(ident, base_ty, expr);
2984
2985 match *base_ty.peel_refs().kind() {
2986 ty::Array(_, len) => {
2987 self.maybe_suggest_array_indexing(&mut err, base, ident, len);
2988 }
2989 ty::RawPtr(..) => {
2990 self.suggest_first_deref_field(&mut err, base, ident);
2991 }
2992 ty::Param(param_ty) => {
2993 err.span_label(ident.span, "unknown field");
2994 self.point_at_param_definition(&mut err, param_ty);
2995 }
2996 ty::Alias(ty::Opaque, _) => {
2997 self.suggest_await_on_field_access(&mut err, ident, base, base_ty.peel_refs());
2998 }
2999 _ => {
3000 err.span_label(ident.span, "unknown field");
3001 }
3002 }
3003
3004 self.suggest_fn_call(&mut err, base, base_ty, |output_ty| {
3005 if let ty::Adt(def, _) = output_ty.kind()
3006 && !def.is_enum()
3007 {
3008 def.non_enum_variant().fields.iter().any(|field| {
3009 field.ident(self.tcx) == ident
3010 && field.vis.is_accessible_from(expr.hir_id.owner.def_id, self.tcx)
3011 })
3012 } else if let ty::Tuple(tys) = output_ty.kind()
3013 && let Ok(idx) = ident.as_str().parse::<usize>()
3014 {
3015 idx < tys.len()
3016 } else {
3017 false
3018 }
3019 });
3020
3021 if ident.name == kw::Await {
3022 err.note("to `.await` a `Future`, switch to Rust 2018 or later");
3025 HelpUseLatestEdition::new().add_to_diag(&mut err);
3026 }
3027
3028 err.emit()
3029 }
3030
3031 fn ban_private_field_access(
3032 &self,
3033 expr: &hir::Expr<'tcx>,
3034 expr_t: Ty<'tcx>,
3035 field: Ident,
3036 base_did: DefId,
3037 return_ty: Option<Ty<'tcx>>,
3038 ) -> ErrorGuaranteed {
3039 let mut err = self.private_field_err(field, base_did);
3040
3041 if self.method_exists_for_diagnostic(field, expr_t, expr.hir_id, return_ty)
3043 && !self.expr_in_place(expr.hir_id)
3044 {
3045 self.suggest_method_call(
3046 &mut err,
3047 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("a method `{0}` also exists, call it with parentheses",
field))
})format!("a method `{field}` also exists, call it with parentheses"),
3048 field,
3049 expr_t,
3050 expr,
3051 None,
3052 );
3053 }
3054 err.emit()
3055 }
3056
3057 fn ban_take_value_of_method(
3058 &self,
3059 expr: &hir::Expr<'tcx>,
3060 expr_t: Ty<'tcx>,
3061 field: Ident,
3062 ) -> ErrorGuaranteed {
3063 let mut err = {
let mut err =
{
self.dcx().struct_span_err(field.span,
::alloc::__export::must_use({
::alloc::fmt::format(format_args!("attempted to take value of method `{0}` on type `{1}`",
field, expr_t))
})).with_code(E0615)
};
if expr_t.references_error() { err.downgrade_to_delayed_bug(); }
err
}type_error_struct!(
3064 self.dcx(),
3065 field.span,
3066 expr_t,
3067 E0615,
3068 "attempted to take value of method `{field}` on type `{expr_t}`",
3069 );
3070 err.span_label(field.span, "method, not a field");
3071 let expr_is_call =
3072 if let hir::Node::Expr(hir::Expr { kind: ExprKind::Call(callee, _args), .. }) =
3073 self.tcx.parent_hir_node(expr.hir_id)
3074 {
3075 expr.hir_id == callee.hir_id
3076 } else {
3077 false
3078 };
3079 let expr_snippet =
3080 self.tcx.sess.source_map().span_to_snippet(expr.span).unwrap_or_default();
3081 let is_wrapped = expr_snippet.starts_with('(') && expr_snippet.ends_with(')');
3082 let after_open = expr.span.lo() + rustc_span::BytePos(1);
3083 let before_close = expr.span.hi() - rustc_span::BytePos(1);
3084
3085 if expr_is_call && is_wrapped {
3086 err.multipart_suggestion(
3087 "remove wrapping parentheses to call the method",
3088 ::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[(expr.span.with_hi(after_open), String::new()),
(expr.span.with_lo(before_close), String::new())]))vec![
3089 (expr.span.with_hi(after_open), String::new()),
3090 (expr.span.with_lo(before_close), String::new()),
3091 ],
3092 Applicability::MachineApplicable,
3093 );
3094 } else if !self.expr_in_place(expr.hir_id) {
3095 let span = if is_wrapped {
3097 expr.span.with_lo(after_open).with_hi(before_close)
3098 } else {
3099 expr.span
3100 };
3101 self.suggest_method_call(
3102 &mut err,
3103 "use parentheses to call the method",
3104 field,
3105 expr_t,
3106 expr,
3107 Some(span),
3108 );
3109 } else if let ty::RawPtr(ptr_ty, _) = expr_t.kind()
3110 && let ty::Adt(adt_def, _) = ptr_ty.kind()
3111 && let ExprKind::Field(base_expr, _) = expr.kind
3112 && let [variant] = &adt_def.variants().raw
3113 && variant.fields.iter().any(|f| f.ident(self.tcx) == field)
3114 {
3115 err.multipart_suggestion(
3116 "to access the field, dereference first",
3117 ::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[(base_expr.span.shrink_to_lo(), "(*".to_string()),
(base_expr.span.shrink_to_hi(), ")".to_string())]))vec![
3118 (base_expr.span.shrink_to_lo(), "(*".to_string()),
3119 (base_expr.span.shrink_to_hi(), ")".to_string()),
3120 ],
3121 Applicability::MaybeIncorrect,
3122 );
3123 } else {
3124 err.help("methods are immutable and cannot be assigned to");
3125 }
3126
3127 self.dcx().try_steal_replace_and_emit_err(field.span, StashKey::GenericInFieldExpr, err)
3129 }
3130
3131 fn point_at_param_definition(&self, err: &mut Diag<'_>, param: ty::ParamTy) {
3132 let generics = self.tcx.generics_of(self.body_id);
3133 let generic_param = generics.type_param(param, self.tcx);
3134 if let ty::GenericParamDefKind::Type { synthetic: true, .. } = generic_param.kind {
3135 return;
3136 }
3137 let param_def_id = generic_param.def_id;
3138 let param_hir_id = match param_def_id.as_local() {
3139 Some(x) => self.tcx.local_def_id_to_hir_id(x),
3140 None => return,
3141 };
3142 let param_span = self.tcx.hir_span(param_hir_id);
3143 let param_name = self.tcx.hir_ty_param_name(param_def_id.expect_local());
3144
3145 err.span_label(param_span, ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("type parameter \'{0}\' declared here",
param_name))
})format!("type parameter '{param_name}' declared here"));
3146 }
3147
3148 fn maybe_suggest_array_indexing(
3149 &self,
3150 err: &mut Diag<'_>,
3151 base: &hir::Expr<'_>,
3152 field: Ident,
3153 len: ty::Const<'tcx>,
3154 ) {
3155 err.span_label(field.span, "unknown field");
3156 if let (Some(len), Ok(user_index)) = (
3157 self.try_structurally_resolve_const(base.span, len).try_to_target_usize(self.tcx),
3158 field.as_str().parse::<u64>(),
3159 ) {
3160 let help = "instead of using tuple indexing, use array indexing";
3161 let applicability = if len < user_index {
3162 Applicability::MachineApplicable
3163 } else {
3164 Applicability::MaybeIncorrect
3165 };
3166 err.multipart_suggestion(
3167 help,
3168 ::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[(base.span.between(field.span), "[".to_string()),
(field.span.shrink_to_hi(), "]".to_string())]))vec![
3169 (base.span.between(field.span), "[".to_string()),
3170 (field.span.shrink_to_hi(), "]".to_string()),
3171 ],
3172 applicability,
3173 );
3174 }
3175 }
3176
3177 fn suggest_first_deref_field(&self, err: &mut Diag<'_>, base: &hir::Expr<'_>, field: Ident) {
3178 err.span_label(field.span, "unknown field");
3179 let val = if let Ok(base) = self.tcx.sess.source_map().span_to_snippet(base.span)
3180 && base.len() < 20
3181 {
3182 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("`{0}`", base))
})format!("`{base}`")
3183 } else {
3184 "the value".to_string()
3185 };
3186 err.multipart_suggestion(
3187 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{0} is a raw pointer; try dereferencing it",
val))
})format!("{val} is a raw pointer; try dereferencing it"),
3188 ::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[(base.span.shrink_to_lo(), "(*".into()),
(base.span.between(field.span),
::alloc::__export::must_use({
::alloc::fmt::format(format_args!(")."))
}))]))vec![
3189 (base.span.shrink_to_lo(), "(*".into()),
3190 (base.span.between(field.span), format!(").")),
3191 ],
3192 Applicability::MaybeIncorrect,
3193 );
3194 }
3195
3196 fn no_such_field_err(
3197 &self,
3198 field: Ident,
3199 base_ty: Ty<'tcx>,
3200 expr: &hir::Expr<'tcx>,
3201 ) -> Diag<'_> {
3202 let span = field.span;
3203 {
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_hir_typeck/src/expr.rs:3203",
"rustc_hir_typeck::expr", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/expr.rs"),
::tracing_core::__macro_support::Option::Some(3203u32),
::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::expr"),
::tracing_core::field::FieldSet::new(&["message"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::EVENT)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let enabled =
::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
&&
::tracing::Level::DEBUG <=
::tracing::level_filters::LevelFilter::current() &&
{
let interest = __CALLSITE.interest();
!interest.is_never() &&
::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
interest)
};
if enabled {
(|value_set: ::tracing::field::ValueSet|
{
let meta = __CALLSITE.metadata();
::tracing::Event::dispatch(meta, &value_set);
;
})({
#[allow(unused_imports)]
use ::tracing::field::{debug, display, Value};
let mut iter = __CALLSITE.metadata().fields().iter();
__CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&format_args!("no_such_field_err(span: {0:?}, field: {1:?}, expr_t: {2:?})",
span, field, base_ty) as &dyn Value))])
});
} else { ; }
};debug!("no_such_field_err(span: {:?}, field: {:?}, expr_t: {:?})", span, field, base_ty);
3204
3205 let mut err = self.dcx().create_err(NoFieldOnType { span, ty: base_ty, field });
3206 if base_ty.references_error() {
3207 err.downgrade_to_delayed_bug();
3208 }
3209
3210 if let Some(within_macro_span) = span.within_macro(expr.span, self.tcx.sess.source_map()) {
3211 err.span_label(within_macro_span, "due to this macro variable");
3212 }
3213
3214 if let Some(def_id) = base_ty.peel_refs().ty_adt_def().map(|d| d.did()) {
3216 for &impl_def_id in self.tcx.inherent_impls(def_id) {
3217 for item in self.tcx.associated_items(impl_def_id).in_definition_order() {
3218 if let ExprKind::Field(base_expr, _) = expr.kind
3219 && item.name() == field.name
3220 && #[allow(non_exhaustive_omitted_patterns)] match item.kind {
ty::AssocKind::Fn { has_self: false, .. } => true,
_ => false,
}matches!(item.kind, ty::AssocKind::Fn { has_self: false, .. })
3221 {
3222 err.span_label(field.span, "this is an associated function, not a method");
3223 err.note("found the following associated function; to be used as method, it must have a `self` parameter");
3224 let impl_ty = self.tcx.type_of(impl_def_id).instantiate_identity();
3225 err.span_note(
3226 self.tcx.def_span(item.def_id),
3227 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("the candidate is defined in an impl for the type `{0}`",
impl_ty))
})format!("the candidate is defined in an impl for the type `{impl_ty}`"),
3228 );
3229
3230 let ty_str = match base_ty.peel_refs().kind() {
3231 ty::Adt(def, args) => self.tcx.def_path_str_with_args(def.did(), args),
3232 _ => base_ty.peel_refs().to_string(),
3233 };
3234 err.multipart_suggestion(
3235 "use associated function syntax instead",
3236 ::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[(base_expr.span, ty_str),
(base_expr.span.between(field.span), "::".to_string())]))vec![
3237 (base_expr.span, ty_str),
3238 (base_expr.span.between(field.span), "::".to_string()),
3239 ],
3240 Applicability::MaybeIncorrect,
3241 );
3242 return err;
3243 }
3244 }
3245 }
3246 }
3247
3248 let mod_id = self.tcx.parent_module(expr.hir_id).to_def_id();
3250 let (ty, unwrap) = if let ty::Adt(def, args) = base_ty.kind()
3251 && (self.tcx.is_diagnostic_item(sym::Result, def.did())
3252 || self.tcx.is_diagnostic_item(sym::Option, def.did()))
3253 && let Some(arg) = args.get(0)
3254 && let Some(ty) = arg.as_type()
3255 {
3256 (ty, "unwrap().")
3257 } else {
3258 (base_ty, "")
3259 };
3260 for found_fields in
3261 self.get_field_candidates_considering_privacy_for_diag(span, ty, mod_id, expr.hir_id)
3262 {
3263 let field_names = found_fields.iter().map(|field| field.0.name).collect::<Vec<_>>();
3264 let mut candidate_fields: Vec<_> = found_fields
3265 .into_iter()
3266 .filter_map(|candidate_field| {
3267 self.check_for_nested_field_satisfying_condition_for_diag(
3268 span,
3269 &|candidate_field, _| candidate_field == field,
3270 candidate_field,
3271 ::alloc::vec::Vec::new()vec![],
3272 mod_id,
3273 expr.hir_id,
3274 )
3275 })
3276 .map(|mut field_path| {
3277 field_path.pop();
3278 field_path.iter().map(|id| ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{0}.", id))
})format!("{}.", id)).collect::<String>()
3279 })
3280 .collect::<Vec<_>>();
3281 candidate_fields.sort();
3282
3283 let len = candidate_fields.len();
3284 if len > 0 && expr.span.eq_ctxt(field.span) {
3287 err.span_suggestions(
3288 field.span.shrink_to_lo(),
3289 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{0} of the expressions\' fields {1} a field of the same name",
if len > 1 { "some" } else { "one" },
if len > 1 { "have" } else { "has" }))
})format!(
3290 "{} of the expressions' fields {} a field of the same name",
3291 if len > 1 { "some" } else { "one" },
3292 if len > 1 { "have" } else { "has" },
3293 ),
3294 candidate_fields.iter().map(|path| ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{0}{1}", unwrap, path))
})format!("{unwrap}{path}")),
3295 Applicability::MaybeIncorrect,
3296 );
3297 } else if let Some(field_name) =
3298 find_best_match_for_name(&field_names, field.name, None)
3299 && !(field.name.as_str().parse::<usize>().is_ok()
3300 && field_name.as_str().parse::<usize>().is_ok())
3301 {
3302 err.span_suggestion_verbose(
3303 field.span,
3304 "a field with a similar name exists",
3305 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{1}{0}", field_name, unwrap))
})format!("{unwrap}{}", field_name),
3306 Applicability::MaybeIncorrect,
3307 );
3308 } else if !field_names.is_empty() {
3309 let is = if field_names.len() == 1 { " is" } else { "s are" };
3310 err.note(
3311 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("available field{1}: {0}",
self.name_series_display(field_names), is))
})format!("available field{is}: {}", self.name_series_display(field_names),),
3312 );
3313 }
3314 }
3315 err
3316 }
3317
3318 fn private_field_err(&self, field: Ident, base_did: DefId) -> Diag<'_> {
3319 let struct_path = self.tcx().def_path_str(base_did);
3320 let kind_name = self.tcx().def_descr(base_did);
3321 {
self.dcx().struct_span_err(field.span,
::alloc::__export::must_use({
::alloc::fmt::format(format_args!("field `{0}` of {1} `{2}` is private",
field, kind_name, struct_path))
})).with_code(E0616)
}struct_span_code_err!(
3322 self.dcx(),
3323 field.span,
3324 E0616,
3325 "field `{field}` of {kind_name} `{struct_path}` is private",
3326 )
3327 .with_span_label(field.span, "private field")
3328 }
3329
3330 pub(crate) fn get_field_candidates_considering_privacy_for_diag(
3331 &self,
3332 span: Span,
3333 base_ty: Ty<'tcx>,
3334 mod_id: DefId,
3335 hir_id: HirId,
3336 ) -> Vec<Vec<(Ident, Ty<'tcx>)>> {
3337 {
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_hir_typeck/src/expr.rs:3337",
"rustc_hir_typeck::expr", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/expr.rs"),
::tracing_core::__macro_support::Option::Some(3337u32),
::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::expr"),
::tracing_core::field::FieldSet::new(&["message"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::EVENT)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let enabled =
::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
&&
::tracing::Level::DEBUG <=
::tracing::level_filters::LevelFilter::current() &&
{
let interest = __CALLSITE.interest();
!interest.is_never() &&
::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
interest)
};
if enabled {
(|value_set: ::tracing::field::ValueSet|
{
let meta = __CALLSITE.metadata();
::tracing::Event::dispatch(meta, &value_set);
;
})({
#[allow(unused_imports)]
use ::tracing::field::{debug, display, Value};
let mut iter = __CALLSITE.metadata().fields().iter();
__CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&format_args!("get_field_candidates(span: {0:?}, base_t: {1:?}",
span, base_ty) as &dyn Value))])
});
} else { ; }
};debug!("get_field_candidates(span: {:?}, base_t: {:?}", span, base_ty);
3338
3339 let mut autoderef = self.autoderef(span, base_ty).silence_errors();
3340 let deref_chain: Vec<_> = autoderef.by_ref().collect();
3341
3342 if autoderef.reached_recursion_limit() {
3346 return ::alloc::vec::Vec::new()vec![];
3347 }
3348
3349 deref_chain
3350 .into_iter()
3351 .filter_map(move |(base_t, _)| {
3352 match base_t.kind() {
3353 ty::Adt(base_def, args) if !base_def.is_enum() => {
3354 let tcx = self.tcx;
3355 let fields = &base_def.non_enum_variant().fields;
3356 if fields.iter().all(|field| !field.vis.is_accessible_from(mod_id, tcx)) {
3360 return None;
3361 }
3362 return Some(
3363 fields
3364 .iter()
3365 .filter(move |field| {
3366 field.vis.is_accessible_from(mod_id, tcx)
3367 && self.is_field_suggestable(field, hir_id, span)
3368 })
3369 .take(100)
3371 .map(|field_def| {
3372 (
3373 field_def.ident(self.tcx).normalize_to_macros_2_0(),
3374 field_def.ty(self.tcx, args),
3375 )
3376 })
3377 .collect::<Vec<_>>(),
3378 );
3379 }
3380 ty::Tuple(types) => {
3381 return Some(
3382 types
3383 .iter()
3384 .enumerate()
3385 .take(100)
3387 .map(|(i, ty)| (Ident::from_str(&i.to_string()), ty))
3388 .collect::<Vec<_>>(),
3389 );
3390 }
3391 _ => None,
3392 }
3393 })
3394 .collect()
3395 }
3396
3397 #[allow(clippy :: suspicious_else_formatting)]
{
let __tracing_attr_span;
let __tracing_attr_guard;
if ::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
&&
::tracing::Level::DEBUG <=
::tracing::level_filters::LevelFilter::current() ||
{ false } {
__tracing_attr_span =
{
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("check_for_nested_field_satisfying_condition_for_diag",
"rustc_hir_typeck::expr", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/expr.rs"),
::tracing_core::__macro_support::Option::Some(3399u32),
::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::expr"),
::tracing_core::field::FieldSet::new(&["span",
"candidate_name", "candidate_ty", "field_path"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::SPAN)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let mut interest = ::tracing::subscriber::Interest::never();
if ::tracing::Level::DEBUG <=
::tracing::level_filters::STATIC_MAX_LEVEL &&
::tracing::Level::DEBUG <=
::tracing::level_filters::LevelFilter::current() &&
{ interest = __CALLSITE.interest(); !interest.is_never() }
&&
::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
interest) {
let meta = __CALLSITE.metadata();
::tracing::Span::new(meta,
&{
#[allow(unused_imports)]
use ::tracing::field::{debug, display, Value};
let mut iter = meta.fields().iter();
meta.fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&::tracing::field::debug(&span)
as &dyn Value)),
(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&::tracing::field::debug(&candidate_name)
as &dyn Value)),
(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&::tracing::field::debug(&candidate_ty)
as &dyn Value)),
(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&::tracing::field::debug(&field_path)
as &dyn Value))])
})
} else {
let span =
::tracing::__macro_support::__disabled_span(__CALLSITE.metadata());
{};
span
}
};
__tracing_attr_guard = __tracing_attr_span.enter();
}
#[warn(clippy :: suspicious_else_formatting)]
{
#[allow(unknown_lints, unreachable_code, clippy ::
diverging_sub_expression, clippy :: empty_loop, clippy ::
let_unit_value, clippy :: let_with_type_underscore, clippy ::
needless_return, clippy :: unreachable)]
if false {
let __tracing_attr_fake_return: Option<Vec<Ident>> = loop {};
return __tracing_attr_fake_return;
}
{
if field_path.len() > 3 { return None; }
field_path.push(candidate_name);
if matches(candidate_name, candidate_ty) {
return Some(field_path);
}
for nested_fields in
self.get_field_candidates_considering_privacy_for_diag(span,
candidate_ty, mod_id, hir_id) {
for field in nested_fields {
if let Some(field_path) =
self.check_for_nested_field_satisfying_condition_for_diag(span,
matches, field, field_path.clone(), mod_id, hir_id) {
return Some(field_path);
}
}
}
None
}
}
}#[instrument(skip(self, matches, mod_id, hir_id), level = "debug")]
3400 pub(crate) fn check_for_nested_field_satisfying_condition_for_diag(
3401 &self,
3402 span: Span,
3403 matches: &impl Fn(Ident, Ty<'tcx>) -> bool,
3404 (candidate_name, candidate_ty): (Ident, Ty<'tcx>),
3405 mut field_path: Vec<Ident>,
3406 mod_id: DefId,
3407 hir_id: HirId,
3408 ) -> Option<Vec<Ident>> {
3409 if field_path.len() > 3 {
3410 return None;
3413 }
3414 field_path.push(candidate_name);
3415 if matches(candidate_name, candidate_ty) {
3416 return Some(field_path);
3417 }
3418 for nested_fields in self.get_field_candidates_considering_privacy_for_diag(
3419 span,
3420 candidate_ty,
3421 mod_id,
3422 hir_id,
3423 ) {
3424 for field in nested_fields {
3426 if let Some(field_path) = self.check_for_nested_field_satisfying_condition_for_diag(
3427 span,
3428 matches,
3429 field,
3430 field_path.clone(),
3431 mod_id,
3432 hir_id,
3433 ) {
3434 return Some(field_path);
3435 }
3436 }
3437 }
3438 None
3439 }
3440
3441 fn check_expr_index(
3442 &self,
3443 base: &'tcx hir::Expr<'tcx>,
3444 idx: &'tcx hir::Expr<'tcx>,
3445 expr: &'tcx hir::Expr<'tcx>,
3446 brackets_span: Span,
3447 ) -> Ty<'tcx> {
3448 let base_t = self.check_expr(base);
3449 let idx_t = self.check_expr(idx);
3450
3451 if base_t.references_error() {
3452 base_t
3453 } else if idx_t.references_error() {
3454 idx_t
3455 } else {
3456 let base_t = self.structurally_resolve_type(base.span, base_t);
3457 match self.lookup_indexing(expr, base, base_t, idx, idx_t) {
3458 Some((index_ty, element_ty)) => {
3459 self.demand_coerce(idx, idx_t, index_ty, None, AllowTwoPhase::No);
3461 self.select_obligations_where_possible(|errors| {
3462 self.point_at_index(errors, idx.span);
3463 });
3464 element_ty
3465 }
3466 None => {
3467 for (base_t, _) in self.autoderef(base.span, base_t).silence_errors() {
3470 if let Some((_, index_ty, element_ty)) =
3471 self.find_and_report_unsatisfied_index_impl(base, base_t)
3472 {
3473 self.demand_coerce(idx, idx_t, index_ty, None, AllowTwoPhase::No);
3474 return element_ty;
3475 }
3476 }
3477
3478 let mut err = {
let mut err =
{
self.dcx().struct_span_err(brackets_span,
::alloc::__export::must_use({
::alloc::fmt::format(format_args!("cannot index into a value of type `{0}`",
base_t))
})).with_code(E0608)
};
if base_t.references_error() { err.downgrade_to_delayed_bug(); }
err
}type_error_struct!(
3479 self.dcx(),
3480 brackets_span,
3481 base_t,
3482 E0608,
3483 "cannot index into a value of type `{base_t}`",
3484 );
3485 if let ty::Tuple(types) = base_t.kind() {
3487 err.help(
3488 "tuples are indexed with a dot and a literal index: `tuple.0`, `tuple.1`, etc.",
3489 );
3490 if let ExprKind::Lit(lit) = idx.kind
3492 && let ast::LitKind::Int(i, ast::LitIntType::Unsuffixed) = lit.node
3493 && i.get() < types.len().try_into().expect("tuple length fits in u128")
3494 {
3495 err.span_suggestion(
3496 brackets_span,
3497 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("to access tuple element `{0}`, use",
i))
})format!("to access tuple element `{i}`, use"),
3498 ::alloc::__export::must_use({ ::alloc::fmt::format(format_args!(".{0}", i)) })format!(".{i}"),
3499 Applicability::MachineApplicable,
3500 );
3501 }
3502 }
3503
3504 if base_t.is_raw_ptr() && idx_t.is_integral() {
3505 err.multipart_suggestion(
3506 "consider using `wrapping_add` or `add` for indexing into raw pointer",
3507 ::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[(base.span.between(idx.span), ".wrapping_add(".to_owned()),
(idx.span.shrink_to_hi().until(expr.span.shrink_to_hi()),
")".to_owned())]))vec![
3508 (base.span.between(idx.span), ".wrapping_add(".to_owned()),
3509 (
3510 idx.span.shrink_to_hi().until(expr.span.shrink_to_hi()),
3511 ")".to_owned(),
3512 ),
3513 ],
3514 Applicability::MaybeIncorrect,
3515 );
3516 }
3517
3518 let reported = err.emit();
3519 Ty::new_error(self.tcx, reported)
3520 }
3521 }
3522 }
3523 }
3524
3525 fn find_and_report_unsatisfied_index_impl(
3533 &self,
3534 base_expr: &hir::Expr<'_>,
3535 base_ty: Ty<'tcx>,
3536 ) -> Option<(ErrorGuaranteed, Ty<'tcx>, Ty<'tcx>)> {
3537 let index_trait_def_id = self.tcx.lang_items().index_trait()?;
3538 let index_trait_output_def_id = self.tcx.get_diagnostic_item(sym::IndexOutput)?;
3539
3540 let mut relevant_impls = ::alloc::vec::Vec::new()vec![];
3541 self.tcx.for_each_relevant_impl(index_trait_def_id, base_ty, |impl_def_id| {
3542 relevant_impls.push(impl_def_id);
3543 });
3544 let [impl_def_id] = relevant_impls[..] else {
3545 return None;
3547 };
3548
3549 self.commit_if_ok(|snapshot| {
3550 let outer_universe = self.universe();
3551
3552 let ocx = ObligationCtxt::new_with_diagnostics(self);
3553 let impl_args = self.fresh_args_for_item(base_expr.span, impl_def_id);
3554 let impl_trait_ref =
3555 self.tcx.impl_trait_ref(impl_def_id).instantiate(self.tcx, impl_args);
3556 let cause = self.misc(base_expr.span);
3557
3558 let impl_trait_ref = ocx.normalize(&cause, self.param_env, impl_trait_ref);
3561 ocx.eq(&cause, self.param_env, base_ty, impl_trait_ref.self_ty())?;
3562
3563 ocx.register_obligations(traits::predicates_for_generics(
3567 |idx, span| {
3568 cause.clone().derived_cause(
3569 ty::Binder::dummy(ty::TraitPredicate {
3570 trait_ref: impl_trait_ref,
3571 polarity: ty::PredicatePolarity::Positive,
3572 }),
3573 |derived| {
3574 ObligationCauseCode::ImplDerived(Box::new(traits::ImplDerivedCause {
3575 derived,
3576 impl_or_alias_def_id: impl_def_id,
3577 impl_def_predicate_index: Some(idx),
3578 span,
3579 }))
3580 },
3581 )
3582 },
3583 self.param_env,
3584 self.tcx.predicates_of(impl_def_id).instantiate(self.tcx, impl_args),
3585 ));
3586
3587 let element_ty = ocx.normalize(
3590 &cause,
3591 self.param_env,
3592 Ty::new_projection_from_args(
3593 self.tcx,
3594 index_trait_output_def_id,
3595 impl_trait_ref.args,
3596 ),
3597 );
3598
3599 let true_errors = ocx.try_evaluate_obligations();
3600
3601 self.leak_check(outer_universe, Some(snapshot))?;
3605
3606 let ambiguity_errors = ocx.evaluate_obligations_error_on_ambiguity();
3608 if true_errors.is_empty() && !ambiguity_errors.is_empty() {
3609 return Err(NoSolution);
3610 }
3611
3612 Ok::<_, NoSolution>((
3615 self.err_ctxt().report_fulfillment_errors(true_errors),
3616 impl_trait_ref.args.type_at(1),
3617 element_ty,
3618 ))
3619 })
3620 .ok()
3621 }
3622
3623 fn point_at_index(&self, errors: &mut Vec<traits::FulfillmentError<'tcx>>, span: Span) {
3624 let mut seen_preds = FxHashSet::default();
3625 errors.sort_by_key(|error| error.root_obligation.recursion_depth);
3629 for error in errors {
3630 match (
3631 error.root_obligation.predicate.kind().skip_binder(),
3632 error.obligation.predicate.kind().skip_binder(),
3633 ) {
3634 (ty::PredicateKind::Clause(ty::ClauseKind::Trait(predicate)), _)
3635 if self.tcx.is_lang_item(predicate.trait_ref.def_id, LangItem::Index) =>
3636 {
3637 seen_preds.insert(error.obligation.predicate.kind().skip_binder());
3638 }
3639 (_, ty::PredicateKind::Clause(ty::ClauseKind::Trait(predicate)))
3640 if self.tcx.is_diagnostic_item(sym::SliceIndex, predicate.trait_ref.def_id) =>
3641 {
3642 seen_preds.insert(error.obligation.predicate.kind().skip_binder());
3643 }
3644 (root, pred) if seen_preds.contains(&pred) || seen_preds.contains(&root) => {}
3645 _ => continue,
3646 }
3647 error.obligation.cause.span = span;
3648 }
3649 }
3650
3651 fn check_expr_yield(
3652 &self,
3653 value: &'tcx hir::Expr<'tcx>,
3654 expr: &'tcx hir::Expr<'tcx>,
3655 ) -> Ty<'tcx> {
3656 match self.coroutine_types {
3657 Some(CoroutineTypes { resume_ty, yield_ty }) => {
3658 self.check_expr_coercible_to_type(value, yield_ty, None);
3659
3660 resume_ty
3661 }
3662 _ => {
3663 self.dcx().emit_err(YieldExprOutsideOfCoroutine { span: expr.span });
3664 self.check_expr(value);
3666 self.tcx.types.unit
3667 }
3668 }
3669 }
3670
3671 fn check_expr_asm_operand(&self, expr: &'tcx hir::Expr<'tcx>, is_input: bool) {
3672 let needs = if is_input { Needs::None } else { Needs::MutPlace };
3673 let ty = self.check_expr_with_needs(expr, needs);
3674 self.require_type_is_sized(ty, expr.span, ObligationCauseCode::InlineAsmSized);
3675
3676 if !is_input && !expr.is_syntactic_place_expr() {
3677 self.dcx()
3678 .struct_span_err(expr.span, "invalid asm output")
3679 .with_span_label(expr.span, "cannot assign to this expression")
3680 .emit();
3681 }
3682
3683 if is_input {
3691 let ty = self.structurally_resolve_type(expr.span, ty);
3692 match *ty.kind() {
3693 ty::FnDef(..) => {
3694 let fnptr_ty = Ty::new_fn_ptr(self.tcx, ty.fn_sig(self.tcx));
3695 self.demand_coerce(expr, ty, fnptr_ty, None, AllowTwoPhase::No);
3696 }
3697 ty::Ref(_, base_ty, mutbl) => {
3698 let ptr_ty = Ty::new_ptr(self.tcx, base_ty, mutbl);
3699 self.demand_coerce(expr, ty, ptr_ty, None, AllowTwoPhase::No);
3700 }
3701 _ => {}
3702 }
3703 }
3704 }
3705
3706 fn check_expr_asm(&self, asm: &'tcx hir::InlineAsm<'tcx>, span: Span) -> Ty<'tcx> {
3707 if let rustc_ast::AsmMacro::NakedAsm = asm.asm_macro {
3708 if !{
#[allow(deprecated)]
{
{
'done:
{
for i in self.tcx.get_all_attrs(self.body_id) {
#[allow(unused_imports)]
use rustc_hir::attrs::AttributeKind::*;
let i: &rustc_hir::Attribute = i;
match i {
rustc_hir::Attribute::Parsed(Naked(..)) => {
break 'done Some(());
}
rustc_hir::Attribute::Unparsed(..) =>
{}
#[deny(unreachable_patterns)]
_ => {}
}
}
None
}
}
}
}.is_some()find_attr!(self.tcx, self.body_id, Naked(..)) {
3709 self.tcx.dcx().emit_err(NakedAsmOutsideNakedFn { span });
3710 }
3711 }
3712
3713 let mut diverge = asm.asm_macro.diverges(asm.options);
3714
3715 for (op, _op_sp) in asm.operands {
3716 match *op {
3717 hir::InlineAsmOperand::In { expr, .. } => {
3718 self.check_expr_asm_operand(expr, true);
3719 }
3720 hir::InlineAsmOperand::Out { expr: Some(expr), .. }
3721 | hir::InlineAsmOperand::InOut { expr, .. } => {
3722 self.check_expr_asm_operand(expr, false);
3723 }
3724 hir::InlineAsmOperand::Out { expr: None, .. } => {}
3725 hir::InlineAsmOperand::SplitInOut { in_expr, out_expr, .. } => {
3726 self.check_expr_asm_operand(in_expr, true);
3727 if let Some(out_expr) = out_expr {
3728 self.check_expr_asm_operand(out_expr, false);
3729 }
3730 }
3731 hir::InlineAsmOperand::Const { ref anon_const } => {
3732 self.check_expr_const_block(anon_const, Expectation::NoExpectation);
3733 }
3734 hir::InlineAsmOperand::SymFn { expr } => {
3735 self.check_expr(expr);
3736 }
3737 hir::InlineAsmOperand::SymStatic { .. } => {}
3738 hir::InlineAsmOperand::Label { block } => {
3739 let previous_diverges = self.diverges.get();
3740
3741 let ty = self.check_expr_block(block, ExpectHasType(self.tcx.types.unit));
3743 if !ty.is_never() {
3744 self.demand_suptype(block.span, self.tcx.types.unit, ty);
3745 diverge = false;
3746 }
3747
3748 self.diverges.set(previous_diverges);
3750 }
3751 }
3752 }
3753
3754 if diverge { self.tcx.types.never } else { self.tcx.types.unit }
3755 }
3756
3757 fn check_expr_offset_of(
3758 &self,
3759 container: &'tcx hir::Ty<'tcx>,
3760 fields: &[Ident],
3761 expr: &'tcx hir::Expr<'tcx>,
3762 ) -> Ty<'tcx> {
3763 let mut current_container = self.lower_ty(container).normalized;
3764 let mut field_indices = Vec::with_capacity(fields.len());
3765 let mut fields = fields.into_iter();
3766
3767 while let Some(&field) = fields.next() {
3768 let container = self.structurally_resolve_type(expr.span, current_container);
3769
3770 match container.kind() {
3771 ty::Adt(container_def, args) if container_def.is_enum() => {
3772 let block = self.tcx.local_def_id_to_hir_id(self.body_id);
3773 let (ident, _def_scope) =
3774 self.tcx.adjust_ident_and_get_scope(field, container_def.did(), block);
3775
3776 if !self.tcx.features().offset_of_enum() {
3777 rustc_session::parse::feature_err(
3778 &self.tcx.sess,
3779 sym::offset_of_enum,
3780 ident.span,
3781 "using enums in offset_of is experimental",
3782 )
3783 .emit();
3784 }
3785
3786 let Some((index, variant)) = container_def
3787 .variants()
3788 .iter_enumerated()
3789 .find(|(_, v)| v.ident(self.tcx).normalize_to_macros_2_0() == ident)
3790 else {
3791 self.dcx()
3792 .create_err(NoVariantNamed { span: ident.span, ident, ty: container })
3793 .with_span_label(field.span, "variant not found")
3794 .emit_unless_delay(container.references_error());
3795 break;
3796 };
3797 let Some(&subfield) = fields.next() else {
3798 {
let mut err =
{
self.dcx().struct_span_err(ident.span,
::alloc::__export::must_use({
::alloc::fmt::format(format_args!("`{0}` is an enum variant; expected field at end of `offset_of`",
ident))
})).with_code(E0795)
};
if container.references_error() { err.downgrade_to_delayed_bug(); }
err
}type_error_struct!(
3799 self.dcx(),
3800 ident.span,
3801 container,
3802 E0795,
3803 "`{ident}` is an enum variant; expected field at end of `offset_of`",
3804 )
3805 .with_span_label(field.span, "enum variant")
3806 .emit();
3807 break;
3808 };
3809 let (subident, sub_def_scope) =
3810 self.tcx.adjust_ident_and_get_scope(subfield, variant.def_id, block);
3811
3812 let Some((subindex, field)) = variant
3813 .fields
3814 .iter_enumerated()
3815 .find(|(_, f)| f.ident(self.tcx).normalize_to_macros_2_0() == subident)
3816 else {
3817 self.dcx()
3818 .create_err(NoFieldOnVariant {
3819 span: ident.span,
3820 container,
3821 ident,
3822 field: subfield,
3823 enum_span: field.span,
3824 field_span: subident.span,
3825 })
3826 .emit_unless_delay(container.references_error());
3827 break;
3828 };
3829
3830 let field_ty = self.field_ty(expr.span, field, args);
3831
3832 self.require_type_is_sized(
3835 field_ty,
3836 expr.span,
3837 ObligationCauseCode::FieldSized {
3838 adt_kind: AdtKind::Enum,
3839 span: self.tcx.def_span(field.did),
3840 last: false,
3841 },
3842 );
3843
3844 if field.vis.is_accessible_from(sub_def_scope, self.tcx) {
3845 self.tcx.check_stability(field.did, Some(expr.hir_id), expr.span, None);
3846 } else {
3847 self.private_field_err(ident, container_def.did()).emit();
3848 }
3849
3850 field_indices.push((current_container, index, subindex));
3853 current_container = field_ty;
3854
3855 continue;
3856 }
3857 ty::Adt(container_def, args) => {
3858 let block = self.tcx.local_def_id_to_hir_id(self.body_id);
3859 let (ident, def_scope) =
3860 self.tcx.adjust_ident_and_get_scope(field, container_def.did(), block);
3861
3862 let fields = &container_def.non_enum_variant().fields;
3863 if let Some((index, field)) = fields
3864 .iter_enumerated()
3865 .find(|(_, f)| f.ident(self.tcx).normalize_to_macros_2_0() == ident)
3866 {
3867 let field_ty = self.field_ty(expr.span, field, args);
3868
3869 if self.tcx.features().offset_of_slice() {
3870 self.require_type_has_static_alignment(field_ty, expr.span);
3871 } else {
3872 self.require_type_is_sized(
3873 field_ty,
3874 expr.span,
3875 ObligationCauseCode::Misc,
3876 );
3877 }
3878
3879 if field.vis.is_accessible_from(def_scope, self.tcx) {
3880 self.tcx.check_stability(field.did, Some(expr.hir_id), expr.span, None);
3881 } else {
3882 self.private_field_err(ident, container_def.did()).emit();
3883 }
3884
3885 field_indices.push((current_container, FIRST_VARIANT, index));
3888 current_container = field_ty;
3889
3890 continue;
3891 }
3892 }
3893 ty::Tuple(tys) => {
3894 if let Ok(index) = field.as_str().parse::<usize>()
3895 && field.name == sym::integer(index)
3896 {
3897 if let Some(&field_ty) = tys.get(index) {
3898 if self.tcx.features().offset_of_slice() {
3899 self.require_type_has_static_alignment(field_ty, expr.span);
3900 } else {
3901 self.require_type_is_sized(
3902 field_ty,
3903 expr.span,
3904 ObligationCauseCode::Misc,
3905 );
3906 }
3907
3908 field_indices.push((current_container, FIRST_VARIANT, index.into()));
3909 current_container = field_ty;
3910
3911 continue;
3912 }
3913 }
3914 }
3915 _ => (),
3916 };
3917
3918 self.no_such_field_err(field, container, expr).emit();
3919
3920 break;
3921 }
3922
3923 self.typeck_results.borrow_mut().offset_of_data_mut().insert(expr.hir_id, field_indices);
3924
3925 self.tcx.types.usize
3926 }
3927}