2014-12-17 14:16:28 -05:00
|
|
|
// 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 <LICENSE-APACHE or
|
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
|
|
|
|
|
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
|
|
|
|
|
// option. This file may not be copied, modified, or distributed
|
|
|
|
|
// except according to those terms.
|
|
|
|
|
|
|
|
|
|
//! Code for projecting associated types out of trait references.
|
|
|
|
|
|
|
|
|
|
use super::elaborate_predicates;
|
2016-02-16 10:36:47 -08:00
|
|
|
use super::specialization_graph;
|
|
|
|
|
use super::translate_substs;
|
2014-12-17 14:16:28 -05:00
|
|
|
use super::Obligation;
|
2014-12-30 17:42:02 -05:00
|
|
|
use super::ObligationCause;
|
2014-12-17 14:16:28 -05:00
|
|
|
use super::PredicateObligation;
|
|
|
|
|
use super::SelectionContext;
|
|
|
|
|
use super::SelectionError;
|
2015-06-10 00:09:37 +03:00
|
|
|
use super::VtableClosureData;
|
2014-12-17 14:16:28 -05:00
|
|
|
use super::VtableImplData;
|
2015-01-10 11:54:15 -05:00
|
|
|
use super::util;
|
2014-12-17 14:16:28 -05:00
|
|
|
|
2016-03-29 12:54:26 +03:00
|
|
|
use hir::def_id::DefId;
|
2016-03-29 20:06:42 -07:00
|
|
|
use infer::{self, InferOk, TypeOrigin};
|
2016-03-22 17:30:57 +02:00
|
|
|
use ty::subst::Subst;
|
|
|
|
|
use ty::{self, ToPredicate, ToPolyTraitRef, Ty, TyCtxt};
|
|
|
|
|
use ty::fold::{TypeFoldable, TypeFolder};
|
2015-01-10 11:54:15 -05:00
|
|
|
use syntax::parse::token;
|
2016-02-16 10:36:47 -08:00
|
|
|
use syntax::ast;
|
2015-01-10 11:54:15 -05:00
|
|
|
use util::common::FN_OUTPUT_NAME;
|
2014-12-17 14:16:28 -05:00
|
|
|
|
2016-02-16 10:36:47 -08:00
|
|
|
use std::rc::Rc;
|
|
|
|
|
|
2016-02-23 12:47:09 -08:00
|
|
|
/// Depending on the stage of compilation, we want projection to be
|
|
|
|
|
/// more or less conservative.
|
|
|
|
|
#[derive(Debug, Copy, Clone, PartialEq, Eq)]
|
|
|
|
|
pub enum ProjectionMode {
|
2016-03-11 14:47:29 -08:00
|
|
|
/// FIXME (#32205)
|
2016-02-23 12:47:09 -08:00
|
|
|
/// At coherence-checking time, we're still constructing the
|
2016-05-05 21:11:41 +02:00
|
|
|
/// specialization graph, and thus we only project
|
2016-02-23 12:47:09 -08:00
|
|
|
/// non-`default` associated types that are defined directly in
|
|
|
|
|
/// the applicable impl. (This behavior should be improved over
|
|
|
|
|
/// time, to allow for successful projections modulo cycles
|
|
|
|
|
/// between different impls).
|
|
|
|
|
///
|
|
|
|
|
/// Here's an example that will fail due to the restriction:
|
|
|
|
|
///
|
|
|
|
|
/// ```
|
|
|
|
|
/// trait Assoc {
|
|
|
|
|
/// type Output;
|
|
|
|
|
/// }
|
|
|
|
|
///
|
|
|
|
|
/// impl<T> Assoc for T {
|
|
|
|
|
/// type Output = bool;
|
|
|
|
|
/// }
|
|
|
|
|
///
|
|
|
|
|
/// impl Assoc for u8 {} // <- inherits the non-default type from above
|
|
|
|
|
///
|
|
|
|
|
/// trait Foo {}
|
|
|
|
|
/// impl Foo for u32 {}
|
|
|
|
|
/// impl Foo for <u8 as Assoc>::Output {} // <- this projection will fail
|
|
|
|
|
/// ```
|
|
|
|
|
///
|
|
|
|
|
/// The projection would succeed if `Output` had been defined
|
|
|
|
|
/// directly in the impl for `u8`.
|
|
|
|
|
Topmost,
|
|
|
|
|
|
|
|
|
|
/// At type-checking time, we refuse to project any associated
|
|
|
|
|
/// type that is marked `default`. Non-`default` ("final") types
|
|
|
|
|
/// are always projected. This is necessary in general for
|
|
|
|
|
/// soundness of specialization. However, we *could* allow
|
|
|
|
|
/// projections in fully-monomorphic cases. We choose not to,
|
|
|
|
|
/// because we prefer for `default type` to force the type
|
|
|
|
|
/// definition to be treated abstractly by any consumers of the
|
|
|
|
|
/// impl. Concretely, that means that the following example will
|
|
|
|
|
/// fail to compile:
|
|
|
|
|
///
|
|
|
|
|
/// ```
|
|
|
|
|
/// trait Assoc {
|
|
|
|
|
/// type Output;
|
|
|
|
|
/// }
|
|
|
|
|
///
|
|
|
|
|
/// impl<T> Assoc for T {
|
|
|
|
|
/// default type Output = bool;
|
|
|
|
|
/// }
|
|
|
|
|
///
|
|
|
|
|
/// fn main() {
|
|
|
|
|
/// let <() as Assoc>::Output = true;
|
|
|
|
|
/// }
|
|
|
|
|
AnyFinal,
|
|
|
|
|
|
|
|
|
|
/// At trans time, all projections will succeed.
|
|
|
|
|
Any,
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
impl ProjectionMode {
|
2016-03-08 15:23:52 -08:00
|
|
|
pub fn is_topmost(&self) -> bool {
|
2016-02-23 12:47:09 -08:00
|
|
|
match *self {
|
|
|
|
|
ProjectionMode::Topmost => true,
|
|
|
|
|
_ => false,
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2016-03-08 15:23:52 -08:00
|
|
|
pub fn is_any_final(&self) -> bool {
|
2016-02-23 12:47:09 -08:00
|
|
|
match *self {
|
|
|
|
|
ProjectionMode::AnyFinal => true,
|
|
|
|
|
_ => false,
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2016-03-08 15:23:52 -08:00
|
|
|
pub fn is_any(&self) -> bool {
|
2016-02-23 12:47:09 -08:00
|
|
|
match *self {
|
|
|
|
|
ProjectionMode::Any => true,
|
|
|
|
|
_ => false,
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2014-12-17 14:16:28 -05:00
|
|
|
pub type PolyProjectionObligation<'tcx> =
|
|
|
|
|
Obligation<'tcx, ty::PolyProjectionPredicate<'tcx>>;
|
|
|
|
|
|
|
|
|
|
pub type ProjectionObligation<'tcx> =
|
|
|
|
|
Obligation<'tcx, ty::ProjectionPredicate<'tcx>>;
|
|
|
|
|
|
|
|
|
|
pub type ProjectionTyObligation<'tcx> =
|
|
|
|
|
Obligation<'tcx, ty::ProjectionTy<'tcx>>;
|
|
|
|
|
|
2014-12-30 17:42:02 -05:00
|
|
|
/// When attempting to resolve `<T as TraitRef>::Name` ...
|
2015-06-18 08:51:23 +03:00
|
|
|
#[derive(Debug)]
|
2014-12-30 17:42:02 -05:00
|
|
|
pub enum ProjectionTyError<'tcx> {
|
2014-12-30 08:59:33 -05:00
|
|
|
/// ...we found multiple sources of information and couldn't resolve the ambiguity.
|
2014-12-17 14:16:28 -05:00
|
|
|
TooManyCandidates,
|
|
|
|
|
|
|
|
|
|
/// ...an error occurred matching `T : TraitRef`
|
|
|
|
|
TraitSelectionError(SelectionError<'tcx>),
|
|
|
|
|
}
|
|
|
|
|
|
2015-01-03 22:54:18 -05:00
|
|
|
#[derive(Clone)]
|
2014-12-17 14:16:28 -05:00
|
|
|
pub struct MismatchedProjectionTypes<'tcx> {
|
2015-09-06 21:51:58 +03:00
|
|
|
pub err: ty::error::TypeError<'tcx>
|
2014-12-17 14:16:28 -05:00
|
|
|
}
|
|
|
|
|
|
2015-06-18 08:51:23 +03:00
|
|
|
#[derive(PartialEq, Eq, Debug)]
|
2014-12-17 14:16:28 -05:00
|
|
|
enum ProjectionTyCandidate<'tcx> {
|
2015-10-22 12:28:47 -04:00
|
|
|
// from a where-clause in the env or object type
|
2014-12-17 14:16:28 -05:00
|
|
|
ParamEnv(ty::PolyProjectionPredicate<'tcx>),
|
2015-10-22 12:28:47 -04:00
|
|
|
|
|
|
|
|
// from the definition of `Trait` when you have something like <<A as Trait>::B as Trait2>::C
|
|
|
|
|
TraitDef(ty::PolyProjectionPredicate<'tcx>),
|
|
|
|
|
|
|
|
|
|
// defined in an impl
|
2014-12-17 14:16:28 -05:00
|
|
|
Impl(VtableImplData<'tcx, PredicateObligation<'tcx>>),
|
2015-10-22 12:28:47 -04:00
|
|
|
|
|
|
|
|
// closure return type
|
2015-06-10 00:09:37 +03:00
|
|
|
Closure(VtableClosureData<'tcx, PredicateObligation<'tcx>>),
|
2015-10-22 12:28:47 -04:00
|
|
|
|
|
|
|
|
// fn pointer return type
|
2015-01-10 11:54:15 -05:00
|
|
|
FnPointer(Ty<'tcx>),
|
2014-12-17 14:16:28 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
struct ProjectionTyCandidateSet<'tcx> {
|
|
|
|
|
vec: Vec<ProjectionTyCandidate<'tcx>>,
|
|
|
|
|
ambiguous: bool
|
|
|
|
|
}
|
|
|
|
|
|
2015-01-26 14:20:38 -05:00
|
|
|
/// Evaluates constraints of the form:
|
|
|
|
|
///
|
|
|
|
|
/// for<...> <T as Trait>::U == V
|
|
|
|
|
///
|
|
|
|
|
/// If successful, this may result in additional obligations.
|
2016-04-29 06:00:23 +03:00
|
|
|
pub fn poly_project_and_unify_type<'cx, 'gcx, 'tcx>(
|
|
|
|
|
selcx: &mut SelectionContext<'cx, 'gcx, 'tcx>,
|
2014-12-17 14:16:28 -05:00
|
|
|
obligation: &PolyProjectionObligation<'tcx>)
|
2014-12-30 17:42:02 -05:00
|
|
|
-> Result<Option<Vec<PredicateObligation<'tcx>>>, MismatchedProjectionTypes<'tcx>>
|
2014-12-17 14:16:28 -05:00
|
|
|
{
|
2015-06-18 20:25:05 +03:00
|
|
|
debug!("poly_project_and_unify_type(obligation={:?})",
|
|
|
|
|
obligation);
|
2014-12-17 14:16:28 -05:00
|
|
|
|
|
|
|
|
let infcx = selcx.infcx();
|
2015-03-10 07:02:27 -04:00
|
|
|
infcx.commit_if_ok(|snapshot| {
|
2014-12-17 14:16:28 -05:00
|
|
|
let (skol_predicate, skol_map) =
|
|
|
|
|
infcx.skolemize_late_bound_regions(&obligation.predicate, snapshot);
|
|
|
|
|
|
|
|
|
|
let skol_obligation = obligation.with(skol_predicate);
|
2014-12-30 17:42:02 -05:00
|
|
|
match project_and_unify_type(selcx, &skol_obligation) {
|
2015-02-03 06:12:43 -05:00
|
|
|
Ok(result) => {
|
2016-03-17 00:15:31 +02:00
|
|
|
match infcx.leak_check(false, &skol_map, snapshot) {
|
2015-02-03 06:12:43 -05:00
|
|
|
Ok(()) => Ok(infcx.plug_leaks(skol_map, snapshot, &result)),
|
|
|
|
|
Err(e) => Err(MismatchedProjectionTypes { err: e }),
|
2014-12-30 17:42:02 -05:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
Err(e) => {
|
2015-02-03 06:12:43 -05:00
|
|
|
Err(e)
|
2014-12-30 17:42:02 -05:00
|
|
|
}
|
2014-12-17 14:16:28 -05:00
|
|
|
}
|
2015-02-03 06:12:43 -05:00
|
|
|
})
|
2014-12-17 14:16:28 -05:00
|
|
|
}
|
|
|
|
|
|
2015-01-26 14:20:38 -05:00
|
|
|
/// Evaluates constraints of the form:
|
|
|
|
|
///
|
|
|
|
|
/// <T as Trait>::U == V
|
|
|
|
|
///
|
|
|
|
|
/// If successful, this may result in additional obligations.
|
2016-04-29 06:00:23 +03:00
|
|
|
fn project_and_unify_type<'cx, 'gcx, 'tcx>(
|
|
|
|
|
selcx: &mut SelectionContext<'cx, 'gcx, 'tcx>,
|
2014-12-17 14:16:28 -05:00
|
|
|
obligation: &ProjectionObligation<'tcx>)
|
2014-12-30 17:42:02 -05:00
|
|
|
-> Result<Option<Vec<PredicateObligation<'tcx>>>, MismatchedProjectionTypes<'tcx>>
|
2014-12-17 14:16:28 -05:00
|
|
|
{
|
2015-06-18 20:25:05 +03:00
|
|
|
debug!("project_and_unify_type(obligation={:?})",
|
|
|
|
|
obligation);
|
2014-12-17 14:16:28 -05:00
|
|
|
|
2014-12-30 17:42:02 -05:00
|
|
|
let Normalized { value: normalized_ty, obligations } =
|
|
|
|
|
match opt_normalize_projection_type(selcx,
|
|
|
|
|
obligation.predicate.projection_ty.clone(),
|
|
|
|
|
obligation.cause.clone(),
|
|
|
|
|
obligation.recursion_depth) {
|
|
|
|
|
Some(n) => n,
|
2015-02-03 06:12:43 -05:00
|
|
|
None => {
|
|
|
|
|
consider_unification_despite_ambiguity(selcx, obligation);
|
|
|
|
|
return Ok(None);
|
|
|
|
|
}
|
2014-12-30 17:42:02 -05:00
|
|
|
};
|
|
|
|
|
|
2015-06-18 20:25:05 +03:00
|
|
|
debug!("project_and_unify_type: normalized_ty={:?} obligations={:?}",
|
|
|
|
|
normalized_ty,
|
|
|
|
|
obligations);
|
2014-12-30 17:42:02 -05:00
|
|
|
|
2014-12-17 14:16:28 -05:00
|
|
|
let infcx = selcx.infcx();
|
2015-11-25 12:41:09 +01:00
|
|
|
let origin = TypeOrigin::RelateOutputImplTypes(obligation.cause.span);
|
2016-03-17 00:15:31 +02:00
|
|
|
match infcx.eq_types(true, origin, normalized_ty, obligation.predicate.ty) {
|
2016-03-29 20:06:42 -07:00
|
|
|
Ok(InferOk { obligations: inferred_obligations, .. }) => {
|
2016-04-04 12:41:05 -07:00
|
|
|
// FIXME(#32730) propagate obligations
|
2016-03-29 20:06:42 -07:00
|
|
|
assert!(inferred_obligations.is_empty());
|
|
|
|
|
Ok(Some(obligations))
|
|
|
|
|
},
|
2014-12-30 17:42:02 -05:00
|
|
|
Err(err) => Err(MismatchedProjectionTypes { err: err }),
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2016-04-29 06:00:23 +03:00
|
|
|
fn consider_unification_despite_ambiguity<'cx, 'gcx, 'tcx>(
|
|
|
|
|
selcx: &mut SelectionContext<'cx, 'gcx, 'tcx>,
|
|
|
|
|
obligation: &ProjectionObligation<'tcx>)
|
|
|
|
|
{
|
2015-06-18 20:25:05 +03:00
|
|
|
debug!("consider_unification_despite_ambiguity(obligation={:?})",
|
|
|
|
|
obligation);
|
2015-02-03 06:12:43 -05:00
|
|
|
|
|
|
|
|
let def_id = obligation.predicate.projection_ty.trait_ref.def_id;
|
|
|
|
|
match selcx.tcx().lang_items.fn_trait_kind(def_id) {
|
|
|
|
|
Some(_) => { }
|
|
|
|
|
None => { return; }
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
let infcx = selcx.infcx();
|
|
|
|
|
let self_ty = obligation.predicate.projection_ty.trait_ref.self_ty();
|
|
|
|
|
let self_ty = infcx.shallow_resolve(self_ty);
|
|
|
|
|
debug!("consider_unification_despite_ambiguity: self_ty.sty={:?}",
|
|
|
|
|
self_ty.sty);
|
|
|
|
|
match self_ty.sty {
|
2016-04-29 08:30:54 +03:00
|
|
|
ty::TyClosure(closure_def_id, substs) => {
|
2015-02-03 06:12:43 -05:00
|
|
|
let closure_typer = selcx.closure_typer();
|
|
|
|
|
let closure_type = closure_typer.closure_type(closure_def_id, substs);
|
|
|
|
|
let ty::Binder((_, ret_type)) =
|
2016-03-17 00:15:31 +02:00
|
|
|
infcx.tcx.closure_trait_ref_and_return_type(def_id,
|
|
|
|
|
self_ty,
|
|
|
|
|
&closure_type.sig,
|
|
|
|
|
util::TupleArgumentsFlag::No);
|
2015-06-10 00:09:37 +03:00
|
|
|
// We don't have to normalize the return type here - this is only
|
|
|
|
|
// reached for TyClosure: Fn inputs where the closure kind is
|
|
|
|
|
// still unknown, which should only occur in typeck where the
|
|
|
|
|
// closure type is already normalized.
|
2015-02-03 06:12:43 -05:00
|
|
|
let (ret_type, _) =
|
|
|
|
|
infcx.replace_late_bound_regions_with_fresh_var(
|
|
|
|
|
obligation.cause.span,
|
|
|
|
|
infer::AssocTypeProjection(obligation.predicate.projection_ty.item_name),
|
|
|
|
|
&ty::Binder(ret_type));
|
2015-06-10 00:09:37 +03:00
|
|
|
|
2015-02-03 06:12:43 -05:00
|
|
|
debug!("consider_unification_despite_ambiguity: ret_type={:?}",
|
2015-06-18 20:25:05 +03:00
|
|
|
ret_type);
|
2015-11-25 12:41:09 +01:00
|
|
|
let origin = TypeOrigin::RelateOutputImplTypes(obligation.cause.span);
|
2015-02-03 06:12:43 -05:00
|
|
|
let obligation_ty = obligation.predicate.ty;
|
2016-03-17 00:15:31 +02:00
|
|
|
match infcx.eq_types(true, origin, obligation_ty, ret_type) {
|
2016-03-29 20:06:42 -07:00
|
|
|
Ok(InferOk { obligations, .. }) => {
|
2016-04-04 12:41:05 -07:00
|
|
|
// FIXME(#32730) propagate obligations
|
2016-03-29 20:06:42 -07:00
|
|
|
assert!(obligations.is_empty());
|
|
|
|
|
}
|
2015-02-03 06:12:43 -05:00
|
|
|
Err(_) => { /* ignore errors */ }
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
_ => { }
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2015-01-26 14:20:38 -05:00
|
|
|
/// Normalizes any associated type projections in `value`, replacing
|
|
|
|
|
/// them with a fully resolved type where possible. The return value
|
|
|
|
|
/// combines the normalized result and any additional obligations that
|
|
|
|
|
/// were incurred as result.
|
2016-04-29 06:00:23 +03:00
|
|
|
pub fn normalize<'a, 'b, 'gcx, 'tcx, T>(selcx: &'a mut SelectionContext<'b, 'gcx, 'tcx>,
|
|
|
|
|
cause: ObligationCause<'tcx>,
|
|
|
|
|
value: &T)
|
|
|
|
|
-> Normalized<'tcx, T>
|
2015-11-18 09:38:57 +00:00
|
|
|
where T : TypeFoldable<'tcx>
|
2014-12-30 17:42:02 -05:00
|
|
|
{
|
2015-01-01 13:15:14 -05:00
|
|
|
normalize_with_depth(selcx, cause, 0, value)
|
|
|
|
|
}
|
|
|
|
|
|
2015-01-26 14:20:38 -05:00
|
|
|
/// As `normalize`, but with a custom depth.
|
2016-04-29 06:00:23 +03:00
|
|
|
pub fn normalize_with_depth<'a, 'b, 'gcx, 'tcx, T>(
|
|
|
|
|
selcx: &'a mut SelectionContext<'b, 'gcx, 'tcx>,
|
|
|
|
|
cause: ObligationCause<'tcx>,
|
|
|
|
|
depth: usize,
|
|
|
|
|
value: &T)
|
|
|
|
|
-> Normalized<'tcx, T>
|
|
|
|
|
|
2015-11-18 09:38:57 +00:00
|
|
|
where T : TypeFoldable<'tcx>
|
2015-01-01 13:15:14 -05:00
|
|
|
{
|
|
|
|
|
let mut normalizer = AssociatedTypeNormalizer::new(selcx, cause, depth);
|
2014-12-30 17:42:02 -05:00
|
|
|
let result = normalizer.fold(value);
|
2015-06-03 02:14:45 +03:00
|
|
|
|
2014-12-30 17:42:02 -05:00
|
|
|
Normalized {
|
|
|
|
|
value: result,
|
|
|
|
|
obligations: normalizer.obligations,
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2016-04-29 06:00:23 +03:00
|
|
|
struct AssociatedTypeNormalizer<'a, 'b: 'a, 'gcx: 'b+'tcx, 'tcx: 'b> {
|
|
|
|
|
selcx: &'a mut SelectionContext<'b, 'gcx, 'tcx>,
|
2014-12-30 17:42:02 -05:00
|
|
|
cause: ObligationCause<'tcx>,
|
|
|
|
|
obligations: Vec<PredicateObligation<'tcx>>,
|
2015-03-25 17:06:52 -07:00
|
|
|
depth: usize,
|
2014-12-30 17:42:02 -05:00
|
|
|
}
|
|
|
|
|
|
2016-04-29 06:00:23 +03:00
|
|
|
impl<'a, 'b, 'gcx, 'tcx> AssociatedTypeNormalizer<'a, 'b, 'gcx, 'tcx> {
|
|
|
|
|
fn new(selcx: &'a mut SelectionContext<'b, 'gcx, 'tcx>,
|
2014-12-30 17:42:02 -05:00
|
|
|
cause: ObligationCause<'tcx>,
|
2015-03-25 17:06:52 -07:00
|
|
|
depth: usize)
|
2016-04-29 06:00:23 +03:00
|
|
|
-> AssociatedTypeNormalizer<'a, 'b, 'gcx, 'tcx>
|
2014-12-30 17:42:02 -05:00
|
|
|
{
|
|
|
|
|
AssociatedTypeNormalizer {
|
|
|
|
|
selcx: selcx,
|
|
|
|
|
cause: cause,
|
|
|
|
|
obligations: vec!(),
|
|
|
|
|
depth: depth,
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2015-11-18 09:38:57 +00:00
|
|
|
fn fold<T:TypeFoldable<'tcx>>(&mut self, value: &T) -> T {
|
2014-12-30 17:42:02 -05:00
|
|
|
let value = self.selcx.infcx().resolve_type_vars_if_possible(value);
|
|
|
|
|
|
|
|
|
|
if !value.has_projection_types() {
|
|
|
|
|
value.clone()
|
|
|
|
|
} else {
|
|
|
|
|
value.fold_with(self)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2016-04-29 06:00:23 +03:00
|
|
|
impl<'a, 'b, 'gcx, 'tcx> TypeFolder<'gcx, 'tcx> for AssociatedTypeNormalizer<'a, 'b, 'gcx, 'tcx> {
|
|
|
|
|
fn tcx<'c>(&'c self) -> TyCtxt<'c, 'gcx, 'tcx> {
|
2014-12-30 17:42:02 -05:00
|
|
|
self.selcx.tcx()
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fn fold_ty(&mut self, ty: Ty<'tcx>) -> Ty<'tcx> {
|
|
|
|
|
// We don't want to normalize associated types that occur inside of region
|
|
|
|
|
// binders, because they may contain bound regions, and we can't cope with that.
|
|
|
|
|
//
|
|
|
|
|
// Example:
|
|
|
|
|
//
|
|
|
|
|
// for<'a> fn(<T as Foo<&'a>>::A)
|
|
|
|
|
//
|
|
|
|
|
// Instead of normalizing `<T as Foo<&'a>>::A` here, we'll
|
|
|
|
|
// normalize it when we instantiate those bound regions (which
|
|
|
|
|
// should occur eventually).
|
|
|
|
|
|
2016-01-06 02:01:28 +00:00
|
|
|
let ty = ty.super_fold_with(self);
|
2014-12-30 17:42:02 -05:00
|
|
|
match ty.sty {
|
2015-06-11 16:21:46 -07:00
|
|
|
ty::TyProjection(ref data) if !data.has_escaping_regions() => { // (*)
|
2014-12-30 17:42:02 -05:00
|
|
|
|
|
|
|
|
// (*) This is kind of hacky -- we need to be able to
|
|
|
|
|
// handle normalization within binders because
|
|
|
|
|
// otherwise we wind up a need to normalize when doing
|
|
|
|
|
// trait matching (since you can have a trait
|
|
|
|
|
// obligation like `for<'a> T::B : Fn(&'a int)`), but
|
|
|
|
|
// we can't normalize with bound regions in scope. So
|
|
|
|
|
// far now we just ignore binders but only normalize
|
|
|
|
|
// if all bound regions are gone (and then we still
|
|
|
|
|
// have to renormalize whenever we instantiate a
|
|
|
|
|
// binder). It would be better to normalize in a
|
|
|
|
|
// binding-aware fashion.
|
|
|
|
|
|
|
|
|
|
let Normalized { value: ty, obligations } =
|
|
|
|
|
normalize_projection_type(self.selcx,
|
|
|
|
|
data.clone(),
|
|
|
|
|
self.cause.clone(),
|
|
|
|
|
self.depth);
|
2015-06-10 17:22:20 +01:00
|
|
|
self.obligations.extend(obligations);
|
2014-12-30 17:42:02 -05:00
|
|
|
ty
|
|
|
|
|
}
|
2015-01-08 12:02:15 -05:00
|
|
|
|
2014-12-30 17:42:02 -05:00
|
|
|
_ => {
|
2015-01-08 12:02:15 -05:00
|
|
|
ty
|
2014-12-30 17:42:02 -05:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2015-03-22 15:11:56 -04:00
|
|
|
#[derive(Clone)]
|
2014-12-30 17:42:02 -05:00
|
|
|
pub struct Normalized<'tcx,T> {
|
|
|
|
|
pub value: T,
|
|
|
|
|
pub obligations: Vec<PredicateObligation<'tcx>>,
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pub type NormalizedTy<'tcx> = Normalized<'tcx, Ty<'tcx>>;
|
|
|
|
|
|
2015-01-08 06:58:41 -05:00
|
|
|
impl<'tcx,T> Normalized<'tcx,T> {
|
2015-01-08 10:45:56 -05:00
|
|
|
pub fn with<U>(self, value: U) -> Normalized<'tcx,U> {
|
2015-01-08 06:58:41 -05:00
|
|
|
Normalized { value: value, obligations: self.obligations }
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2015-01-26 14:20:38 -05:00
|
|
|
/// The guts of `normalize`: normalize a specific projection like `<T
|
|
|
|
|
/// as Trait>::Item`. The result is always a type (and possibly
|
|
|
|
|
/// additional obligations). If ambiguity arises, which implies that
|
|
|
|
|
/// there are unresolved type variables in the projection, we will
|
|
|
|
|
/// substitute a fresh type variable `$X` and generate a new
|
|
|
|
|
/// obligation `<T as Trait>::Item == $X` for later.
|
2016-04-29 06:00:23 +03:00
|
|
|
pub fn normalize_projection_type<'a, 'b, 'gcx, 'tcx>(
|
|
|
|
|
selcx: &'a mut SelectionContext<'b, 'gcx, 'tcx>,
|
2014-12-30 17:42:02 -05:00
|
|
|
projection_ty: ty::ProjectionTy<'tcx>,
|
|
|
|
|
cause: ObligationCause<'tcx>,
|
2015-03-25 17:06:52 -07:00
|
|
|
depth: usize)
|
2014-12-30 17:42:02 -05:00
|
|
|
-> NormalizedTy<'tcx>
|
|
|
|
|
{
|
|
|
|
|
opt_normalize_projection_type(selcx, projection_ty.clone(), cause.clone(), depth)
|
|
|
|
|
.unwrap_or_else(move || {
|
|
|
|
|
// if we bottom out in ambiguity, create a type variable
|
|
|
|
|
// and a deferred predicate to resolve this when more type
|
|
|
|
|
// information is available.
|
|
|
|
|
|
|
|
|
|
let ty_var = selcx.infcx().next_ty_var();
|
|
|
|
|
let projection = ty::Binder(ty::ProjectionPredicate {
|
|
|
|
|
projection_ty: projection_ty,
|
|
|
|
|
ty: ty_var
|
|
|
|
|
});
|
2015-10-24 18:37:28 +03:00
|
|
|
let obligation = Obligation::with_depth(
|
|
|
|
|
cause, depth + 1, projection.to_predicate());
|
2014-12-30 17:42:02 -05:00
|
|
|
Normalized {
|
|
|
|
|
value: ty_var,
|
|
|
|
|
obligations: vec!(obligation)
|
|
|
|
|
}
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
2015-01-26 14:20:38 -05:00
|
|
|
/// The guts of `normalize`: normalize a specific projection like `<T
|
|
|
|
|
/// as Trait>::Item`. The result is always a type (and possibly
|
|
|
|
|
/// additional obligations). Returns `None` in the case of ambiguity,
|
|
|
|
|
/// which indicates that there are unbound type variables.
|
2016-04-29 06:00:23 +03:00
|
|
|
fn opt_normalize_projection_type<'a, 'b, 'gcx, 'tcx>(
|
|
|
|
|
selcx: &'a mut SelectionContext<'b, 'gcx, 'tcx>,
|
2014-12-30 17:42:02 -05:00
|
|
|
projection_ty: ty::ProjectionTy<'tcx>,
|
|
|
|
|
cause: ObligationCause<'tcx>,
|
2015-03-25 17:06:52 -07:00
|
|
|
depth: usize)
|
2014-12-30 17:42:02 -05:00
|
|
|
-> Option<NormalizedTy<'tcx>>
|
|
|
|
|
{
|
|
|
|
|
debug!("normalize_projection_type(\
|
2015-06-18 20:25:05 +03:00
|
|
|
projection_ty={:?}, \
|
2014-12-30 17:42:02 -05:00
|
|
|
depth={})",
|
2015-06-18 20:25:05 +03:00
|
|
|
projection_ty,
|
2014-12-30 17:42:02 -05:00
|
|
|
depth);
|
|
|
|
|
|
|
|
|
|
let obligation = Obligation::with_depth(cause.clone(), depth, projection_ty.clone());
|
|
|
|
|
match project_type(selcx, &obligation) {
|
|
|
|
|
Ok(ProjectedTy::Progress(projected_ty, mut obligations)) => {
|
|
|
|
|
// if projection succeeded, then what we get out of this
|
|
|
|
|
// is also non-normalized (consider: it was derived from
|
|
|
|
|
// an impl, where-clause etc) and hence we must
|
|
|
|
|
// re-normalize it
|
|
|
|
|
|
2015-06-18 20:25:05 +03:00
|
|
|
debug!("normalize_projection_type: projected_ty={:?} depth={} obligations={:?}",
|
|
|
|
|
projected_ty,
|
2015-01-01 13:15:14 -05:00
|
|
|
depth,
|
2015-06-18 20:25:05 +03:00
|
|
|
obligations);
|
2014-12-30 17:42:02 -05:00
|
|
|
|
2015-06-24 02:54:32 +03:00
|
|
|
if projected_ty.has_projection_types() {
|
2015-10-24 18:37:28 +03:00
|
|
|
let mut normalizer = AssociatedTypeNormalizer::new(selcx, cause, depth+1);
|
2014-12-30 17:42:02 -05:00
|
|
|
let normalized_ty = normalizer.fold(&projected_ty);
|
|
|
|
|
|
2015-06-18 20:25:05 +03:00
|
|
|
debug!("normalize_projection_type: normalized_ty={:?} depth={}",
|
|
|
|
|
normalized_ty,
|
2014-12-30 17:42:02 -05:00
|
|
|
depth);
|
|
|
|
|
|
2015-06-10 17:22:20 +01:00
|
|
|
obligations.extend(normalizer.obligations);
|
2014-12-30 17:42:02 -05:00
|
|
|
Some(Normalized {
|
|
|
|
|
value: normalized_ty,
|
|
|
|
|
obligations: obligations,
|
|
|
|
|
})
|
|
|
|
|
} else {
|
|
|
|
|
Some(Normalized {
|
|
|
|
|
value: projected_ty,
|
|
|
|
|
obligations: obligations,
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
Ok(ProjectedTy::NoProgress(projected_ty)) => {
|
2015-06-18 20:25:05 +03:00
|
|
|
debug!("normalize_projection_type: projected_ty={:?} no progress",
|
|
|
|
|
projected_ty);
|
2014-12-30 17:42:02 -05:00
|
|
|
Some(Normalized {
|
|
|
|
|
value: projected_ty,
|
|
|
|
|
obligations: vec!()
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
Err(ProjectionTyError::TooManyCandidates) => {
|
2015-06-14 02:27:18 +03:00
|
|
|
debug!("normalize_projection_type: too many candidates");
|
2014-12-30 17:42:02 -05:00
|
|
|
None
|
|
|
|
|
}
|
|
|
|
|
Err(ProjectionTyError::TraitSelectionError(_)) => {
|
2015-06-14 02:27:18 +03:00
|
|
|
debug!("normalize_projection_type: ERROR");
|
2014-12-30 17:42:02 -05:00
|
|
|
// if we got an error processing the `T as Trait` part,
|
|
|
|
|
// just return `ty::err` but add the obligation `T :
|
|
|
|
|
// Trait`, which when processed will cause the error to be
|
|
|
|
|
// reported later
|
|
|
|
|
|
|
|
|
|
Some(normalize_to_error(selcx, projection_ty, cause, depth))
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2016-01-07 13:51:16 -05:00
|
|
|
/// If we are projecting `<T as Trait>::Item`, but `T: Trait` does not
|
|
|
|
|
/// hold. In various error cases, we cannot generate a valid
|
|
|
|
|
/// normalized projection. Therefore, we create an inference variable
|
|
|
|
|
/// return an associated obligation that, when fulfilled, will lead to
|
|
|
|
|
/// an error.
|
2015-05-05 22:09:17 +03:00
|
|
|
///
|
2016-01-07 13:51:16 -05:00
|
|
|
/// Note that we used to return `TyError` here, but that was quite
|
|
|
|
|
/// dubious -- the premise was that an error would *eventually* be
|
|
|
|
|
/// reported, when the obligation was processed. But in general once
|
|
|
|
|
/// you see a `TyError` you are supposed to be able to assume that an
|
|
|
|
|
/// error *has been* reported, so that you can take whatever heuristic
|
|
|
|
|
/// paths you want to take. To make things worse, it was possible for
|
|
|
|
|
/// cycles to arise, where you basically had a setup like `<MyType<$0>
|
|
|
|
|
/// as Trait>::Foo == $0`. Here, normalizing `<MyType<$0> as
|
|
|
|
|
/// Trait>::Foo> to `[type error]` would lead to an obligation of
|
|
|
|
|
/// `<MyType<[type error]> as Trait>::Foo`. We are supposed to report
|
|
|
|
|
/// an error for this obligation, but we legitimately should not,
|
|
|
|
|
/// because it contains `[type error]`. Yuck! (See issue #29857 for
|
|
|
|
|
/// one case where this arose.)
|
2016-04-29 06:00:23 +03:00
|
|
|
fn normalize_to_error<'a, 'gcx, 'tcx>(selcx: &mut SelectionContext<'a, 'gcx, 'tcx>,
|
|
|
|
|
projection_ty: ty::ProjectionTy<'tcx>,
|
|
|
|
|
cause: ObligationCause<'tcx>,
|
|
|
|
|
depth: usize)
|
|
|
|
|
-> NormalizedTy<'tcx>
|
2014-12-30 17:42:02 -05:00
|
|
|
{
|
|
|
|
|
let trait_ref = projection_ty.trait_ref.to_poly_trait_ref();
|
|
|
|
|
let trait_obligation = Obligation { cause: cause,
|
|
|
|
|
recursion_depth: depth,
|
2015-06-23 11:50:50 -07:00
|
|
|
predicate: trait_ref.to_predicate() };
|
2016-01-07 13:51:16 -05:00
|
|
|
let new_value = selcx.infcx().next_ty_var();
|
2014-12-30 17:42:02 -05:00
|
|
|
Normalized {
|
2016-01-07 13:51:16 -05:00
|
|
|
value: new_value,
|
2014-12-30 17:42:02 -05:00
|
|
|
obligations: vec!(trait_obligation)
|
2014-12-17 14:16:28 -05:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2014-12-30 17:42:02 -05:00
|
|
|
enum ProjectedTy<'tcx> {
|
|
|
|
|
Progress(Ty<'tcx>, Vec<PredicateObligation<'tcx>>),
|
|
|
|
|
NoProgress(Ty<'tcx>),
|
|
|
|
|
}
|
|
|
|
|
|
2014-12-17 14:16:28 -05:00
|
|
|
/// Compute the result of a projection type (if we can).
|
2016-04-29 06:00:23 +03:00
|
|
|
fn project_type<'cx, 'gcx, 'tcx>(
|
|
|
|
|
selcx: &mut SelectionContext<'cx, 'gcx, 'tcx>,
|
2014-12-17 14:16:28 -05:00
|
|
|
obligation: &ProjectionTyObligation<'tcx>)
|
2014-12-30 17:42:02 -05:00
|
|
|
-> Result<ProjectedTy<'tcx>, ProjectionTyError<'tcx>>
|
2014-12-17 14:16:28 -05:00
|
|
|
{
|
2015-06-18 20:25:05 +03:00
|
|
|
debug!("project(obligation={:?})",
|
|
|
|
|
obligation);
|
2014-12-17 14:16:28 -05:00
|
|
|
|
2014-12-30 17:42:02 -05:00
|
|
|
let recursion_limit = selcx.tcx().sess.recursion_limit.get();
|
|
|
|
|
if obligation.recursion_depth >= recursion_limit {
|
|
|
|
|
debug!("project: overflow!");
|
2016-03-17 00:15:31 +02:00
|
|
|
selcx.infcx().report_overflow_error(&obligation, true);
|
2014-12-30 17:42:02 -05:00
|
|
|
}
|
|
|
|
|
|
2015-01-02 04:20:34 -05:00
|
|
|
let obligation_trait_ref =
|
|
|
|
|
selcx.infcx().resolve_type_vars_if_possible(&obligation.predicate.trait_ref);
|
|
|
|
|
|
2015-06-18 20:25:05 +03:00
|
|
|
debug!("project: obligation_trait_ref={:?}", obligation_trait_ref);
|
2015-01-02 04:20:34 -05:00
|
|
|
|
|
|
|
|
if obligation_trait_ref.references_error() {
|
|
|
|
|
return Ok(ProjectedTy::Progress(selcx.tcx().types.err, vec!()));
|
|
|
|
|
}
|
|
|
|
|
|
2014-12-17 14:16:28 -05:00
|
|
|
let mut candidates = ProjectionTyCandidateSet {
|
|
|
|
|
vec: Vec::new(),
|
|
|
|
|
ambiguous: false,
|
|
|
|
|
};
|
|
|
|
|
|
2014-12-23 05:26:34 -05:00
|
|
|
assemble_candidates_from_param_env(selcx,
|
|
|
|
|
obligation,
|
2015-01-02 12:16:41 -08:00
|
|
|
&obligation_trait_ref,
|
2014-12-23 05:26:34 -05:00
|
|
|
&mut candidates);
|
|
|
|
|
|
2015-01-27 14:52:54 -05:00
|
|
|
assemble_candidates_from_trait_def(selcx,
|
|
|
|
|
obligation,
|
|
|
|
|
&obligation_trait_ref,
|
|
|
|
|
&mut candidates);
|
|
|
|
|
|
2014-12-23 05:26:34 -05:00
|
|
|
if let Err(e) = assemble_candidates_from_impls(selcx,
|
|
|
|
|
obligation,
|
2015-01-02 12:16:41 -08:00
|
|
|
&obligation_trait_ref,
|
2014-12-23 05:26:34 -05:00
|
|
|
&mut candidates) {
|
|
|
|
|
return Err(ProjectionTyError::TraitSelectionError(e));
|
2014-12-26 07:07:55 -05:00
|
|
|
}
|
2014-12-17 14:16:28 -05:00
|
|
|
|
|
|
|
|
debug!("{} candidates, ambiguous={}",
|
|
|
|
|
candidates.vec.len(),
|
|
|
|
|
candidates.ambiguous);
|
|
|
|
|
|
2015-10-22 12:28:47 -04:00
|
|
|
// Inherent ambiguity that prevents us from even enumerating the
|
|
|
|
|
// candidates.
|
|
|
|
|
if candidates.ambiguous {
|
|
|
|
|
return Err(ProjectionTyError::TooManyCandidates);
|
|
|
|
|
}
|
2014-12-17 14:16:28 -05:00
|
|
|
|
2015-02-12 01:12:57 +08:00
|
|
|
// Drop duplicates.
|
|
|
|
|
//
|
|
|
|
|
// Note: `candidates.vec` seems to be on the critical path of the
|
|
|
|
|
// compiler. Replacing it with an hash set was also tried, which would
|
|
|
|
|
// render the following dedup unnecessary. It led to cleaner code but
|
|
|
|
|
// prolonged compiling time of `librustc` from 5m30s to 6m in one test, or
|
|
|
|
|
// ~9% performance lost.
|
|
|
|
|
if candidates.vec.len() > 1 {
|
|
|
|
|
let mut i = 0;
|
|
|
|
|
while i < candidates.vec.len() {
|
|
|
|
|
let has_dup = (0..i).any(|j| candidates.vec[i] == candidates.vec[j]);
|
|
|
|
|
if has_dup {
|
|
|
|
|
candidates.vec.swap_remove(i);
|
|
|
|
|
} else {
|
|
|
|
|
i += 1;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2015-10-22 12:28:47 -04:00
|
|
|
// Prefer where-clauses. As in select, if there are multiple
|
|
|
|
|
// candidates, we prefer where-clause candidates over impls. This
|
|
|
|
|
// may seem a bit surprising, since impls are the source of
|
|
|
|
|
// "truth" in some sense, but in fact some of the impls that SEEM
|
|
|
|
|
// applicable are not, because of nested obligations. Where
|
|
|
|
|
// clauses are the safer choice. See the comment on
|
|
|
|
|
// `select::SelectionCandidate` and #21974 for more details.
|
|
|
|
|
if candidates.vec.len() > 1 {
|
|
|
|
|
debug!("retaining param-env candidates only from {:?}", candidates.vec);
|
|
|
|
|
candidates.vec.retain(|c| match *c {
|
|
|
|
|
ProjectionTyCandidate::ParamEnv(..) => true,
|
|
|
|
|
ProjectionTyCandidate::Impl(..) |
|
|
|
|
|
ProjectionTyCandidate::Closure(..) |
|
|
|
|
|
ProjectionTyCandidate::TraitDef(..) |
|
|
|
|
|
ProjectionTyCandidate::FnPointer(..) => false,
|
|
|
|
|
});
|
|
|
|
|
debug!("resulting candidate set: {:?}", candidates.vec);
|
|
|
|
|
if candidates.vec.len() != 1 {
|
|
|
|
|
return Err(ProjectionTyError::TooManyCandidates);
|
|
|
|
|
}
|
2014-12-17 14:16:28 -05:00
|
|
|
}
|
|
|
|
|
|
2015-10-22 12:28:47 -04:00
|
|
|
assert!(candidates.vec.len() <= 1);
|
|
|
|
|
|
2016-02-16 10:36:47 -08:00
|
|
|
let possible_candidate = candidates.vec.pop().and_then(|candidate| {
|
|
|
|
|
// In Any (i.e. trans) mode, all projections succeed;
|
|
|
|
|
// otherwise, we need to be sensitive to `default` and
|
|
|
|
|
// specialization.
|
2016-03-08 15:23:52 -08:00
|
|
|
if !selcx.projection_mode().is_any() {
|
2016-02-16 10:36:47 -08:00
|
|
|
if let ProjectionTyCandidate::Impl(ref impl_data) = candidate {
|
|
|
|
|
if let Some(node_item) = assoc_ty_def(selcx,
|
|
|
|
|
impl_data.impl_def_id,
|
|
|
|
|
obligation.predicate.item_name) {
|
|
|
|
|
if node_item.node.is_from_trait() {
|
|
|
|
|
if node_item.item.ty.is_some() {
|
|
|
|
|
// If the associated type has a default from the
|
|
|
|
|
// trait, that should be considered `default` and
|
|
|
|
|
// hence not projected.
|
|
|
|
|
//
|
|
|
|
|
// Note, however, that we allow a projection from
|
|
|
|
|
// the trait specifically in the case that the trait
|
|
|
|
|
// does *not* give a default. This is purely to
|
|
|
|
|
// avoid spurious errors: the situation can only
|
|
|
|
|
// arise when *no* impl in the specialization chain
|
|
|
|
|
// has provided a definition for the type. When we
|
|
|
|
|
// confirm the candidate, we'll turn the projection
|
|
|
|
|
// into a TyError, since the actual error will be
|
|
|
|
|
// reported in `check_impl_items_against_trait`.
|
|
|
|
|
return None;
|
|
|
|
|
}
|
|
|
|
|
} else if node_item.item.defaultness.is_default() {
|
|
|
|
|
return None;
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
// Normally this situation could only arise througha
|
|
|
|
|
// compiler bug, but at coherence-checking time we only look
|
|
|
|
|
// at the topmost impl (we don't even consider the trait
|
|
|
|
|
// itself) for the definition -- so we can fail to find a
|
|
|
|
|
// definition of the type even if it exists.
|
2016-03-11 14:47:29 -08:00
|
|
|
|
|
|
|
|
// For now, we just unconditionally ICE, because otherwise,
|
|
|
|
|
// examples like the following will succeed:
|
|
|
|
|
//
|
|
|
|
|
// ```
|
|
|
|
|
// trait Assoc {
|
|
|
|
|
// type Output;
|
|
|
|
|
// }
|
|
|
|
|
//
|
|
|
|
|
// impl<T> Assoc for T {
|
|
|
|
|
// default type Output = bool;
|
|
|
|
|
// }
|
|
|
|
|
//
|
|
|
|
|
// impl Assoc for u8 {}
|
|
|
|
|
// impl Assoc for u16 {}
|
|
|
|
|
//
|
|
|
|
|
// trait Foo {}
|
|
|
|
|
// impl Foo for <u8 as Assoc>::Output {}
|
|
|
|
|
// impl Foo for <u16 as Assoc>::Output {}
|
|
|
|
|
// return None;
|
|
|
|
|
// }
|
|
|
|
|
// ```
|
|
|
|
|
//
|
|
|
|
|
// The essential problem here is that the projection fails,
|
|
|
|
|
// leaving two unnormalized types, which appear not to unify
|
|
|
|
|
// -- so the overlap check succeeds, when it should fail.
|
2016-03-25 01:14:29 +01:00
|
|
|
bug!("Tried to project an inherited associated type during \
|
|
|
|
|
coherence checking, which is currently not supported.");
|
2016-02-16 10:36:47 -08:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
Some(candidate)
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
match possible_candidate {
|
2014-12-17 14:16:28 -05:00
|
|
|
Some(candidate) => {
|
2014-12-30 17:42:02 -05:00
|
|
|
let (ty, obligations) = confirm_candidate(selcx, obligation, candidate);
|
|
|
|
|
Ok(ProjectedTy::Progress(ty, obligations))
|
2014-12-17 14:16:28 -05:00
|
|
|
}
|
|
|
|
|
None => {
|
2015-06-25 04:09:46 +03:00
|
|
|
Ok(ProjectedTy::NoProgress(selcx.tcx().mk_projection(
|
|
|
|
|
obligation.predicate.trait_ref.clone(),
|
|
|
|
|
obligation.predicate.item_name)))
|
2014-12-17 14:16:28 -05:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// The first thing we have to do is scan through the parameter
|
|
|
|
|
/// environment to see whether there are any projection predicates
|
|
|
|
|
/// there that can answer this question.
|
2016-04-29 06:00:23 +03:00
|
|
|
fn assemble_candidates_from_param_env<'cx, 'gcx, 'tcx>(
|
|
|
|
|
selcx: &mut SelectionContext<'cx, 'gcx, 'tcx>,
|
2015-01-02 04:20:34 -05:00
|
|
|
obligation: &ProjectionTyObligation<'tcx>,
|
2015-04-21 18:59:58 +03:00
|
|
|
obligation_trait_ref: &ty::TraitRef<'tcx>,
|
2014-12-17 14:16:28 -05:00
|
|
|
candidate_set: &mut ProjectionTyCandidateSet<'tcx>)
|
|
|
|
|
{
|
2015-10-22 12:28:47 -04:00
|
|
|
debug!("assemble_candidates_from_param_env(..)");
|
2015-05-11 17:02:56 -04:00
|
|
|
let env_predicates = selcx.param_env().caller_bounds.iter().cloned();
|
2015-10-22 12:28:47 -04:00
|
|
|
assemble_candidates_from_predicates(selcx,
|
|
|
|
|
obligation,
|
|
|
|
|
obligation_trait_ref,
|
|
|
|
|
candidate_set,
|
|
|
|
|
ProjectionTyCandidate::ParamEnv,
|
|
|
|
|
env_predicates);
|
2014-12-26 07:07:55 -05:00
|
|
|
}
|
|
|
|
|
|
2015-01-27 14:52:54 -05:00
|
|
|
/// In the case of a nested projection like <<A as Foo>::FooT as Bar>::BarT, we may find
|
|
|
|
|
/// that the definition of `Foo` has some clues:
|
|
|
|
|
///
|
2015-03-12 22:42:38 -04:00
|
|
|
/// ```
|
2015-01-27 14:52:54 -05:00
|
|
|
/// trait Foo {
|
|
|
|
|
/// type FooT : Bar<BarT=i32>
|
|
|
|
|
/// }
|
|
|
|
|
/// ```
|
|
|
|
|
///
|
|
|
|
|
/// Here, for example, we could conclude that the result is `i32`.
|
2016-04-29 06:00:23 +03:00
|
|
|
fn assemble_candidates_from_trait_def<'cx, 'gcx, 'tcx>(
|
|
|
|
|
selcx: &mut SelectionContext<'cx, 'gcx, 'tcx>,
|
2015-01-27 14:52:54 -05:00
|
|
|
obligation: &ProjectionTyObligation<'tcx>,
|
2015-04-21 18:59:58 +03:00
|
|
|
obligation_trait_ref: &ty::TraitRef<'tcx>,
|
2015-01-27 14:52:54 -05:00
|
|
|
candidate_set: &mut ProjectionTyCandidateSet<'tcx>)
|
|
|
|
|
{
|
2015-10-22 12:28:47 -04:00
|
|
|
debug!("assemble_candidates_from_trait_def(..)");
|
|
|
|
|
|
2015-01-27 14:52:54 -05:00
|
|
|
// Check whether the self-type is itself a projection.
|
|
|
|
|
let trait_ref = match obligation_trait_ref.self_ty().sty {
|
2015-06-11 16:21:46 -07:00
|
|
|
ty::TyProjection(ref data) => data.trait_ref.clone(),
|
|
|
|
|
ty::TyInfer(ty::TyVar(_)) => {
|
2015-01-27 14:52:54 -05:00
|
|
|
// If the self-type is an inference variable, then it MAY wind up
|
|
|
|
|
// being a projected type, so induce an ambiguity.
|
|
|
|
|
candidate_set.ambiguous = true;
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
_ => { return; }
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
// If so, extract what we know from the trait and try to come up with a good answer.
|
2015-06-25 23:42:17 +03:00
|
|
|
let trait_predicates = selcx.tcx().lookup_predicates(trait_ref.def_id);
|
2015-02-11 10:28:52 -05:00
|
|
|
let bounds = trait_predicates.instantiate(selcx.tcx(), trait_ref.substs);
|
2015-05-11 17:02:56 -04:00
|
|
|
let bounds = elaborate_predicates(selcx.tcx(), bounds.predicates.into_vec());
|
2015-10-22 12:28:47 -04:00
|
|
|
assemble_candidates_from_predicates(selcx,
|
|
|
|
|
obligation,
|
|
|
|
|
obligation_trait_ref,
|
|
|
|
|
candidate_set,
|
|
|
|
|
ProjectionTyCandidate::TraitDef,
|
|
|
|
|
bounds)
|
2015-01-27 14:52:54 -05:00
|
|
|
}
|
|
|
|
|
|
2016-04-29 06:00:23 +03:00
|
|
|
fn assemble_candidates_from_predicates<'cx, 'gcx, 'tcx, I>(
|
|
|
|
|
selcx: &mut SelectionContext<'cx, 'gcx, 'tcx>,
|
2015-01-02 04:20:34 -05:00
|
|
|
obligation: &ProjectionTyObligation<'tcx>,
|
2015-04-21 18:59:58 +03:00
|
|
|
obligation_trait_ref: &ty::TraitRef<'tcx>,
|
2014-12-26 07:07:55 -05:00
|
|
|
candidate_set: &mut ProjectionTyCandidateSet<'tcx>,
|
2015-10-22 12:28:47 -04:00
|
|
|
ctor: fn(ty::PolyProjectionPredicate<'tcx>) -> ProjectionTyCandidate<'tcx>,
|
2015-05-11 17:02:56 -04:00
|
|
|
env_predicates: I)
|
|
|
|
|
where I: Iterator<Item=ty::Predicate<'tcx>>
|
2014-12-26 07:07:55 -05:00
|
|
|
{
|
2015-06-18 20:25:05 +03:00
|
|
|
debug!("assemble_candidates_from_predicates(obligation={:?})",
|
|
|
|
|
obligation);
|
2014-12-26 07:07:55 -05:00
|
|
|
let infcx = selcx.infcx();
|
2015-05-11 17:02:56 -04:00
|
|
|
for predicate in env_predicates {
|
2015-06-18 20:25:05 +03:00
|
|
|
debug!("assemble_candidates_from_predicates: predicate={:?}",
|
|
|
|
|
predicate);
|
2014-12-17 14:16:28 -05:00
|
|
|
match predicate {
|
|
|
|
|
ty::Predicate::Projection(ref data) => {
|
2015-01-06 13:41:14 -05:00
|
|
|
let same_name = data.item_name() == obligation.predicate.item_name;
|
|
|
|
|
|
|
|
|
|
let is_match = same_name && infcx.probe(|_| {
|
2015-11-25 12:41:09 +01:00
|
|
|
let origin = TypeOrigin::Misc(obligation.cause.span);
|
2014-12-17 14:16:28 -05:00
|
|
|
let data_poly_trait_ref =
|
|
|
|
|
data.to_poly_trait_ref();
|
2015-01-10 11:54:15 -05:00
|
|
|
let obligation_poly_trait_ref =
|
|
|
|
|
obligation_trait_ref.to_poly_trait_ref();
|
2014-12-17 14:16:28 -05:00
|
|
|
infcx.sub_poly_trait_refs(false,
|
|
|
|
|
origin,
|
2015-01-10 11:54:15 -05:00
|
|
|
data_poly_trait_ref,
|
2016-03-29 20:06:42 -07:00
|
|
|
obligation_poly_trait_ref)
|
2016-04-04 12:41:05 -07:00
|
|
|
// FIXME(#32730) propagate obligations
|
2016-03-29 20:06:42 -07:00
|
|
|
.map(|InferOk { obligations, .. }| assert!(obligations.is_empty()))
|
|
|
|
|
.is_ok()
|
2014-12-17 14:16:28 -05:00
|
|
|
});
|
|
|
|
|
|
2015-06-18 20:25:05 +03:00
|
|
|
debug!("assemble_candidates_from_predicates: candidate={:?} \
|
|
|
|
|
is_match={} same_name={}",
|
|
|
|
|
data, is_match, same_name);
|
2015-01-06 13:41:14 -05:00
|
|
|
|
2015-01-10 11:54:15 -05:00
|
|
|
if is_match {
|
2015-10-22 12:28:47 -04:00
|
|
|
candidate_set.vec.push(ctor(data.clone()));
|
2014-12-17 14:16:28 -05:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
_ => { }
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2016-04-29 06:00:23 +03:00
|
|
|
fn assemble_candidates_from_object_type<'cx, 'gcx, 'tcx>(
|
|
|
|
|
selcx: &mut SelectionContext<'cx, 'gcx, 'tcx>,
|
2014-12-26 07:07:55 -05:00
|
|
|
obligation: &ProjectionTyObligation<'tcx>,
|
2015-04-21 18:59:58 +03:00
|
|
|
obligation_trait_ref: &ty::TraitRef<'tcx>,
|
2015-07-04 05:46:54 +03:00
|
|
|
candidate_set: &mut ProjectionTyCandidateSet<'tcx>)
|
2014-12-26 07:07:55 -05:00
|
|
|
{
|
2015-07-04 05:46:54 +03:00
|
|
|
let self_ty = obligation_trait_ref.self_ty();
|
|
|
|
|
let object_ty = selcx.infcx().shallow_resolve(self_ty);
|
2015-06-18 20:25:05 +03:00
|
|
|
debug!("assemble_candidates_from_object_type(object_ty={:?})",
|
|
|
|
|
object_ty);
|
2014-12-23 05:26:34 -05:00
|
|
|
let data = match object_ty.sty {
|
2015-06-11 16:21:46 -07:00
|
|
|
ty::TyTrait(ref data) => data,
|
2014-12-23 05:26:34 -05:00
|
|
|
_ => {
|
2016-03-25 18:31:27 +01:00
|
|
|
span_bug!(
|
2014-12-23 05:26:34 -05:00
|
|
|
obligation.cause.span,
|
2016-03-25 18:31:27 +01:00
|
|
|
"assemble_candidates_from_object_type called with non-object: {:?}",
|
|
|
|
|
object_ty);
|
2014-12-23 05:26:34 -05:00
|
|
|
}
|
2014-12-26 07:07:55 -05:00
|
|
|
};
|
2014-12-23 05:26:34 -05:00
|
|
|
let projection_bounds = data.projection_bounds_with_self_ty(selcx.tcx(), object_ty);
|
2014-12-29 13:41:25 -05:00
|
|
|
let env_predicates = projection_bounds.iter()
|
2015-06-23 11:50:50 -07:00
|
|
|
.map(|p| p.to_predicate())
|
2014-12-29 13:41:25 -05:00
|
|
|
.collect();
|
2015-05-11 17:02:56 -04:00
|
|
|
let env_predicates = elaborate_predicates(selcx.tcx(), env_predicates);
|
2015-10-22 12:28:47 -04:00
|
|
|
assemble_candidates_from_predicates(selcx,
|
|
|
|
|
obligation,
|
|
|
|
|
obligation_trait_ref,
|
|
|
|
|
candidate_set,
|
|
|
|
|
ProjectionTyCandidate::ParamEnv,
|
|
|
|
|
env_predicates)
|
2014-12-26 07:07:55 -05:00
|
|
|
}
|
|
|
|
|
|
2016-04-29 06:00:23 +03:00
|
|
|
fn assemble_candidates_from_impls<'cx, 'gcx, 'tcx>(
|
|
|
|
|
selcx: &mut SelectionContext<'cx, 'gcx, 'tcx>,
|
2014-12-30 17:42:02 -05:00
|
|
|
obligation: &ProjectionTyObligation<'tcx>,
|
2015-04-21 18:59:58 +03:00
|
|
|
obligation_trait_ref: &ty::TraitRef<'tcx>,
|
2014-12-17 14:16:28 -05:00
|
|
|
candidate_set: &mut ProjectionTyCandidateSet<'tcx>)
|
2014-12-30 17:42:02 -05:00
|
|
|
-> Result<(), SelectionError<'tcx>>
|
2014-12-17 14:16:28 -05:00
|
|
|
{
|
|
|
|
|
// If we are resolving `<T as TraitRef<...>>::Item == Type`,
|
|
|
|
|
// start out by selecting the predicate `T as TraitRef<...>`:
|
2015-01-02 04:20:34 -05:00
|
|
|
let poly_trait_ref = obligation_trait_ref.to_poly_trait_ref();
|
|
|
|
|
let trait_obligation = obligation.with(poly_trait_ref.to_poly_trait_predicate());
|
2014-12-17 14:16:28 -05:00
|
|
|
let vtable = match selcx.select(&trait_obligation) {
|
|
|
|
|
Ok(Some(vtable)) => vtable,
|
|
|
|
|
Ok(None) => {
|
|
|
|
|
candidate_set.ambiguous = true;
|
|
|
|
|
return Ok(());
|
|
|
|
|
}
|
|
|
|
|
Err(e) => {
|
2015-06-18 20:25:05 +03:00
|
|
|
debug!("assemble_candidates_from_impls: selection error {:?}",
|
|
|
|
|
e);
|
2014-12-30 17:42:02 -05:00
|
|
|
return Err(e);
|
2014-12-17 14:16:28 -05:00
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
match vtable {
|
|
|
|
|
super::VtableImpl(data) => {
|
2015-06-18 20:25:05 +03:00
|
|
|
debug!("assemble_candidates_from_impls: impl candidate {:?}",
|
|
|
|
|
data);
|
2015-01-06 13:41:14 -05:00
|
|
|
|
2014-12-17 14:16:28 -05:00
|
|
|
candidate_set.vec.push(
|
|
|
|
|
ProjectionTyCandidate::Impl(data));
|
|
|
|
|
}
|
2015-07-04 05:46:54 +03:00
|
|
|
super::VtableObject(_) => {
|
2014-12-23 05:26:34 -05:00
|
|
|
assemble_candidates_from_object_type(
|
2015-07-04 05:46:54 +03:00
|
|
|
selcx, obligation, obligation_trait_ref, candidate_set);
|
2014-12-23 05:26:34 -05:00
|
|
|
}
|
2015-06-10 00:09:37 +03:00
|
|
|
super::VtableClosure(data) => {
|
2015-01-10 11:54:15 -05:00
|
|
|
candidate_set.vec.push(
|
2015-06-10 00:09:37 +03:00
|
|
|
ProjectionTyCandidate::Closure(data));
|
2015-01-10 11:54:15 -05:00
|
|
|
}
|
|
|
|
|
super::VtableFnPointer(fn_type) => {
|
|
|
|
|
candidate_set.vec.push(
|
|
|
|
|
ProjectionTyCandidate::FnPointer(fn_type));
|
|
|
|
|
}
|
2014-12-17 14:16:28 -05:00
|
|
|
super::VtableParam(..) => {
|
|
|
|
|
// This case tell us nothing about the value of an
|
|
|
|
|
// associated type. Consider:
|
|
|
|
|
//
|
|
|
|
|
// ```
|
|
|
|
|
// trait SomeTrait { type Foo; }
|
|
|
|
|
// fn foo<T:SomeTrait>(...) { }
|
|
|
|
|
// ```
|
|
|
|
|
//
|
|
|
|
|
// If the user writes `<T as SomeTrait>::Foo`, then the `T
|
|
|
|
|
// : SomeTrait` binding does not help us decide what the
|
|
|
|
|
// type `Foo` is (at least, not more specifically than
|
|
|
|
|
// what we already knew).
|
|
|
|
|
//
|
|
|
|
|
// But wait, you say! What about an example like this:
|
|
|
|
|
//
|
|
|
|
|
// ```
|
2015-03-25 17:06:52 -07:00
|
|
|
// fn bar<T:SomeTrait<Foo=usize>>(...) { ... }
|
2014-12-17 14:16:28 -05:00
|
|
|
// ```
|
|
|
|
|
//
|
2015-03-25 17:06:52 -07:00
|
|
|
// Doesn't the `T : Sometrait<Foo=usize>` predicate help
|
2014-12-17 14:16:28 -05:00
|
|
|
// resolve `T::Foo`? And of course it does, but in fact
|
|
|
|
|
// that single predicate is desugared into two predicates
|
|
|
|
|
// in the compiler: a trait predicate (`T : SomeTrait`) and a
|
|
|
|
|
// projection. And the projection where clause is handled
|
|
|
|
|
// in `assemble_candidates_from_param_env`.
|
|
|
|
|
}
|
2015-02-07 14:24:34 +01:00
|
|
|
super::VtableDefaultImpl(..) |
|
2015-01-10 11:54:15 -05:00
|
|
|
super::VtableBuiltin(..) => {
|
2014-12-17 14:16:28 -05:00
|
|
|
// These traits have no associated types.
|
2016-03-25 18:31:27 +01:00
|
|
|
span_bug!(
|
2014-12-17 14:16:28 -05:00
|
|
|
obligation.cause.span,
|
2016-03-25 18:31:27 +01:00
|
|
|
"Cannot project an associated type from `{:?}`",
|
|
|
|
|
vtable);
|
2014-12-17 14:16:28 -05:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
|
}
|
|
|
|
|
|
2016-04-29 06:00:23 +03:00
|
|
|
fn confirm_candidate<'cx, 'gcx, 'tcx>(
|
|
|
|
|
selcx: &mut SelectionContext<'cx, 'gcx, 'tcx>,
|
2014-12-30 17:42:02 -05:00
|
|
|
obligation: &ProjectionTyObligation<'tcx>,
|
2014-12-17 14:16:28 -05:00
|
|
|
candidate: ProjectionTyCandidate<'tcx>)
|
2014-12-30 17:42:02 -05:00
|
|
|
-> (Ty<'tcx>, Vec<PredicateObligation<'tcx>>)
|
2014-12-17 14:16:28 -05:00
|
|
|
{
|
2015-06-18 20:25:05 +03:00
|
|
|
debug!("confirm_candidate(candidate={:?}, obligation={:?})",
|
|
|
|
|
candidate,
|
|
|
|
|
obligation);
|
2014-12-17 14:16:28 -05:00
|
|
|
|
2014-12-30 17:42:02 -05:00
|
|
|
match candidate {
|
2015-10-22 12:28:47 -04:00
|
|
|
ProjectionTyCandidate::ParamEnv(poly_projection) |
|
|
|
|
|
ProjectionTyCandidate::TraitDef(poly_projection) => {
|
2015-01-10 11:54:15 -05:00
|
|
|
confirm_param_env_candidate(selcx, obligation, poly_projection)
|
2014-12-17 14:16:28 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ProjectionTyCandidate::Impl(impl_vtable) => {
|
2015-01-10 11:54:15 -05:00
|
|
|
confirm_impl_candidate(selcx, obligation, impl_vtable)
|
|
|
|
|
}
|
2014-12-17 14:16:28 -05:00
|
|
|
|
2015-06-10 00:09:37 +03:00
|
|
|
ProjectionTyCandidate::Closure(closure_vtable) => {
|
|
|
|
|
confirm_closure_candidate(selcx, obligation, closure_vtable)
|
2015-01-10 11:54:15 -05:00
|
|
|
}
|
2014-12-17 14:16:28 -05:00
|
|
|
|
2015-01-10 11:54:15 -05:00
|
|
|
ProjectionTyCandidate::FnPointer(fn_type) => {
|
|
|
|
|
confirm_fn_pointer_candidate(selcx, obligation, fn_type)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2016-04-29 06:00:23 +03:00
|
|
|
fn confirm_fn_pointer_candidate<'cx, 'gcx, 'tcx>(
|
|
|
|
|
selcx: &mut SelectionContext<'cx, 'gcx, 'tcx>,
|
2015-01-10 11:54:15 -05:00
|
|
|
obligation: &ProjectionTyObligation<'tcx>,
|
|
|
|
|
fn_type: Ty<'tcx>)
|
|
|
|
|
-> (Ty<'tcx>, Vec<PredicateObligation<'tcx>>)
|
|
|
|
|
{
|
|
|
|
|
let fn_type = selcx.infcx().shallow_resolve(fn_type);
|
2015-06-24 08:24:13 +03:00
|
|
|
let sig = fn_type.fn_sig();
|
2015-01-10 11:54:15 -05:00
|
|
|
confirm_callable_candidate(selcx, obligation, sig, util::TupleArgumentsFlag::Yes)
|
|
|
|
|
}
|
|
|
|
|
|
2016-04-29 06:00:23 +03:00
|
|
|
fn confirm_closure_candidate<'cx, 'gcx, 'tcx>(
|
|
|
|
|
selcx: &mut SelectionContext<'cx, 'gcx, 'tcx>,
|
2015-01-10 11:54:15 -05:00
|
|
|
obligation: &ProjectionTyObligation<'tcx>,
|
2015-06-10 00:09:37 +03:00
|
|
|
vtable: VtableClosureData<'tcx, PredicateObligation<'tcx>>)
|
2015-01-10 11:54:15 -05:00
|
|
|
-> (Ty<'tcx>, Vec<PredicateObligation<'tcx>>)
|
|
|
|
|
{
|
|
|
|
|
let closure_typer = selcx.closure_typer();
|
2016-04-29 08:30:54 +03:00
|
|
|
let closure_type = closure_typer.closure_type(vtable.closure_def_id, vtable.substs);
|
2015-06-10 00:09:37 +03:00
|
|
|
let Normalized {
|
|
|
|
|
value: closure_type,
|
|
|
|
|
mut obligations
|
|
|
|
|
} = normalize_with_depth(selcx,
|
|
|
|
|
obligation.cause.clone(),
|
|
|
|
|
obligation.recursion_depth+1,
|
|
|
|
|
&closure_type);
|
|
|
|
|
let (ty, mut cc_obligations) = confirm_callable_candidate(selcx,
|
|
|
|
|
obligation,
|
|
|
|
|
&closure_type.sig,
|
|
|
|
|
util::TupleArgumentsFlag::No);
|
|
|
|
|
obligations.append(&mut cc_obligations);
|
|
|
|
|
(ty, obligations)
|
2015-01-10 11:54:15 -05:00
|
|
|
}
|
|
|
|
|
|
2016-04-29 06:00:23 +03:00
|
|
|
fn confirm_callable_candidate<'cx, 'gcx, 'tcx>(
|
|
|
|
|
selcx: &mut SelectionContext<'cx, 'gcx, 'tcx>,
|
2015-01-10 11:54:15 -05:00
|
|
|
obligation: &ProjectionTyObligation<'tcx>,
|
|
|
|
|
fn_sig: &ty::PolyFnSig<'tcx>,
|
|
|
|
|
flag: util::TupleArgumentsFlag)
|
|
|
|
|
-> (Ty<'tcx>, Vec<PredicateObligation<'tcx>>)
|
|
|
|
|
{
|
|
|
|
|
let tcx = selcx.tcx();
|
|
|
|
|
|
2015-06-18 20:25:05 +03:00
|
|
|
debug!("confirm_callable_candidate({:?},{:?})",
|
|
|
|
|
obligation,
|
|
|
|
|
fn_sig);
|
2015-01-10 11:54:15 -05:00
|
|
|
|
2015-02-15 15:09:26 -05:00
|
|
|
// the `Output` associated type is declared on `FnOnce`
|
|
|
|
|
let fn_once_def_id = tcx.lang_items.fn_once_trait().unwrap();
|
|
|
|
|
|
2015-01-10 11:54:15 -05:00
|
|
|
// Note: we unwrap the binder here but re-create it below (1)
|
|
|
|
|
let ty::Binder((trait_ref, ret_type)) =
|
2016-03-17 00:15:31 +02:00
|
|
|
tcx.closure_trait_ref_and_return_type(fn_once_def_id,
|
|
|
|
|
obligation.predicate.trait_ref.self_ty(),
|
|
|
|
|
fn_sig,
|
|
|
|
|
flag);
|
2015-01-10 11:54:15 -05:00
|
|
|
|
|
|
|
|
let predicate = ty::Binder(ty::ProjectionPredicate { // (1) recreate binder here
|
|
|
|
|
projection_ty: ty::ProjectionTy {
|
|
|
|
|
trait_ref: trait_ref,
|
|
|
|
|
item_name: token::intern(FN_OUTPUT_NAME),
|
|
|
|
|
},
|
|
|
|
|
ty: ret_type
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
confirm_param_env_candidate(selcx, obligation, predicate)
|
|
|
|
|
}
|
|
|
|
|
|
2016-04-29 06:00:23 +03:00
|
|
|
fn confirm_param_env_candidate<'cx, 'gcx, 'tcx>(
|
|
|
|
|
selcx: &mut SelectionContext<'cx, 'gcx, 'tcx>,
|
2015-01-10 11:54:15 -05:00
|
|
|
obligation: &ProjectionTyObligation<'tcx>,
|
|
|
|
|
poly_projection: ty::PolyProjectionPredicate<'tcx>)
|
|
|
|
|
-> (Ty<'tcx>, Vec<PredicateObligation<'tcx>>)
|
|
|
|
|
{
|
|
|
|
|
let infcx = selcx.infcx();
|
|
|
|
|
|
|
|
|
|
let projection =
|
|
|
|
|
infcx.replace_late_bound_regions_with_fresh_var(
|
|
|
|
|
obligation.cause.span,
|
|
|
|
|
infer::LateBoundRegionConversionTime::HigherRankedType,
|
|
|
|
|
&poly_projection).0;
|
|
|
|
|
|
|
|
|
|
assert_eq!(projection.projection_ty.item_name,
|
|
|
|
|
obligation.predicate.item_name);
|
|
|
|
|
|
2015-11-25 12:41:09 +01:00
|
|
|
let origin = TypeOrigin::RelateOutputImplTypes(obligation.cause.span);
|
2015-11-20 09:34:33 -08:00
|
|
|
match infcx.eq_trait_refs(false,
|
|
|
|
|
origin,
|
|
|
|
|
obligation.predicate.trait_ref.clone(),
|
|
|
|
|
projection.projection_ty.trait_ref.clone()) {
|
2016-03-29 20:06:42 -07:00
|
|
|
Ok(InferOk { obligations, .. }) => {
|
2016-04-04 12:41:05 -07:00
|
|
|
// FIXME(#32730) propagate obligations
|
2016-03-29 20:06:42 -07:00
|
|
|
assert!(obligations.is_empty());
|
|
|
|
|
}
|
2015-01-10 11:54:15 -05:00
|
|
|
Err(e) => {
|
2016-03-25 18:31:27 +01:00
|
|
|
span_bug!(
|
2015-01-10 11:54:15 -05:00
|
|
|
obligation.cause.span,
|
2016-03-25 18:31:27 +01:00
|
|
|
"Failed to unify `{:?}` and `{:?}` in projection: {}",
|
|
|
|
|
obligation,
|
|
|
|
|
projection,
|
|
|
|
|
e);
|
2015-01-10 11:54:15 -05:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
(projection.ty, vec!())
|
|
|
|
|
}
|
|
|
|
|
|
2016-04-29 06:00:23 +03:00
|
|
|
fn confirm_impl_candidate<'cx, 'gcx, 'tcx>(
|
|
|
|
|
selcx: &mut SelectionContext<'cx, 'gcx, 'tcx>,
|
2015-01-10 11:54:15 -05:00
|
|
|
obligation: &ProjectionTyObligation<'tcx>,
|
|
|
|
|
impl_vtable: VtableImplData<'tcx, PredicateObligation<'tcx>>)
|
|
|
|
|
-> (Ty<'tcx>, Vec<PredicateObligation<'tcx>>)
|
|
|
|
|
{
|
2015-12-29 13:37:34 -08:00
|
|
|
let VtableImplData { substs, nested, impl_def_id } = impl_vtable;
|
2015-01-10 11:54:15 -05:00
|
|
|
|
2016-02-16 10:36:47 -08:00
|
|
|
let tcx = selcx.tcx();
|
|
|
|
|
let trait_ref = obligation.predicate.trait_ref;
|
|
|
|
|
let assoc_ty = assoc_ty_def(selcx, impl_def_id, obligation.predicate.item_name);
|
|
|
|
|
|
|
|
|
|
match assoc_ty {
|
|
|
|
|
Some(node_item) => {
|
|
|
|
|
let ty = node_item.item.ty.unwrap_or_else(|| {
|
|
|
|
|
// This means that the impl is missing a definition for the
|
|
|
|
|
// associated type. This error will be reported by the type
|
|
|
|
|
// checker method `check_impl_items_against_trait`, so here we
|
|
|
|
|
// just return TyError.
|
|
|
|
|
debug!("confirm_impl_candidate: no associated type {:?} for {:?}",
|
|
|
|
|
node_item.item.name,
|
|
|
|
|
obligation.predicate.trait_ref);
|
|
|
|
|
tcx.types.err
|
|
|
|
|
});
|
2016-02-19 10:49:14 -08:00
|
|
|
let substs = translate_substs(selcx.infcx(), impl_def_id, substs, node_item.node);
|
2016-03-11 15:51:35 -08:00
|
|
|
(ty.subst(tcx, substs), nested)
|
2016-02-16 10:36:47 -08:00
|
|
|
}
|
|
|
|
|
None => {
|
2016-03-25 18:31:27 +01:00
|
|
|
span_bug!(obligation.cause.span,
|
|
|
|
|
"No associated type for {:?}",
|
|
|
|
|
trait_ref);
|
2016-02-16 10:36:47 -08:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Locate the definition of an associated type in the specialization hierarchy,
|
|
|
|
|
/// starting from the given impl.
|
|
|
|
|
///
|
|
|
|
|
/// Based on the "projection mode", this lookup may in fact only examine the
|
|
|
|
|
/// topmost impl. See the comments for `ProjectionMode` for more details.
|
2016-04-29 06:00:23 +03:00
|
|
|
fn assoc_ty_def<'cx, 'gcx, 'tcx>(
|
|
|
|
|
selcx: &SelectionContext<'cx, 'gcx, 'tcx>,
|
|
|
|
|
impl_def_id: DefId,
|
|
|
|
|
assoc_ty_name: ast::Name)
|
|
|
|
|
-> Option<specialization_graph::NodeItem<Rc<ty::AssociatedType<'tcx>>>>
|
2016-02-16 10:36:47 -08:00
|
|
|
{
|
|
|
|
|
let trait_def_id = selcx.tcx().impl_trait_ref(impl_def_id).unwrap().def_id;
|
|
|
|
|
|
2016-03-08 15:23:52 -08:00
|
|
|
if selcx.projection_mode().is_topmost() {
|
2016-02-16 10:36:47 -08:00
|
|
|
let impl_node = specialization_graph::Node::Impl(impl_def_id);
|
|
|
|
|
for item in impl_node.items(selcx.tcx()) {
|
|
|
|
|
if let ty::TypeTraitItem(assoc_ty) = item {
|
|
|
|
|
if assoc_ty.name == assoc_ty_name {
|
|
|
|
|
return Some(specialization_graph::NodeItem {
|
|
|
|
|
node: specialization_graph::Node::Impl(impl_def_id),
|
|
|
|
|
item: assoc_ty,
|
|
|
|
|
});
|
2015-05-26 17:12:39 +03:00
|
|
|
}
|
|
|
|
|
}
|
2014-12-17 14:16:28 -05:00
|
|
|
}
|
2015-12-29 13:37:34 -08:00
|
|
|
None
|
2016-02-16 10:36:47 -08:00
|
|
|
} else {
|
|
|
|
|
selcx.tcx().lookup_trait_def(trait_def_id)
|
|
|
|
|
.ancestors(impl_def_id)
|
|
|
|
|
.type_defs(selcx.tcx(), assoc_ty_name)
|
|
|
|
|
.next()
|
|
|
|
|
}
|
2014-12-17 14:16:28 -05:00
|
|
|
}
|