summaryrefslogtreecommitdiffstats
path: root/compiler/rustc_ty_utils/src/instance.rs
diff options
context:
space:
mode:
Diffstat (limited to 'compiler/rustc_ty_utils/src/instance.rs')
-rw-r--r--compiler/rustc_ty_utils/src/instance.rs20
1 files changed, 19 insertions, 1 deletions
diff --git a/compiler/rustc_ty_utils/src/instance.rs b/compiler/rustc_ty_utils/src/instance.rs
index 2eaeca73d..0a6c11809 100644
--- a/compiler/rustc_ty_utils/src/instance.rs
+++ b/compiler/rustc_ty_utils/src/instance.rs
@@ -8,6 +8,8 @@ use rustc_span::sym;
use rustc_trait_selection::traits;
use traits::{translate_substs, Reveal};
+use crate::errors::UnexpectedFnPtrAssociatedItem;
+
fn resolve_instance<'tcx>(
tcx: TyCtxt<'tcx>,
key: ty::ParamEnvAnd<'tcx, (DefId, SubstsRef<'tcx>)>,
@@ -243,7 +245,8 @@ fn resolve_associated_item<'tcx>(
}
}
traits::ImplSource::Builtin(..) => {
- if Some(trait_ref.def_id) == tcx.lang_items().clone_trait() {
+ let lang_items = tcx.lang_items();
+ if Some(trait_ref.def_id) == lang_items.clone_trait() {
// FIXME(eddyb) use lang items for methods instead of names.
let name = tcx.item_name(trait_item_id);
if name == sym::clone {
@@ -270,6 +273,21 @@ fn resolve_associated_item<'tcx>(
let substs = tcx.erase_regions(rcvr_substs);
Some(ty::Instance::new(trait_item_id, substs))
}
+ } else if Some(trait_ref.def_id) == lang_items.fn_ptr_trait() {
+ if lang_items.fn_ptr_addr() == Some(trait_item_id) {
+ let self_ty = trait_ref.self_ty();
+ if !matches!(self_ty.kind(), ty::FnPtr(..)) {
+ return Ok(None);
+ }
+ Some(Instance {
+ def: ty::InstanceDef::FnPtrAddrShim(trait_item_id, self_ty),
+ substs: rcvr_substs,
+ })
+ } else {
+ tcx.sess.emit_fatal(UnexpectedFnPtrAssociatedItem {
+ span: tcx.def_span(trait_item_id),
+ })
+ }
} else {
None
}