1use std::cell::LazyCell;
2use std::ops::ControlFlow;
3
4use rustc_abi::{ExternAbi, FieldIdx, ScalableElt};
5use rustc_data_structures::unord::{UnordMap, UnordSet};
6use rustc_errors::codes::*;
7use rustc_errors::{Diag, DiagCtxtHandle, Diagnostic, EmissionGuarantee, Level, MultiSpan};
8use rustc_hir as hir;
9use rustc_hir::attrs::ReprAttr::ReprPacked;
10use rustc_hir::def::{CtorKind, DefKind};
11use rustc_hir::{LangItem, Node, find_attr, intravisit};
12use rustc_infer::infer::{RegionVariableOrigin, TyCtxtInferExt};
13use rustc_infer::traits::{Obligation, ObligationCauseCode, WellFormedLoc};
14use rustc_lint_defs::builtin::UNSUPPORTED_CALLING_CONVENTIONS;
15use rustc_macros::Diagnostic;
16use rustc_middle::hir::nested_filter;
17use rustc_middle::middle::resolve_bound_vars::ResolvedArg;
18use rustc_middle::middle::stability::EvalResult;
19use rustc_middle::ty::error::TypeErrorToStringExt;
20use rustc_middle::ty::layout::{LayoutError, MAX_SIMD_LANES};
21use rustc_middle::ty::util::Discr;
22use rustc_middle::ty::{
23 AdtDef, BottomUpFolder, FnSig, GenericArgKind, RegionKind, TypeFoldable, TypeSuperVisitable,
24 TypeVisitable, TypeVisitableExt, Unnormalized, fold_regions,
25};
26use rustc_session::lint::builtin::UNINHABITED_STATIC;
27use rustc_span::{DesugaringKind, sym};
28use rustc_target::spec::{AbiMap, AbiMapping};
29use rustc_trait_selection::error_reporting::InferCtxtErrorExt;
30use rustc_trait_selection::traits;
31use rustc_trait_selection::traits::query::evaluate_obligation::InferCtxtExt;
32use tracing::{debug, instrument};
33use ty::TypingMode;
34
35use super::compare_impl_item::check_type_bounds;
36use super::*;
37use crate::check::wfcheck::{
38 check_associated_item, check_trait_item, check_type_defn, check_variances_for_type_defn,
39 check_where_clauses, enter_wf_checking_ctxt,
40};
41use crate::diagnostics;
42
43fn add_abi_diag_help<T: EmissionGuarantee>(abi: ExternAbi, diag: &mut Diag<'_, T>) {
44 if let ExternAbi::Cdecl { unwind } = abi {
45 let c_abi = ExternAbi::C { unwind };
46 diag.help(::alloc::__export::must_use({
::alloc::fmt::format(format_args!("use `extern {0}` instead", c_abi))
})format!("use `extern {c_abi}` instead",));
47 } else if let ExternAbi::Stdcall { unwind } = abi {
48 let c_abi = ExternAbi::C { unwind };
49 let system_abi = ExternAbi::System { unwind };
50 diag.help(::alloc::__export::must_use({
::alloc::fmt::format(format_args!("if you need `extern {0}` on win32 and `extern {1}` everywhere else, use `extern {2}`",
abi, c_abi, system_abi))
})format!(
51 "if you need `extern {abi}` on win32 and `extern {c_abi}` everywhere else, \
52 use `extern {system_abi}`"
53 ));
54 }
55}
56
57pub fn check_abi(tcx: TyCtxt<'_>, hir_id: hir::HirId, span: Span, abi: ExternAbi) {
58 struct UnsupportedCallingConventions {
59 abi: ExternAbi,
60 }
61
62 impl<'a> Diagnostic<'a, ()> for UnsupportedCallingConventions {
63 fn into_diag(self, dcx: DiagCtxtHandle<'a>, level: Level) -> Diag<'a, ()> {
64 let Self { abi } = self;
65 let mut lint = Diag::new(
66 dcx,
67 level,
68 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{0} is not a supported ABI for the current target",
abi))
})format!("{abi} is not a supported ABI for the current target"),
69 );
70 add_abi_diag_help(abi, &mut lint);
71 lint
72 }
73 }
74 match AbiMap::from_target(&tcx.sess.target).canonize_abi(abi, false) {
79 AbiMapping::Direct(..) => (),
80 AbiMapping::Invalid => {
82 tcx.dcx().span_delayed_bug(span, ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{0} should be rejected in ast_lowering",
abi))
})format!("{abi} should be rejected in ast_lowering"));
83 }
84 AbiMapping::Deprecated(..) => {
85 tcx.emit_node_span_lint(
86 UNSUPPORTED_CALLING_CONVENTIONS,
87 hir_id,
88 span,
89 UnsupportedCallingConventions { abi },
90 );
91 }
92 }
93}
94
95pub fn check_custom_abi(tcx: TyCtxt<'_>, def_id: LocalDefId, fn_sig: FnSig<'_>, fn_sig_span: Span) {
96 if fn_sig.abi() == ExternAbi::Custom {
97 if !{
{
'done:
{
for i in ::rustc_hir::attrs::HasAttrs::get_attrs(def_id, &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!(tcx, def_id, Naked(_)) {
99 tcx.dcx().emit_err(crate::diagnostics::AbiCustomClothedFunction {
100 span: fn_sig_span,
101 naked_span: tcx.def_span(def_id).shrink_to_lo(),
102 });
103 }
104 }
105}
106
107fn check_struct(tcx: TyCtxt<'_>, def_id: LocalDefId) -> Result<(), ErrorGuaranteed> {
108 let def = tcx.adt_def(def_id);
109 let span = tcx.def_span(def_id);
110 def.destructor(tcx); if let Some(scalable) = def.repr().scalable {
113 check_scalable_vector(tcx, span, def_id, scalable);
114 } else if def.repr().simd() {
115 check_simd(tcx, span, def_id);
116 }
117
118 check_transparent(tcx, def);
119 check_packed(tcx, span, def);
120 check_type_defn(tcx, def_id, false)
121}
122
123fn check_union(tcx: TyCtxt<'_>, def_id: LocalDefId) -> Result<(), ErrorGuaranteed> {
124 let def = tcx.adt_def(def_id);
125 let span = tcx.def_span(def_id);
126 def.destructor(tcx); check_transparent(tcx, def);
128 check_union_fields(tcx, span, def_id);
129 check_packed(tcx, span, def);
130 check_type_defn(tcx, def_id, true)
131}
132
133fn allowed_union_or_unsafe_field<'tcx>(
134 tcx: TyCtxt<'tcx>,
135 ty: Ty<'tcx>,
136 typing_env: ty::TypingEnv<'tcx>,
137 span: Span,
138) -> bool {
139 if ty.is_trivially_pure_clone_copy() {
144 return true;
145 }
146 let def_id = tcx
149 .lang_items()
150 .get(LangItem::BikeshedGuaranteedNoDrop)
151 .unwrap_or_else(|| tcx.require_lang_item(LangItem::Copy, span));
152 let Ok(ty) = tcx.try_normalize_erasing_regions(typing_env, Unnormalized::new_wip(ty)) else {
153 tcx.dcx().span_delayed_bug(span, "could not normalize field type");
154 return true;
155 };
156 let (infcx, param_env) = tcx.infer_ctxt().build_with_typing_env(typing_env);
157 infcx.predicate_must_hold_modulo_regions(&Obligation::new(
158 tcx,
159 ObligationCause::dummy_with_span(span),
160 param_env,
161 ty::TraitRef::new(tcx, def_id, [ty]),
162 ))
163}
164
165fn check_union_fields(tcx: TyCtxt<'_>, span: Span, item_def_id: LocalDefId) -> bool {
167 let def = tcx.adt_def(item_def_id);
168 if !def.is_union() {
::core::panicking::panic("assertion failed: def.is_union()")
};assert!(def.is_union());
169
170 let typing_env = ty::TypingEnv::non_body_analysis(tcx, item_def_id);
171 let args = ty::GenericArgs::identity_for_item(tcx, item_def_id);
172
173 for field in &def.non_enum_variant().fields {
174 if !allowed_union_or_unsafe_field(
175 tcx,
176 field.ty(tcx, args).skip_norm_wip(),
177 typing_env,
178 span,
179 ) {
180 let (field_span, ty_span) = match tcx.hir_get_if_local(field.did) {
181 Some(Node::Field(field)) => (field.span, field.ty.span),
183 _ => {
::core::panicking::panic_fmt(format_args!("internal error: entered unreachable code: {0}",
format_args!("mir field has to correspond to hir field")));
}unreachable!("mir field has to correspond to hir field"),
184 };
185 tcx.dcx().emit_err(diagnostics::InvalidUnionField {
186 field_span,
187 sugg: diagnostics::InvalidUnionFieldSuggestion {
188 lo: ty_span.shrink_to_lo(),
189 hi: ty_span.shrink_to_hi(),
190 },
191 note: (),
192 });
193 return false;
194 }
195 }
196
197 true
198}
199
200fn check_static_inhabited(tcx: TyCtxt<'_>, def_id: LocalDefId) {
202 #[derive(const _: () =
{
impl<'_sess, G> rustc_errors::Diagnostic<'_sess, G> for
StaticOfUninhabitedType where G: rustc_errors::EmissionGuarantee {
#[track_caller]
fn into_diag(self, dcx: rustc_errors::DiagCtxtHandle<'_sess>,
level: rustc_errors::Level) -> rustc_errors::Diag<'_sess, G> {
match self {
StaticOfUninhabitedType => {
let mut diag =
rustc_errors::Diag::new(dcx, level,
rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("static of uninhabited type")));
diag.note(rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("uninhabited statics cannot be initialized, and any access would be an immediate error")));
;
diag
}
}
}
}
};Diagnostic)]
203 #[diag("static of uninhabited type")]
204 #[note("uninhabited statics cannot be initialized, and any access would be an immediate error")]
205 struct StaticOfUninhabitedType;
206
207 let ty = tcx.type_of(def_id).instantiate_identity().skip_norm_wip();
213 let span = tcx.def_span(def_id);
214 let layout = match tcx.layout_of(ty::TypingEnv::fully_monomorphized().as_query_input(ty)) {
215 Ok(l) => l,
216 Err(LayoutError::SizeOverflow(_))
218 if #[allow(non_exhaustive_omitted_patterns)] match tcx.def_kind(def_id) {
DefKind::Static { .. } if
tcx.def_kind(tcx.local_parent(def_id)) == DefKind::ForeignMod => true,
_ => false,
}matches!(tcx.def_kind(def_id), DefKind::Static{ .. }
219 if tcx.def_kind(tcx.local_parent(def_id)) == DefKind::ForeignMod) =>
220 {
221 tcx.dcx().emit_err(diagnostics::TooLargeStatic { span });
222 return;
223 }
224 Err(e @ LayoutError::InvalidSimd { .. }) => {
226 let ty_span = tcx.ty_span(def_id);
227 tcx.dcx().span_err(ty_span, e.to_string());
228 return;
229 }
230 Err(e) => {
232 tcx.dcx().span_delayed_bug(span, ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{0:?}", e))
})format!("{e:?}"));
233 return;
234 }
235 };
236 if layout.is_uninhabited() {
237 tcx.emit_node_span_lint(
238 UNINHABITED_STATIC,
239 tcx.local_def_id_to_hir_id(def_id),
240 span,
241 StaticOfUninhabitedType,
242 );
243 }
244}
245
246fn check_opaque(tcx: TyCtxt<'_>, def_id: LocalDefId) {
249 let hir::OpaqueTy { origin, .. } = *tcx.hir_expect_opaque_ty(def_id);
250
251 if tcx.sess.opts.actually_rustdoc {
256 return;
257 }
258
259 if tcx.type_of(def_id).instantiate_identity().skip_norm_wip().references_error() {
260 return;
261 }
262 if check_opaque_for_cycles(tcx, def_id).is_err() {
263 return;
264 }
265
266 let _ = check_opaque_meets_bounds(tcx, def_id, origin);
267}
268
269pub(super) fn check_opaque_for_cycles<'tcx>(
271 tcx: TyCtxt<'tcx>,
272 def_id: LocalDefId,
273) -> Result<(), ErrorGuaranteed> {
274 let args = GenericArgs::identity_for_item(tcx, def_id);
275
276 if tcx.try_expand_impl_trait_type(def_id.to_def_id(), args).is_err() {
279 let reported = opaque_type_cycle_error(tcx, def_id);
280 return Err(reported);
281 }
282
283 Ok(())
284}
285
286#[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_opaque_meets_bounds",
"rustc_hir_analysis::check::check", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_analysis/src/check/check.rs"),
::tracing_core::__macro_support::Option::Some(301u32),
::tracing_core::__macro_support::Option::Some("rustc_hir_analysis::check::check"),
::tracing_core::field::FieldSet::new(&["def_id", "origin"],
::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(&def_id)
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(&origin)
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: Result<(), ErrorGuaranteed> =
loop {};
return __tracing_attr_fake_return;
}
{
let (span, definition_def_id) =
if let Some((span, def_id)) =
best_definition_site_of_opaque(tcx, def_id, origin) {
(span, Some(def_id))
} else { (tcx.def_span(def_id), None) };
let defining_use_anchor =
match origin {
hir::OpaqueTyOrigin::FnReturn { parent, .. } |
hir::OpaqueTyOrigin::AsyncFn { parent, .. } |
hir::OpaqueTyOrigin::TyAlias { parent, .. } => parent,
};
let param_env = tcx.param_env(defining_use_anchor);
let infcx =
tcx.infer_ctxt().build(if tcx.next_trait_solver_globally() {
TypingMode::post_borrowck_analysis(tcx, defining_use_anchor)
} else {
TypingMode::analysis_in_body(tcx, defining_use_anchor)
});
let ocx = ObligationCtxt::new_with_diagnostics(&infcx);
let args =
match origin {
hir::OpaqueTyOrigin::FnReturn { parent, .. } |
hir::OpaqueTyOrigin::AsyncFn { parent, .. } |
hir::OpaqueTyOrigin::TyAlias { parent, .. } =>
GenericArgs::identity_for_item(tcx,
parent).extend_to(tcx, def_id.to_def_id(),
|param, _|
{
tcx.map_opaque_lifetime_to_parent_lifetime(param.def_id.expect_local()).into()
}),
};
let opaque_ty = Ty::new_opaque(tcx, def_id.to_def_id(), args);
let hidden_ty =
tcx.type_of(def_id.to_def_id()).instantiate(tcx,
args).skip_norm_wip();
let hidden_ty =
fold_regions(tcx, hidden_ty,
|re, _dbi|
match re.kind() {
ty::ReErased =>
infcx.next_region_var(RegionVariableOrigin::Misc(span)),
_ => re,
});
for (predicate, pred_span) in
tcx.explicit_item_bounds(def_id).iter_instantiated_copied(tcx,
args).map(Unnormalized::skip_norm_wip) {
let predicate =
predicate.fold_with(&mut BottomUpFolder {
tcx,
ty_op: |ty| if ty == opaque_ty { hidden_ty } else { ty },
lt_op: |lt| lt,
ct_op: |ct| ct,
});
ocx.register_obligation(Obligation::new(tcx,
ObligationCause::new(span, def_id,
ObligationCauseCode::OpaqueTypeBound(pred_span,
definition_def_id)), param_env, predicate));
}
let misc_cause = ObligationCause::misc(span, def_id);
match ocx.eq(&misc_cause, param_env, opaque_ty, hidden_ty) {
Ok(()) => {}
Err(ty_err) => {
let ty_err = ty_err.to_string(tcx);
let guar =
tcx.dcx().span_delayed_bug(span,
::alloc::__export::must_use({
::alloc::fmt::format(format_args!("could not unify `{0}` with revealed type:\n{1}",
hidden_ty, ty_err))
}));
return Err(guar);
}
}
let predicate =
ty::Binder::dummy(ty::PredicateKind::Clause(ty::ClauseKind::WellFormed(hidden_ty.into())));
ocx.register_obligation(Obligation::new(tcx, misc_cause.clone(),
param_env, predicate));
let errors = ocx.evaluate_obligations_error_on_ambiguity();
if !errors.is_empty() {
let guar = infcx.err_ctxt().report_fulfillment_errors(errors);
return Err(guar);
}
let wf_tys =
ocx.assumed_wf_types_and_report_errors(param_env,
defining_use_anchor)?;
ocx.resolve_regions_and_report_errors(defining_use_anchor,
param_env, wf_tys)?;
if infcx.next_trait_solver() {
Ok(())
} else if let hir::OpaqueTyOrigin::FnReturn { .. } |
hir::OpaqueTyOrigin::AsyncFn { .. } = origin {
let _ = infcx.take_opaque_types();
Ok(())
} else {
for (mut key, mut ty) in infcx.take_opaque_types() {
ty.ty = infcx.resolve_vars_if_possible(ty.ty);
key = infcx.resolve_vars_if_possible(key);
sanity_check_found_hidden_type(tcx, key, ty)?;
}
Ok(())
}
}
}
}#[instrument(level = "debug", skip(tcx))]
302fn check_opaque_meets_bounds<'tcx>(
303 tcx: TyCtxt<'tcx>,
304 def_id: LocalDefId,
305 origin: hir::OpaqueTyOrigin<LocalDefId>,
306) -> Result<(), ErrorGuaranteed> {
307 let (span, definition_def_id) =
308 if let Some((span, def_id)) = best_definition_site_of_opaque(tcx, def_id, origin) {
309 (span, Some(def_id))
310 } else {
311 (tcx.def_span(def_id), None)
312 };
313
314 let defining_use_anchor = match origin {
315 hir::OpaqueTyOrigin::FnReturn { parent, .. }
316 | hir::OpaqueTyOrigin::AsyncFn { parent, .. }
317 | hir::OpaqueTyOrigin::TyAlias { parent, .. } => parent,
318 };
319 let param_env = tcx.param_env(defining_use_anchor);
320
321 let infcx = tcx.infer_ctxt().build(if tcx.next_trait_solver_globally() {
323 TypingMode::post_borrowck_analysis(tcx, defining_use_anchor)
324 } else {
325 TypingMode::analysis_in_body(tcx, defining_use_anchor)
326 });
327 let ocx = ObligationCtxt::new_with_diagnostics(&infcx);
328
329 let args = match origin {
330 hir::OpaqueTyOrigin::FnReturn { parent, .. }
331 | hir::OpaqueTyOrigin::AsyncFn { parent, .. }
332 | hir::OpaqueTyOrigin::TyAlias { parent, .. } => GenericArgs::identity_for_item(
333 tcx, parent,
334 )
335 .extend_to(tcx, def_id.to_def_id(), |param, _| {
336 tcx.map_opaque_lifetime_to_parent_lifetime(param.def_id.expect_local()).into()
337 }),
338 };
339
340 let opaque_ty = Ty::new_opaque(tcx, def_id.to_def_id(), args);
341
342 let hidden_ty = tcx.type_of(def_id.to_def_id()).instantiate(tcx, args).skip_norm_wip();
349 let hidden_ty = fold_regions(tcx, hidden_ty, |re, _dbi| match re.kind() {
350 ty::ReErased => infcx.next_region_var(RegionVariableOrigin::Misc(span)),
351 _ => re,
352 });
353
354 for (predicate, pred_span) in tcx
358 .explicit_item_bounds(def_id)
359 .iter_instantiated_copied(tcx, args)
360 .map(Unnormalized::skip_norm_wip)
361 {
362 let predicate = predicate.fold_with(&mut BottomUpFolder {
363 tcx,
364 ty_op: |ty| if ty == opaque_ty { hidden_ty } else { ty },
365 lt_op: |lt| lt,
366 ct_op: |ct| ct,
367 });
368
369 ocx.register_obligation(Obligation::new(
370 tcx,
371 ObligationCause::new(
372 span,
373 def_id,
374 ObligationCauseCode::OpaqueTypeBound(pred_span, definition_def_id),
375 ),
376 param_env,
377 predicate,
378 ));
379 }
380
381 let misc_cause = ObligationCause::misc(span, def_id);
382 match ocx.eq(&misc_cause, param_env, opaque_ty, hidden_ty) {
386 Ok(()) => {}
387 Err(ty_err) => {
388 let ty_err = ty_err.to_string(tcx);
394 let guar = tcx.dcx().span_delayed_bug(
395 span,
396 format!("could not unify `{hidden_ty}` with revealed type:\n{ty_err}"),
397 );
398 return Err(guar);
399 }
400 }
401
402 let predicate =
406 ty::Binder::dummy(ty::PredicateKind::Clause(ty::ClauseKind::WellFormed(hidden_ty.into())));
407 ocx.register_obligation(Obligation::new(tcx, misc_cause.clone(), param_env, predicate));
408
409 let errors = ocx.evaluate_obligations_error_on_ambiguity();
412 if !errors.is_empty() {
413 let guar = infcx.err_ctxt().report_fulfillment_errors(errors);
414 return Err(guar);
415 }
416
417 let wf_tys = ocx.assumed_wf_types_and_report_errors(param_env, defining_use_anchor)?;
418 ocx.resolve_regions_and_report_errors(defining_use_anchor, param_env, wf_tys)?;
419
420 if infcx.next_trait_solver() {
421 Ok(())
422 } else if let hir::OpaqueTyOrigin::FnReturn { .. } | hir::OpaqueTyOrigin::AsyncFn { .. } =
423 origin
424 {
425 let _ = infcx.take_opaque_types();
431 Ok(())
432 } else {
433 for (mut key, mut ty) in infcx.take_opaque_types() {
435 ty.ty = infcx.resolve_vars_if_possible(ty.ty);
436 key = infcx.resolve_vars_if_possible(key);
437 sanity_check_found_hidden_type(tcx, key, ty)?;
438 }
439 Ok(())
440 }
441}
442
443fn best_definition_site_of_opaque<'tcx>(
444 tcx: TyCtxt<'tcx>,
445 opaque_def_id: LocalDefId,
446 origin: hir::OpaqueTyOrigin<LocalDefId>,
447) -> Option<(Span, LocalDefId)> {
448 struct TaitConstraintLocator<'tcx> {
449 opaque_def_id: LocalDefId,
450 tcx: TyCtxt<'tcx>,
451 }
452 impl<'tcx> TaitConstraintLocator<'tcx> {
453 fn check(&self, item_def_id: LocalDefId) -> ControlFlow<(Span, LocalDefId)> {
454 if !self.tcx.has_typeck_results(item_def_id) {
455 return ControlFlow::Continue(());
456 }
457
458 let opaque_types_defined_by = self.tcx.opaque_types_defined_by(item_def_id);
459 if !opaque_types_defined_by.contains(&self.opaque_def_id) {
461 return ControlFlow::Continue(());
462 }
463
464 if let Some(hidden_ty) = self
465 .tcx
466 .mir_borrowck(item_def_id)
467 .ok()
468 .and_then(|opaque_types| opaque_types.get(&self.opaque_def_id))
469 {
470 ControlFlow::Break((hidden_ty.span, item_def_id))
471 } else {
472 ControlFlow::Continue(())
473 }
474 }
475 }
476 impl<'tcx> intravisit::Visitor<'tcx> for TaitConstraintLocator<'tcx> {
477 type NestedFilter = nested_filter::All;
478 type Result = ControlFlow<(Span, LocalDefId)>;
479 fn maybe_tcx(&mut self) -> Self::MaybeTyCtxt {
480 self.tcx
481 }
482 fn visit_expr(&mut self, ex: &'tcx hir::Expr<'tcx>) -> Self::Result {
483 intravisit::walk_expr(self, ex)
484 }
485 fn visit_item(&mut self, it: &'tcx hir::Item<'tcx>) -> Self::Result {
486 self.check(it.owner_id.def_id)?;
487 intravisit::walk_item(self, it)
488 }
489 fn visit_impl_item(&mut self, it: &'tcx hir::ImplItem<'tcx>) -> Self::Result {
490 self.check(it.owner_id.def_id)?;
491 intravisit::walk_impl_item(self, it)
492 }
493 fn visit_trait_item(&mut self, it: &'tcx hir::TraitItem<'tcx>) -> Self::Result {
494 self.check(it.owner_id.def_id)?;
495 intravisit::walk_trait_item(self, it)
496 }
497 fn visit_foreign_item(&mut self, it: &'tcx hir::ForeignItem<'tcx>) -> Self::Result {
498 intravisit::walk_foreign_item(self, it)
499 }
500 }
501
502 let mut locator = TaitConstraintLocator { tcx, opaque_def_id };
503 match origin {
504 hir::OpaqueTyOrigin::FnReturn { parent, .. }
505 | hir::OpaqueTyOrigin::AsyncFn { parent, .. } => locator.check(parent).break_value(),
506 hir::OpaqueTyOrigin::TyAlias { parent, in_assoc_ty: true } => {
507 let impl_def_id = tcx.local_parent(parent);
508 for assoc in tcx.associated_items(impl_def_id).in_definition_order() {
509 match assoc.kind {
510 ty::AssocKind::Const { .. } | ty::AssocKind::Fn { .. } => {
511 if let ControlFlow::Break(span) = locator.check(assoc.def_id.expect_local())
512 {
513 return Some(span);
514 }
515 }
516 ty::AssocKind::Type { .. } => {}
517 }
518 }
519
520 None
521 }
522 hir::OpaqueTyOrigin::TyAlias { in_assoc_ty: false, .. } => {
523 tcx.hir_walk_toplevel_module(&mut locator).break_value()
524 }
525 }
526}
527
528fn sanity_check_found_hidden_type<'tcx>(
529 tcx: TyCtxt<'tcx>,
530 key: ty::OpaqueTypeKey<'tcx>,
531 mut ty: ty::ProvisionalHiddenType<'tcx>,
532) -> Result<(), ErrorGuaranteed> {
533 if ty.ty.is_ty_var() {
534 return Ok(());
536 }
537 if let &ty::Alias(ty::AliasTy { kind: ty::Opaque { def_id }, args, .. }) = ty.ty.kind() {
538 if def_id == key.def_id.to_def_id() && args == key.args {
539 return Ok(());
542 }
543 }
544 let erase_re_vars = |ty: Ty<'tcx>| {
545 fold_regions(tcx, ty, |r, _| match r.kind() {
546 RegionKind::ReVar(_) => tcx.lifetimes.re_erased,
547 _ => r,
548 })
549 };
550 ty.ty = erase_re_vars(ty.ty);
553 let hidden_ty = tcx.type_of(key.def_id).instantiate(tcx, key.args).skip_norm_wip();
555 let hidden_ty = erase_re_vars(hidden_ty);
556
557 if hidden_ty == ty.ty {
559 Ok(())
560 } else {
561 let span = tcx.def_span(key.def_id);
562 let other = ty::ProvisionalHiddenType { ty: hidden_ty, span };
563 Err(ty.build_mismatch_error(&other, tcx)?.emit())
564 }
565}
566
567fn check_opaque_precise_captures<'tcx>(tcx: TyCtxt<'tcx>, opaque_def_id: LocalDefId) {
576 let hir::OpaqueTy { bounds, .. } = *tcx.hir_node_by_def_id(opaque_def_id).expect_opaque_ty();
577 let Some(precise_capturing_args) = bounds.iter().find_map(|bound| match *bound {
578 hir::GenericBound::Use(bounds, ..) => Some(bounds),
579 _ => None,
580 }) else {
581 return;
583 };
584
585 let mut expected_captures = UnordSet::default();
586 let mut shadowed_captures = UnordSet::default();
587 let mut seen_params = UnordMap::default();
588 let mut prev_non_lifetime_param = None;
589 for arg in precise_capturing_args {
590 let (hir_id, ident) = match *arg {
591 hir::PreciseCapturingArg::Param(hir::PreciseCapturingNonLifetimeArg {
592 hir_id,
593 ident,
594 ..
595 }) => {
596 if prev_non_lifetime_param.is_none() {
597 prev_non_lifetime_param = Some(ident);
598 }
599 (hir_id, ident)
600 }
601 hir::PreciseCapturingArg::Lifetime(&hir::Lifetime { hir_id, ident, .. }) => {
602 if let Some(prev_non_lifetime_param) = prev_non_lifetime_param {
603 tcx.dcx().emit_err(diagnostics::LifetimesMustBeFirst {
604 lifetime_span: ident.span,
605 name: ident.name,
606 other_span: prev_non_lifetime_param.span,
607 });
608 }
609 (hir_id, ident)
610 }
611 };
612
613 let ident = ident.normalize_to_macros_2_0();
614 if let Some(span) = seen_params.insert(ident, ident.span) {
615 tcx.dcx().emit_err(diagnostics::DuplicatePreciseCapture {
616 name: ident.name,
617 first_span: span,
618 second_span: ident.span,
619 });
620 }
621
622 match tcx.named_bound_var(hir_id) {
623 Some(ResolvedArg::EarlyBound(def_id)) => {
624 expected_captures.insert(def_id.to_def_id());
625
626 if let DefKind::LifetimeParam = tcx.def_kind(def_id)
632 && let Some(def_id) = tcx
633 .map_opaque_lifetime_to_parent_lifetime(def_id)
634 .opt_param_def_id(tcx, tcx.parent(opaque_def_id.to_def_id()))
635 {
636 shadowed_captures.insert(def_id);
637 }
638 }
639 _ => {
640 tcx.dcx()
641 .span_delayed_bug(tcx.hir_span(hir_id), "parameter should have been resolved");
642 }
643 }
644 }
645
646 let variances = tcx.variances_of(opaque_def_id);
647 let mut def_id = Some(opaque_def_id.to_def_id());
648 while let Some(generics) = def_id {
649 let generics = tcx.generics_of(generics);
650 def_id = generics.parent;
651
652 for param in &generics.own_params {
653 if expected_captures.contains(¶m.def_id) {
654 match (&variances[param.index as usize], &ty::Invariant) {
(left_val, right_val) => {
if !(*left_val == *right_val) {
let kind = ::core::panicking::AssertKind::Eq;
::core::panicking::assert_failed(kind, &*left_val, &*right_val,
::core::option::Option::Some(format_args!("precise captured param should be invariant")));
}
}
};assert_eq!(
655 variances[param.index as usize],
656 ty::Invariant,
657 "precise captured param should be invariant"
658 );
659 continue;
660 }
661 if shadowed_captures.contains(¶m.def_id) {
665 continue;
666 }
667
668 match param.kind {
669 ty::GenericParamDefKind::Lifetime => {
670 let use_span = tcx.def_span(param.def_id);
671 let opaque_span = tcx.def_span(opaque_def_id);
672 if variances[param.index as usize] == ty::Invariant {
674 if let DefKind::OpaqueTy = tcx.def_kind(tcx.parent(param.def_id))
675 && let Some(def_id) = tcx
676 .map_opaque_lifetime_to_parent_lifetime(param.def_id.expect_local())
677 .opt_param_def_id(tcx, tcx.parent(opaque_def_id.to_def_id()))
678 {
679 tcx.dcx().emit_err(diagnostics::LifetimeNotCaptured {
680 opaque_span,
681 use_span,
682 param_span: tcx.def_span(def_id),
683 });
684 } else {
685 if tcx.def_kind(tcx.parent(param.def_id)) == DefKind::Trait {
686 tcx.dcx().emit_err(diagnostics::LifetimeImplicitlyCaptured {
687 opaque_span,
688 param_span: tcx.def_span(param.def_id),
689 });
690 } else {
691 tcx.dcx().emit_err(diagnostics::LifetimeNotCaptured {
696 opaque_span,
697 use_span: opaque_span,
698 param_span: use_span,
699 });
700 }
701 }
702 continue;
703 }
704 }
705 ty::GenericParamDefKind::Type { .. } => {
706 if #[allow(non_exhaustive_omitted_patterns)] match tcx.def_kind(param.def_id) {
DefKind::Trait | DefKind::TraitAlias => true,
_ => false,
}matches!(tcx.def_kind(param.def_id), DefKind::Trait | DefKind::TraitAlias) {
707 tcx.dcx().emit_err(diagnostics::SelfTyNotCaptured {
709 trait_span: tcx.def_span(param.def_id),
710 opaque_span: tcx.def_span(opaque_def_id),
711 });
712 } else {
713 tcx.dcx().emit_err(diagnostics::ParamNotCaptured {
715 param_span: tcx.def_span(param.def_id),
716 opaque_span: tcx.def_span(opaque_def_id),
717 kind: "type",
718 });
719 }
720 }
721 ty::GenericParamDefKind::Const { .. } => {
722 tcx.dcx().emit_err(diagnostics::ParamNotCaptured {
724 param_span: tcx.def_span(param.def_id),
725 opaque_span: tcx.def_span(opaque_def_id),
726 kind: "const",
727 });
728 }
729 }
730 }
731 }
732}
733
734fn is_enum_of_nonnullable_ptr<'tcx>(
735 tcx: TyCtxt<'tcx>,
736 adt_def: AdtDef<'tcx>,
737 args: GenericArgsRef<'tcx>,
738) -> bool {
739 if adt_def.repr().inhibit_enum_layout_opt() {
740 return false;
741 }
742
743 let [var_one, var_two] = &adt_def.variants().raw[..] else {
744 return false;
745 };
746 let (([], [field]) | ([field], [])) = (&var_one.fields.raw[..], &var_two.fields.raw[..]) else {
747 return false;
748 };
749 #[allow(non_exhaustive_omitted_patterns)] match field.ty(tcx,
args).skip_norm_wip().kind() {
ty::FnPtr(..) | ty::Ref(..) => true,
_ => false,
}matches!(field.ty(tcx, args).skip_norm_wip().kind(), ty::FnPtr(..) | ty::Ref(..))
750}
751
752fn check_static_linkage(tcx: TyCtxt<'_>, def_id: LocalDefId) {
753 if tcx.codegen_fn_attrs(def_id).import_linkage.is_some() {
754 if match tcx.type_of(def_id).instantiate_identity().skip_norm_wip().kind() {
755 ty::RawPtr(_, _) => false,
756 ty::Adt(adt_def, args) => !is_enum_of_nonnullable_ptr(tcx, *adt_def, *args),
757 _ => true,
758 } {
759 tcx.dcx().emit_err(diagnostics::LinkageType { span: tcx.def_span(def_id) });
760 }
761 }
762}
763
764pub(crate) fn check_item_type(tcx: TyCtxt<'_>, def_id: LocalDefId) -> Result<(), ErrorGuaranteed> {
765 let mut res = Ok(());
766 let generics = tcx.generics_of(def_id);
767
768 for param in &generics.own_params {
769 match param.kind {
770 ty::GenericParamDefKind::Lifetime { .. } => {}
771 ty::GenericParamDefKind::Type { has_default, .. } => {
772 if has_default {
773 tcx.ensure_ok().type_of(param.def_id);
774 }
775 }
776 ty::GenericParamDefKind::Const { has_default, .. } => {
777 tcx.ensure_ok().type_of(param.def_id);
778 if has_default {
779 let ct = tcx.const_param_default(param.def_id).skip_binder();
781 if let ty::ConstKind::Unevaluated(uv) = ct.kind()
782 && let Some(def_id) = uv.kind.opt_def_id()
783 {
784 tcx.ensure_ok().type_of(def_id);
785 }
786 }
787 }
788 }
789 }
790
791 match tcx.def_kind(def_id) {
792 DefKind::Static { .. } => {
793 tcx.ensure_ok().generics_of(def_id);
794 tcx.ensure_ok().type_of(def_id);
795 tcx.ensure_ok().predicates_of(def_id);
796
797 check_static_inhabited(tcx, def_id);
798 check_static_linkage(tcx, def_id);
799 let ty = tcx.type_of(def_id).instantiate_identity().skip_norm_wip();
800 res = res.and(wfcheck::check_static_item(
801 tcx, def_id, ty, true,
802 ));
803
804 return res;
808 }
809 DefKind::Enum => {
810 tcx.ensure_ok().generics_of(def_id);
811 tcx.ensure_ok().type_of(def_id);
812 tcx.ensure_ok().predicates_of(def_id);
813 crate::collect::check_enum_variant_types(tcx, def_id);
814 check_enum(tcx, def_id);
815 check_variances_for_type_defn(tcx, def_id);
816 res = res.and(check_type_defn(tcx, def_id, true));
817 return res;
819 }
820 DefKind::Fn => {
821 tcx.ensure_ok().generics_of(def_id);
822 tcx.ensure_ok().type_of(def_id);
823 tcx.ensure_ok().predicates_of(def_id);
824 tcx.ensure_ok().fn_sig(def_id);
825 tcx.ensure_ok().codegen_fn_attrs(def_id);
826 if let Some(i) = tcx.intrinsic(def_id) {
827 intrinsic::check_intrinsic_type(
828 tcx,
829 def_id,
830 tcx.def_ident_span(def_id).unwrap(),
831 i.name,
832 )
833 }
834 }
835 DefKind::Impl { of_trait } => {
836 tcx.ensure_ok().generics_of(def_id);
837 tcx.ensure_ok().type_of(def_id);
838 tcx.ensure_ok().predicates_of(def_id);
839 tcx.ensure_ok().associated_items(def_id);
840 if of_trait {
841 let impl_trait_header = tcx.impl_trait_header(def_id);
842 res = res.and(tcx.ensure_result().coherent_trait(
843 impl_trait_header.trait_ref.instantiate_identity().skip_norm_wip().def_id,
844 ));
845
846 if res.is_ok() {
847 check_impl_items_against_trait(tcx, def_id, impl_trait_header);
851 }
852 }
853 }
854 DefKind::Trait => {
855 tcx.ensure_ok().generics_of(def_id);
856 tcx.ensure_ok().trait_def(def_id);
857 tcx.ensure_ok().explicit_super_predicates_of(def_id);
858 tcx.ensure_ok().predicates_of(def_id);
859 tcx.ensure_ok().associated_items(def_id);
860 let assoc_items = tcx.associated_items(def_id);
861
862 for &assoc_item in assoc_items.in_definition_order() {
863 match assoc_item.kind {
864 ty::AssocKind::Type { .. } if assoc_item.defaultness(tcx).has_value() => {
865 let trait_args = GenericArgs::identity_for_item(tcx, def_id);
866 let _: Result<_, rustc_errors::ErrorGuaranteed> = check_type_bounds(
867 tcx,
868 assoc_item,
869 assoc_item,
870 ty::TraitRef::new_from_args(tcx, def_id.to_def_id(), trait_args),
871 );
872 }
873 _ => {}
874 }
875 }
876 res = res.and(wfcheck::check_trait(tcx, def_id));
877 wfcheck::check_gat_where_clauses(tcx, def_id);
878 return res;
880 }
881 DefKind::TraitAlias => {
882 tcx.ensure_ok().generics_of(def_id);
883 tcx.ensure_ok().explicit_implied_predicates_of(def_id);
884 tcx.ensure_ok().explicit_super_predicates_of(def_id);
885 tcx.ensure_ok().predicates_of(def_id);
886 res = res.and(wfcheck::check_trait(tcx, def_id));
887 return res;
889 }
890 def_kind @ (DefKind::Struct | DefKind::Union) => {
891 tcx.ensure_ok().generics_of(def_id);
892 tcx.ensure_ok().type_of(def_id);
893 tcx.ensure_ok().predicates_of(def_id);
894
895 let adt = tcx.adt_def(def_id).non_enum_variant();
896 for f in adt.fields.iter() {
897 tcx.ensure_ok().generics_of(f.did);
898 tcx.ensure_ok().type_of(f.did);
899 tcx.ensure_ok().predicates_of(f.did);
900 }
901
902 if let Some((_, ctor_def_id)) = adt.ctor {
903 crate::collect::check_ctor(tcx, ctor_def_id.expect_local());
904 }
905 check_variances_for_type_defn(tcx, def_id);
906 res = res.and(match def_kind {
907 DefKind::Struct => check_struct(tcx, def_id),
908 DefKind::Union => check_union(tcx, def_id),
909 _ => ::core::panicking::panic("internal error: entered unreachable code")unreachable!(),
910 });
911 return res;
913 }
914 DefKind::OpaqueTy => {
915 check_opaque_precise_captures(tcx, def_id);
916
917 let origin = tcx.local_opaque_ty_origin(def_id);
918 if let hir::OpaqueTyOrigin::FnReturn { parent: fn_def_id, .. }
919 | hir::OpaqueTyOrigin::AsyncFn { parent: fn_def_id, .. } = origin
920 && let hir::Node::TraitItem(trait_item) = tcx.hir_node_by_def_id(fn_def_id)
921 && let (_, hir::TraitFn::Required(..)) = trait_item.expect_fn()
922 {
923 } else {
925 check_opaque(tcx, def_id);
926 }
927
928 tcx.ensure_ok().predicates_of(def_id);
929 tcx.ensure_ok().explicit_item_bounds(def_id);
930 tcx.ensure_ok().explicit_item_self_bounds(def_id);
931 if tcx.is_conditionally_const(def_id) {
932 tcx.ensure_ok().explicit_implied_const_bounds(def_id);
933 tcx.ensure_ok().const_conditions(def_id);
934 }
935
936 return res;
940 }
941 DefKind::Const { .. } => {
942 tcx.ensure_ok().generics_of(def_id);
943 tcx.ensure_ok().type_of(def_id);
944 tcx.ensure_ok().predicates_of(def_id);
945
946 res = res.and(enter_wf_checking_ctxt(tcx, def_id, |wfcx| {
947 let ty = tcx.type_of(def_id).instantiate_identity();
948 let ty_span = tcx.ty_span(def_id);
949 let ty = wfcx.deeply_normalize(ty_span, Some(WellFormedLoc::Ty(def_id)), ty);
950 wfcx.register_wf_obligation(ty_span, Some(WellFormedLoc::Ty(def_id)), ty.into());
951 wfcx.register_bound(
952 traits::ObligationCause::new(
953 ty_span,
954 def_id,
955 ObligationCauseCode::SizedConstOrStatic,
956 ),
957 tcx.param_env(def_id),
958 ty,
959 tcx.require_lang_item(LangItem::Sized, ty_span),
960 );
961 check_where_clauses(wfcx, def_id);
962
963 if tcx.is_type_const(def_id) {
964 wfcheck::check_type_const(wfcx, def_id, ty, true)?;
965 }
966 Ok(())
967 }));
968
969 return res;
973 }
974 DefKind::TyAlias => {
975 tcx.ensure_ok().generics_of(def_id);
976 tcx.ensure_ok().type_of(def_id);
977 tcx.ensure_ok().predicates_of(def_id);
978 let ty = tcx.type_of(def_id).instantiate_identity();
979 let span = tcx.def_span(def_id);
980 if tcx.type_alias_is_lazy(def_id) {
981 res = res.and(enter_wf_checking_ctxt(tcx, def_id, |wfcx| {
982 let item_ty = wfcx.deeply_normalize(span, Some(WellFormedLoc::Ty(def_id)), ty);
983 wfcx.register_wf_obligation(
984 span,
985 Some(WellFormedLoc::Ty(def_id)),
986 item_ty.into(),
987 );
988 check_where_clauses(wfcx, def_id);
989 Ok(())
990 }));
991 } else {
992 check_type_alias_type_params_are_used(tcx, def_id);
993 res = res.and(enter_wf_checking_ctxt(tcx, def_id, |wfcx| {
994 if let Some(unnormalized_obligations) = wfcx.unnormalized_obligations(span, ty.skip_norm_wip())
1005 {
1006 let filtered_obligations =
1007 unnormalized_obligations.into_iter().filter(|o| {
1008 #[allow(non_exhaustive_omitted_patterns)] match o.predicate.kind().skip_binder()
{
ty::PredicateKind::Clause(ty::ClauseKind::ConstArgHasType(ct, _)) if
#[allow(non_exhaustive_omitted_patterns)] match ct.kind() {
ty::ConstKind::Param(..) => true,
_ => false,
} => true,
_ => false,
}matches!(o.predicate.kind().skip_binder(),
1009 ty::PredicateKind::Clause(ty::ClauseKind::ConstArgHasType(ct, _))
1010 if matches!(ct.kind(), ty::ConstKind::Param(..)))
1011 });
1012 wfcx.ocx.register_obligations(filtered_obligations)
1013 }
1014 Ok(())
1015 }));
1016 }
1017
1018 return res;
1022 }
1023 DefKind::ForeignMod => {
1024 let it = tcx.hir_expect_item(def_id);
1025 let hir::ItemKind::ForeignMod { abi, items } = it.kind else {
1026 return Ok(());
1027 };
1028
1029 check_abi(tcx, it.hir_id(), it.span, abi);
1030
1031 for &item in items {
1032 let def_id = item.owner_id.def_id;
1033
1034 let generics = tcx.generics_of(def_id);
1035 let own_counts = generics.own_counts();
1036 if generics.own_params.len() - own_counts.lifetimes != 0 {
1037 let (kinds, kinds_pl, egs) = match (own_counts.types, own_counts.consts) {
1038 (_, 0) => ("type", "types", Some("u32")),
1039 (0, _) => ("const", "consts", None),
1042 _ => ("type or const", "types or consts", None),
1043 };
1044 let name = if {
{
'done:
{
for i in ::rustc_hir::attrs::HasAttrs::get_attrs(def_id, &tcx)
{
#[allow(unused_imports)]
use rustc_hir::attrs::AttributeKind::*;
let i: &rustc_hir::Attribute = i;
match i {
rustc_hir::Attribute::Parsed(RustcEiiForeignItem) => {
break 'done Some(());
}
rustc_hir::Attribute::Unparsed(..) =>
{}
#[deny(unreachable_patterns)]
_ => {}
}
}
None
}
}
}.is_some()find_attr!(tcx, def_id, RustcEiiForeignItem) {
1045 "externally implementable items"
1046 } else {
1047 "foreign items"
1048 };
1049
1050 let span = tcx.def_span(def_id);
1051 {
tcx.dcx().struct_span_err(span,
::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{0} may not have {1} parameters",
name, kinds))
})).with_code(E0044)
}struct_span_code_err!(
1052 tcx.dcx(),
1053 span,
1054 E0044,
1055 "{name} may not have {kinds} parameters",
1056 )
1057 .with_span_label(span, ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("can\'t have {0} parameters",
kinds))
})format!("can't have {kinds} parameters"))
1058 .with_help(
1059 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("replace the {0} parameters with concrete {1}{2}",
kinds, kinds_pl,
egs.map(|egs|
::alloc::__export::must_use({
::alloc::fmt::format(format_args!(" like `{0}`", egs))
})).unwrap_or_default()))
})format!(
1062 "replace the {} parameters with concrete {}{}",
1063 kinds,
1064 kinds_pl,
1065 egs.map(|egs| format!(" like `{egs}`")).unwrap_or_default(),
1066 ),
1067 )
1068 .emit();
1069 }
1070
1071 tcx.ensure_ok().generics_of(def_id);
1072 tcx.ensure_ok().type_of(def_id);
1073 tcx.ensure_ok().predicates_of(def_id);
1074 if tcx.is_conditionally_const(def_id) {
1075 tcx.ensure_ok().explicit_implied_const_bounds(def_id);
1076 tcx.ensure_ok().const_conditions(def_id);
1077 }
1078 match tcx.def_kind(def_id) {
1079 DefKind::Fn => {
1080 tcx.ensure_ok().codegen_fn_attrs(def_id);
1081 tcx.ensure_ok().fn_sig(def_id);
1082 let item = tcx.hir_foreign_item(item);
1083 let hir::ForeignItemKind::Fn(sig, ..) = item.kind else { ::rustc_middle::util::bug::bug_fmt(format_args!("impossible case reached"))bug!() };
1084 check_c_variadic_abi(tcx, sig.decl, abi, item.span);
1085 }
1086 DefKind::Static { .. } => {
1087 tcx.ensure_ok().codegen_fn_attrs(def_id);
1088 }
1089 _ => (),
1090 }
1091 }
1092 return res;
1094 }
1095 DefKind::Closure => {
1096 tcx.ensure_ok().codegen_fn_attrs(def_id);
1100 return res;
1108 }
1109 DefKind::AssocFn => {
1110 tcx.ensure_ok().codegen_fn_attrs(def_id);
1111 tcx.ensure_ok().type_of(def_id);
1112 tcx.ensure_ok().fn_sig(def_id);
1113 tcx.ensure_ok().predicates_of(def_id);
1114 res = res.and(check_associated_item(tcx, def_id));
1115 let assoc_item = tcx.associated_item(def_id);
1116 match assoc_item.container {
1117 ty::AssocContainer::InherentImpl | ty::AssocContainer::TraitImpl(_) => {}
1118 ty::AssocContainer::Trait => {
1119 res = res.and(check_trait_item(tcx, def_id));
1120 }
1121 }
1122
1123 return res;
1127 }
1128 DefKind::AssocConst { .. } => {
1129 tcx.ensure_ok().type_of(def_id);
1130 tcx.ensure_ok().predicates_of(def_id);
1131 res = res.and(check_associated_item(tcx, def_id));
1132 let assoc_item = tcx.associated_item(def_id);
1133 match assoc_item.container {
1134 ty::AssocContainer::InherentImpl | ty::AssocContainer::TraitImpl(_) => {}
1135 ty::AssocContainer::Trait => {
1136 res = res.and(check_trait_item(tcx, def_id));
1137 }
1138 }
1139
1140 return res;
1144 }
1145 DefKind::AssocTy => {
1146 tcx.ensure_ok().predicates_of(def_id);
1147 res = res.and(check_associated_item(tcx, def_id));
1148
1149 let assoc_item = tcx.associated_item(def_id);
1150 let has_type = match assoc_item.container {
1151 ty::AssocContainer::InherentImpl | ty::AssocContainer::TraitImpl(_) => true,
1152 ty::AssocContainer::Trait => {
1153 tcx.ensure_ok().explicit_item_bounds(def_id);
1154 tcx.ensure_ok().explicit_item_self_bounds(def_id);
1155 if tcx.is_conditionally_const(def_id) {
1156 tcx.ensure_ok().explicit_implied_const_bounds(def_id);
1157 tcx.ensure_ok().const_conditions(def_id);
1158 }
1159 res = res.and(check_trait_item(tcx, def_id));
1160 assoc_item.defaultness(tcx).has_value()
1161 }
1162 };
1163 if has_type {
1164 tcx.ensure_ok().type_of(def_id);
1165 }
1166
1167 return res;
1171 }
1172
1173 DefKind::AnonConst
1175 | DefKind::InlineConst
1176 | DefKind::ExternCrate
1177 | DefKind::Macro(..)
1178 | DefKind::Use
1179 | DefKind::GlobalAsm
1180 | DefKind::Mod => return res,
1181 _ => {}
1182 }
1183 let node = tcx.hir_node_by_def_id(def_id);
1184 res.and(match node {
1185 hir::Node::Crate(_) => ::rustc_middle::util::bug::bug_fmt(format_args!("check_well_formed cannot be applied to the crate root"))bug!("check_well_formed cannot be applied to the crate root"),
1186 hir::Node::Item(item) => wfcheck::check_item(tcx, item),
1187 hir::Node::ForeignItem(item) => wfcheck::check_foreign_item(tcx, item),
1188 _ => {
::core::panicking::panic_fmt(format_args!("internal error: entered unreachable code: {0}",
format_args!("{0:?}", node)));
}unreachable!("{node:?}"),
1189 })
1190}
1191
1192pub(super) fn check_specialization_validity<'tcx>(
1193 tcx: TyCtxt<'tcx>,
1194 trait_def: &ty::TraitDef,
1195 trait_item: ty::AssocItem,
1196 impl_id: DefId,
1197 impl_item: DefId,
1198) {
1199 let Ok(ancestors) = trait_def.ancestors(tcx, impl_id) else { return };
1200 let mut ancestor_impls = ancestors.skip(1).filter_map(|parent| {
1201 if parent.is_from_trait() {
1202 None
1203 } else {
1204 Some((parent, parent.item(tcx, trait_item.def_id)))
1205 }
1206 });
1207
1208 let opt_result = ancestor_impls.find_map(|(parent_impl, parent_item)| {
1209 match parent_item {
1210 Some(parent_item) if traits::impl_item_is_final(tcx, &parent_item) => {
1213 Some(Err(parent_impl.def_id()))
1214 }
1215
1216 Some(_) => Some(Ok(())),
1218
1219 None => {
1223 if tcx.defaultness(parent_impl.def_id()).is_default() {
1224 None
1225 } else {
1226 Some(Err(parent_impl.def_id()))
1227 }
1228 }
1229 }
1230 });
1231
1232 let result = opt_result.unwrap_or(Ok(()));
1235
1236 if let Err(parent_impl) = result {
1237 if !tcx.is_impl_trait_in_trait(impl_item) {
1238 let span = tcx.def_span(impl_item);
1239 let ident = tcx.item_ident(impl_item);
1240
1241 let err = match tcx.span_of_impl(parent_impl) {
1242 Ok(sp) => diagnostics::ImplNotMarkedDefault::Ok { span, ident, ok_label: sp },
1243 Err(cname) => diagnostics::ImplNotMarkedDefault::Err { span, ident, cname },
1244 };
1245
1246 tcx.dcx().emit_err(err);
1247 } else {
1248 tcx.dcx().delayed_bug(::alloc::__export::must_use({
::alloc::fmt::format(format_args!("parent item: {0:?} not marked as default",
parent_impl))
})format!("parent item: {parent_impl:?} not marked as default"));
1249 }
1250 }
1251}
1252
1253fn check_overriding_final_trait_item<'tcx>(
1254 tcx: TyCtxt<'tcx>,
1255 trait_item: ty::AssocItem,
1256 impl_item: ty::AssocItem,
1257) {
1258 if trait_item.defaultness(tcx).is_final() {
1259 tcx.dcx().emit_err(diagnostics::OverridingFinalTraitFunction {
1260 impl_span: tcx.def_span(impl_item.def_id),
1261 trait_span: tcx.def_span(trait_item.def_id),
1262 ident: tcx.item_ident(impl_item.def_id),
1263 });
1264 }
1265}
1266
1267fn check_impl_items_against_trait<'tcx>(
1268 tcx: TyCtxt<'tcx>,
1269 impl_id: LocalDefId,
1270 impl_trait_header: ty::ImplTraitHeader<'tcx>,
1271) {
1272 let trait_ref = impl_trait_header.trait_ref.instantiate_identity().skip_norm_wip();
1273 if trait_ref.references_error() {
1277 return;
1278 }
1279
1280 let impl_item_refs = tcx.associated_item_def_ids(impl_id);
1281
1282 match impl_trait_header.polarity {
1284 ty::ImplPolarity::Reservation | ty::ImplPolarity::Positive => {}
1285 ty::ImplPolarity::Negative => {
1286 if let [first_item_ref, ..] = *impl_item_refs {
1287 let first_item_span = tcx.def_span(first_item_ref);
1288 {
tcx.dcx().struct_span_err(first_item_span,
::alloc::__export::must_use({
::alloc::fmt::format(format_args!("negative impls cannot have any items"))
})).with_code(E0749)
}struct_span_code_err!(
1289 tcx.dcx(),
1290 first_item_span,
1291 E0749,
1292 "negative impls cannot have any items"
1293 )
1294 .emit();
1295 }
1296 return;
1297 }
1298 }
1299
1300 let trait_def = tcx.trait_def(trait_ref.def_id);
1301
1302 let self_is_guaranteed_unsize_self = tcx.impl_self_is_guaranteed_unsized(impl_id);
1303
1304 for &impl_item in impl_item_refs {
1305 let ty_impl_item = tcx.associated_item(impl_item);
1306 let ty_trait_item = match ty_impl_item.expect_trait_impl() {
1307 Ok(trait_item_id) => tcx.associated_item(trait_item_id),
1308 Err(ErrorGuaranteed { .. }) => continue,
1309 };
1310
1311 let res = tcx.ensure_result().compare_impl_item(impl_item.expect_local());
1312 if res.is_ok() {
1313 match ty_impl_item.kind {
1314 ty::AssocKind::Fn { .. } => {
1315 compare_impl_item::refine::check_refining_return_position_impl_trait_in_trait(
1316 tcx,
1317 ty_impl_item,
1318 ty_trait_item,
1319 tcx.impl_trait_ref(ty_impl_item.container_id(tcx))
1320 .instantiate_identity()
1321 .skip_norm_wip(),
1322 );
1323 }
1324 ty::AssocKind::Const { .. } => {}
1325 ty::AssocKind::Type { .. } => {}
1326 }
1327 }
1328
1329 if self_is_guaranteed_unsize_self && tcx.generics_require_sized_self(ty_trait_item.def_id) {
1330 tcx.emit_node_span_lint(
1331 rustc_lint_defs::builtin::DEAD_CODE,
1332 tcx.local_def_id_to_hir_id(ty_impl_item.def_id.expect_local()),
1333 tcx.def_span(ty_impl_item.def_id),
1334 diagnostics::UselessImplItem,
1335 )
1336 }
1337
1338 check_specialization_validity(
1339 tcx,
1340 trait_def,
1341 ty_trait_item,
1342 impl_id.to_def_id(),
1343 impl_item,
1344 );
1345
1346 check_overriding_final_trait_item(tcx, ty_trait_item, ty_impl_item);
1347 }
1348
1349 if let Ok(ancestors) = trait_def.ancestors(tcx, impl_id.to_def_id()) {
1350 let mut missing_items = Vec::new();
1352
1353 let mut must_implement_one_of: Option<&[Ident]> =
1354 trait_def.must_implement_one_of.as_deref();
1355
1356 for &trait_item_id in tcx.associated_item_def_ids(trait_ref.def_id) {
1357 let leaf_def = ancestors.leaf_def(tcx, trait_item_id);
1358
1359 let is_implemented = leaf_def
1360 .as_ref()
1361 .is_some_and(|node_item| node_item.item.defaultness(tcx).has_value());
1362
1363 if !is_implemented
1364 && tcx.defaultness(impl_id).is_final()
1365 && !(self_is_guaranteed_unsize_self && tcx.generics_require_sized_self(trait_item_id))
1367 {
1368 missing_items.push(tcx.associated_item(trait_item_id));
1369 }
1370
1371 let is_implemented_here =
1373 leaf_def.as_ref().is_some_and(|node_item| !node_item.defining_node.is_from_trait());
1374
1375 if !is_implemented_here {
1376 let full_impl_span = tcx.hir_span_with_body(tcx.local_def_id_to_hir_id(impl_id));
1377 match tcx.eval_default_body_stability(trait_item_id, full_impl_span) {
1378 EvalResult::Deny { .. }
1381 if !tcx.features().pin_ergonomics()
1382 && tcx.is_lang_item(trait_ref.def_id, hir::LangItem::Drop)
1383 && tcx.item_name(trait_item_id) == sym::drop =>
1384 {
1385 missing_items.push(tcx.associated_item(trait_item_id));
1386 }
1387 EvalResult::Deny { feature, reason, issue, .. } => default_body_is_unstable(
1388 tcx,
1389 full_impl_span,
1390 trait_item_id,
1391 feature,
1392 reason,
1393 issue,
1394 ),
1395
1396 EvalResult::Allow | EvalResult::Unmarked => {}
1398 }
1399 }
1400
1401 if let Some(required_items) = &must_implement_one_of {
1402 if is_implemented_here {
1403 let trait_item = tcx.associated_item(trait_item_id);
1404 if required_items.contains(&trait_item.ident(tcx)) {
1405 must_implement_one_of = None;
1406 }
1407 }
1408 }
1409
1410 if let Some(leaf_def) = &leaf_def
1411 && !leaf_def.is_final()
1412 && let def_id = leaf_def.item.def_id
1413 && tcx.impl_method_has_trait_impl_trait_tys(def_id)
1414 {
1415 let def_kind = tcx.def_kind(def_id);
1416 let descr = tcx.def_kind_descr(def_kind, def_id);
1417 let (msg, feature) = if tcx.asyncness(def_id).is_async() {
1418 (
1419 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("async {0} in trait cannot be specialized",
descr))
})format!("async {descr} in trait cannot be specialized"),
1420 "async functions in traits",
1421 )
1422 } else {
1423 (
1424 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{0} with return-position `impl Trait` in trait cannot be specialized",
descr))
})format!(
1425 "{descr} with return-position `impl Trait` in trait cannot be specialized"
1426 ),
1427 "return position `impl Trait` in traits",
1428 )
1429 };
1430 tcx.dcx()
1431 .struct_span_err(tcx.def_span(def_id), msg)
1432 .with_note(::alloc::__export::must_use({
::alloc::fmt::format(format_args!("specialization behaves in inconsistent and surprising ways with {0}, and for now is disallowed",
feature))
})format!(
1433 "specialization behaves in inconsistent and surprising ways with \
1434 {feature}, and for now is disallowed"
1435 ))
1436 .emit();
1437 }
1438 }
1439
1440 if !missing_items.is_empty() {
1441 let full_impl_span = tcx.hir_span_with_body(tcx.local_def_id_to_hir_id(impl_id));
1442 missing_items_err(tcx, impl_id, &missing_items, full_impl_span);
1443 }
1444
1445 if let Some(missing_items) = must_implement_one_of {
1446 let attr_span = {
{
'done:
{
for i in
::rustc_hir::attrs::HasAttrs::get_attrs(trait_ref.def_id,
&tcx) {
#[allow(unused_imports)]
use rustc_hir::attrs::AttributeKind::*;
let i: &rustc_hir::Attribute = i;
match i {
rustc_hir::Attribute::Parsed(RustcMustImplementOneOf {
attr_span, .. }) => {
break 'done Some(*attr_span);
}
rustc_hir::Attribute::Unparsed(..) =>
{}
#[deny(unreachable_patterns)]
_ => {}
}
}
None
}
}
}find_attr!(tcx, trait_ref.def_id, RustcMustImplementOneOf {attr_span, ..} => *attr_span);
1447
1448 missing_items_must_implement_one_of_err(
1449 tcx,
1450 tcx.def_span(impl_id),
1451 missing_items,
1452 attr_span,
1453 );
1454 }
1455 }
1456}
1457
1458fn check_simd(tcx: TyCtxt<'_>, sp: Span, def_id: LocalDefId) {
1459 let t = tcx.type_of(def_id).instantiate_identity().skip_norm_wip();
1460 if let ty::Adt(def, args) = t.kind()
1461 && def.is_struct()
1462 {
1463 let fields = &def.non_enum_variant().fields;
1464 if fields.is_empty() {
1465 {
tcx.dcx().struct_span_err(sp,
::alloc::__export::must_use({
::alloc::fmt::format(format_args!("SIMD vector cannot be empty"))
})).with_code(E0075)
}struct_span_code_err!(tcx.dcx(), sp, E0075, "SIMD vector cannot be empty").emit();
1466 return;
1467 }
1468
1469 let array_field = &fields[FieldIdx::ZERO];
1470 let array_ty = array_field.ty(tcx, args).skip_norm_wip();
1471 let ty::Array(element_ty, len_const) = array_ty.kind() else {
1472 {
tcx.dcx().struct_span_err(sp,
::alloc::__export::must_use({
::alloc::fmt::format(format_args!("SIMD vector\'s only field must be an array"))
})).with_code(E0076)
}struct_span_code_err!(
1473 tcx.dcx(),
1474 sp,
1475 E0076,
1476 "SIMD vector's only field must be an array"
1477 )
1478 .with_span_label(tcx.def_span(array_field.did), "not an array")
1479 .emit();
1480 return;
1481 };
1482
1483 if let Some(second_field) = fields.get(FieldIdx::ONE) {
1484 {
tcx.dcx().struct_span_err(sp,
::alloc::__export::must_use({
::alloc::fmt::format(format_args!("SIMD vector cannot have multiple fields"))
})).with_code(E0075)
}struct_span_code_err!(tcx.dcx(), sp, E0075, "SIMD vector cannot have multiple fields")
1485 .with_span_label(tcx.def_span(second_field.did), "excess field")
1486 .emit();
1487 return;
1488 }
1489
1490 if let Some(len) = len_const.try_to_target_usize(tcx) {
1495 if len == 0 {
1496 {
tcx.dcx().struct_span_err(sp,
::alloc::__export::must_use({
::alloc::fmt::format(format_args!("SIMD vector cannot be empty"))
})).with_code(E0075)
}struct_span_code_err!(tcx.dcx(), sp, E0075, "SIMD vector cannot be empty").emit();
1497 return;
1498 } else if len > MAX_SIMD_LANES {
1499 {
tcx.dcx().struct_span_err(sp,
::alloc::__export::must_use({
::alloc::fmt::format(format_args!("SIMD vector cannot have more than {0} elements",
MAX_SIMD_LANES))
})).with_code(E0075)
}struct_span_code_err!(
1500 tcx.dcx(),
1501 sp,
1502 E0075,
1503 "SIMD vector cannot have more than {MAX_SIMD_LANES} elements",
1504 )
1505 .emit();
1506 return;
1507 }
1508 }
1509
1510 match element_ty.kind() {
1515 ty::Param(_) => (), ty::Int(_) | ty::Uint(_) | ty::Float(_) | ty::RawPtr(_, _) => (), _ => {
1518 {
tcx.dcx().struct_span_err(sp,
::alloc::__export::must_use({
::alloc::fmt::format(format_args!("SIMD vector element type should be a primitive scalar (integer/float/pointer) type"))
})).with_code(E0077)
}struct_span_code_err!(
1519 tcx.dcx(),
1520 sp,
1521 E0077,
1522 "SIMD vector element type should be a \
1523 primitive scalar (integer/float/pointer) type"
1524 )
1525 .emit();
1526 return;
1527 }
1528 }
1529 }
1530}
1531
1532#[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_scalable_vector",
"rustc_hir_analysis::check::check", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_analysis/src/check/check.rs"),
::tracing_core::__macro_support::Option::Some(1532u32),
::tracing_core::__macro_support::Option::Some("rustc_hir_analysis::check::check"),
::tracing_core::field::FieldSet::new(&["span", "def_id",
"scalable"],
::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(&def_id)
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(&scalable)
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: () = loop {};
return __tracing_attr_fake_return;
}
{
let ty =
tcx.type_of(def_id).instantiate_identity().skip_norm_wip();
let ty::Adt(def, args) = ty.kind() else { return };
if !def.is_struct() {
tcx.dcx().delayed_bug("`rustc_scalable_vector` applied to non-struct");
return;
}
let fields = &def.non_enum_variant().fields;
match scalable {
ScalableElt::ElementCount(..) if fields.is_empty() => {
let mut err =
tcx.dcx().struct_span_err(span,
"scalable vectors must have a single field");
err.help("scalable vector types' only field must be a primitive scalar type");
err.emit();
return;
}
ScalableElt::ElementCount(..) if fields.len() >= 2 => {
tcx.dcx().struct_span_err(span,
"scalable vectors cannot have multiple fields").emit();
return;
}
ScalableElt::Container if fields.is_empty() => {
let mut err =
tcx.dcx().struct_span_err(span,
"scalable vector tuples must have at least one field");
err.help("tuples of scalable vectors can only contain multiple of the same scalable vector type");
err.emit();
return;
}
ScalableElt::Container if fields.len() > 8 => {
let mut err =
tcx.dcx().struct_span_err(span,
"scalable vector tuples can have at most eight fields");
err.help("tuples of scalable vectors can only contain multiple of the same scalable vector type");
err.emit();
return;
}
_ => {}
}
match scalable {
ScalableElt::ElementCount(..) => {
let element_ty =
&fields[FieldIdx::ZERO].ty(tcx, args).skip_norm_wip();
match element_ty.kind() {
ty::Int(_) | ty::Uint(_) | ty::Float(_) | ty::Bool => (),
_ => {
let mut err =
tcx.dcx().struct_span_err(span,
"element type of a scalable vector must be a primitive scalar");
err.help("only `u*`, `i*`, `f*` and `bool` types are accepted");
err.emit();
}
}
}
ScalableElt::Container => {
let mut prev_field_ty = None;
for field in fields.iter() {
let element_ty = field.ty(tcx, args).skip_norm_wip();
if let ty::Adt(def, _) = element_ty.kind() &&
def.repr().scalable() {
match def.repr().scalable.expect("`repr().scalable.is_some()` != `repr().scalable()`")
{
ScalableElt::ElementCount(_) => {}
ScalableElt::Container => {
tcx.dcx().span_err(tcx.def_span(field.did),
"scalable vector structs cannot contain other scalable vector structs");
break;
}
}
} else {
tcx.dcx().span_err(tcx.def_span(field.did),
"scalable vector structs can only have scalable vector fields");
break;
}
if let Some(prev_ty) = prev_field_ty.replace(element_ty) &&
prev_ty != element_ty {
tcx.dcx().span_err(tcx.def_span(field.did),
"all fields in a scalable vector struct must be the same type");
break;
}
}
}
}
}
}
}#[tracing::instrument(skip(tcx), level = "debug")]
1533fn check_scalable_vector(tcx: TyCtxt<'_>, span: Span, def_id: LocalDefId, scalable: ScalableElt) {
1534 let ty = tcx.type_of(def_id).instantiate_identity().skip_norm_wip();
1535 let ty::Adt(def, args) = ty.kind() else { return };
1536 if !def.is_struct() {
1537 tcx.dcx().delayed_bug("`rustc_scalable_vector` applied to non-struct");
1538 return;
1539 }
1540
1541 let fields = &def.non_enum_variant().fields;
1542 match scalable {
1543 ScalableElt::ElementCount(..) if fields.is_empty() => {
1544 let mut err =
1545 tcx.dcx().struct_span_err(span, "scalable vectors must have a single field");
1546 err.help("scalable vector types' only field must be a primitive scalar type");
1547 err.emit();
1548 return;
1549 }
1550 ScalableElt::ElementCount(..) if fields.len() >= 2 => {
1551 tcx.dcx().struct_span_err(span, "scalable vectors cannot have multiple fields").emit();
1552 return;
1553 }
1554 ScalableElt::Container if fields.is_empty() => {
1555 let mut err = tcx
1556 .dcx()
1557 .struct_span_err(span, "scalable vector tuples must have at least one field");
1558 err.help("tuples of scalable vectors can only contain multiple of the same scalable vector type");
1559 err.emit();
1560 return;
1561 }
1562 ScalableElt::Container if fields.len() > 8 => {
1563 let mut err = tcx
1564 .dcx()
1565 .struct_span_err(span, "scalable vector tuples can have at most eight fields");
1566 err.help("tuples of scalable vectors can only contain multiple of the same scalable vector type");
1567 err.emit();
1568 return;
1569 }
1570 _ => {}
1571 }
1572
1573 match scalable {
1574 ScalableElt::ElementCount(..) => {
1575 let element_ty = &fields[FieldIdx::ZERO].ty(tcx, args).skip_norm_wip();
1576
1577 match element_ty.kind() {
1581 ty::Int(_) | ty::Uint(_) | ty::Float(_) | ty::Bool => (),
1582 _ => {
1583 let mut err = tcx.dcx().struct_span_err(
1584 span,
1585 "element type of a scalable vector must be a primitive scalar",
1586 );
1587 err.help("only `u*`, `i*`, `f*` and `bool` types are accepted");
1588 err.emit();
1589 }
1590 }
1591 }
1592 ScalableElt::Container => {
1593 let mut prev_field_ty = None;
1594 for field in fields.iter() {
1595 let element_ty = field.ty(tcx, args).skip_norm_wip();
1596 if let ty::Adt(def, _) = element_ty.kind()
1597 && def.repr().scalable()
1598 {
1599 match def
1600 .repr()
1601 .scalable
1602 .expect("`repr().scalable.is_some()` != `repr().scalable()`")
1603 {
1604 ScalableElt::ElementCount(_) => { }
1605 ScalableElt::Container => {
1606 tcx.dcx().span_err(
1607 tcx.def_span(field.did),
1608 "scalable vector structs cannot contain other scalable vector structs",
1609 );
1610 break;
1611 }
1612 }
1613 } else {
1614 tcx.dcx().span_err(
1615 tcx.def_span(field.did),
1616 "scalable vector structs can only have scalable vector fields",
1617 );
1618 break;
1619 }
1620
1621 if let Some(prev_ty) = prev_field_ty.replace(element_ty)
1622 && prev_ty != element_ty
1623 {
1624 tcx.dcx().span_err(
1625 tcx.def_span(field.did),
1626 "all fields in a scalable vector struct must be the same type",
1627 );
1628 break;
1629 }
1630 }
1631 }
1632 }
1633}
1634
1635pub(super) fn check_packed(tcx: TyCtxt<'_>, sp: Span, def: ty::AdtDef<'_>) {
1636 let repr = def.repr();
1637 if repr.packed() {
1638 if def.is_pin_project() {
1642 tcx.dcx().emit_err(diagnostics::PinV2OnPacked {
1643 span: sp,
1644 pin_v2_span: {
{
'done:
{
for i in ::rustc_hir::attrs::HasAttrs::get_attrs(def.did(), &tcx)
{
#[allow(unused_imports)]
use rustc_hir::attrs::AttributeKind::*;
let i: &rustc_hir::Attribute = i;
match i {
rustc_hir::Attribute::Parsed(PinV2(span)) => {
break 'done Some(*span);
}
rustc_hir::Attribute::Unparsed(..) =>
{}
#[deny(unreachable_patterns)]
_ => {}
}
}
None
}
}
}find_attr!(tcx, def.did(), PinV2(span) => *span),
1645 adt_name: tcx.item_name(def.did()),
1646 });
1647 }
1648 if let Some(reprs) = {
{
'done:
{
for i in ::rustc_hir::attrs::HasAttrs::get_attrs(def.did(), &tcx)
{
#[allow(unused_imports)]
use rustc_hir::attrs::AttributeKind::*;
let i: &rustc_hir::Attribute = i;
match i {
rustc_hir::Attribute::Parsed(Repr { reprs, .. }) => {
break 'done Some(reprs);
}
rustc_hir::Attribute::Unparsed(..) =>
{}
#[deny(unreachable_patterns)]
_ => {}
}
}
None
}
}
}find_attr!(tcx, def.did(), Repr { reprs, .. } => reprs) {
1649 for (r, _) in reprs {
1650 if let ReprPacked(pack) = r
1651 && let Some(repr_pack) = repr.pack
1652 && pack != &repr_pack
1653 {
1654 {
tcx.dcx().struct_span_err(sp,
::alloc::__export::must_use({
::alloc::fmt::format(format_args!("type has conflicting packed representation hints"))
})).with_code(E0634)
}struct_span_code_err!(
1655 tcx.dcx(),
1656 sp,
1657 E0634,
1658 "type has conflicting packed representation hints"
1659 )
1660 .emit();
1661 }
1662 }
1663 }
1664 if repr.align.is_some() {
1665 {
tcx.dcx().struct_span_err(sp,
::alloc::__export::must_use({
::alloc::fmt::format(format_args!("type has conflicting packed and align representation hints"))
})).with_code(E0587)
}struct_span_code_err!(
1666 tcx.dcx(),
1667 sp,
1668 E0587,
1669 "type has conflicting packed and align representation hints"
1670 )
1671 .emit();
1672 } else if let Some(def_spans) = check_packed_inner(tcx, def.did(), &mut ::alloc::vec::Vec::new()vec![]) {
1673 let mut err = {
tcx.dcx().struct_span_err(sp,
::alloc::__export::must_use({
::alloc::fmt::format(format_args!("packed type cannot transitively contain a `#[repr(align)]` type"))
})).with_code(E0588)
}struct_span_code_err!(
1674 tcx.dcx(),
1675 sp,
1676 E0588,
1677 "packed type cannot transitively contain a `#[repr(align)]` type"
1678 );
1679
1680 err.span_note(
1681 tcx.def_span(def_spans[0].0),
1682 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("`{0}` has a `#[repr(align)]` attribute",
tcx.item_name(def_spans[0].0)))
})format!("`{}` has a `#[repr(align)]` attribute", tcx.item_name(def_spans[0].0)),
1683 );
1684
1685 if def_spans.len() > 2 {
1686 let mut first = true;
1687 for (adt_def, span) in def_spans.iter().skip(1).rev() {
1688 let ident = tcx.item_name(*adt_def);
1689 err.span_note(
1690 *span,
1691 if first {
1692 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("`{0}` contains a field of type `{1}`",
tcx.type_of(def.did()).instantiate_identity().skip_norm_wip(),
ident))
})format!(
1693 "`{}` contains a field of type `{}`",
1694 tcx.type_of(def.did()).instantiate_identity().skip_norm_wip(),
1695 ident
1696 )
1697 } else {
1698 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("...which contains a field of type `{0}`",
ident))
})format!("...which contains a field of type `{ident}`")
1699 },
1700 );
1701 first = false;
1702 }
1703 }
1704
1705 err.emit();
1706 }
1707 }
1708}
1709
1710pub(super) fn check_packed_inner(
1711 tcx: TyCtxt<'_>,
1712 def_id: DefId,
1713 stack: &mut Vec<DefId>,
1714) -> Option<Vec<(DefId, Span)>> {
1715 if let ty::Adt(def, args) = tcx.type_of(def_id).instantiate_identity().skip_norm_wip().kind() {
1716 if def.is_struct() || def.is_union() {
1717 if def.repr().align.is_some() {
1718 return Some(::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[(def.did(), DUMMY_SP)]))vec![(def.did(), DUMMY_SP)]);
1719 }
1720
1721 stack.push(def_id);
1722 for field in &def.non_enum_variant().fields {
1723 if let ty::Adt(def, _) = field.ty(tcx, args).skip_norm_wip().kind()
1724 && !stack.contains(&def.did())
1725 && let Some(mut defs) = check_packed_inner(tcx, def.did(), stack)
1726 {
1727 defs.push((def.did(), field.ident(tcx).span));
1728 return Some(defs);
1729 }
1730 }
1731 stack.pop();
1732 }
1733 }
1734
1735 None
1736}
1737
1738pub(super) fn check_transparent<'tcx>(tcx: TyCtxt<'tcx>, adt: ty::AdtDef<'tcx>) {
1739 if !adt.repr().transparent() {
1740 return;
1741 }
1742
1743 if adt.is_union() && !tcx.features().transparent_unions() {
1744 feature_err(
1745 &tcx.sess,
1746 sym::transparent_unions,
1747 tcx.def_span(adt.did()),
1748 "transparent unions are unstable",
1749 )
1750 .emit();
1751 }
1752
1753 if adt.variants().len() != 1 {
1754 bad_variant_count(tcx, adt, tcx.def_span(adt.did()), adt.did());
1755 return;
1757 }
1758 let variant = adt.variant(VariantIdx::ZERO);
1759
1760 if variant.fields.len() <= 1 {
1761 return;
1763 }
1764
1765 let typing_env = ty::TypingEnv::non_body_analysis(tcx, adt.did());
1766
1767 enum NonTrivialReason<'tcx> {
1771 UnknownLayout,
1772 NonZeroSized,
1773 NonTrivialAlignment,
1774 PrivateField { inside: Ty<'tcx> },
1775 NonExhaustive { ty: Ty<'tcx> },
1776 ReprC { ty: Ty<'tcx> },
1777 }
1778 struct NonTrivialFieldInfo<'tcx> {
1779 span: Span,
1780 reason: NonTrivialReason<'tcx>,
1781 }
1782
1783 fn is_trivial<'tcx>(
1786 tcx: TyCtxt<'tcx>,
1787 typing_env: ty::TypingEnv<'tcx>,
1788 ty: Ty<'tcx>,
1789 ) -> ControlFlow<NonTrivialReason<'tcx>> {
1790 let ty =
1792 tcx.try_normalize_erasing_regions(typing_env, Unnormalized::new_wip(ty)).unwrap_or(ty);
1793 match ty.kind() {
1794 ty::Tuple(list) => list.iter().try_for_each(|t| is_trivial(tcx, typing_env, t)),
1795 ty::Array(ty, _) => is_trivial(tcx, typing_env, *ty),
1796 ty::Adt(def, args) => {
1797 if !def.did().is_local() && !{
{
'done:
{
for i in
::rustc_hir::attrs::HasAttrs::get_attrs(def.did(), &tcx) {
#[allow(unused_imports)]
use rustc_hir::attrs::AttributeKind::*;
let i: &rustc_hir::Attribute = i;
match i {
rustc_hir::Attribute::Parsed(RustcPubTransparent(_)) => {
break 'done Some(());
}
rustc_hir::Attribute::Unparsed(..) =>
{}
#[deny(unreachable_patterns)]
_ => {}
}
}
None
}
}
}.is_some()find_attr!(tcx, def.did(), RustcPubTransparent(_)) {
1798 let non_exhaustive = def.is_variant_list_non_exhaustive()
1799 || def.variants().iter().any(ty::VariantDef::is_field_list_non_exhaustive);
1800 if non_exhaustive {
1801 return ControlFlow::Break(NonTrivialReason::NonExhaustive { ty });
1802 }
1803 let has_priv = def.all_fields().any(|f| !f.vis.is_public());
1804 if has_priv {
1805 return ControlFlow::Break(NonTrivialReason::PrivateField { inside: ty });
1806 }
1807 }
1808 if def.repr().c() {
1809 return ControlFlow::Break(NonTrivialReason::ReprC { ty });
1810 }
1811 def.all_fields()
1812 .map(|field| field.ty(tcx, args).skip_norm_wip())
1813 .try_for_each(|t| is_trivial(tcx, typing_env, t))
1814 }
1815 _ => ControlFlow::Continue(()),
1816 }
1817 }
1818
1819 let non_trivial_fields = variant
1820 .fields
1821 .iter()
1822 .filter_map(|field| {
1823 let ty = field.ty(tcx, GenericArgs::identity_for_item(tcx, field.did)).skip_norm_wip();
1824 let layout = tcx.layout_of(typing_env.as_query_input(ty));
1825 let span = tcx.hir_span_if_local(field.did).unwrap();
1827 if !layout.is_ok_and(|layout| layout.is_1zst()) {
1829 let reason = match layout {
1830 Err(_) => NonTrivialReason::UnknownLayout,
1831 Ok(layout) => {
1832 if !(layout.is_sized() && layout.size.bytes() == 0) {
1833 NonTrivialReason::NonZeroSized
1834 } else {
1835 NonTrivialReason::NonTrivialAlignment
1836 }
1837 }
1838 };
1839 return Some(NonTrivialFieldInfo { span, reason });
1840 }
1841 if let Some(reason) = is_trivial(tcx, typing_env, ty).break_value() {
1843 return Some(NonTrivialFieldInfo { span, reason });
1844 }
1845 None
1847 })
1848 .collect::<Vec<_>>();
1849
1850 if non_trivial_fields.len() > 1 {
1851 let count = non_trivial_fields.len();
1852 let desc = if adt.is_enum() {
1853 format_args!("the variant of a transparent {0}", adt.descr())format_args!("the variant of a transparent {}", adt.descr())
1854 } else {
1855 format_args!("transparent {0}", adt.descr())format_args!("transparent {}", adt.descr())
1856 };
1857 let ty_span = tcx.def_span(adt.did());
1858 let mut diag = tcx.dcx().struct_span_err(
1859 ty_span,
1860 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{0} needs at most one non-trivial field, but has {1}",
desc, count))
})format!("{desc} needs at most one non-trivial field, but has {count}"),
1861 );
1862 diag.code(E0690);
1863
1864 diag.span_label(ty_span, ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("needs at most one non-trivial field, but has {0}",
count))
})format!("needs at most one non-trivial field, but has {count}"));
1866 for field in non_trivial_fields {
1868 let msg = match field.reason {
1869 NonTrivialReason::UnknownLayout => {
1870 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("this field is generic and hence may have non-zero size"))
})format!("this field is generic and hence may have non-zero size")
1871 }
1872 NonTrivialReason::NonZeroSized => ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("this field has non-zero size"))
})format!("this field has non-zero size"),
1873 NonTrivialReason::NonTrivialAlignment => ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("this field requires alignment"))
})format!("this field requires alignment"),
1874 NonTrivialReason::PrivateField { inside } => ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("this field contains `{0}`, which has private fields, so it could become non-zero-sized in the future",
inside))
})format!(
1875 "this field contains `{inside}`, which has private fields, so it could become non-zero-sized in the future"
1876 ),
1877 NonTrivialReason::NonExhaustive { ty } => ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("this field contains `{0}`, which is marked with `#[non_exhaustive]`, so it could become non-zero-sized in the future",
ty))
})format!(
1878 "this field contains `{ty}`, which is marked with `#[non_exhaustive]`, so it could become non-zero-sized in the future"
1879 ),
1880 NonTrivialReason::ReprC { ty } => ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("this field contains `{0}`, which is a `#[repr(C)]` type, so it is not guaranteed to be zero-sized on all targets",
ty))
})format!(
1881 "this field contains `{ty}`, which is a `#[repr(C)]` type, so it is not guaranteed to be zero-sized on all targets"
1882 ),
1883 };
1884 diag.span_label(field.span, msg);
1885 }
1886
1887 diag.emit();
1888 return;
1889 }
1890}
1891
1892#[allow(trivial_numeric_casts)]
1893fn check_enum(tcx: TyCtxt<'_>, def_id: LocalDefId) {
1894 let def = tcx.adt_def(def_id);
1895 def.destructor(tcx); if def.variants().is_empty() {
1898 {
{
'done:
{
for i in ::rustc_hir::attrs::HasAttrs::get_attrs(def_id, &tcx) {
#[allow(unused_imports)]
use rustc_hir::attrs::AttributeKind::*;
let i: &rustc_hir::Attribute = i;
match i {
rustc_hir::Attribute::Parsed(Repr { reprs, first_span }) =>
{
break 'done
Some({
{
tcx.dcx().struct_span_err(reprs.first().map(|repr|
repr.1).unwrap_or(*first_span),
::alloc::__export::must_use({
::alloc::fmt::format(format_args!("unsupported representation for zero-variant enum"))
})).with_code(E0084)
}.with_span_label(tcx.def_span(def_id),
"zero-variant enum").emit();
});
}
rustc_hir::Attribute::Unparsed(..) =>
{}
#[deny(unreachable_patterns)]
_ => {}
}
}
None
}
}
};find_attr!(tcx, def_id, Repr { reprs, first_span } => {
1899 struct_span_code_err!(
1900 tcx.dcx(),
1901 reprs.first().map(|repr| repr.1).unwrap_or(*first_span),
1902 E0084,
1903 "unsupported representation for zero-variant enum"
1904 )
1905 .with_span_label(tcx.def_span(def_id), "zero-variant enum")
1906 .emit();
1907 });
1908 }
1909
1910 for v in def.variants() {
1911 if let ty::VariantDiscr::Explicit(discr_def_id) = v.discr {
1912 tcx.ensure_ok().typeck(discr_def_id.expect_local());
1913 }
1914 }
1915
1916 if def.repr().int.is_none() {
1917 let is_unit = |var: &ty::VariantDef| #[allow(non_exhaustive_omitted_patterns)] match var.ctor_kind() {
Some(CtorKind::Const) => true,
_ => false,
}matches!(var.ctor_kind(), Some(CtorKind::Const));
1918 let get_disr = |var: &ty::VariantDef| match var.discr {
1919 ty::VariantDiscr::Explicit(disr) => Some(disr),
1920 ty::VariantDiscr::Relative(_) => None,
1921 };
1922
1923 let non_unit = def.variants().iter().find(|var| !is_unit(var));
1924 let disr_unit =
1925 def.variants().iter().filter(|var| is_unit(var)).find_map(|var| get_disr(var));
1926 let disr_non_unit =
1927 def.variants().iter().filter(|var| !is_unit(var)).find_map(|var| get_disr(var));
1928
1929 if disr_non_unit.is_some() || (disr_unit.is_some() && non_unit.is_some()) {
1930 let mut err = {
tcx.dcx().struct_span_err(tcx.def_span(def_id),
::alloc::__export::must_use({
::alloc::fmt::format(format_args!("`#[repr(inttype)]` must be specified for enums with explicit discriminants and non-unit variants"))
})).with_code(E0732)
}struct_span_code_err!(
1931 tcx.dcx(),
1932 tcx.def_span(def_id),
1933 E0732,
1934 "`#[repr(inttype)]` must be specified for enums with explicit discriminants and non-unit variants"
1935 );
1936 if let Some(disr_non_unit) = disr_non_unit {
1937 err.span_label(
1938 tcx.def_span(disr_non_unit),
1939 "explicit discriminant on non-unit variant specified here",
1940 );
1941 } else {
1942 err.span_label(
1943 tcx.def_span(disr_unit.unwrap()),
1944 "explicit discriminant specified here",
1945 );
1946 err.span_label(
1947 tcx.def_span(non_unit.unwrap().def_id),
1948 "non-unit discriminant declared here",
1949 );
1950 }
1951 err.emit();
1952 }
1953 }
1954
1955 detect_discriminant_duplicate(tcx, def);
1956 check_transparent(tcx, def);
1957}
1958
1959fn detect_discriminant_duplicate<'tcx>(tcx: TyCtxt<'tcx>, adt: ty::AdtDef<'tcx>) {
1961 let report = |dis: Discr<'tcx>, idx, err: &mut Diag<'_>| {
1964 let var = adt.variant(idx); let (span, display_discr) = match var.discr {
1966 ty::VariantDiscr::Explicit(discr_def_id) => {
1967 if let hir::Node::AnonConst(expr) =
1969 tcx.hir_node_by_def_id(discr_def_id.expect_local())
1970 && let hir::ExprKind::Lit(lit) = &tcx.hir_body(expr.body).value.kind
1971 && let rustc_ast::LitKind::Int(lit_value, _int_kind) = &lit.node
1972 && *lit_value != dis.val
1973 {
1974 (tcx.def_span(discr_def_id), ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("`{0}` (overflowed from `{1}`)",
dis, lit_value))
})format!("`{dis}` (overflowed from `{lit_value}`)"))
1975 } else {
1976 (tcx.def_span(discr_def_id), ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("`{0}`", dis))
})format!("`{dis}`"))
1978 }
1979 }
1980 ty::VariantDiscr::Relative(0) => (tcx.def_span(var.def_id), ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("`{0}`", dis))
})format!("`{dis}`")),
1982 ty::VariantDiscr::Relative(distance_to_explicit) => {
1983 if let Some(explicit_idx) =
1988 idx.as_u32().checked_sub(distance_to_explicit).map(VariantIdx::from_u32)
1989 {
1990 let explicit_variant = adt.variant(explicit_idx);
1991 let ve_ident = var.name;
1992 let ex_ident = explicit_variant.name;
1993 let sp = if distance_to_explicit > 1 { "variants" } else { "variant" };
1994
1995 err.span_label(
1996 tcx.def_span(explicit_variant.def_id),
1997 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("discriminant for `{0}` incremented from this startpoint (`{1}` + {2} {3} later => `{0}` = {4})",
ve_ident, ex_ident, distance_to_explicit, sp, dis))
})format!(
1998 "discriminant for `{ve_ident}` incremented from this startpoint \
1999 (`{ex_ident}` + {distance_to_explicit} {sp} later \
2000 => `{ve_ident}` = {dis})"
2001 ),
2002 );
2003 }
2004
2005 (tcx.def_span(var.def_id), ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("`{0}`", dis))
})format!("`{dis}`"))
2006 }
2007 };
2008
2009 err.span_label(span, ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{0} assigned here", display_discr))
})format!("{display_discr} assigned here"));
2010 };
2011
2012 let mut discrs = adt.discriminants(tcx).collect::<Vec<_>>();
2013
2014 let mut i = 0;
2021 while i < discrs.len() {
2022 let var_i_idx = discrs[i].0;
2023 let mut error: Option<Diag<'_, _>> = None;
2024
2025 let mut o = i + 1;
2026 while o < discrs.len() {
2027 let var_o_idx = discrs[o].0;
2028
2029 if discrs[i].1.val == discrs[o].1.val {
2030 let err = error.get_or_insert_with(|| {
2031 let mut ret = {
tcx.dcx().struct_span_err(tcx.def_span(adt.did()),
::alloc::__export::must_use({
::alloc::fmt::format(format_args!("discriminant value `{0}` assigned more than once",
discrs[i].1))
})).with_code(E0081)
}struct_span_code_err!(
2032 tcx.dcx(),
2033 tcx.def_span(adt.did()),
2034 E0081,
2035 "discriminant value `{}` assigned more than once",
2036 discrs[i].1,
2037 );
2038
2039 report(discrs[i].1, var_i_idx, &mut ret);
2040
2041 ret
2042 });
2043
2044 report(discrs[o].1, var_o_idx, err);
2045
2046 discrs[o] = *discrs.last().unwrap();
2048 discrs.pop();
2049 } else {
2050 o += 1;
2051 }
2052 }
2053
2054 if let Some(e) = error {
2055 e.emit();
2056 }
2057
2058 i += 1;
2059 }
2060}
2061
2062fn check_type_alias_type_params_are_used<'tcx>(tcx: TyCtxt<'tcx>, def_id: LocalDefId) {
2063 let generics = tcx.generics_of(def_id);
2064 if generics.own_counts().types == 0 {
2065 return;
2066 }
2067
2068 let ty = tcx.type_of(def_id).instantiate_identity().skip_norm_wip();
2069 if ty.references_error() {
2070 return;
2072 }
2073
2074 let bounded_params = LazyCell::new(|| {
2076 tcx.explicit_predicates_of(def_id)
2077 .predicates
2078 .iter()
2079 .filter_map(|(predicate, span)| {
2080 let bounded_ty = match predicate.kind().skip_binder() {
2081 ty::ClauseKind::Trait(pred) => pred.trait_ref.self_ty(),
2082 ty::ClauseKind::TypeOutlives(pred) => pred.0,
2083 _ => return None,
2084 };
2085 if let ty::Param(param) = bounded_ty.kind() {
2086 Some((param.index, span))
2087 } else {
2088 None
2089 }
2090 })
2091 .collect::<FxIndexMap<_, _>>()
2097 });
2098
2099 let mut params_used = DenseBitSet::new_empty(generics.own_params.len());
2100 for leaf in ty.walk() {
2101 if let GenericArgKind::Type(leaf_ty) = leaf.kind()
2102 && let ty::Param(param) = leaf_ty.kind()
2103 {
2104 {
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_analysis/src/check/check.rs:2104",
"rustc_hir_analysis::check::check", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_analysis/src/check/check.rs"),
::tracing_core::__macro_support::Option::Some(2104u32),
::tracing_core::__macro_support::Option::Some("rustc_hir_analysis::check::check"),
::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!("found use of ty param {0:?}",
param) as &dyn Value))])
});
} else { ; }
};debug!("found use of ty param {:?}", param);
2105 params_used.insert(param.index);
2106 }
2107 }
2108
2109 for param in &generics.own_params {
2110 if !params_used.contains(param.index)
2111 && let ty::GenericParamDefKind::Type { .. } = param.kind
2112 {
2113 let span = tcx.def_span(param.def_id);
2114 let param_name = Ident::new(param.name, span);
2115
2116 let has_explicit_bounds = bounded_params.is_empty()
2120 || (*bounded_params).get(¶m.index).is_some_and(|&&pred_sp| {
2121 !pred_sp.is_desugaring(DesugaringKind::DefaultBound { def: param.def_id })
2122 });
2123 let const_param_help = !has_explicit_bounds;
2124
2125 let mut diag = tcx.dcx().create_err(diagnostics::UnusedGenericParameter {
2126 span,
2127 param_name,
2128 param_def_kind: tcx.def_descr(param.def_id),
2129 help: diagnostics::UnusedGenericParameterHelp::TyAlias { param_name },
2130 usage_spans: ::alloc::vec::Vec::new()vec![],
2131 const_param_help,
2132 });
2133 diag.code(E0091);
2134 diag.emit();
2135 }
2136 }
2137}
2138
2139fn opaque_type_cycle_error(tcx: TyCtxt<'_>, opaque_def_id: LocalDefId) -> ErrorGuaranteed {
2148 let span = tcx.def_span(opaque_def_id);
2149 let mut err = {
tcx.dcx().struct_span_err(span,
::alloc::__export::must_use({
::alloc::fmt::format(format_args!("cannot resolve opaque type"))
})).with_code(E0720)
}struct_span_code_err!(tcx.dcx(), span, E0720, "cannot resolve opaque type");
2150
2151 let mut label = false;
2152 if let Some((def_id, visitor)) = get_owner_return_paths(tcx, opaque_def_id) {
2153 let typeck_results = tcx.typeck(def_id);
2154 if visitor
2155 .returns
2156 .iter()
2157 .filter_map(|expr| typeck_results.node_type_opt(expr.hir_id))
2158 .all(|ty| #[allow(non_exhaustive_omitted_patterns)] match ty.kind() {
ty::Never => true,
_ => false,
}matches!(ty.kind(), ty::Never))
2159 {
2160 let spans = visitor
2161 .returns
2162 .iter()
2163 .filter(|expr| typeck_results.node_type_opt(expr.hir_id).is_some())
2164 .map(|expr| expr.span)
2165 .collect::<Vec<Span>>();
2166 let span_len = spans.len();
2167 if span_len == 1 {
2168 err.span_label(spans[0], "this returned value is of `!` type");
2169 } else {
2170 let mut multispan: MultiSpan = spans.clone().into();
2171 for span in spans {
2172 multispan.push_span_label(span, "this returned value is of `!` type");
2173 }
2174 err.span_note(multispan, "these returned values have a concrete \"never\" type");
2175 }
2176 err.help("this error will resolve once the item's body returns a concrete type");
2177 } else {
2178 let mut seen = FxHashSet::default();
2179 seen.insert(span);
2180 err.span_label(span, "recursive opaque type");
2181 label = true;
2182 for (sp, ty) in visitor
2183 .returns
2184 .iter()
2185 .filter_map(|e| typeck_results.node_type_opt(e.hir_id).map(|t| (e.span, t)))
2186 .filter(|(_, ty)| !#[allow(non_exhaustive_omitted_patterns)] match ty.kind() {
ty::Never => true,
_ => false,
}matches!(ty.kind(), ty::Never))
2187 {
2188 #[derive(#[automatically_derived]
impl ::core::default::Default for OpaqueTypeCollector {
#[inline]
fn default() -> OpaqueTypeCollector {
OpaqueTypeCollector {
opaques: ::core::default::Default::default(),
closures: ::core::default::Default::default(),
}
}
}Default)]
2189 struct OpaqueTypeCollector {
2190 opaques: Vec<DefId>,
2191 closures: Vec<DefId>,
2192 }
2193 impl<'tcx> ty::TypeVisitor<TyCtxt<'tcx>> for OpaqueTypeCollector {
2194 fn visit_ty(&mut self, t: Ty<'tcx>) {
2195 match *t.kind() {
2196 ty::Alias(ty::AliasTy { kind: ty::Opaque { def_id: def }, .. }) => {
2197 self.opaques.push(def);
2198 }
2199 ty::Closure(def_id, ..) | ty::Coroutine(def_id, ..) => {
2200 self.closures.push(def_id);
2201 t.super_visit_with(self);
2202 }
2203 _ => t.super_visit_with(self),
2204 }
2205 }
2206 }
2207
2208 let mut visitor = OpaqueTypeCollector::default();
2209 ty.visit_with(&mut visitor);
2210 for def_id in visitor.opaques {
2211 let ty_span = tcx.def_span(def_id);
2212 if !seen.contains(&ty_span) {
2213 let descr = if ty.is_opaque() { "opaque " } else { "" };
2214 err.span_label(ty_span, ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("returning this {0}type `{1}`",
descr, ty))
})format!("returning this {descr}type `{ty}`"));
2215 seen.insert(ty_span);
2216 }
2217 err.span_label(sp, ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("returning here with type `{0}`",
ty))
})format!("returning here with type `{ty}`"));
2218 }
2219
2220 for closure_def_id in visitor.closures {
2221 let Some(closure_local_did) = closure_def_id.as_local() else {
2222 continue;
2223 };
2224 let typeck_results = tcx.typeck(closure_local_did);
2225
2226 let mut label_match = |ty: Ty<'_>, span| {
2227 for arg in ty.walk() {
2228 if let ty::GenericArgKind::Type(ty) = arg.kind()
2229 && let ty::Alias(ty::AliasTy {
2230 kind: ty::Opaque { def_id: captured_def_id },
2231 ..
2232 }) = *ty.kind()
2233 && captured_def_id == opaque_def_id.to_def_id()
2234 {
2235 err.span_label(
2236 span,
2237 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{0} captures itself here",
tcx.def_descr(closure_def_id)))
})format!(
2238 "{} captures itself here",
2239 tcx.def_descr(closure_def_id)
2240 ),
2241 );
2242 }
2243 }
2244 };
2245
2246 for capture in typeck_results.closure_min_captures_flattened(closure_local_did)
2248 {
2249 label_match(capture.place.ty(), capture.get_path_span(tcx));
2250 }
2251 if tcx.is_coroutine(closure_def_id)
2253 && let Some(coroutine_layout) = tcx.mir_coroutine_witnesses(closure_def_id)
2254 {
2255 for interior_ty in &coroutine_layout.field_tys {
2256 label_match(interior_ty.ty, interior_ty.source_info.span);
2257 }
2258 }
2259 }
2260 }
2261 }
2262 }
2263 if !label {
2264 err.span_label(span, "cannot resolve opaque type");
2265 }
2266 err.emit()
2267}
2268
2269pub(super) fn check_coroutine_obligations(
2270 tcx: TyCtxt<'_>,
2271 def_id: LocalDefId,
2272) -> Result<(), ErrorGuaranteed> {
2273 if true {
if !!tcx.is_typeck_child(def_id.to_def_id()) {
::core::panicking::panic("assertion failed: !tcx.is_typeck_child(def_id.to_def_id())")
};
};debug_assert!(!tcx.is_typeck_child(def_id.to_def_id()));
2274
2275 let typeck_results = tcx.typeck(def_id);
2276 let param_env = tcx.param_env(def_id);
2277
2278 {
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_analysis/src/check/check.rs:2278",
"rustc_hir_analysis::check::check", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_analysis/src/check/check.rs"),
::tracing_core::__macro_support::Option::Some(2278u32),
::tracing_core::__macro_support::Option::Some("rustc_hir_analysis::check::check"),
::tracing_core::field::FieldSet::new(&["typeck_results.coroutine_stalled_predicates"],
::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(&typeck_results.coroutine_stalled_predicates)
as &dyn Value))])
});
} else { ; }
};debug!(?typeck_results.coroutine_stalled_predicates);
2279
2280 let mode = if tcx.next_trait_solver_globally() {
2281 TypingMode::borrowck(tcx, def_id)
2285 } else {
2286 TypingMode::analysis_in_body(tcx, def_id)
2287 };
2288
2289 let infcx = tcx.infer_ctxt().ignoring_regions().build(mode);
2294
2295 let ocx = ObligationCtxt::new_with_diagnostics(&infcx);
2296 for (predicate, cause) in &typeck_results.coroutine_stalled_predicates {
2297 ocx.register_obligation(Obligation::new(tcx, cause.clone(), param_env, *predicate));
2298 }
2299
2300 let errors = ocx.evaluate_obligations_error_on_ambiguity();
2301 {
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_analysis/src/check/check.rs:2301",
"rustc_hir_analysis::check::check", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_analysis/src/check/check.rs"),
::tracing_core::__macro_support::Option::Some(2301u32),
::tracing_core::__macro_support::Option::Some("rustc_hir_analysis::check::check"),
::tracing_core::field::FieldSet::new(&["errors"],
::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(&errors) as
&dyn Value))])
});
} else { ; }
};debug!(?errors);
2302 if !errors.is_empty() {
2303 return Err(infcx.err_ctxt().report_fulfillment_errors(errors));
2304 }
2305
2306 if !tcx.next_trait_solver_globally() {
2307 for (key, ty) in infcx.take_opaque_types() {
2310 let hidden_type = infcx.resolve_vars_if_possible(ty);
2311 let key = infcx.resolve_vars_if_possible(key);
2312 sanity_check_found_hidden_type(tcx, key, hidden_type)?;
2313 }
2314 } else {
2315 let _ = infcx.take_opaque_types();
2318 }
2319
2320 Ok(())
2321}
2322
2323pub(super) fn check_potentially_region_dependent_goals<'tcx>(
2324 tcx: TyCtxt<'tcx>,
2325 def_id: LocalDefId,
2326) -> Result<(), ErrorGuaranteed> {
2327 if !tcx.next_trait_solver_globally() {
2328 return Ok(());
2329 }
2330 let typeck_results = tcx.typeck(def_id);
2331 let param_env = tcx.param_env(def_id);
2332
2333 let typing_mode = TypingMode::borrowck(tcx, def_id);
2335 let infcx = tcx.infer_ctxt().ignoring_regions().build(typing_mode);
2336 let ocx = ObligationCtxt::new_with_diagnostics(&infcx);
2337 for (predicate, cause) in &typeck_results.potentially_region_dependent_goals {
2338 let predicate = fold_regions(tcx, *predicate, |_, _| {
2339 infcx.next_region_var(RegionVariableOrigin::Misc(cause.span))
2340 });
2341 ocx.register_obligation(Obligation::new(tcx, cause.clone(), param_env, predicate));
2342 }
2343
2344 let errors = ocx.evaluate_obligations_error_on_ambiguity();
2345 {
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_analysis/src/check/check.rs:2345",
"rustc_hir_analysis::check::check", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_analysis/src/check/check.rs"),
::tracing_core::__macro_support::Option::Some(2345u32),
::tracing_core::__macro_support::Option::Some("rustc_hir_analysis::check::check"),
::tracing_core::field::FieldSet::new(&["errors"],
::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(&errors) as
&dyn Value))])
});
} else { ; }
};debug!(?errors);
2346 if errors.is_empty() { Ok(()) } else { Err(infcx.err_ctxt().report_fulfillment_errors(errors)) }
2347}