Skip to main content

rustc_ty_utils/
instance.rs

1use rustc_errors::ErrorGuaranteed;
2use rustc_hir::LangItem;
3use rustc_hir::def_id::DefId;
4use rustc_infer::infer::TyCtxtInferExt;
5use rustc_middle::bug;
6use rustc_middle::query::Providers;
7use rustc_middle::traits::{BuiltinImplSource, CodegenObligationError};
8use rustc_middle::ty::{
9    self, ClosureKind, GenericArgsRef, Instance, PseudoCanonicalInput, TyCtxt, TypeVisitableExt,
10    Unnormalized,
11};
12use rustc_span::sym;
13use rustc_trait_selection::traits;
14use tracing::debug;
15use traits::translate_args;
16
17use crate::diagnostics::UnexpectedFnPtrAssociatedItem;
18
19fn resolve_instance_raw<'tcx>(
20    tcx: TyCtxt<'tcx>,
21    key: ty::PseudoCanonicalInput<'tcx, (DefId, GenericArgsRef<'tcx>)>,
22) -> Result<Option<Instance<'tcx>>, ErrorGuaranteed> {
23    let PseudoCanonicalInput { typing_env, value: (def_id, args) } = key;
24
25    let result = if let Some(trait_def_id) = tcx.trait_of_assoc(def_id) {
26        {
    use ::tracing::__macro_support::Callsite as _;
    static __CALLSITE: ::tracing::callsite::DefaultCallsite =
        {
            static META: ::tracing::Metadata<'static> =
                {
                    ::tracing_core::metadata::Metadata::new("event compiler/rustc_ty_utils/src/instance.rs:26",
                        "rustc_ty_utils::instance", ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_ty_utils/src/instance.rs"),
                        ::tracing_core::__macro_support::Option::Some(26u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_ty_utils::instance"),
                        ::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!(" => associated item, attempting to find impl in typing_env {0:#?}",
                                                    typing_env) as &dyn Value))])
            });
    } else { ; }
};debug!(" => associated item, attempting to find impl in typing_env {:#?}", typing_env);
27        resolve_associated_item(
28            tcx,
29            def_id,
30            typing_env,
31            trait_def_id,
32            tcx.normalize_erasing_regions(typing_env, Unnormalized::new_wip(args)),
33        )
34    } else {
35        let def = if tcx.intrinsic(def_id).is_some() {
36            {
    use ::tracing::__macro_support::Callsite as _;
    static __CALLSITE: ::tracing::callsite::DefaultCallsite =
        {
            static META: ::tracing::Metadata<'static> =
                {
                    ::tracing_core::metadata::Metadata::new("event compiler/rustc_ty_utils/src/instance.rs:36",
                        "rustc_ty_utils::instance", ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_ty_utils/src/instance.rs"),
                        ::tracing_core::__macro_support::Option::Some(36u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_ty_utils::instance"),
                        ::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!(" => intrinsic")
                                            as &dyn Value))])
            });
    } else { ; }
};debug!(" => intrinsic");
37            ty::InstanceKind::Intrinsic(def_id)
38        } else if tcx.is_lang_item(def_id, LangItem::DropGlue) {
39            let ty = args.type_at(0);
40
41            let shim = if ty.needs_drop(tcx, typing_env) {
42                {
    use ::tracing::__macro_support::Callsite as _;
    static __CALLSITE: ::tracing::callsite::DefaultCallsite =
        {
            static META: ::tracing::Metadata<'static> =
                {
                    ::tracing_core::metadata::Metadata::new("event compiler/rustc_ty_utils/src/instance.rs:42",
                        "rustc_ty_utils::instance", ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_ty_utils/src/instance.rs"),
                        ::tracing_core::__macro_support::Option::Some(42u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_ty_utils::instance"),
                        ::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!(" => nontrivial drop glue")
                                            as &dyn Value))])
            });
    } else { ; }
};debug!(" => nontrivial drop glue");
43                match *ty.kind() {
44                    ty::Coroutine(coroutine_def_id, ..) => {
45                        // FIXME: sync drop of coroutine with async drop (generate both versions?)
46                        // Currently just ignored
47                        if tcx.optimized_mir(coroutine_def_id).coroutine_drop_async().is_some() {
48                            ty::ShimKind::DropGlue(def_id, None)
49                        } else {
50                            ty::ShimKind::DropGlue(def_id, Some(ty))
51                        }
52                    }
53                    ty::Closure(..)
54                    | ty::CoroutineClosure(..)
55                    | ty::Tuple(..)
56                    | ty::Adt(..)
57                    | ty::Dynamic(..)
58                    | ty::Array(..)
59                    | ty::Slice(..)
60                    | ty::UnsafeBinder(..) => ty::ShimKind::DropGlue(def_id, Some(ty)),
61                    // Drop shims can only be built from ADTs.
62                    _ => return Ok(None),
63                }
64            } else {
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_ty_utils/src/instance.rs:65",
                        "rustc_ty_utils::instance", ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_ty_utils/src/instance.rs"),
                        ::tracing_core::__macro_support::Option::Some(65u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_ty_utils::instance"),
                        ::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!(" => trivial drop glue")
                                            as &dyn Value))])
            });
    } else { ; }
};debug!(" => trivial drop glue");
66                ty::ShimKind::DropGlue(def_id, None)
67            };
68            ty::InstanceKind::Shim(shim)
69        } else if tcx.is_lang_item(def_id, LangItem::AsyncDropInPlace) {
70            let ty = args.type_at(0);
71
72            if ty.needs_async_drop(tcx, typing_env) {
73                match *ty.kind() {
74                    ty::Closure(..)
75                    | ty::CoroutineClosure(..)
76                    | ty::Coroutine(..)
77                    | ty::Tuple(..)
78                    | ty::Adt(..)
79                    | ty::Dynamic(..)
80                    | ty::Array(..)
81                    | ty::Slice(..) => {}
82                    // Async destructor ctor shims can only be built from ADTs.
83                    _ => return Ok(None),
84                }
85                {
    use ::tracing::__macro_support::Callsite as _;
    static __CALLSITE: ::tracing::callsite::DefaultCallsite =
        {
            static META: ::tracing::Metadata<'static> =
                {
                    ::tracing_core::metadata::Metadata::new("event compiler/rustc_ty_utils/src/instance.rs:85",
                        "rustc_ty_utils::instance", ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_ty_utils/src/instance.rs"),
                        ::tracing_core::__macro_support::Option::Some(85u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_ty_utils::instance"),
                        ::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!(" => nontrivial async drop glue ctor")
                                            as &dyn Value))])
            });
    } else { ; }
};debug!(" => nontrivial async drop glue ctor");
86                ty::InstanceKind::Shim(ty::ShimKind::AsyncDropGlueCtor(def_id, ty))
87            } else {
88                {
    use ::tracing::__macro_support::Callsite as _;
    static __CALLSITE: ::tracing::callsite::DefaultCallsite =
        {
            static META: ::tracing::Metadata<'static> =
                {
                    ::tracing_core::metadata::Metadata::new("event compiler/rustc_ty_utils/src/instance.rs:88",
                        "rustc_ty_utils::instance", ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_ty_utils/src/instance.rs"),
                        ::tracing_core::__macro_support::Option::Some(88u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_ty_utils::instance"),
                        ::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!(" => trivial async drop glue ctor")
                                            as &dyn Value))])
            });
    } else { ; }
};debug!(" => trivial async drop glue ctor");
89                ty::InstanceKind::Shim(ty::ShimKind::AsyncDropGlueCtor(def_id, ty))
90            }
91        } else if tcx.is_async_drop_in_place_coroutine(def_id) {
92            let ty = args.type_at(0);
93            ty::InstanceKind::Shim(ty::ShimKind::AsyncDropGlue(def_id, ty))
94        } else {
95            {
    use ::tracing::__macro_support::Callsite as _;
    static __CALLSITE: ::tracing::callsite::DefaultCallsite =
        {
            static META: ::tracing::Metadata<'static> =
                {
                    ::tracing_core::metadata::Metadata::new("event compiler/rustc_ty_utils/src/instance.rs:95",
                        "rustc_ty_utils::instance", ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_ty_utils/src/instance.rs"),
                        ::tracing_core::__macro_support::Option::Some(95u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_ty_utils::instance"),
                        ::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!(" => free item")
                                            as &dyn Value))])
            });
    } else { ; }
};debug!(" => free item");
96            ty::InstanceKind::Item(def_id)
97        };
98
99        Ok(Some(Instance { def, args }))
100    };
101    {
    use ::tracing::__macro_support::Callsite as _;
    static __CALLSITE: ::tracing::callsite::DefaultCallsite =
        {
            static META: ::tracing::Metadata<'static> =
                {
                    ::tracing_core::metadata::Metadata::new("event compiler/rustc_ty_utils/src/instance.rs:101",
                        "rustc_ty_utils::instance", ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_ty_utils/src/instance.rs"),
                        ::tracing_core::__macro_support::Option::Some(101u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_ty_utils::instance"),
                        ::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!("resolve_instance: result={0:?}",
                                                    result) as &dyn Value))])
            });
    } else { ; }
};debug!("resolve_instance: result={:?}", result);
102    result
103}
104
105fn resolve_associated_item<'tcx>(
106    tcx: TyCtxt<'tcx>,
107    trait_item_id: DefId,
108    typing_env: ty::TypingEnv<'tcx>,
109    trait_id: DefId,
110    rcvr_args: GenericArgsRef<'tcx>,
111) -> Result<Option<Instance<'tcx>>, ErrorGuaranteed> {
112    {
    use ::tracing::__macro_support::Callsite as _;
    static __CALLSITE: ::tracing::callsite::DefaultCallsite =
        {
            static META: ::tracing::Metadata<'static> =
                {
                    ::tracing_core::metadata::Metadata::new("event compiler/rustc_ty_utils/src/instance.rs:112",
                        "rustc_ty_utils::instance", ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_ty_utils/src/instance.rs"),
                        ::tracing_core::__macro_support::Option::Some(112u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_ty_utils::instance"),
                        ::tracing_core::field::FieldSet::new(&["message",
                                        "trait_item_id", "typing_env", "trait_id", "rcvr_args"],
                            ::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!("resolve_associated_item")
                                            as &dyn Value)),
                                (&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
                                    ::tracing::__macro_support::Option::Some(&debug(&trait_item_id)
                                            as &dyn Value)),
                                (&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
                                    ::tracing::__macro_support::Option::Some(&debug(&typing_env)
                                            as &dyn Value)),
                                (&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
                                    ::tracing::__macro_support::Option::Some(&debug(&trait_id)
                                            as &dyn Value)),
                                (&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
                                    ::tracing::__macro_support::Option::Some(&debug(&rcvr_args)
                                            as &dyn Value))])
            });
    } else { ; }
};debug!(?trait_item_id, ?typing_env, ?trait_id, ?rcvr_args, "resolve_associated_item");
113
114    let trait_ref = ty::TraitRef::from_assoc(tcx, trait_id, rcvr_args);
115
116    let input = typing_env.as_query_input(trait_ref);
117    let vtbl = match tcx.codegen_select_candidate(input) {
118        Ok(vtbl) => vtbl,
119        Err(CodegenObligationError::Ambiguity | CodegenObligationError::Unimplemented) => {
120            return Ok(None);
121        }
122        Err(CodegenObligationError::UnconstrainedParam(guar)) => return Err(guar),
123    };
124
125    // Now that we know which impl is being used, we can dispatch to
126    // the actual function:
127    Ok(match vtbl {
128        traits::ImplSource::UserDefined(impl_data) => {
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_ty_utils/src/instance.rs:129",
                        "rustc_ty_utils::instance", ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_ty_utils/src/instance.rs"),
                        ::tracing_core::__macro_support::Option::Some(129u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_ty_utils::instance"),
                        ::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!("resolving ImplSource::UserDefined: {0:?}, {1:?}, {2:?}, {3:?}",
                                                    typing_env, trait_item_id, rcvr_args, impl_data) as
                                            &dyn Value))])
            });
    } else { ; }
};debug!(
130                "resolving ImplSource::UserDefined: {:?}, {:?}, {:?}, {:?}",
131                typing_env, trait_item_id, rcvr_args, impl_data
132            );
133            if !!rcvr_args.has_infer() {
    ::core::panicking::panic("assertion failed: !rcvr_args.has_infer()")
};assert!(!rcvr_args.has_infer());
134            if !!trait_ref.has_infer() {
    ::core::panicking::panic("assertion failed: !trait_ref.has_infer()")
};assert!(!trait_ref.has_infer());
135
136            let trait_def_id = tcx.impl_trait_id(impl_data.impl_def_id);
137            let trait_def = tcx.trait_def(trait_def_id);
138            let leaf_def = trait_def
139                .ancestors(tcx, impl_data.impl_def_id)?
140                .leaf_def(tcx, trait_item_id)
141                .unwrap_or_else(|| {
142                    ::rustc_middle::util::bug::bug_fmt(format_args!("{0:?} not found in {1:?}",
        trait_item_id, impl_data.impl_def_id));bug!("{:?} not found in {:?}", trait_item_id, impl_data.impl_def_id);
143                });
144
145            // Since this is a trait item, we need to see if the item is either a trait
146            // default item or a specialization because we can't resolve those until we're
147            // in `TypingMode::PostAnalysis`.
148            //
149            // NOTE: This should be kept in sync with the similar code in
150            // `rustc_trait_selection::traits::project::assemble_candidates_from_impls()`.
151            let eligible = if leaf_def.is_final() {
152                // Non-specializable items are always projectable.
153                true
154            } else {
155                // Only reveal a specializable default if we're past type-checking
156                // and the obligation is monomorphic, otherwise passes such as
157                // transmute checking and polymorphic MIR optimizations could
158                // get a result which isn't correct for all monomorphizations.
159                match typing_env.typing_mode().assert_not_erased() {
160                    ty::TypingMode::Coherence
161                    | ty::TypingMode::Typeck { .. }
162                    | ty::TypingMode::PostTypeckUntilBorrowck { .. }
163                    | ty::TypingMode::PostBorrowck { .. } => false,
164                    ty::TypingMode::PostAnalysis | ty::TypingMode::Codegen => {
165                        !trait_ref.still_further_specializable()
166                    }
167                }
168            };
169            if !eligible {
170                return Ok(None);
171            }
172
173            let typing_env = typing_env.with_post_analysis_normalized(tcx);
174            let (infcx, param_env) = tcx.infer_ctxt().build_with_typing_env(typing_env);
175            let args = rcvr_args.rebase_onto(tcx, trait_def_id, impl_data.args);
176            let args = translate_args(
177                &infcx,
178                param_env,
179                impl_data.impl_def_id,
180                args,
181                leaf_def.defining_node,
182            );
183            let args = infcx.tcx.erase_and_anonymize_regions(args);
184
185            // HACK: We may have overlapping `dyn Trait` built-in impls and
186            // user-provided blanket impls. Detect that case here, and return
187            // ambiguity.
188            //
189            // This should not affect totally monomorphized contexts, only
190            // resolve calls that happen polymorphically, such as the mir-inliner
191            // and const-prop (and also some lints).
192            let self_ty = rcvr_args.type_at(0);
193            if !self_ty.is_known_rigid() {
194                let predicates = tcx
195                    .predicates_of(impl_data.impl_def_id)
196                    .instantiate(tcx, impl_data.args)
197                    .predicates;
198                let sized_def_id = tcx.lang_items().sized_trait();
199                // If we find a `Self: Sized` bound on the item, then we know
200                // that `dyn Trait` can certainly never apply here.
201                if !predicates.into_iter().filter_map(|p| p.as_trait_clause()).any(|clause| {
202                    Some(clause.def_id()) == sized_def_id
203                        && clause.skip_binder().self_ty() == self_ty
204                }) {
205                    return Ok(None);
206                }
207            }
208
209            // Any final impl is required to define all associated items.
210            if !leaf_def.item.defaultness(tcx).has_value() {
211                let guar = tcx.dcx().span_delayed_bug(
212                    tcx.def_span(leaf_def.item.def_id),
213                    "missing value for assoc item in impl",
214                );
215                return Err(guar);
216            }
217
218            // Make sure that we're projecting to an item that has compatible args.
219            // This may happen if we are resolving an instance before codegen, such
220            // as during inlining. This check is also done in projection.
221            if !tcx.check_args_compatible(leaf_def.item.def_id, args) {
222                let guar = tcx.dcx().span_delayed_bug(
223                    tcx.def_span(leaf_def.item.def_id),
224                    "missing value for assoc item in impl",
225                );
226                return Err(guar);
227            }
228
229            // We check that the impl item is compatible with the trait item
230            // because otherwise we may ICE in const eval due to type mismatches,
231            // signature incompatibilities, etc.
232            // NOTE: We could also only enforce this in `PostAnalysis`, which
233            // is what CTFE and MIR inlining would care about anyways.
234            if trait_item_id != leaf_def.item.def_id
235                && let Some(leaf_def_item) = leaf_def.item.def_id.as_local()
236            {
237                tcx.ensure_result().compare_impl_item(leaf_def_item)?;
238            }
239
240            Some(ty::Instance::new_raw(leaf_def.item.def_id, args))
241        }
242        traits::ImplSource::Builtin(BuiltinImplSource::Object(_), _) => {
243            let trait_ref = ty::TraitRef::from_assoc(tcx, trait_id, rcvr_args);
244
245            // `final` methods should be called directly.
246            if tcx.defaultness(trait_item_id).is_final() {
247                return Ok(Some(ty::Instance::new_raw(trait_item_id, rcvr_args)));
248            }
249
250            if trait_ref.has_non_region_infer() || trait_ref.has_non_region_param() {
251                // We only resolve totally substituted vtable entries.
252                None
253            } else {
254                let vtable_base = tcx.first_method_vtable_slot(trait_ref);
255                let offset = tcx
256                    .own_existential_vtable_entries(trait_id)
257                    .iter()
258                    .copied()
259                    .position(|def_id| def_id == trait_item_id);
260                offset.map(|offset| Instance {
261                    def: ty::InstanceKind::Virtual(trait_item_id, vtable_base + offset),
262                    args: rcvr_args,
263                })
264            }
265        }
266        traits::ImplSource::Builtin(BuiltinImplSource::Misc | BuiltinImplSource::Trivial, _) => {
267            if tcx.is_lang_item(trait_ref.def_id, LangItem::Clone) {
268                // FIXME(eddyb) use lang items for methods instead of names.
269                let name = tcx.item_name(trait_item_id);
270                if name == sym::clone {
271                    let self_ty = trait_ref.self_ty();
272                    match self_ty.kind() {
273                        ty::FnDef(..) | ty::FnPtr(..) => (),
274                        ty::Coroutine(..)
275                        | ty::CoroutineWitness(..)
276                        | ty::Closure(..)
277                        | ty::CoroutineClosure(..)
278                        | ty::Tuple(..) => {}
279                        _ => return Ok(None),
280                    };
281
282                    Some(Instance {
283                        def: ty::InstanceKind::Shim(ty::ShimKind::Clone(trait_item_id, self_ty)),
284                        args: rcvr_args,
285                    })
286                } else {
287                    {
    match (&name, &sym::clone_from) {
        (left_val, right_val) => {
            if !(*left_val == *right_val) {
                let kind = ::core::panicking::AssertKind::Eq;
                ::core::panicking::assert_failed(kind, &*left_val,
                    &*right_val, ::core::option::Option::None);
            }
        }
    }
};assert_eq!(name, sym::clone_from);
288
289                    // Use the default `fn clone_from` from `trait Clone`.
290                    let args = tcx.erase_and_anonymize_regions(rcvr_args);
291                    Some(ty::Instance::new_raw(trait_item_id, args))
292                }
293            } else if tcx.is_lang_item(trait_ref.def_id, LangItem::FnPtrTrait) {
294                if tcx.is_lang_item(trait_item_id, LangItem::FnPtrAddr) {
295                    let self_ty = trait_ref.self_ty();
296                    if !#[allow(non_exhaustive_omitted_patterns)] match self_ty.kind() {
    ty::FnPtr(..) => true,
    _ => false,
}matches!(self_ty.kind(), ty::FnPtr(..)) {
297                        return Ok(None);
298                    }
299                    Some(Instance {
300                        def: ty::InstanceKind::Shim(ty::ShimKind::FnPtrAddr(
301                            trait_item_id,
302                            self_ty,
303                        )),
304                        args: rcvr_args,
305                    })
306                } else {
307                    tcx.dcx().emit_fatal(UnexpectedFnPtrAssociatedItem {
308                        span: tcx.def_span(trait_item_id),
309                    })
310                }
311            } else if let Some(target_kind) = tcx.fn_trait_kind_from_def_id(trait_ref.def_id) {
312                // FIXME: This doesn't check for malformed libcore that defines, e.g.,
313                // `trait Fn { fn call_once(&self) { .. } }`. This is mostly for extension
314                // methods.
315                if truecfg!(debug_assertions)
316                    && ![sym::call, sym::call_mut, sym::call_once]
317                        .contains(&tcx.item_name(trait_item_id))
318                {
319                    // For compiler developers who'd like to add new items to `Fn`/`FnMut`/`FnOnce`,
320                    // you either need to generate a shim body, or perhaps return
321                    // `InstanceKind::Item` pointing to a trait default method body if
322                    // it is given a default implementation by the trait.
323                    ::rustc_middle::util::bug::bug_fmt(format_args!("no definition for `{1}::{0}` for built-in callable type",
        tcx.item_name(trait_item_id), trait_ref))bug!(
324                        "no definition for `{trait_ref}::{}` for built-in callable type",
325                        tcx.item_name(trait_item_id)
326                    )
327                }
328                match *rcvr_args.type_at(0).kind() {
329                    ty::Closure(closure_def_id, args) => {
330                        Some(Instance::resolve_closure(tcx, closure_def_id, args, target_kind))
331                    }
332                    ty::FnDef(..) | ty::FnPtr(..) => Some(Instance {
333                        def: ty::InstanceKind::Shim(ty::ShimKind::FnPtr(
334                            trait_item_id,
335                            rcvr_args.type_at(0),
336                        )),
337                        args: rcvr_args,
338                    }),
339                    ty::CoroutineClosure(coroutine_closure_def_id, args) => {
340                        // When a coroutine-closure implements the `Fn` traits, then it
341                        // always dispatches to the `FnOnce` implementation. This is to
342                        // ensure that the `closure_kind` of the resulting closure is in
343                        // sync with the built-in trait implementations (since all of the
344                        // implementations return `FnOnce::Output`).
345                        if ty::ClosureKind::FnOnce == args.as_coroutine_closure().kind() {
346                            Some(Instance::new_raw(coroutine_closure_def_id, args))
347                        } else {
348                            Some(Instance {
349                                def: ty::InstanceKind::Shim(
350                                    ty::ShimKind::ConstructCoroutineInClosure {
351                                        coroutine_closure_def_id,
352                                        receiver_by_ref: target_kind != ty::ClosureKind::FnOnce,
353                                    },
354                                ),
355                                args,
356                            })
357                        }
358                    }
359                    _ => ::rustc_middle::util::bug::bug_fmt(format_args!("no built-in definition for `{1}::{0}` for non-fn type",
        tcx.item_name(trait_item_id), trait_ref))bug!(
360                        "no built-in definition for `{trait_ref}::{}` for non-fn type",
361                        tcx.item_name(trait_item_id)
362                    ),
363                }
364            } else if let Some(target_kind) = tcx.async_fn_trait_kind_from_def_id(trait_ref.def_id)
365            {
366                match *rcvr_args.type_at(0).kind() {
367                    ty::CoroutineClosure(coroutine_closure_def_id, args) => {
368                        if target_kind == ClosureKind::FnOnce
369                            && args.as_coroutine_closure().kind() != ClosureKind::FnOnce
370                        {
371                            // If we're computing `AsyncFnOnce` for a by-ref closure then
372                            // construct a new body that has the right return types.
373                            Some(Instance {
374                                def: ty::InstanceKind::Shim(
375                                    ty::ShimKind::ConstructCoroutineInClosure {
376                                        coroutine_closure_def_id,
377                                        receiver_by_ref: false,
378                                    },
379                                ),
380                                args,
381                            })
382                        } else {
383                            Some(Instance::new_raw(coroutine_closure_def_id, args))
384                        }
385                    }
386                    ty::Closure(closure_def_id, args) => {
387                        Some(Instance::resolve_closure(tcx, closure_def_id, args, target_kind))
388                    }
389                    ty::FnDef(..) | ty::FnPtr(..) => Some(Instance {
390                        def: ty::InstanceKind::Shim(ty::ShimKind::FnPtr(
391                            trait_item_id,
392                            rcvr_args.type_at(0),
393                        )),
394                        args: rcvr_args,
395                    }),
396                    _ => ::rustc_middle::util::bug::bug_fmt(format_args!("no built-in definition for `{1}::{0}` for non-lending-closure type",
        tcx.item_name(trait_item_id), trait_ref))bug!(
397                        "no built-in definition for `{trait_ref}::{}` for non-lending-closure type",
398                        tcx.item_name(trait_item_id)
399                    ),
400                }
401            } else if tcx.is_lang_item(trait_ref.def_id, LangItem::TransmuteTrait) {
402                let name = tcx.item_name(trait_item_id);
403                {
    match (&name, &sym::transmute) {
        (left_val, right_val) => {
            if !(*left_val == *right_val) {
                let kind = ::core::panicking::AssertKind::Eq;
                ::core::panicking::assert_failed(kind, &*left_val,
                    &*right_val, ::core::option::Option::None);
            }
        }
    }
};assert_eq!(name, sym::transmute);
404                let args = tcx.erase_and_anonymize_regions(rcvr_args);
405                Some(ty::Instance::new_raw(trait_item_id, args))
406            } else if tcx.is_lang_item(trait_ref.def_id, LangItem::Field) {
407                if tcx.is_lang_item(trait_item_id, LangItem::FieldOffset) {
408                    let self_ty = trait_ref.self_ty();
409                    match self_ty.kind() {
410                        ty::Adt(def, _) if def.is_field_representing_type() => {}
411                        _ => ::rustc_middle::util::bug::bug_fmt(format_args!("expected field representing type, found {0}",
        self_ty))bug!("expected field representing type, found {self_ty}"),
412                    }
413                    Some(Instance {
414                        def: ty::InstanceKind::Item(
415                            tcx.lang_items().get(LangItem::FieldOffset).unwrap(),
416                        ),
417                        args: rcvr_args,
418                    })
419                } else {
420                    ::rustc_middle::util::bug::bug_fmt(format_args!("unexpected associated associated item"))bug!("unexpected associated associated item")
421                }
422            } else {
423                Instance::try_resolve_item_for_coroutine(tcx, trait_item_id, trait_id, rcvr_args)
424            }
425        }
426        traits::ImplSource::Param(..)
427        | traits::ImplSource::Builtin(BuiltinImplSource::TraitUpcasting { .. }, _) => None,
428    })
429}
430
431pub(crate) fn provide(providers: &mut Providers) {
432    *providers = Providers { resolve_instance_raw, ..*providers };
433}