1use std::cell::OnceCell;
2use std::ops::ControlFlow;
34use rustc_data_structures::fx::FxHashSet;
5use rustc_data_structures::graph;
6use rustc_data_structures::graph::vec_graph::VecGraph;
7use rustc_data_structures::unord::{UnordMap, UnordSet};
8use rustc_hir::attrs::DivergingFallbackBehavior;
9use rustc_hir::def::{DefKind, Res};
10use rustc_hir::def_id::DefId;
11use rustc_hir::intravisit::{InferKind, Visitor};
12use rustc_hir::{selfas hir, CRATE_HIR_ID, HirId};
13use rustc_lint::builtin::FLOAT_LITERAL_F32_FALLBACK;
14use rustc_middle::ty::{self, FloatVid, Ty, TyCtxt, TypeSuperVisitable, TypeVisitable};
15use rustc_session::lint;
16use rustc_span::def_id::LocalDefId;
17use rustc_span::{DUMMY_SP, Span};
18use rustc_trait_selection::traits::{ObligationCause, ObligationCtxt};
19use tracing::debug;
2021use crate::{FnCtxt, diagnostics};
2223impl<'tcx> FnCtxt<'_, 'tcx> {
24/// Performs type inference fallback, setting [`FnCtxt::diverging_fallback_has_occurred`]
25 /// if the never type fallback has occurred.
26pub(super) fn type_inference_fallback(&self) {
27{
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_hir_typeck/src/fallback.rs:27",
"rustc_hir_typeck::fallback", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/fallback.rs"),
::tracing_core::__macro_support::Option::Some(27u32),
::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::fallback"),
::tracing_core::field::FieldSet::new(&["message"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::EVENT)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let enabled =
::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
&&
::tracing::Level::DEBUG <=
::tracing::level_filters::LevelFilter::current() &&
{
let interest = __CALLSITE.interest();
!interest.is_never() &&
::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
interest)
};
if enabled {
(|value_set: ::tracing::field::ValueSet|
{
let meta = __CALLSITE.metadata();
::tracing::Event::dispatch(meta, &value_set);
;
})({
#[allow(unused_imports)]
use ::tracing::field::{debug, display, Value};
let mut iter = __CALLSITE.metadata().fields().iter();
__CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&format_args!("type-inference-fallback start obligations: {0:#?}",
self.fulfillment_cx.borrow_mut().pending_obligations()) as
&dyn Value))])
});
} else { ; }
};debug!(
28"type-inference-fallback start obligations: {:#?}",
29self.fulfillment_cx.borrow_mut().pending_obligations()
30 );
3132// All type checking constraints were added, try to fallback unsolved variables.
33self.select_obligations_where_possible(|_| {});
3435{
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_hir_typeck/src/fallback.rs:35",
"rustc_hir_typeck::fallback", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/fallback.rs"),
::tracing_core::__macro_support::Option::Some(35u32),
::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::fallback"),
::tracing_core::field::FieldSet::new(&["message"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::EVENT)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let enabled =
::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
&&
::tracing::Level::DEBUG <=
::tracing::level_filters::LevelFilter::current() &&
{
let interest = __CALLSITE.interest();
!interest.is_never() &&
::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
interest)
};
if enabled {
(|value_set: ::tracing::field::ValueSet|
{
let meta = __CALLSITE.metadata();
::tracing::Event::dispatch(meta, &value_set);
;
})({
#[allow(unused_imports)]
use ::tracing::field::{debug, display, Value};
let mut iter = __CALLSITE.metadata().fields().iter();
__CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&format_args!("type-inference-fallback post selection obligations: {0:#?}",
self.fulfillment_cx.borrow_mut().pending_obligations()) as
&dyn Value))])
});
} else { ; }
};debug!(
36"type-inference-fallback post selection obligations: {:#?}",
37self.fulfillment_cx.borrow_mut().pending_obligations()
38 );
3940let fallback_occurred = self.fallback_types();
4142if fallback_occurred {
43// if fallback occurred, previously stalled goals may make progress again
44self.select_obligations_where_possible(|_| {});
45 }
46 }
4748fn fallback_types(&self) -> bool {
49// Check if we have any unresolved variables. If not, no need for fallback.
50let unresolved_variables = self.unresolved_variables();
5152if unresolved_variables.is_empty() {
53return false;
54 }
5556let (diverging_fallback, diverging_fallback_ty) =
57self.calculate_diverging_fallback(&unresolved_variables);
58let fallback_to_f32 = self.calculate_fallback_to_f32(&unresolved_variables);
5960// We do fallback in two passes, to try to generate
61 // better error messages.
62 // The first time, we do *not* replace opaque types.
63let mut fallback_occurred = false;
64for ty in unresolved_variables {
65{
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_hir_typeck/src/fallback.rs:65",
"rustc_hir_typeck::fallback", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/fallback.rs"),
::tracing_core::__macro_support::Option::Some(65u32),
::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::fallback"),
::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!("unsolved_variable = {0:?}",
ty) as &dyn Value))])
});
} else { ; }
};debug!("unsolved_variable = {:?}", ty);
66 fallback_occurred |= self.fallback_if_possible(
67 ty,
68&diverging_fallback,
69 diverging_fallback_ty,
70&fallback_to_f32,
71 );
72 }
7374fallback_occurred75 }
7677/// Tries to apply a fallback to `ty` if it is an unsolved variable.
78 ///
79 /// - Unconstrained ints are replaced with `i32`.
80 ///
81 /// - Unconstrained floats are replaced with `f64`, except when there is a trait predicate
82 /// `f32: From<{float}>`, in which case `f32` is used as the fallback instead.
83 ///
84 /// - Non-numerics may get replaced with `()` or `!`, depending on how they
85 /// were categorized by [`Self::calculate_diverging_fallback`], crate's
86 /// edition, and the setting of `#![rustc_never_type_options(fallback = ...)]`.
87 ///
88 /// Fallback becomes very dubious if we have encountered
89 /// type-checking errors. In that case, fallback to Error.
90 ///
91 /// Sets [`FnCtxt::diverging_fallback_has_occurred`] if never type fallback
92 /// is performed during this call.
93fn fallback_if_possible(
94&self,
95 ty: Ty<'tcx>,
96 diverging_fallback: &UnordSet<Ty<'tcx>>,
97 diverging_fallback_ty: Ty<'tcx>,
98 fallback_to_f32: &UnordSet<FloatVid>,
99 ) -> bool {
100// Careful: we do NOT shallow-resolve `ty`. We know that `ty`
101 // is an unsolved variable, and we determine its fallback
102 // based solely on how it was created, not what other type
103 // variables it may have been unified with since then.
104 //
105 // The reason this matters is that other attempts at fallback
106 // may (in principle) conflict with this fallback, and we wish
107 // to generate a type error in that case. (However, this
108 // actually isn't true right now, because we're only using the
109 // builtin fallback rules. This would be true if we were using
110 // user-supplied fallbacks. But it's still useful to write the
111 // code to detect bugs.)
112 //
113 // (Note though that if we have a general type variable `?T`
114 // that is then unified with an integer type variable `?I`
115 // that ultimately never gets resolved to a special integral
116 // type, `?T` is not considered unsolved, but `?I` is. The
117 // same is true for float variables.)
118let fallback = match ty.kind() {
119_ if let Some(e) = self.tainted_by_errors() => Ty::new_error(self.tcx, e),
120 ty::Infer(ty::IntVar(_)) => self.tcx.types.i32,
121 ty::Infer(ty::FloatVar(vid)) if fallback_to_f32.contains(vid) => self.tcx.types.f32,
122 ty::Infer(ty::FloatVar(_)) => self.tcx.types.f64,
123_ if diverging_fallback.contains(&ty) => {
124self.diverging_fallback_has_occurred.set(true);
125diverging_fallback_ty126 }
127_ => return false,
128 };
129{
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_hir_typeck/src/fallback.rs:129",
"rustc_hir_typeck::fallback", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/fallback.rs"),
::tracing_core::__macro_support::Option::Some(129u32),
::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::fallback"),
::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!("fallback_if_possible(ty={0:?}): defaulting to `{1:?}`",
ty, fallback) as &dyn Value))])
});
} else { ; }
};debug!("fallback_if_possible(ty={:?}): defaulting to `{:?}`", ty, fallback);
130131let span = ty.ty_vid().map_or(DUMMY_SP, |vid| self.infcx.type_var_origin(vid).span);
132self.demand_eqtype(span, ty, fallback);
133true
134}
135136/// Existing code relies on `f32: From<T>` (usually written as `T: Into<f32>`) resolving `T` to
137 /// `f32` when the type of `T` is inferred from an unsuffixed float literal. Using the default
138 /// fallback of `f64`, this would break when adding `impl From<f16> for f32`, as there are now
139 /// two float type which could be `T`, meaning that the fallback of `f64` would be used and
140 /// compilation error would occur as `f32` does not implement `From<f64>`. To avoid breaking
141 /// existing code, we instead fallback `T` to `f32` when there is a trait predicate
142 /// `f32: From<T>`. This means code like the following will continue to compile:
143 ///
144 /// ```rust
145 /// fn foo<T: Into<f32>>(_: T) {}
146 ///
147 /// foo(1.0);
148 /// ```
149fn calculate_fallback_to_f32(&self, unresolved_variables: &[Ty<'tcx>]) -> UnordSet<FloatVid> {
150// Short-circuit: if no unresolved variable is a float, no f32 fallback can apply,
151 // so we can skip the (potentially very expensive) work in `from_float_for_f32_root_vids`.
152 // Under the new solver, that function walks `visit_proof_tree` for every pending
153 // obligation, which is O(N × proof_tree_size) and can dominate type-checking on crates
154 // with many large pending obligations and no f32 involvement.
155if unresolved_variables.iter().all(|ty| ty.float_vid().is_none()) {
156return UnordSet::new();
157 }
158let roots: UnordSet<ty::FloatVid> = self.from_float_for_f32_root_vids();
159if roots.is_empty() {
160// Most functions have no `f32: From<{float}>` predicates, so short-circuit and return
161 // an empty set when this is the case.
162return UnordSet::new();
163 }
164// Calculate all the unresolved variables that need to fallback to `f32` here. This ensures
165 // we don't need to find root variables in `fallback_if_possible`: see the comment at the
166 // top of that function for details.
167let fallback_to_f32 = unresolved_variables168 .iter()
169 .flat_map(|ty| ty.float_vid())
170 .filter(|vid| roots.contains(&self.root_float_var(*vid)))
171 .inspect(|vid| {
172let origin = self.float_var_origin(*vid);
173// Show the entire literal in the suggestion to make it clearer.
174let mut literal = self.tcx.sess.source_map().span_to_snippet(origin.span).ok();
175// A `.` at the end of the literal is no longer necessary if `f32` is explicitly specified
176if let Some(ref mut literal) = literal177 && literal.ends_with('.')
178 {
179literal.pop();
180 }
181self.tcx.emit_node_span_lint(
182FLOAT_LITERAL_F32_FALLBACK,
183origin.lint_id.unwrap_or(CRATE_HIR_ID),
184origin.span,
185 diagnostics::FloatLiteralF32Fallback {
186 span: literal.as_ref().map(|_| origin.span),
187 literal: literal.unwrap_or_default(),
188 },
189 );
190 })
191 .collect();
192{
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_hir_typeck/src/fallback.rs:192",
"rustc_hir_typeck::fallback", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/fallback.rs"),
::tracing_core::__macro_support::Option::Some(192u32),
::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::fallback"),
::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!("calculate_fallback_to_f32: fallback_to_f32={0:?}",
fallback_to_f32) as &dyn Value))])
});
} else { ; }
};debug!("calculate_fallback_to_f32: fallback_to_f32={:?}", fallback_to_f32);
193fallback_to_f32194 }
195196fn calculate_diverging_fallback(
197&self,
198 unresolved_variables: &[Ty<'tcx>],
199 ) -> (UnordSet<Ty<'tcx>>, Ty<'tcx>) {
200{
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_hir_typeck/src/fallback.rs:200",
"rustc_hir_typeck::fallback", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/fallback.rs"),
::tracing_core::__macro_support::Option::Some(200u32),
::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::fallback"),
::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!("calculate_diverging_fallback({0:?})",
unresolved_variables) as &dyn Value))])
});
} else { ; }
};debug!("calculate_diverging_fallback({:?})", unresolved_variables);
201202let diverging_fallback_ty = match self.diverging_fallback_behavior {
203 DivergingFallbackBehavior::ToUnit => self.tcx.types.unit,
204 DivergingFallbackBehavior::ToNever => self.tcx.types.never,
205 DivergingFallbackBehavior::NoFallback => {
206// the type doesn't matter, since no fallback will occur
207return (UnordSet::new(), self.tcx.types.unit);
208 }
209 };
210211// Construct a coercion graph where an edge `A -> B` indicates
212 // a type variable is that is coerced
213let coercion_graph = self.create_coercion_graph();
214215// Extract the unsolved type inference variable vids; note that some
216 // unsolved variables are integer/float variables and are excluded.
217let unsolved_vids = unresolved_variables.iter().filter_map(|ty| ty.ty_vid());
218219// Compute the diverging root vids D -- that is, the root vid of
220 // those type variables that (a) are the target of a coercion from
221 // a `!` type and (b) have not yet been solved.
222 //
223 // These variables are the ones that are targets for fallback to
224 // either `!` or `()`.
225let diverging_roots: UnordSet<ty::TyVid> = self226 .diverging_type_vars
227 .borrow()
228 .iter()
229 .map(|&ty_id| self.shallow_resolve(Ty::new_var(self.tcx, ty_id)))
230 .filter_map(|ty| ty.ty_vid())
231 .map(|vid| self.root_var(vid))
232 .collect();
233{
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_hir_typeck/src/fallback.rs:233",
"rustc_hir_typeck::fallback", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/fallback.rs"),
::tracing_core::__macro_support::Option::Some(233u32),
::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::fallback"),
::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!("calculate_diverging_fallback: diverging_type_vars={0:?}",
self.diverging_type_vars.borrow()) as &dyn Value))])
});
} else { ; }
};debug!(
234"calculate_diverging_fallback: diverging_type_vars={:?}",
235self.diverging_type_vars.borrow()
236 );
237{
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_hir_typeck/src/fallback.rs:237",
"rustc_hir_typeck::fallback", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/fallback.rs"),
::tracing_core::__macro_support::Option::Some(237u32),
::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::fallback"),
::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!("calculate_diverging_fallback: diverging_roots={0:?}",
diverging_roots) as &dyn Value))])
});
} else { ; }
};debug!("calculate_diverging_fallback: diverging_roots={:?}", diverging_roots);
238239// Find all type variables that are reachable from a diverging
240 // type variable. These will typically default to `!`, unless
241 // we find later that they are *also* reachable from some
242 // other type variable outside this set.
243let mut diverging_vids = ::alloc::vec::Vec::new()vec![];
244for unsolved_vid in unsolved_vids {
245let root_vid = self.root_var(unsolved_vid);
246{
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_hir_typeck/src/fallback.rs:246",
"rustc_hir_typeck::fallback", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/fallback.rs"),
::tracing_core::__macro_support::Option::Some(246u32),
::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::fallback"),
::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!("calculate_diverging_fallback: unsolved_vid={0:?} root_vid={1:?} diverges={2:?}",
unsolved_vid, root_vid, diverging_roots.contains(&root_vid))
as &dyn Value))])
});
} else { ; }
};debug!(
247"calculate_diverging_fallback: unsolved_vid={:?} root_vid={:?} diverges={:?}",
248 unsolved_vid,
249 root_vid,
250 diverging_roots.contains(&root_vid),
251 );
252if diverging_roots.contains(&root_vid) {
253 diverging_vids.push(unsolved_vid);
254255{
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_hir_typeck/src/fallback.rs:255",
"rustc_hir_typeck::fallback", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/fallback.rs"),
::tracing_core::__macro_support::Option::Some(255u32),
::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::fallback"),
::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!("calculate_diverging_fallback: root_vid={0:?} reaches {1:?}",
root_vid,
graph::depth_first_search(&coercion_graph,
root_vid).collect::<Vec<_>>()) as &dyn Value))])
});
} else { ; }
};debug!(
256"calculate_diverging_fallback: root_vid={:?} reaches {:?}",
257 root_vid,
258 graph::depth_first_search(&coercion_graph, root_vid).collect::<Vec<_>>()
259 );
260 }
261 }
262263{
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_hir_typeck/src/fallback.rs:263",
"rustc_hir_typeck::fallback", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/fallback.rs"),
::tracing_core::__macro_support::Option::Some(263u32),
::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::fallback"),
::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!("obligations: {0:#?}",
self.fulfillment_cx.borrow_mut().pending_obligations()) as
&dyn Value))])
});
} else { ; }
};debug!("obligations: {:#?}", self.fulfillment_cx.borrow_mut().pending_obligations());
264265let mut diverging_fallback = UnordSet::with_capacity(diverging_vids.len());
266let unsafe_infer_vars = OnceCell::new();
267268self.lint_obligations_broken_by_never_type_fallback_change(
269&diverging_vids,
270&coercion_graph,
271 );
272273for &diverging_vid in &diverging_vids {
274let diverging_ty = Ty::new_var(self.tcx, diverging_vid);
275let root_vid = self.root_var(diverging_vid);
276277self.lint_never_type_fallback_flowing_into_unsafe_code(
278&unsafe_infer_vars,
279&coercion_graph,
280 root_vid,
281 );
282283 diverging_fallback.insert(diverging_ty);
284 }
285286 (diverging_fallback, diverging_fallback_ty)
287 }
288289fn lint_never_type_fallback_flowing_into_unsafe_code(
290&self,
291 unsafe_infer_vars: &OnceCell<UnordMap<ty::TyVid, (HirId, Span, UnsafeUseReason)>>,
292 coercion_graph: &VecGraph<ty::TyVid, true>,
293 root_vid: ty::TyVid,
294 ) {
295let unsafe_infer_vars = unsafe_infer_vars.get_or_init(|| {
296let unsafe_infer_vars = compute_unsafe_infer_vars(self, self.body_id);
297{
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_hir_typeck/src/fallback.rs:297",
"rustc_hir_typeck::fallback", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/fallback.rs"),
::tracing_core::__macro_support::Option::Some(297u32),
::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::fallback"),
::tracing_core::field::FieldSet::new(&["unsafe_infer_vars"],
::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(&unsafe_infer_vars)
as &dyn Value))])
});
} else { ; }
};debug!(?unsafe_infer_vars);
298unsafe_infer_vars299 });
300301let affected_unsafe_infer_vars =
302 graph::depth_first_search_as_undirected(&coercion_graph, root_vid)
303 .filter_map(|x| unsafe_infer_vars.get(&x).copied())
304 .collect::<Vec<_>>();
305306let sugg = self.try_to_suggest_annotations(&[root_vid], coercion_graph);
307308for (hir_id, span, reason) in affected_unsafe_infer_vars {
309self.tcx.emit_node_span_lint(
310 lint::builtin::NEVER_TYPE_FALLBACK_FLOWING_INTO_UNSAFE,
311 hir_id,
312 span,
313match reason {
314 UnsafeUseReason::Call => {
315 diagnostics::NeverTypeFallbackFlowingIntoUnsafe::Call { sugg: sugg.clone() }
316 }
317 UnsafeUseReason::Method => {
318 diagnostics::NeverTypeFallbackFlowingIntoUnsafe::Method {
319 sugg: sugg.clone(),
320 }
321 }
322 UnsafeUseReason::Path => {
323 diagnostics::NeverTypeFallbackFlowingIntoUnsafe::Path { sugg: sugg.clone() }
324 }
325 UnsafeUseReason::UnionField => {
326 diagnostics::NeverTypeFallbackFlowingIntoUnsafe::UnionField {
327 sugg: sugg.clone(),
328 }
329 }
330 UnsafeUseReason::Deref => {
331 diagnostics::NeverTypeFallbackFlowingIntoUnsafe::Deref {
332 sugg: sugg.clone(),
333 }
334 }
335 },
336 );
337 }
338 }
339340fn lint_obligations_broken_by_never_type_fallback_change(
341&self,
342 diverging_vids: &[ty::TyVid],
343 coercions: &VecGraph<ty::TyVid, true>,
344 ) {
345let DivergingFallbackBehavior::ToUnit = self.diverging_fallback_behavior else { return };
346347// Fallback happens if and only if there are diverging variables
348if diverging_vids.is_empty() {
349return;
350 }
351352// Returns errors which happen if fallback is set to `fallback`
353let remaining_errors_if_fallback_to = |fallback| {
354self.probe(|_| {
355let obligations = self.fulfillment_cx.borrow().pending_obligations();
356let ocx = ObligationCtxt::new_with_diagnostics(&self.infcx);
357ocx.register_obligations(obligations.iter().cloned());
358359for &diverging_vid in diverging_vids {
360let diverging_ty = Ty::new_var(self.tcx, diverging_vid);
361362 ocx.eq(&ObligationCause::dummy(), self.param_env, diverging_ty, fallback)
363 .expect("expected diverging var to be unconstrained");
364 }
365366ocx.try_evaluate_obligations()
367 })
368 };
369370// If we have no errors with `fallback = ()`, but *do* have errors with `fallback = !`,
371 // then this code will be broken by the never type fallback change.
372let unit_errors = remaining_errors_if_fallback_to(self.tcx.types.unit);
373if unit_errors.is_empty()
374 && let mut never_errors = remaining_errors_if_fallback_to(self.tcx.types.never)
375 && let [never_error, ..] = never_errors.as_mut_slice()
376 {
377self.adjust_fulfillment_error_for_expr_obligation(never_error);
378let sugg = self.try_to_suggest_annotations(diverging_vids, coercions);
379self.tcx.emit_node_span_lint(
380 lint::builtin::DEPENDENCY_ON_UNIT_NEVER_TYPE_FALLBACK,
381self.tcx.local_def_id_to_hir_id(self.body_id),
382self.tcx.def_span(self.body_id),
383 diagnostics::DependencyOnUnitNeverTypeFallback {
384 obligation_span: never_error.obligation.cause.span,
385 obligation: never_error.obligation.predicate,
386sugg,
387 },
388 )
389 }
390 }
391392/// Returns a graph whose nodes are (unresolved) inference variables and where
393 /// an edge `?A -> ?B` indicates that the variable `?A` is coerced to `?B`.
394fn create_coercion_graph(&self) -> VecGraph<ty::TyVid, true> {
395let pending_obligations = self.fulfillment_cx.borrow_mut().pending_obligations();
396{
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_hir_typeck/src/fallback.rs:396",
"rustc_hir_typeck::fallback", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/fallback.rs"),
::tracing_core::__macro_support::Option::Some(396u32),
::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::fallback"),
::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!("create_coercion_graph: pending_obligations={0:?}",
pending_obligations) as &dyn Value))])
});
} else { ; }
};debug!("create_coercion_graph: pending_obligations={:?}", pending_obligations);
397let coercion_edges: Vec<(ty::TyVid, ty::TyVid)> = pending_obligations398 .into_iter()
399 .filter_map(|obligation| {
400// The predicates we are looking for look like `Coerce(?A -> ?B)`.
401 // They will have no bound variables.
402obligation.predicate.kind().no_bound_vars()
403 })
404 .filter_map(|atom| {
405// We consider both subtyping and coercion to imply 'flow' from
406 // some position in the code `a` to a different position `b`.
407 // This is then used to determine which variables interact with
408 // live code, and as such must fall back to `()` to preserve
409 // soundness.
410 //
411 // In practice currently the two ways that this happens is
412 // coercion and subtyping.
413let (a, b) = match atom {
414 ty::PredicateKind::Coerce(ty::CoercePredicate { a, b }) => (a, b),
415 ty::PredicateKind::Subtype(ty::SubtypePredicate { a_is_expected: _, a, b }) => {
416 (a, b)
417 }
418_ => return None,
419 };
420421let a_vid = self.root_vid(a)?;
422let b_vid = self.root_vid(b)?;
423Some((a_vid, b_vid))
424 })
425 .collect();
426{
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_hir_typeck/src/fallback.rs:426",
"rustc_hir_typeck::fallback", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/fallback.rs"),
::tracing_core::__macro_support::Option::Some(426u32),
::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::fallback"),
::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!("create_coercion_graph: coercion_edges={0:?}",
coercion_edges) as &dyn Value))])
});
} else { ; }
};debug!("create_coercion_graph: coercion_edges={:?}", coercion_edges);
427let num_ty_vars = self.num_ty_vars();
428429VecGraph::new(num_ty_vars, coercion_edges)
430 }
431432/// If `ty` is an unresolved type variable, returns its root vid.
433fn root_vid(&self, ty: Ty<'tcx>) -> Option<ty::TyVid> {
434Some(self.root_var(self.shallow_resolve(ty).ty_vid()?))
435 }
436437/// If `ty` is an unresolved float type variable, returns its root vid.
438pub(crate) fn root_float_vid(&self, ty: Ty<'tcx>) -> Option<ty::FloatVid> {
439Some(self.root_float_var(self.shallow_resolve(ty).float_vid()?))
440 }
441442/// Given a set of diverging vids and coercions, walk the HIR to gather a
443 /// set of suggestions which can be applied to preserve fallback to unit.
444fn try_to_suggest_annotations(
445&self,
446 diverging_vids: &[ty::TyVid],
447 coercions: &VecGraph<ty::TyVid, true>,
448 ) -> diagnostics::SuggestAnnotations {
449let body =
450self.tcx.hir_maybe_body_owned_by(self.body_id).expect("body id must have an owner");
451// For each diverging var, look through the HIR for a place to give it
452 // a type annotation. We do this per var because we only really need one
453 // suggestion to influence a var to be `()`.
454let suggestions = diverging_vids455 .iter()
456 .copied()
457 .filter_map(|vid| {
458let reachable_vids =
459 graph::depth_first_search_as_undirected(coercions, vid).collect();
460AnnotateUnitFallbackVisitor { reachable_vids, fcx: self }
461 .visit_expr(body.value)
462 .break_value()
463 })
464 .collect();
465 diagnostics::SuggestAnnotations { suggestions }
466 }
467}
468469/// Try to walk the HIR to find a place to insert a useful suggestion
470/// to preserve fallback to `()` in 2024.
471struct AnnotateUnitFallbackVisitor<'a, 'tcx> {
472 reachable_vids: FxHashSet<ty::TyVid>,
473 fcx: &'a FnCtxt<'a, 'tcx>,
474}
475impl<'tcx> AnnotateUnitFallbackVisitor<'_, 'tcx> {
476// For a given path segment, if it's missing a turbofish, try to suggest adding
477 // one so we can constrain an argument to `()`. To keep the suggestion simple,
478 // we want to simply suggest `_` for all the other args. This (for now) only
479 // works when there are only type variables (and region variables, since we can
480 // elide them)...
481fn suggest_for_segment(
482&self,
483 arg_segment: &'tcx hir::PathSegment<'tcx>,
484 def_id: DefId,
485 id: HirId,
486 ) -> ControlFlow<diagnostics::SuggestAnnotation> {
487if arg_segment.args.is_none()
488 && let Some(all_args) = self.fcx.typeck_results.borrow().node_args_opt(id)
489 && let generics = self.fcx.tcx.generics_of(def_id)
490 && let args = all_args[generics.parent_count..].iter().zip(&generics.own_params)
491// We can't turbofish consts :(
492 && args.clone().all(|(_, param)| #[allow(non_exhaustive_omitted_patterns)] match param.kind {
ty::GenericParamDefKind::Type { .. } | ty::GenericParamDefKind::Lifetime
=> true,
_ => false,
}matches!(param.kind, ty::GenericParamDefKind::Type { .. } | ty::GenericParamDefKind::Lifetime))
493 {
494// We filter out APITs, which are not turbofished.
495let non_apit_type_args = args.filter(|(_, param)| {
496#[allow(non_exhaustive_omitted_patterns)] match param.kind {
ty::GenericParamDefKind::Type { synthetic: false, .. } => true,
_ => false,
}matches!(param.kind, ty::GenericParamDefKind::Type { synthetic: false, .. })497 });
498let n_tys = non_apit_type_args.clone().count();
499for (idx, (arg, _)) in non_apit_type_args.enumerate() {
500if let Some(ty) = arg.as_type()
501 && let Some(vid) = self.fcx.root_vid(ty)
502 && self.reachable_vids.contains(&vid)
503 {
504return ControlFlow::Break(diagnostics::SuggestAnnotation::Turbo(
505 arg_segment.ident.span.shrink_to_hi(),
506 n_tys,
507 idx,
508 ));
509 }
510 }
511 }
512 ControlFlow::Continue(())
513 }
514}
515impl<'tcx> Visitor<'tcx> for AnnotateUnitFallbackVisitor<'_, 'tcx> {
516type Result = ControlFlow<diagnostics::SuggestAnnotation>;
517518fn visit_infer(
519&mut self,
520 inf_id: HirId,
521 inf_span: Span,
522 _kind: InferKind<'tcx>,
523 ) -> Self::Result {
524// Try to replace `_` with `()`.
525if let Some(ty) = self.fcx.typeck_results.borrow().node_type_opt(inf_id)
526 && let Some(vid) = self.fcx.root_vid(ty)
527 && self.reachable_vids.contains(&vid)
528 && inf_span.can_be_used_for_suggestions()
529 {
530return ControlFlow::Break(diagnostics::SuggestAnnotation::Unit(inf_span));
531 }
532533 ControlFlow::Continue(())
534 }
535536fn visit_qpath(
537&mut self,
538 qpath: &'tcx rustc_hir::QPath<'tcx>,
539 id: HirId,
540 span: Span,
541 ) -> Self::Result {
542let arg_segment = match qpath {
543 hir::QPath::Resolved(_, path) => {
544path.segments.last().expect("paths should have a segment")
545 }
546 hir::QPath::TypeRelative(_, segment) => segment,
547 };
548// Alternatively, try to turbofish `::<_, (), _>`.
549if let Some(def_id) = self.fcx.typeck_results.borrow().qpath_res(qpath, id).opt_def_id()
550 && span.can_be_used_for_suggestions()
551 {
552self.suggest_for_segment(arg_segment, def_id, id)?;
553 }
554 hir::intravisit::walk_qpath(self, qpath, id)
555 }
556557fn visit_expr(&mut self, expr: &'tcx hir::Expr<'tcx>) -> Self::Result {
558if let hir::ExprKind::Closure(&hir::Closure { body, .. })
559 | hir::ExprKind::ConstBlock(hir::ConstBlock { body, .. }) = expr.kind
560 {
561self.visit_body(self.fcx.tcx.hir_body(body))?;
562 }
563564// Try to suggest adding an explicit qself `()` to a trait method path.
565 // i.e. changing `Default::default()` to `<() as Default>::default()`.
566if let hir::ExprKind::Path(hir::QPath::Resolved(None, path)) = expr.kind
567 && let Res::Def(DefKind::AssocFn, def_id) = path.res
568 && self.fcx.tcx.trait_of_assoc(def_id).is_some()
569 && let Some(args) = self.fcx.typeck_results.borrow().node_args_opt(expr.hir_id)
570 && let self_ty = args.type_at(0)
571 && let Some(vid) = self.fcx.root_vid(self_ty)
572 && self.reachable_vids.contains(&vid)
573 && let [.., trait_segment, _method_segment] = path.segments
574 && expr.span.can_be_used_for_suggestions()
575 {
576let span = path.span.shrink_to_lo().to(trait_segment.ident.span);
577return ControlFlow::Break(diagnostics::SuggestAnnotation::Path(span));
578 }
579580// Or else, try suggesting turbofishing the method args.
581if let hir::ExprKind::MethodCall(segment, ..) = expr.kind
582 && let Some(def_id) =
583self.fcx.typeck_results.borrow().type_dependent_def_id(expr.hir_id)
584 && expr.span.can_be_used_for_suggestions()
585 {
586self.suggest_for_segment(segment, def_id, expr.hir_id)?;
587 }
588589 hir::intravisit::walk_expr(self, expr)
590 }
591592fn visit_local(&mut self, local: &'tcx hir::LetStmt<'tcx>) -> Self::Result {
593// For a local, try suggest annotating the type if it's missing.
594if let hir::LocalSource::Normal = local.source
595 && let None = local.ty
596 && let Some(ty) = self.fcx.typeck_results.borrow().node_type_opt(local.hir_id)
597 && let Some(vid) = self.fcx.root_vid(ty)
598 && self.reachable_vids.contains(&vid)
599 && local.span.can_be_used_for_suggestions()
600 {
601return ControlFlow::Break(diagnostics::SuggestAnnotation::Local(
602local.pat.span.shrink_to_hi(),
603 ));
604 }
605 hir::intravisit::walk_local(self, local)
606 }
607}
608609#[derive(#[automatically_derived]
impl ::core::fmt::Debug for UnsafeUseReason {
#[inline]
fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
::core::fmt::Formatter::write_str(f,
match self {
UnsafeUseReason::Call => "Call",
UnsafeUseReason::Method => "Method",
UnsafeUseReason::Path => "Path",
UnsafeUseReason::UnionField => "UnionField",
UnsafeUseReason::Deref => "Deref",
})
}
}Debug, #[automatically_derived]
impl ::core::marker::Copy for UnsafeUseReason { }Copy, #[automatically_derived]
impl ::core::clone::Clone for UnsafeUseReason {
#[inline]
fn clone(&self) -> UnsafeUseReason { *self }
}Clone)]
610pub(crate) enum UnsafeUseReason {
611 Call,
612 Method,
613 Path,
614 UnionField,
615 Deref,
616}
617618/// Finds all type variables which are passed to an `unsafe` operation.
619///
620/// For example, for this function `f`:
621/// ```ignore (demonstrative)
622/// fn f() {
623/// unsafe {
624/// let x /* ?X */ = core::mem::zeroed();
625/// // ^^^^^^^^^^^^^^^^^^^ -- hir_id, span, reason
626///
627/// let y = core::mem::zeroed::<Option<_ /* ?Y */>>();
628/// // ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ -- hir_id, span, reason
629/// }
630/// }
631/// ```
632///
633/// `compute_unsafe_infer_vars` will return `{ id(?X) -> (hir_id, span, Call) }`
634fn compute_unsafe_infer_vars<'a, 'tcx>(
635 fcx: &'a FnCtxt<'a, 'tcx>,
636 body_id: LocalDefId,
637) -> UnordMap<ty::TyVid, (HirId, Span, UnsafeUseReason)> {
638let body = fcx.tcx.hir_maybe_body_owned_by(body_id).expect("body id must have an owner");
639let mut res = UnordMap::default();
640641struct UnsafeInferVarsVisitor<'a, 'tcx> {
642 fcx: &'a FnCtxt<'a, 'tcx>,
643 res: &'a mut UnordMap<ty::TyVid, (HirId, Span, UnsafeUseReason)>,
644 }
645646impl Visitor<'_> for UnsafeInferVarsVisitor<'_, '_> {
647fn visit_expr(&mut self, ex: &'_ hir::Expr<'_>) {
648let typeck_results = self.fcx.typeck_results.borrow();
649650match ex.kind {
651 hir::ExprKind::MethodCall(..) => {
652if let Some(def_id) = typeck_results.type_dependent_def_id(ex.hir_id)
653 && let method_ty =
654self.fcx.tcx.type_of(def_id).instantiate_identity().skip_norm_wip()
655 && let sig = method_ty.fn_sig(self.fcx.tcx)
656 && sig.safety().is_unsafe()
657 {
658let mut collector = InferVarCollector {
659 value: (ex.hir_id, ex.span, UnsafeUseReason::Method),
660 res: self.res,
661 };
662663// Collect generic arguments (incl. `Self`) of the method
664typeck_results665 .node_args(ex.hir_id)
666 .types()
667 .for_each(|t| t.visit_with(&mut collector));
668 }
669 }
670671 hir::ExprKind::Call(func, ..) => {
672let func_ty = typeck_results.expr_ty(func);
673674if func_ty.is_fn()
675 && let sig = func_ty.fn_sig(self.fcx.tcx)
676 && sig.safety().is_unsafe()
677 {
678let mut collector = InferVarCollector {
679 value: (ex.hir_id, ex.span, UnsafeUseReason::Call),
680 res: self.res,
681 };
682683// Try collecting generic arguments of the function.
684 // Note that we do this below for any paths (that don't have to be called),
685 // but there we do it with a different span/reason.
686 // This takes priority.
687typeck_results688 .node_args(func.hir_id)
689 .types()
690 .for_each(|t| t.visit_with(&mut collector));
691692// Also check the return type, for cases like `returns_unsafe_fn_ptr()()`
693sig.output().visit_with(&mut collector);
694 }
695 }
696697// Check paths which refer to functions.
698 // We do this, instead of only checking `Call` to make sure the lint can't be
699 // avoided by storing unsafe function in a variable.
700hir::ExprKind::Path(_) => {
701let ty = typeck_results.expr_ty(ex);
702703// If this path refers to an unsafe function, collect inference variables which may affect it.
704 // `is_fn` excludes closures, but those can't be unsafe.
705if ty.is_fn()
706 && let sig = ty.fn_sig(self.fcx.tcx)
707 && sig.safety().is_unsafe()
708 {
709let mut collector = InferVarCollector {
710 value: (ex.hir_id, ex.span, UnsafeUseReason::Path),
711 res: self.res,
712 };
713714// Collect generic arguments of the function
715typeck_results716 .node_args(ex.hir_id)
717 .types()
718 .for_each(|t| t.visit_with(&mut collector));
719 }
720 }
721722 hir::ExprKind::Unary(hir::UnOp::Deref, pointer) => {
723if let ty::RawPtr(pointee, _) = typeck_results.expr_ty(pointer).kind() {
724pointee.visit_with(&mut InferVarCollector {
725 value: (ex.hir_id, ex.span, UnsafeUseReason::Deref),
726 res: self.res,
727 });
728 }
729 }
730731 hir::ExprKind::Field(base, _) => {
732let base_ty = typeck_results.expr_ty(base);
733734if base_ty.is_union() {
735typeck_results.expr_ty(ex).visit_with(&mut InferVarCollector {
736 value: (ex.hir_id, ex.span, UnsafeUseReason::UnionField),
737 res: self.res,
738 });
739 }
740 }
741742_ => (),
743 };
744745 hir::intravisit::walk_expr(self, ex);
746 }
747 }
748749struct InferVarCollector<'r, V> {
750 value: V,
751 res: &'r mut UnordMap<ty::TyVid, V>,
752 }
753754impl<'tcx, V: Copy> ty::TypeVisitor<TyCtxt<'tcx>> for InferVarCollector<'_, V> {
755fn visit_ty(&mut self, t: Ty<'tcx>) {
756if let Some(vid) = t.ty_vid() {
757_ = self.res.try_insert(vid, self.value);
758 } else {
759t.super_visit_with(self)
760 }
761 }
762 }
763764UnsafeInferVarsVisitor { fcx, res: &mut res }.visit_expr(&body.value);
765766{
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_hir_typeck/src/fallback.rs:766",
"rustc_hir_typeck::fallback", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/fallback.rs"),
::tracing_core::__macro_support::Option::Some(766u32),
::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::fallback"),
::tracing_core::field::FieldSet::new(&["message", "res"],
::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!("collected the following unsafe vars for {0:?}",
body_id) as &dyn Value)),
(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&debug(&res) as
&dyn Value))])
});
} else { ; }
};debug!(?res, "collected the following unsafe vars for {body_id:?}");
767768res769}