Skip to main content

rustc_hir_typeck/
fallback.rs

1use std::cell::OnceCell;
2use std::ops::ControlFlow;
3
4use 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::{self as 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;
20
21use crate::{FnCtxt, diagnostics};
22
23impl<'tcx> FnCtxt<'_, 'tcx> {
24    /// Performs type inference fallback, setting [`FnCtxt::diverging_fallback_has_occurred`]
25    /// if the never type fallback has occurred.
26    pub(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: {:#?}",
29            self.fulfillment_cx.borrow_mut().pending_obligations()
30        );
31
32        // All type checking constraints were added, try to fallback unsolved variables.
33        self.select_obligations_where_possible(|_| {});
34
35        {
    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: {:#?}",
37            self.fulfillment_cx.borrow_mut().pending_obligations()
38        );
39
40        let fallback_occurred = self.fallback_types();
41
42        if fallback_occurred {
43            // if fallback occurred, previously stalled goals may make progress again
44            self.select_obligations_where_possible(|_| {});
45        }
46    }
47
48    fn fallback_types(&self) -> bool {
49        // Check if we have any unresolved variables. If not, no need for fallback.
50        let unresolved_variables = self.unresolved_variables();
51
52        if unresolved_variables.is_empty() {
53            return false;
54        }
55
56        let (diverging_fallback, diverging_fallback_ty) =
57            self.calculate_diverging_fallback(&unresolved_variables);
58        let fallback_to_f32 = self.calculate_fallback_to_f32(&unresolved_variables);
59
60        // We do fallback in two passes, to try to generate
61        // better error messages.
62        // The first time, we do *not* replace opaque types.
63        let mut fallback_occurred = false;
64        for 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        }
73
74        fallback_occurred
75    }
76
77    /// 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.
93    fn 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.)
118        let 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) => {
124                self.diverging_fallback_has_occurred.set(true);
125                diverging_fallback_ty
126            }
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);
130
131        let span = ty.ty_vid().map_or(DUMMY_SP, |vid| self.infcx.type_var_origin(vid).span);
132        self.demand_eqtype(span, ty, fallback);
133        true
134    }
135
136    /// 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    /// ```
149    fn 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.
155        if unresolved_variables.iter().all(|ty| ty.float_vid().is_none()) {
156            return UnordSet::new();
157        }
158        let roots: UnordSet<ty::FloatVid> = self.from_float_for_f32_root_vids();
159        if 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.
162            return 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.
167        let fallback_to_f32 = unresolved_variables
168            .iter()
169            .flat_map(|ty| ty.float_vid())
170            .filter(|vid| roots.contains(&self.root_float_var(*vid)))
171            .inspect(|vid| {
172                let origin = self.float_var_origin(*vid);
173                // Show the entire literal in the suggestion to make it clearer.
174                let 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
176                if let Some(ref mut literal) = literal
177                    && literal.ends_with('.')
178                {
179                    literal.pop();
180                }
181                self.tcx.emit_node_span_lint(
182                    FLOAT_LITERAL_F32_FALLBACK,
183                    origin.lint_id.unwrap_or(CRATE_HIR_ID),
184                    origin.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);
193        fallback_to_f32
194    }
195
196    fn 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);
201
202        let 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
207                return (UnordSet::new(), self.tcx.types.unit);
208            }
209        };
210
211        // Construct a coercion graph where an edge `A -> B` indicates
212        // a type variable is that is coerced
213        let coercion_graph = self.create_coercion_graph();
214
215        // Extract the unsolved type inference variable vids; note that some
216        // unsolved variables are integer/float variables and are excluded.
217        let unsolved_vids = unresolved_variables.iter().filter_map(|ty| ty.ty_vid());
218
219        // 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 `()`.
225        let diverging_roots: UnordSet<ty::TyVid> = self
226            .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={:?}",
235            self.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);
238
239        // 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.
243        let mut diverging_vids = ::alloc::vec::Vec::new()vec![];
244        for unsolved_vid in unsolved_vids {
245            let 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            );
252            if diverging_roots.contains(&root_vid) {
253                diverging_vids.push(unsolved_vid);
254
255                {
    use ::tracing::__macro_support::Callsite as _;
    static __CALLSITE: ::tracing::callsite::DefaultCallsite =
        {
            static META: ::tracing::Metadata<'static> =
                {
                    ::tracing_core::metadata::Metadata::new("event compiler/rustc_hir_typeck/src/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        }
262
263        {
    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());
264
265        let mut diverging_fallback = UnordSet::with_capacity(diverging_vids.len());
266        let unsafe_infer_vars = OnceCell::new();
267
268        self.lint_obligations_broken_by_never_type_fallback_change(
269            &diverging_vids,
270            &coercion_graph,
271        );
272
273        for &diverging_vid in &diverging_vids {
274            let diverging_ty = Ty::new_var(self.tcx, diverging_vid);
275            let root_vid = self.root_var(diverging_vid);
276
277            self.lint_never_type_fallback_flowing_into_unsafe_code(
278                &unsafe_infer_vars,
279                &coercion_graph,
280                root_vid,
281            );
282
283            diverging_fallback.insert(diverging_ty);
284        }
285
286        (diverging_fallback, diverging_fallback_ty)
287    }
288
289    fn 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    ) {
295        let unsafe_infer_vars = unsafe_infer_vars.get_or_init(|| {
296            let 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);
298            unsafe_infer_vars
299        });
300
301        let 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<_>>();
305
306        let sugg = self.try_to_suggest_annotations(&[root_vid], coercion_graph);
307
308        for (hir_id, span, reason) in affected_unsafe_infer_vars {
309            self.tcx.emit_node_span_lint(
310                lint::builtin::NEVER_TYPE_FALLBACK_FLOWING_INTO_UNSAFE,
311                hir_id,
312                span,
313                match 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    }
339
340    fn lint_obligations_broken_by_never_type_fallback_change(
341        &self,
342        diverging_vids: &[ty::TyVid],
343        coercions: &VecGraph<ty::TyVid, true>,
344    ) {
345        let DivergingFallbackBehavior::ToUnit = self.diverging_fallback_behavior else { return };
346
347        // Fallback happens if and only if there are diverging variables
348        if diverging_vids.is_empty() {
349            return;
350        }
351
352        // Returns errors which happen if fallback is set to `fallback`
353        let remaining_errors_if_fallback_to = |fallback| {
354            self.probe(|_| {
355                let obligations = self.fulfillment_cx.borrow().pending_obligations();
356                let ocx = ObligationCtxt::new_with_diagnostics(&self.infcx);
357                ocx.register_obligations(obligations.iter().cloned());
358
359                for &diverging_vid in diverging_vids {
360                    let diverging_ty = Ty::new_var(self.tcx, diverging_vid);
361
362                    ocx.eq(&ObligationCause::dummy(), self.param_env, diverging_ty, fallback)
363                        .expect("expected diverging var to be unconstrained");
364                }
365
366                ocx.try_evaluate_obligations()
367            })
368        };
369
370        // 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.
372        let unit_errors = remaining_errors_if_fallback_to(self.tcx.types.unit);
373        if 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        {
377            self.adjust_fulfillment_error_for_expr_obligation(never_error);
378            let sugg = self.try_to_suggest_annotations(diverging_vids, coercions);
379            self.tcx.emit_node_span_lint(
380                lint::builtin::DEPENDENCY_ON_UNIT_NEVER_TYPE_FALLBACK,
381                self.tcx.local_def_id_to_hir_id(self.body_id),
382                self.tcx.def_span(self.body_id),
383                diagnostics::DependencyOnUnitNeverTypeFallback {
384                    obligation_span: never_error.obligation.cause.span,
385                    obligation: never_error.obligation.predicate,
386                    sugg,
387                },
388            )
389        }
390    }
391
392    /// 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`.
394    fn create_coercion_graph(&self) -> VecGraph<ty::TyVid, true> {
395        let 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);
397        let coercion_edges: Vec<(ty::TyVid, ty::TyVid)> = pending_obligations
398            .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.
402                obligation.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.
413                let (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                };
420
421                let a_vid = self.root_vid(a)?;
422                let b_vid = self.root_vid(b)?;
423                Some((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);
427        let num_ty_vars = self.num_ty_vars();
428
429        VecGraph::new(num_ty_vars, coercion_edges)
430    }
431
432    /// If `ty` is an unresolved type variable, returns its root vid.
433    fn root_vid(&self, ty: Ty<'tcx>) -> Option<ty::TyVid> {
434        Some(self.root_var(self.shallow_resolve(ty).ty_vid()?))
435    }
436
437    /// If `ty` is an unresolved float type variable, returns its root vid.
438    pub(crate) fn root_float_vid(&self, ty: Ty<'tcx>) -> Option<ty::FloatVid> {
439        Some(self.root_float_var(self.shallow_resolve(ty).float_vid()?))
440    }
441
442    /// 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.
444    fn try_to_suggest_annotations(
445        &self,
446        diverging_vids: &[ty::TyVid],
447        coercions: &VecGraph<ty::TyVid, true>,
448    ) -> diagnostics::SuggestAnnotations {
449        let body =
450            self.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 `()`.
454        let suggestions = diverging_vids
455            .iter()
456            .copied()
457            .filter_map(|vid| {
458                let reachable_vids =
459                    graph::depth_first_search_as_undirected(coercions, vid).collect();
460                AnnotateUnitFallbackVisitor { reachable_vids, fcx: self }
461                    .visit_expr(body.value)
462                    .break_value()
463            })
464            .collect();
465        diagnostics::SuggestAnnotations { suggestions }
466    }
467}
468
469/// 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)...
481    fn suggest_for_segment(
482        &self,
483        arg_segment: &'tcx hir::PathSegment<'tcx>,
484        def_id: DefId,
485        id: HirId,
486    ) -> ControlFlow<diagnostics::SuggestAnnotation> {
487        if 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.
495            let 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            });
498            let n_tys = non_apit_type_args.clone().count();
499            for (idx, (arg, _)) in non_apit_type_args.enumerate() {
500                if let Some(ty) = arg.as_type()
501                    && let Some(vid) = self.fcx.root_vid(ty)
502                    && self.reachable_vids.contains(&vid)
503                {
504                    return 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> {
516    type Result = ControlFlow<diagnostics::SuggestAnnotation>;
517
518    fn 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 `()`.
525        if 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        {
530            return ControlFlow::Break(diagnostics::SuggestAnnotation::Unit(inf_span));
531        }
532
533        ControlFlow::Continue(())
534    }
535
536    fn visit_qpath(
537        &mut self,
538        qpath: &'tcx rustc_hir::QPath<'tcx>,
539        id: HirId,
540        span: Span,
541    ) -> Self::Result {
542        let arg_segment = match qpath {
543            hir::QPath::Resolved(_, path) => {
544                path.segments.last().expect("paths should have a segment")
545            }
546            hir::QPath::TypeRelative(_, segment) => segment,
547        };
548        // Alternatively, try to turbofish `::<_, (), _>`.
549        if let Some(def_id) = self.fcx.typeck_results.borrow().qpath_res(qpath, id).opt_def_id()
550            && span.can_be_used_for_suggestions()
551        {
552            self.suggest_for_segment(arg_segment, def_id, id)?;
553        }
554        hir::intravisit::walk_qpath(self, qpath, id)
555    }
556
557    fn visit_expr(&mut self, expr: &'tcx hir::Expr<'tcx>) -> Self::Result {
558        if let hir::ExprKind::Closure(&hir::Closure { body, .. })
559        | hir::ExprKind::ConstBlock(hir::ConstBlock { body, .. }) = expr.kind
560        {
561            self.visit_body(self.fcx.tcx.hir_body(body))?;
562        }
563
564        // Try to suggest adding an explicit qself `()` to a trait method path.
565        // i.e. changing `Default::default()` to `<() as Default>::default()`.
566        if 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        {
576            let span = path.span.shrink_to_lo().to(trait_segment.ident.span);
577            return ControlFlow::Break(diagnostics::SuggestAnnotation::Path(span));
578        }
579
580        // Or else, try suggesting turbofishing the method args.
581        if let hir::ExprKind::MethodCall(segment, ..) = expr.kind
582            && let Some(def_id) =
583                self.fcx.typeck_results.borrow().type_dependent_def_id(expr.hir_id)
584            && expr.span.can_be_used_for_suggestions()
585        {
586            self.suggest_for_segment(segment, def_id, expr.hir_id)?;
587        }
588
589        hir::intravisit::walk_expr(self, expr)
590    }
591
592    fn visit_local(&mut self, local: &'tcx hir::LetStmt<'tcx>) -> Self::Result {
593        // For a local, try suggest annotating the type if it's missing.
594        if 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        {
601            return ControlFlow::Break(diagnostics::SuggestAnnotation::Local(
602                local.pat.span.shrink_to_hi(),
603            ));
604        }
605        hir::intravisit::walk_local(self, local)
606    }
607}
608
609#[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}
617
618/// 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)> {
638    let body = fcx.tcx.hir_maybe_body_owned_by(body_id).expect("body id must have an owner");
639    let mut res = UnordMap::default();
640
641    struct UnsafeInferVarsVisitor<'a, 'tcx> {
642        fcx: &'a FnCtxt<'a, 'tcx>,
643        res: &'a mut UnordMap<ty::TyVid, (HirId, Span, UnsafeUseReason)>,
644    }
645
646    impl Visitor<'_> for UnsafeInferVarsVisitor<'_, '_> {
647        fn visit_expr(&mut self, ex: &'_ hir::Expr<'_>) {
648            let typeck_results = self.fcx.typeck_results.borrow();
649
650            match ex.kind {
651                hir::ExprKind::MethodCall(..) => {
652                    if let Some(def_id) = typeck_results.type_dependent_def_id(ex.hir_id)
653                        && let method_ty =
654                            self.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                    {
658                        let mut collector = InferVarCollector {
659                            value: (ex.hir_id, ex.span, UnsafeUseReason::Method),
660                            res: self.res,
661                        };
662
663                        // Collect generic arguments (incl. `Self`) of the method
664                        typeck_results
665                            .node_args(ex.hir_id)
666                            .types()
667                            .for_each(|t| t.visit_with(&mut collector));
668                    }
669                }
670
671                hir::ExprKind::Call(func, ..) => {
672                    let func_ty = typeck_results.expr_ty(func);
673
674                    if func_ty.is_fn()
675                        && let sig = func_ty.fn_sig(self.fcx.tcx)
676                        && sig.safety().is_unsafe()
677                    {
678                        let mut collector = InferVarCollector {
679                            value: (ex.hir_id, ex.span, UnsafeUseReason::Call),
680                            res: self.res,
681                        };
682
683                        // 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.
687                        typeck_results
688                            .node_args(func.hir_id)
689                            .types()
690                            .for_each(|t| t.visit_with(&mut collector));
691
692                        // Also check the return type, for cases like `returns_unsafe_fn_ptr()()`
693                        sig.output().visit_with(&mut collector);
694                    }
695                }
696
697                // 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.
700                hir::ExprKind::Path(_) => {
701                    let ty = typeck_results.expr_ty(ex);
702
703                    // 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.
705                    if ty.is_fn()
706                        && let sig = ty.fn_sig(self.fcx.tcx)
707                        && sig.safety().is_unsafe()
708                    {
709                        let mut collector = InferVarCollector {
710                            value: (ex.hir_id, ex.span, UnsafeUseReason::Path),
711                            res: self.res,
712                        };
713
714                        // Collect generic arguments of the function
715                        typeck_results
716                            .node_args(ex.hir_id)
717                            .types()
718                            .for_each(|t| t.visit_with(&mut collector));
719                    }
720                }
721
722                hir::ExprKind::Unary(hir::UnOp::Deref, pointer) => {
723                    if let ty::RawPtr(pointee, _) = typeck_results.expr_ty(pointer).kind() {
724                        pointee.visit_with(&mut InferVarCollector {
725                            value: (ex.hir_id, ex.span, UnsafeUseReason::Deref),
726                            res: self.res,
727                        });
728                    }
729                }
730
731                hir::ExprKind::Field(base, _) => {
732                    let base_ty = typeck_results.expr_ty(base);
733
734                    if base_ty.is_union() {
735                        typeck_results.expr_ty(ex).visit_with(&mut InferVarCollector {
736                            value: (ex.hir_id, ex.span, UnsafeUseReason::UnionField),
737                            res: self.res,
738                        });
739                    }
740                }
741
742                _ => (),
743            };
744
745            hir::intravisit::walk_expr(self, ex);
746        }
747    }
748
749    struct InferVarCollector<'r, V> {
750        value: V,
751        res: &'r mut UnordMap<ty::TyVid, V>,
752    }
753
754    impl<'tcx, V: Copy> ty::TypeVisitor<TyCtxt<'tcx>> for InferVarCollector<'_, V> {
755        fn visit_ty(&mut self, t: Ty<'tcx>) {
756            if let Some(vid) = t.ty_vid() {
757                _ = self.res.try_insert(vid, self.value);
758            } else {
759                t.super_visit_with(self)
760            }
761        }
762    }
763
764    UnsafeInferVarsVisitor { fcx, res: &mut res }.visit_expr(&body.value);
765
766    {
    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:?}");
767
768    res
769}