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