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 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 _ => 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 _ => 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 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 let eligible = if leaf_def.is_final() {
152 true
154 } else {
155 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 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 !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 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 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 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 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 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 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 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 if truecfg!(debug_assertions)
316 && ![sym::call, sym::call_mut, sym::call_once]
317 .contains(&tcx.item_name(trait_item_id))
318 {
319 ::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 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 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}