// Copyright 2014 The Rust Project Developers. See the COPYRIGHT // file at the top-level directory of this distribution and at // http://rust-lang.org/COPYRIGHT. // // Licensed under the Apache License, Version 2.0 or the MIT license // , at your // option. This file may not be copied, modified, or distributed // except according to those terms. /*! See `doc.rs` for high-level documentation */ use super::SelectionContext; use super::Obligation; use super::util; use middle::subst; use middle::subst::Subst; use middle::ty; use middle::typeck::infer::InferCtxt; use syntax::ast; use syntax::codemap::DUMMY_SP; use util::ppaux::Repr; pub fn impl_can_satisfy(infcx: &InferCtxt, impl1_def_id: ast::DefId, impl2_def_id: ast::DefId) -> bool { debug!("impl_can_satisfy(\ impl1_def_id={}, \ impl2_def_id={})", impl1_def_id.repr(infcx.tcx), impl2_def_id.repr(infcx.tcx)); // `impl1` provides an implementation of `Foo for Z`. let impl1_substs = util::fresh_substs_for_impl(infcx, DUMMY_SP, impl1_def_id); let impl1_trait_ref = ty::impl_trait_ref(infcx.tcx, impl1_def_id).unwrap() .subst(infcx.tcx, &impl1_substs); // Determine whether `impl2` can provide an implementation for those // same types. let param_env = ty::empty_parameter_environment(); let mut selcx = SelectionContext::intercrate(infcx, ¶m_env, infcx.tcx); let obligation = Obligation::misc(DUMMY_SP, impl1_trait_ref); debug!("impl_can_satisfy obligation={}", obligation.repr(infcx.tcx)); selcx.evaluate_impl(impl2_def_id, &obligation) } pub fn impl_is_local(tcx: &ty::ctxt, impl_def_id: ast::DefId) -> bool { debug!("impl_is_local({})", impl_def_id.repr(tcx)); // We only except this routine to be invoked on implementations // of a trait, not inherent implementations. let trait_ref = ty::impl_trait_ref(tcx, impl_def_id).unwrap(); debug!("trait_ref={}", trait_ref.repr(tcx)); // If the trait is local to the crate, ok. if trait_ref.def_id.krate == ast::LOCAL_CRATE { debug!("trait {} is local to current crate", trait_ref.def_id.repr(tcx)); return true; } // Otherwise, at least one of the input types must be local to the // crate. trait_ref.input_types().iter().any(|&t| ty_is_local(tcx, t)) } pub fn ty_is_local(tcx: &ty::ctxt, ty: ty::t) -> bool { debug!("ty_is_local({})", ty.repr(tcx)); match ty::get(ty).sty { ty::ty_bool | ty::ty_char | ty::ty_int(..) | ty::ty_uint(..) | ty::ty_float(..) | ty::ty_str(..) => { false } ty::ty_unboxed_closure(..) => { // This routine is invoked on types specified by users as // part of an impl and hence an unboxed closure type // cannot appear. tcx.sess.bug("ty_is_local applied to unboxed closure type") } ty::ty_bare_fn(..) | ty::ty_closure(..) => { false } ty::ty_uniq(t) => { let krate = tcx.lang_items.owned_box().map(|d| d.krate); krate == Some(ast::LOCAL_CRATE) || ty_is_local(tcx, t) } ty::ty_vec(t, _) | ty::ty_ptr(ty::mt { ty: t, .. }) | ty::ty_rptr(_, ty::mt { ty: t, .. }) => { ty_is_local(tcx, t) } ty::ty_tup(ref ts) => { ts.iter().any(|&t| ty_is_local(tcx, t)) } ty::ty_enum(def_id, ref substs) | ty::ty_struct(def_id, ref substs) => { def_id.krate == ast::LOCAL_CRATE || { let variances = ty::item_variances(tcx, def_id); subst::ParamSpace::all().iter().any(|&space| { substs.types.get_slice(space).iter().enumerate().any( |(i, &t)| { match *variances.types.get(space, i) { ty::Bivariant => { // If Foo is bivariant with respect to // T, then it doesn't matter whether T is // local or not, because `Foo` for any // U will be a subtype of T. false } ty::Contravariant | ty::Covariant | ty::Invariant => { ty_is_local(tcx, t) } } }) }) } } ty::ty_trait(ref tt) => { tt.principal.def_id.krate == ast::LOCAL_CRATE } // Type parameters may be bound to types that are not local to // the crate. ty::ty_param(..) => { false } ty::ty_infer(..) | ty::ty_open(..) | ty::ty_err => { tcx.sess.bug( format!("ty_is_local invoked on unexpected type: {}", ty.repr(tcx)).as_slice()) } } }