// 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. use middle::subst; use middle::subst::{ParamSpace, Substs, VecPerParamSpace, Subst}; use middle::infer::InferCtxt; use middle::ty::{mod, Ty}; use std::collections::HashSet; use std::fmt; use std::rc::Rc; use syntax::ast; use syntax::codemap::Span; use util::common::ErrorReported; use util::ppaux::Repr; use super::{Obligation, ObligationCause, TraitObligation, VtableImpl, VtableParam, VtableParamData, VtableImplData}; /////////////////////////////////////////////////////////////////////////// // Supertrait iterator pub struct Supertraits<'cx, 'tcx:'cx> { tcx: &'cx ty::ctxt<'tcx>, stack: Vec>, visited: HashSet>>, } struct SupertraitEntry<'tcx> { position: uint, supertraits: Vec>>, } pub fn supertraits<'cx, 'tcx>(tcx: &'cx ty::ctxt<'tcx>, trait_ref: Rc>) -> Supertraits<'cx, 'tcx> { //! Returns an iterator over the trait reference `T` and all of its supertrait references. May //! contain duplicates. In general the ordering is not defined. //! //! Example: //! //! ``` //! trait Foo { ... } //! trait Bar : Foo { ... } //! trait Baz : Bar+Foo { ... } //! ``` //! //! `supertraits(Baz)` yields `[Baz, Bar, Foo, Foo]` in some order. transitive_bounds(tcx, &[trait_ref]) } pub fn transitive_bounds<'cx, 'tcx>(tcx: &'cx ty::ctxt<'tcx>, bounds: &[Rc>]) -> Supertraits<'cx, 'tcx> { let bounds = Vec::from_fn(bounds.len(), |i| bounds[i].clone()); let visited: HashSet> = bounds.iter() .map(|b| (*b).clone()) .collect(); let entry = SupertraitEntry { position: 0, supertraits: bounds }; Supertraits { tcx: tcx, stack: vec![entry], visited: visited } } impl<'cx, 'tcx> Supertraits<'cx, 'tcx> { fn push(&mut self, trait_ref: &ty::TraitRef<'tcx>) { let ty::ParamBounds { builtin_bounds, mut trait_bounds, .. } = ty::bounds_for_trait_ref(self.tcx, trait_ref); for builtin_bound in builtin_bounds.iter() { let bound_trait_ref = trait_ref_for_builtin_bound(self.tcx, builtin_bound, trait_ref.self_ty()); bound_trait_ref.map(|trait_ref| trait_bounds.push(trait_ref)); } // Only keep those bounds that we haven't already seen. This // is necessary to prevent infinite recursion in some cases. // One common case is when people define `trait Sized { }` // rather than `trait Sized for Sized? { }`. trait_bounds.retain(|r| self.visited.insert((*r).clone())); let entry = SupertraitEntry { position: 0, supertraits: trait_bounds }; self.stack.push(entry); } /// Returns the path taken through the trait supertraits to reach the current point. pub fn indices(&self) -> Vec { self.stack.iter().map(|e| e.position).collect() } } impl<'cx, 'tcx> Iterator>> for Supertraits<'cx, 'tcx> { fn next(&mut self) -> Option>> { loop { // Extract next item from top-most stack frame, if any. let next_trait = match self.stack.last_mut() { None => { // No more stack frames. Done. return None; } Some(entry) => { let p = entry.position; if p < entry.supertraits.len() { // Still more supertraits left in the top stack frame. entry.position += 1; let next_trait = entry.supertraits[p].clone(); Some(next_trait) } else { None } } }; match next_trait { Some(next_trait) => { self.push(&*next_trait); return Some(next_trait); } None => { // Top stack frame is exhausted, pop it. self.stack.pop(); } } } } } // determine the `self` type, using fresh variables for all variables // declared on the impl declaration e.g., `impl for Box<[(A,B)]>` // would return ($0, $1) where $0 and $1 are freshly instantiated type // variables. pub fn fresh_substs_for_impl<'a, 'tcx>(infcx: &InferCtxt<'a, 'tcx>, span: Span, impl_def_id: ast::DefId) -> Substs<'tcx> { let tcx = infcx.tcx; let impl_generics = ty::lookup_item_type(tcx, impl_def_id).generics; let input_substs = infcx.fresh_substs_for_generics(span, &impl_generics); // Add substs for the associated types bound in the impl. let ref items = tcx.impl_items.borrow()[impl_def_id]; let mut assoc_tys = Vec::new(); for item in items.iter() { if let &ty::ImplOrTraitItemId::TypeTraitItemId(id) = item { assoc_tys.push(tcx.tcache.borrow()[id].ty.subst(tcx, &input_substs)); } } input_substs.with_assoc_tys(assoc_tys) } impl<'tcx, N> fmt::Show for VtableImplData<'tcx, N> { fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { write!(f, "VtableImpl({})", self.impl_def_id) } } impl<'tcx> fmt::Show for VtableParamData<'tcx> { fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { write!(f, "VtableParam(...)") } } /// See `super::obligations_for_generics` pub fn obligations_for_generics<'tcx>(tcx: &ty::ctxt<'tcx>, cause: ObligationCause<'tcx>, recursion_depth: uint, generic_bounds: &ty::GenericBounds<'tcx>, type_substs: &VecPerParamSpace>) -> VecPerParamSpace> { debug!("obligations_for_generics(generic_bounds={}, type_substs={})", generic_bounds.repr(tcx), type_substs.repr(tcx)); let mut obligations = VecPerParamSpace::empty(); for (space, index, bounds) in generic_bounds.types.iter_enumerated() { push_obligations_for_param_bounds(tcx, cause, recursion_depth, space, index, bounds, type_substs, &mut obligations); } debug!("obligations() ==> {}", obligations.repr(tcx)); return obligations; } fn push_obligations_for_param_bounds<'tcx>( tcx: &ty::ctxt<'tcx>, cause: ObligationCause<'tcx>, recursion_depth: uint, space: subst::ParamSpace, index: uint, param_bounds: &ty::ParamBounds<'tcx>, param_type_substs: &VecPerParamSpace>, obligations: &mut VecPerParamSpace>) { let param_ty = *param_type_substs.get(space, index); for builtin_bound in param_bounds.builtin_bounds.iter() { let obligation = obligation_for_builtin_bound(tcx, cause, builtin_bound, recursion_depth, param_ty); if let Ok(ob) = obligation { obligations.push(space, ob); } } for bound_trait_ref in param_bounds.trait_bounds.iter() { obligations.push( space, Obligation { cause: cause, recursion_depth: recursion_depth, trait_ref: (*bound_trait_ref).clone() }); } } pub fn trait_ref_for_builtin_bound<'tcx>( tcx: &ty::ctxt<'tcx>, builtin_bound: ty::BuiltinBound, param_ty: Ty<'tcx>) -> Option>> { match tcx.lang_items.from_builtin_kind(builtin_bound) { Ok(def_id) => { Some(Rc::new(ty::TraitRef { def_id: def_id, substs: Substs::empty().with_self_ty(param_ty) })) } Err(e) => { tcx.sess.err(e.as_slice()); None } } } pub fn obligation_for_builtin_bound<'tcx>( tcx: &ty::ctxt<'tcx>, cause: ObligationCause<'tcx>, builtin_bound: ty::BuiltinBound, recursion_depth: uint, param_ty: Ty<'tcx>) -> Result, ErrorReported> { let trait_ref = trait_ref_for_builtin_bound(tcx, builtin_bound, param_ty); match trait_ref { Some(trait_ref) => Ok(Obligation { cause: cause, recursion_depth: recursion_depth, trait_ref: trait_ref }), None => Err(ErrorReported) } } /// Starting from a caller obligation `caller_bound` (which has coordinates `space`/`i` in the list /// of caller obligations), search through the trait and supertraits to find one where `test(d)` is /// true, where `d` is the def-id of the trait/supertrait. If any is found, return `Some(p)` where /// `p` is the path to that trait/supertrait. Else `None`. pub fn search_trait_and_supertraits_from_bound<'tcx>(tcx: &ty::ctxt<'tcx>, caller_bound: Rc>, test: |ast::DefId| -> bool) -> Option> { for bound in transitive_bounds(tcx, &[caller_bound]) { if test(bound.def_id) { let vtable_param = VtableParamData { bound: bound }; return Some(vtable_param); } } return None; } impl<'tcx,O:Repr<'tcx>> Repr<'tcx> for super::Obligation<'tcx, O> { fn repr(&self, tcx: &ty::ctxt<'tcx>) -> String { format!("Obligation(trait_ref={},depth={})", self.trait_ref.repr(tcx), self.recursion_depth) } } impl<'tcx, N:Repr<'tcx>> Repr<'tcx> for super::Vtable<'tcx, N> { fn repr(&self, tcx: &ty::ctxt<'tcx>) -> String { match *self { super::VtableImpl(ref v) => v.repr(tcx), super::VtableUnboxedClosure(ref d, ref s) => format!("VtableUnboxedClosure({},{})", d.repr(tcx), s.repr(tcx)), super::VtableFnPointer(ref d) => format!("VtableFnPointer({})", d.repr(tcx)), super::VtableParam(ref v) => format!("VtableParam({})", v.repr(tcx)), super::VtableBuiltin(ref d) => d.repr(tcx) } } } impl<'tcx, N:Repr<'tcx>> Repr<'tcx> for super::VtableImplData<'tcx, N> { fn repr(&self, tcx: &ty::ctxt<'tcx>) -> String { format!("VtableImpl(impl_def_id={}, substs={}, nested={})", self.impl_def_id.repr(tcx), self.substs.repr(tcx), self.nested.repr(tcx)) } } impl<'tcx, N:Repr<'tcx>> Repr<'tcx> for super::VtableBuiltinData { fn repr(&self, tcx: &ty::ctxt<'tcx>) -> String { format!("VtableBuiltin(nested={})", self.nested.repr(tcx)) } } impl<'tcx> Repr<'tcx> for super::VtableParamData<'tcx> { fn repr(&self, tcx: &ty::ctxt<'tcx>) -> String { format!("VtableParam(bound={})", self.bound.repr(tcx)) } } impl<'tcx> Repr<'tcx> for super::SelectionError<'tcx> { fn repr(&self, tcx: &ty::ctxt<'tcx>) -> String { match *self { super::Overflow => format!("Overflow"), super::Unimplemented => format!("Unimplemented"), super::OutputTypeParameterMismatch(ref t, ref e) => format!("OutputTypeParameterMismatch({}, {})", t.repr(tcx), e.repr(tcx)), } } } impl<'tcx> Repr<'tcx> for super::FulfillmentError<'tcx> { fn repr(&self, tcx: &ty::ctxt<'tcx>) -> String { format!("FulfillmentError({},{})", self.obligation.repr(tcx), self.code.repr(tcx)) } } impl<'tcx> Repr<'tcx> for super::FulfillmentErrorCode<'tcx> { fn repr(&self, tcx: &ty::ctxt<'tcx>) -> String { match *self { super::CodeSelectionError(ref o) => o.repr(tcx), super::CodeAmbiguity => format!("Ambiguity") } } } impl<'tcx> fmt::Show for super::FulfillmentErrorCode<'tcx> { fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { match *self { super::CodeSelectionError(ref e) => write!(f, "{}", e), super::CodeAmbiguity => write!(f, "Ambiguity") } } } impl<'tcx> Repr<'tcx> for ty::type_err<'tcx> { fn repr(&self, tcx: &ty::ctxt<'tcx>) -> String { ty::type_err_to_str(tcx, self) } }